diff --git a/.eslintrc b/.eslintrc index 8ff3679c2..c4daa00c9 100644 --- a/.eslintrc +++ b/.eslintrc @@ -1,14 +1,12 @@ { - "extends": "airbnb/legacy", + "extends": "mgcrea/legacy", "rules": { - "strict": [2, "global"], - "comma-dangle": [2, "never"], - "func-names": 0, + "consistent-return": 1, "padded-blocks": 0, - "no-use-before-define": [2, "nofunc"], - "no-unused-vars": 1, - "vars-on-top": 1, - "id-length": [2, {"properties": "never", "exceptions": ["i", "j", "k", "l", "x", "y", "_"]}] + "no-param-reassign": 1, + "no-shadow": 1, + "no-underscore-dangle": 0, + "space-before-function-paren": [2, "always"] }, "globals": { "angular": 1 diff --git a/.github/lock.yml b/.github/lock.yml new file mode 100644 index 000000000..46bb6dbb9 --- /dev/null +++ b/.github/lock.yml @@ -0,0 +1,39 @@ +# Configuration for Lock Threads - https://github.com/dessant/lock-threads + +# Number of days of inactivity before a closed issue or pull request is locked +daysUntilLock: 90 + +# Skip issues and pull requests created before a given timestamp. Timestamp must +# follow ISO 8601 (`YYYY-MM-DD`). Set to `false` to disable +skipCreatedBefore: false + +# Issues and pull requests with these labels will be ignored. Set to `[]` to disable +exemptLabels: [] + +# Label to add before locking, such as `outdated`. Set to `false` to disable +lockLabel: 'outdated' + +# Comment to post before locking. Set to `false` to disable +lockComment: > + This thread has been automatically locked since there has not been + any recent activity after it was closed. Please open a new issue for + related bugs. + +# Assign `resolved` as the reason for locking. Set to `false` to disable +setLockReason: true + +# Limit to only `issues` or `pulls` +only: issues + +# Optionally, specify configuration settings just for `issues` or `pulls` +# issues: +# exemptLabels: +# - help-wanted +# lockLabel: outdated + +# pulls: +# daysUntilLock: 30 + +# Repository to extend settings from +# _extends: repo + diff --git a/.github/stale.yml b/.github/stale.yml new file mode 100644 index 000000000..e556fa985 --- /dev/null +++ b/.github/stale.yml @@ -0,0 +1,17 @@ +# Number of days of inactivity before an issue becomes stale +daysUntilStale: 60 +# Number of days of inactivity before a stale issue is closed +daysUntilClose: 7 +# Issues with these labels will never be considered stale +exemptLabels: + - pinned + - security +# Label to use when marking an issue as stale +staleLabel: stale +# Comment to post when marking an issue as stale. Set to `false` to disable +markComment: > + This issue has been automatically marked as stale because it has not had + recent activity. It will be closed if no further activity occurs. Thank you + for your contributions. +# Comment to post when closing a stale issue. Set to `false` to disable +closeComment: false diff --git a/.gitignore b/.gitignore index ea26cbcda..551af5d8c 100644 --- a/.gitignore +++ b/.gitignore @@ -2,6 +2,7 @@ .tmp/ .DS_Store .idea +*.log *.sublime-project *.sublime-workspace bower_components/ @@ -11,4 +12,4 @@ node_modules/ /test/coverage/ !.gitignore !dist/ -*.nupkg \ No newline at end of file +*.nupkg diff --git a/.jshintrc b/.jshintrc deleted file mode 100644 index 352f27876..000000000 --- a/.jshintrc +++ /dev/null @@ -1,28 +0,0 @@ -{ - "node": true, - "browser": true, - "devel": false, - "esnext": true, - "bitwise": true, - "camelcase": true, - "curly": false, - "eqeqeq": true, - "immed": true, - "indent": 2, - "latedef": false, - "newcap": true, - "noarg": true, - "quotmark": "single", - "regexp": true, - "undef": true, - "unused": false, - "strict": true, - "trailing": true, - "smarttabs": true, - "boss": false, - "eqnull": false, - "expr": true, - "globals": { - "angular": false - } -} diff --git a/.npmignore b/.npmignore index e69de29bb..7255efa0b 100644 --- a/.npmignore +++ b/.npmignore @@ -0,0 +1 @@ +yarn.lock \ No newline at end of file diff --git a/.travis.yml b/.travis.yml index 49576c924..305cf62f7 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,7 +1,7 @@ language: node_js sudo: false node_js: - - "stable" + - "4" env: global: @@ -11,7 +11,6 @@ env: before_script: - export DISPLAY=:99.0 - - export PHANTOMJS_BIN=/usr/local/phantomjs/bin/phantomjs - export CHROME_BIN=chromium-browser - sh -e /etc/init.d/xvfb start - sleep 3 # give xvfb some time to start @@ -22,12 +21,16 @@ before_script: - date --rfc-2822 script: - - $(npm bin)/gulp jshint karma:travis + - npm run lint + - $(npm bin)/gulp karma:travis + - $(npm bin)/gulp karma:travis~1.6.0 + - $(npm bin)/gulp karma:travis~1.5.0 + - $(npm bin)/gulp karma:travis~1.4.0 - $(npm bin)/gulp karma:travis~1.3.0 - $(npm bin)/gulp karma:travis~1.2.0 after_script: - - cat test/coverage/*/lcov.info | codeclimate + - cat test/coverage/*/lcov.info | codeclimate-test-reporter addons: code_climate: diff --git a/README.md b/README.md index f64f35c32..ac13988f5 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ # [AngularStrap](http://mgcrea.github.io/angular-strap) -[![Bower Version](http://img.shields.io/bower/v/angular-strap.svg?style=flat)](https://github.com/mgcrea/angular-strap/releases) [![Build Status](http://img.shields.io/travis/mgcrea/angular-strap/master.svg?style=flat)](http://travis-ci.org/mgcrea/angular-strap) [![devDependency Status](http://img.shields.io/david/dev/mgcrea/angular-strap.svg?style=flat)](https://david-dm.org/mgcrea/angular-strap#info=devDependencies) [![Coverage Status](http://img.shields.io/codeclimate/coverage/github/mgcrea/angular-strap.svg?style=flat)](https://codeclimate.com/github/mgcrea/angular-strap) [![Climate Status](https://img.shields.io/codeclimate/github/mgcrea/angular-strap.svg?style=flat)](https://codeclimate.com/github/mgcrea/angular-strap) +[![Bower Version](http://img.shields.io/bower/v/angular-strap.svg?style=flat)](https://github.com/mgcrea/angular-strap/releases) [![status](https://img.shields.io/badge/status-deprecated-red.svg)](https://github.com/mgcrea/angular-strap/issues/2256) [![Build Status](http://img.shields.io/travis/mgcrea/angular-strap/master.svg?style=flat)](http://travis-ci.org/mgcrea/angular-strap) [![Banner](http://mgcrea.github.io/angular-strap/images/snippet.png)](http://mgcrea.github.io/angular-strap) @@ -8,11 +8,20 @@ AngularStrap is a set of native directives that enables seamless integration of - With no external dependency except the [Bootstrap CSS Styles](https://github.com/twbs/bootstrap/blob/master/dist/css/bootstrap.css), AngularStrap is lighter and faster than ever as it does leverage the power of ngAnimate from AngularJS 1.2+! -- AngularStrap is [tested](https://travis-ci.org/mgcrea/angular-strap) against the latest patch release of the 1.2, 1.3 and 1.4 branches. +- AngularStrap is [tested](https://travis-ci.org/mgcrea/angular-strap) against the latest patch release of the 1.2, 1.3, 1.4 and 1.5 branches. - If you don't want to use `ngAnimate`, you will have to include a tiny [ngAnimate mock](https://github.com/mgcrea/angular-strap/wiki/ngAnimate-mock). +## Looking for maintainers! + +We're currently looking for aspiring maintainers to tackle issues and pull requests! + +I (mgcrea) have not worked on any Angular.js v1 codebase for more than a year now. For me it's time to move on! + +If you have an ongoing project depending on AngularStrap, and would like to become a contributor, please chime in on [issue #2256](https://github.com/mgcrea/angular-strap/issues/2256) + + ## Documentation and examples + Check the [documentation](http://mgcrea.github.io/angular-strap) and [changelog](https://github.com/mgcrea/angular-strap/releases). @@ -76,7 +85,9 @@ $ bower install $ cd docs $ bower install $ cd .. -$ $(npm bin)/gulp test +$ npm test +# or +$ npm run test:watch ``` You can build the latest version using `gulp`. diff --git a/bower.json b/bower.json index 467abbce4..031b9bc40 100644 --- a/bower.json +++ b/bower.json @@ -1,7 +1,7 @@ { "name": "angular-strap", "description": "AngularStrap - AngularJS directives for Bootstrap", - "version": "2.3.5", + "version": "2.3.12", "keywords": [ "angular", "bootstrap" @@ -37,18 +37,18 @@ "angular": "^1.2.21" }, "devDependencies": { - "angular-animate": "~1.4", - "angular-i18n": "~1.4", - "angular-mocks": "~1.4", - "angular-motion": "^0.3.3", - "angular-route": "~1.4", - "angular-sanitize": "~1.4", - "angular-scenario": "~1.4", - "bootstrap": "^3.2.0", - "bootstrap-additions": "^0.2.3", - "fastclick": "^1.0.3", - "font-awesome": "^4.1.0", - "highlightjs": "^8.0.0", - "jquery": "^2.1.1" + "angular-animate": "~1.5", + "angular-i18n": "~1.5", + "angular-mocks": "~1.5", + "angular-motion": "^0.4.3", + "angular-route": "~1.5", + "angular-sanitize": "~1.5", + "angular-scenario": "~1.5", + "bootstrap": "^3.3.6", + "bootstrap-additions": "^0.3.1", + "fastclick": "^1.0.6", + "font-awesome": "^4.5.0", + "highlightjs": "^9.1.0", + "jquery": "^2.2.0" } } diff --git a/dist/angular-strap.compat.js b/dist/angular-strap.compat.js index 935421a20..7f7b7d493 100644 --- a/dist/angular-strap.compat.js +++ b/dist/angular-strap.compat.js @@ -1,12 +1,13 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ (function(window, document, undefined) { 'use strict'; + bsCompilerService.$inject = ["$q", "$http", "$injector", "$compile", "$controller", "$templateCache"]; angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions' ]).provider('$bsTypeahead', function() { var defaults = this.defaults = { animation: 'am-fade', @@ -27,7 +28,6 @@ trimValue: true }; this.$get = ["$window", "$rootScope", "$bsTooltip", "$$rAF", "$timeout", function($window, $rootScope, $tooltip, $$rAF, $timeout) { - var bodyEl = angular.element($window.document.body); function TypeaheadFactory(element, controller, config) { var $typeahead = {}; var options = angular.extend({}, defaults, config); @@ -71,6 +71,9 @@ scope.$resetMatches(); if (parentScope) parentScope.$digest(); scope.$emit(options.prefixEvent + '.select', value, index, $typeahead); + if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) { + options.onSelect(value, index, $typeahead); + } }; $typeahead.$isVisible = function() { if (!options.minLength || !controller) { @@ -79,18 +82,30 @@ return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength; }; $typeahead.$getIndex = function(value) { - var l = scope.$matches.length, i = l; - if (!l) return; - for (i = l; i--; ) { - if (scope.$matches[i].value === value) break; + var index; + for (index = scope.$matches.length; index--; ) { + if (angular.equals(scope.$matches[index].value, value)) break; } - if (i < 0) return; - return i; + return index; }; $typeahead.$onMouseDown = function(evt) { evt.preventDefault(); evt.stopPropagation(); }; + $typeahead.$$updateScrollTop = function(container, index) { + if (index > -1 && index < container.children.length) { + var active = container.children[index]; + var clientTop = active.offsetTop; + var clientBottom = active.offsetTop + active.clientHeight; + var highWatermark = container.scrollTop; + var lowWatermark = container.scrollTop + container.clientHeight; + if (clientBottom >= highWatermark && clientTop < highWatermark) { + container.scrollTop = Math.max(0, container.scrollTop - container.clientHeight); + } else if (clientBottom > lowWatermark) { + container.scrollTop = clientTop; + } + } + }; $typeahead.$onKeyDown = function(evt) { if (!/(38|40|13)/.test(evt.keyCode)) return; if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) { @@ -99,26 +114,37 @@ } if (evt.keyCode === 13 && scope.$matches.length) { $typeahead.select(scope.$activeIndex); - } else if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--; else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++; else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0; + } else if (evt.keyCode === 38 && scope.$activeIndex > 0) { + scope.$activeIndex--; + } else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) { + scope.$activeIndex++; + } else if (angular.isUndefined(scope.$activeIndex)) { + scope.$activeIndex = 0; + } + $typeahead.$$updateScrollTop($typeahead.$element[0], scope.$activeIndex); scope.$digest(); }; var show = $typeahead.show; $typeahead.show = function() { show(); $timeout(function() { - $typeahead.$element && $typeahead.$element.on('mousedown', $typeahead.$onMouseDown); - if (options.keyboard) { - element && element.on('keydown', $typeahead.$onKeyDown); + if ($typeahead.$element) { + $typeahead.$element.on('mousedown', $typeahead.$onMouseDown); + if (options.keyboard) { + if (element) element.on('keydown', $typeahead.$onKeyDown); + } } }, 0, false); }; var hide = $typeahead.hide; $typeahead.hide = function() { - $typeahead.$element && $typeahead.$element.off('mousedown', $typeahead.$onMouseDown); + if ($typeahead.$element) $typeahead.$element.off('mousedown', $typeahead.$onMouseDown); if (options.keyboard) { - element && element.off('keydown', $typeahead.$onKeyDown); + if (element) element.off('keydown', $typeahead.$onKeyDown); + } + if (!options.autoSelect) { + $typeahead.activate(-1); } - if (!options.autoSelect) $typeahead.activate(-1); hide(); }; return $typeahead; @@ -135,9 +161,8 @@ return array.then(function(results) { return $filter('filter')(results, expression, comparator); }); - } else { - return $filter('filter')(array, expression, comparator); } + return $filter('filter')(array, expression, comparator); }; } ]).directive('bsTypeahead', ["$window", "$parse", "$q", "$bsTypeahead", "$bsParseOptions", function($window, $parse, $q, $typeahead, $parseOptions) { var defaults = $typeahead.defaults; @@ -145,6 +170,7 @@ restrict: 'EAC', require: 'ngModel', link: function postLink(scope, element, attr, controller) { + element.off('change'); var options = { scope: scope }; @@ -152,16 +178,24 @@ if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; - angular.forEach([ 'html', 'container', 'trimValue' ], function(key) { + angular.forEach([ 'html', 'container', 'trimValue', 'filter' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); if (!element.attr('autocomplete')) element.attr('autocomplete', 'off'); - var filter = options.filter || defaults.filter; + var filter = angular.isDefined(options.filter) ? options.filter : defaults.filter; var limit = options.limit || defaults.limit; var comparator = options.comparator || defaults.comparator; var bsOptions = attr.bsOptions; - if (filter) bsOptions += ' | ' + filter + ':$viewValue'; - if (comparator) bsOptions += ':' + comparator; + if (filter) { + bsOptions += ' | ' + filter + ':$viewValue'; + if (comparator) bsOptions += ':' + comparator; + } if (limit) bsOptions += ' | limitTo:' + limit; var parsedOptions = $parseOptions(bsOptions); var typeahead = $typeahead(element, controller, options); @@ -191,7 +225,7 @@ if (displayValue) { return displayValue; } - if (modelValue && typeof modelValue !== 'object') { + if (angular.isDefined(modelValue) && typeof modelValue !== 'object') { return modelValue; } return ''; @@ -201,10 +235,13 @@ return element.val(''); } var index = typeahead.$getIndex(controller.$modelValue); - var selected = angular.isDefined(index) ? typeahead.$scope.$matches[index].label : controller.$viewValue; + var selected = index !== -1 ? typeahead.$scope.$matches[index].label : controller.$viewValue; selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected; var value = selected ? selected.toString().replace(/<(?:.|\n)*?>/gm, '') : ''; + var ss = element[0].selectionStart; + var sd = element[0].selectionEnd; element.val(options.trimValue === false ? value : value.trim()); + element[0].setSelectionRange(ss, sd); }; scope.$on('$destroy', function() { if (typeahead) typeahead.destroy(); @@ -225,7 +262,7 @@ placement: 'top', templateUrl: 'tooltip/tooltip.tpl.html', template: '', - contentTemplate: false, + titleTemplate: false, trigger: 'hover focus', keyboard: false, html: false, @@ -235,15 +272,16 @@ delay: 0, autoClose: false, bsEnabled: true, + mouseDownPreventDefault: true, + mouseDownStopPropagation: true, viewport: { selector: 'body', padding: 0 } }; this.$get = ["$window", "$rootScope", "$bsCompiler", "$q", "$templateCache", "$http", "$animate", "$sce", "bsDimensions", "$$rAF", "$timeout", function($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) { - var trim = String.prototype.trim; - var isTouch = 'createTouch' in $window.document; - var htmlReplaceRegExp = /ng-bind="/gi; + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); + var isTouch = 'createTouch' in $window.document && isNative; var $body = angular.element($window.document); function TooltipFactory(element, config) { var $tooltip = {}; @@ -283,8 +321,12 @@ }); }; $tooltip.$isShown = scope.$isShown = false; - var timeout, hoverState; - var compileData, tipElement, tipContainer, tipScope; + var timeout; + var hoverState; + var compileData; + var tipElement; + var tipContainer; + var tipScope; promise.then(function(data) { compileData = data; $tooltip.init(); @@ -309,7 +351,11 @@ } if (options.show) { scope.$$postDigest(function() { - options.trigger === 'focus' ? element[0].focus() : $tooltip.show(); + if (options.trigger === 'focus') { + element[0].focus(); + } else { + $tooltip.show(); + } }); } }; @@ -331,7 +377,11 @@ $tooltip.show = function() { if (!options.bsEnabled || $tooltip.$isShown) return; scope.$emit(options.prefixEvent + '.show.before', $tooltip); - var parent, after; + if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) { + options.onBeforeShow($tooltip); + } + var parent; + var after; if (options.container) { parent = tipContainer; if (tipContainer[0].lastChild) { @@ -356,7 +406,11 @@ if (options.animation) tipElement.addClass(options.animation); if (options.type) tipElement.addClass(options.prefixClass + '-' + options.type); if (options.customClass) tipElement.addClass(options.customClass); - after ? after.after(tipElement) : parent.prepend(tipElement); + if (after) { + after.after(tipElement); + } else { + parent.prepend(tipElement); + } $tooltip.$isShown = scope.$isShown = true; safeDigest(scope); $tooltip.$applyPlacement(); @@ -383,6 +437,9 @@ }; function enterAnimateCallback() { scope.$emit(options.prefixEvent + '.show', $tooltip); + if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) { + options.onShow($tooltip); + } } $tooltip.leave = function() { clearTimeout(timeout); @@ -401,12 +458,17 @@ $tooltip.hide = function(blur) { if (!$tooltip.$isShown) return; scope.$emit(options.prefixEvent + '.hide.before', $tooltip); + if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) { + options.onBeforeHide($tooltip); + } _blur = blur; _tipToHide = tipElement; - if (angular.version.minor <= 2) { - $animate.leave(tipElement, leaveAnimateCallback); - } else { - $animate.leave(tipElement).then(leaveAnimateCallback); + if (tipElement !== null) { + if (angular.version.minor <= 2) { + $animate.leave(tipElement, leaveAnimateCallback); + } else { + $animate.leave(tipElement).then(leaveAnimateCallback); + } } $tooltip.$isShown = scope.$isShown = false; safeDigest(scope); @@ -419,6 +481,9 @@ }; function leaveAnimateCallback() { scope.$emit(options.prefixEvent + '.hide', $tooltip); + if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) { + options.onHide($tooltip); + } if (tipElement === _tipToHide) { if (_blur && options.trigger === 'focus') { return element[0].blur(); @@ -426,8 +491,15 @@ destroyTipElement(); } } - $tooltip.toggle = function() { - $tooltip.$isShown ? $tooltip.leave() : $tooltip.enter(); + $tooltip.toggle = function(evt) { + if (evt) { + evt.preventDefault(); + } + if ($tooltip.$isShown) { + $tooltip.leave(); + } else { + $tooltip.enter(); + } }; $tooltip.focus = function() { tipElement[0].focus(); @@ -440,25 +512,29 @@ }; $tooltip.$applyPlacement = function() { if (!tipElement) return; - var placement = options.placement, autoToken = /\s?auto?\s?/i, autoPlace = autoToken.test(placement); + var placement = options.placement; + var autoToken = /\s?auto?\s?/i; + var autoPlace = autoToken.test(placement); if (autoPlace) { placement = placement.replace(autoToken, '') || defaults.placement; } tipElement.addClass(options.placement); - var elementPosition = getPosition(), tipWidth = tipElement.prop('offsetWidth'), tipHeight = tipElement.prop('offsetHeight'); + var elementPosition = getPosition(); + var tipWidth = tipElement.prop('offsetWidth'); + var tipHeight = tipElement.prop('offsetHeight'); $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport); if (autoPlace) { var originalPlacement = placement; var viewportPosition = getPosition($tooltip.$viewport); - if (originalPlacement.indexOf('bottom') >= 0 && elementPosition.bottom + tipHeight > viewportPosition.bottom) { + if (/bottom/.test(originalPlacement) && elementPosition.bottom + tipHeight > viewportPosition.bottom) { placement = originalPlacement.replace('bottom', 'top'); - } else if (originalPlacement.indexOf('top') >= 0 && elementPosition.top - tipHeight < viewportPosition.top) { + } else if (/top/.test(originalPlacement) && elementPosition.top - tipHeight < viewportPosition.top) { placement = originalPlacement.replace('top', 'bottom'); } - if ((originalPlacement === 'right' || originalPlacement === 'bottom-left' || originalPlacement === 'top-left') && elementPosition.right + tipWidth > viewportPosition.width) { - placement = originalPlacement === 'right' ? 'left' : placement.replace('left', 'right'); - } else if ((originalPlacement === 'left' || originalPlacement === 'bottom-right' || originalPlacement === 'top-right') && elementPosition.left - tipWidth < viewportPosition.left) { - placement = originalPlacement === 'left' ? 'right' : placement.replace('right', 'left'); + if (/left/.test(originalPlacement) && elementPosition.left - tipWidth < viewportPosition.left) { + placement = placement.replace('left', 'right'); + } else if (/right/.test(originalPlacement) && elementPosition.right + tipWidth > viewportPosition.width) { + placement = placement.replace('right', 'left'); } tipElement.removeClass(originalPlacement).addClass(placement); } @@ -478,19 +554,29 @@ } }; $tooltip.$onFocusElementMouseDown = function(evt) { - evt.preventDefault(); - evt.stopPropagation(); - $tooltip.$isShown ? element[0].blur() : element[0].focus(); + if (options.mouseDownPreventDefault) { + evt.preventDefault(); + } + if (options.mouseDownStopPropagation) { + evt.stopPropagation(); + } + if ($tooltip.$isShown) { + element[0].blur(); + } else { + element[0].focus(); + } }; function bindTriggerEvents() { var triggers = options.trigger.split(' '); angular.forEach(triggers, function(trigger) { - if (trigger === 'click') { - element.on('click', $tooltip.toggle); + if (trigger === 'click' || trigger === 'contextmenu') { + element.on(trigger, $tooltip.toggle); } else if (trigger !== 'manual') { element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter); element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave); - nodeName === 'button' && trigger !== 'hover' && element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + if (nodeName === 'button' && trigger !== 'hover') { + element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + } } }); } @@ -498,12 +584,14 @@ var triggers = options.trigger.split(' '); for (var i = triggers.length; i--; ) { var trigger = triggers[i]; - if (trigger === 'click') { - element.off('click', $tooltip.toggle); + if (trigger === 'click' || trigger === 'contextmenu') { + element.off(trigger, $tooltip.toggle); } else if (trigger !== 'manual') { element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter); element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave); - nodeName === 'button' && trigger !== 'hover' && element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + if (nodeName === 'button' && trigger !== 'hover') { + element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + } } } } @@ -541,7 +629,8 @@ } function getPosition($element) { $element = $element || (options.target || element); - var el = $element[0], isBody = el.tagName === 'BODY'; + var el = $element[0]; + var isBody = el.tagName === 'BODY'; var elRect = el.getBoundingClientRect(); var rect = {}; for (var p in elRect) { @@ -556,9 +645,11 @@ var elOffset = isBody ? { top: 0, left: 0 - } : dimensions.offset(el), scroll = { + } : dimensions.offset(el); + var scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0 - }, outerDims = isBody ? { + }; + var outerDims = isBody ? { width: document.documentElement.clientWidth, height: $window.innerHeight } : null; @@ -607,22 +698,33 @@ case 'right': offset.left = position.left + position.width - actualWidth; + break; + + default: + break; } } else if (split[0] === 'left' || split[0] === 'right') { switch (split[1]) { case 'top': - offset.top = position.top - actualHeight; + offset.top = position.top - actualHeight + position.height; break; case 'bottom': - offset.top = position.top + position.height; + offset.top = position.top; + break; + + default: + break; } } return offset; } function applyPlacement(offset, placement) { - var tip = tipElement[0], width = tip.offsetWidth, height = tip.offsetHeight; - var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10), marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10); + var tip = tipElement[0]; + var width = tip.offsetWidth; + var height = tip.offsetHeight; + var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10); + var marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10); if (isNaN(marginTop)) marginTop = 0; if (isNaN(marginLeft)) marginLeft = 0; offset.top = offset.top + marginTop; @@ -636,7 +738,8 @@ }); } }, offset), 0); - var actualWidth = tip.offsetWidth, actualHeight = tip.offsetHeight; + var actualWidth = tip.offsetWidth; + var actualHeight = tip.offsetHeight; if (placement === 'top' && actualHeight !== height) { offset.top = offset.top + height - actualHeight; } @@ -649,7 +752,9 @@ } dimensions.setOffset(tip, offset); if (/top|right|bottom|left/.test(placement)) { - var isVertical = /top|bottom/.test(placement), arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight, arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'; + var isVertical = /top|bottom/.test(placement); + var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight; + var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'; replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical); } } @@ -711,35 +816,39 @@ function findElement(query, element) { return angular.element((element || document).querySelectorAll(query)); } - var fetchPromises = {}; - function fetchTemplate(template) { - if (fetchPromises[template]) return fetchPromises[template]; - return fetchPromises[template] = $http.get(template, { - cache: $templateCache - }).then(function(res) { - return res.data; - }); - } return TooltipFactory; } ]; - }).directive('bsTooltip', ["$window", "$location", "$sce", "$bsTooltip", "$$rAF", function($window, $location, $sce, $tooltip, $$rAF) { + }).directive('bsTooltip', ["$window", "$location", "$sce", "$parse", "$bsTooltip", "$$rAF", function($window, $location, $sce, $parse, $tooltip, $$rAF) { return { restrict: 'EAC', scope: true, link: function postLink(scope, element, attr, transclusion) { + var tooltip; var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'titleTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } + }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); var dataTarget = element.attr('data-target'); if (angular.isDefined(dataTarget)) { - if (falseValueRegExp.test(dataTarget)) options.target = false; else options.target = dataTarget; + if (falseValueRegExp.test(dataTarget)) { + options.target = false; + } else { + options.target = dataTarget; + } } if (!scope.hasOwnProperty('title')) { scope.title = ''; @@ -748,36 +857,61 @@ if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) { var oldValue = scope.title; scope.title = $sce.trustAsHtml(newValue); - angular.isDefined(oldValue) && $$rAF(function() { - tooltip && tooltip.$applyPlacement(); - }); + if (angular.isDefined(oldValue)) { + $$rAF(function() { + if (tooltip) tooltip.$applyPlacement(); + }); + } } }); - attr.bsTooltip && scope.$watch(attr.bsTooltip, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.title = newValue; + attr.$observe('disabled', function(newValue) { + if (newValue && tooltip.$isShown) { + tooltip.hide(); } - angular.isDefined(oldValue) && $$rAF(function() { - tooltip && tooltip.$applyPlacement(); - }); - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i); - newValue === true ? tooltip.show() : tooltip.hide(); - }); - attr.bsEnabled && scope.$watch(attr.bsEnabled, function(newValue, oldValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i); - newValue === false ? tooltip.setEnabled(false) : tooltip.setEnabled(true); }); - attr.viewport && scope.$watch(attr.viewport, function(newValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - tooltip.setViewport(newValue); - }); - var tooltip = $tooltip(element, options); + if (attr.bsTooltip) { + scope.$watch(attr.bsTooltip, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.title = newValue; + } + if (angular.isDefined(oldValue)) { + $$rAF(function() { + if (tooltip) tooltip.$applyPlacement(); + }); + } + }, true); + } + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i); + if (newValue === true) { + tooltip.show(); + } else { + tooltip.hide(); + } + }); + } + if (attr.bsEnabled) { + scope.$watch(attr.bsEnabled, function(newValue, oldValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i); + if (newValue === false) { + tooltip.setEnabled(false); + } else { + tooltip.setEnabled(true); + } + }); + } + if (attr.viewport) { + scope.$watch(attr.viewport, function(newValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + tooltip.setViewport(newValue); + }); + } + tooltip = $tooltip(element, options); scope.$on('$destroy', function() { if (tooltip) tooltip.destroy(); options = null; @@ -789,6 +923,7 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$bsTimepicker', function() { var defaults = this.defaults = { animation: 'am-fade', + defaultDate: 'auto', prefixClass: 'timepicker', placement: 'bottom-left', templateUrl: 'timepicker/timepicker.tpl.html', @@ -815,7 +950,7 @@ arrowBehavior: 'pager' }; this.$get = ["$window", "$document", "$rootScope", "$sce", "$bsDateFormatter", "$bsTooltip", "$timeout", function($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) { - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; if (!defaults.lang) { defaults.lang = $dateFormatter.getDefaultLocale(); @@ -844,7 +979,12 @@ millisecond: startDate.getMilliseconds() }; var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang); - var hoursFormat = $dateFormatter.hoursFormat(format), timeSeparator = $dateFormatter.timeSeparator(format), minutesFormat = $dateFormatter.minutesFormat(format), secondsFormat = $dateFormatter.secondsFormat(format), showSeconds = $dateFormatter.showSeconds(format), showAM = $dateFormatter.showAM(format); + var hoursFormat = $dateFormatter.hoursFormat(format); + var timeSeparator = $dateFormatter.timeSeparator(format); + var minutesFormat = $dateFormatter.minutesFormat(format); + var secondsFormat = $dateFormatter.secondsFormat(format); + var showSeconds = $dateFormatter.showSeconds(format); + var showAM = $dateFormatter.showAM(format); scope.$iconUp = options.iconUp; scope.$iconDown = options.iconDown; scope.$select = function(date, index) { @@ -871,7 +1011,9 @@ } }; $timepicker.select = function(date, index, keep) { - if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) controller.$dateValue = new Date(1970, 0, 1); + if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) { + controller.$dateValue = options.defaultDate === 'today' ? new Date() : new Date(1970, 0, 1); + } if (!angular.isDate(date)) date = new Date(date); if (index === 0) controller.$dateValue.setHours(date.getHours()); else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes()); else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds()); controller.$setViewValue(angular.copy(controller.$dateValue)); @@ -892,8 +1034,10 @@ controller.$render(); }; $timepicker.$build = function() { - var i, midIndex = scope.midIndex = parseInt(options.length / 2, 10); - var hours = [], hour; + var i; + var midIndex = scope.midIndex = parseInt(options.length / 2, 10); + var hours = []; + var hour; for (i = 0; i < options.length; i++) { hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep); hours.push({ @@ -903,7 +1047,8 @@ disabled: $timepicker.$isDisabled(hour, 0) }); } - var minutes = [], minute; + var minutes = []; + var minute; for (i = 0; i < options.length; i++) { minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep); minutes.push({ @@ -913,7 +1058,8 @@ disabled: $timepicker.$isDisabled(minute, 1) }); } - var seconds = [], second; + var seconds = []; + var second; for (i = 0; i < options.length; i++) { second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep); seconds.push({ @@ -1019,9 +1165,12 @@ return; } var newDate = new Date($timepicker.$date); - var hours = newDate.getHours(), hoursLength = formatDate(newDate, hoursFormat).length; - var minutes = newDate.getMinutes(), minutesLength = formatDate(newDate, minutesFormat).length; - var seconds = newDate.getSeconds(), secondsLength = formatDate(newDate, secondsFormat).length; + var hours = newDate.getHours(); + var hoursLength = formatDate(newDate, hoursFormat).length; + var minutes = newDate.getMinutes(); + var minutesLength = formatDate(newDate, minutesFormat).length; + var seconds = newDate.getSeconds(); + var secondsLength = formatDate(newDate, secondsFormat).length; var sepLength = 1; var lateralMove = /(37|39)/.test(evt.keyCode); var count = 2 + showSeconds * 1 + showAM * 1; @@ -1097,18 +1246,18 @@ if (!isTouch && element.attr('readonly') || element.attr('disabled')) return; _show(); $timeout(function() { - $timepicker.$element && $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); + if ($timepicker.$element) $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); if (options.keyboard) { - element && element.on('keydown', $timepicker.$onKeyDown); + if (element) element.on('keydown', $timepicker.$onKeyDown); } }, 0, false); }; var _hide = $timepicker.hide; $timepicker.hide = function(blur) { if (!$timepicker.$isShown) return; - $timepicker.$element && $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); + if ($timepicker.$element) $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); if (options.keyboard) { - element && element.off('keydown', $timepicker.$onKeyDown); + if (element) element.off('keydown', $timepicker.$onKeyDown); } _hide(blur); }; @@ -1119,7 +1268,7 @@ } ]; }).directive('bsTimepicker', ["$window", "$parse", "$q", "$bsDateFormatter", "$bsDateParser", "$bsTimepicker", function($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) { var defaults = $timepicker.defaults; - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); return { restrict: 'EAC', require: 'ngModel', @@ -1127,17 +1276,20 @@ var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent', 'defaultDate' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'autoclose', 'useNative', 'roundDisplay' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } }); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!timepicker || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i); - newValue === true ? timepicker.show() : timepicker.hide(); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm'; var timepicker = $timepicker(element, controller, options); @@ -1146,16 +1298,29 @@ var formatDate = function(date, format, timezone) { return $dateFormatter.formatDate(date, format, lang, timezone); }; + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!timepicker || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i); + if (newValue === true) { + timepicker.show(); + } else { + timepicker.hide(); + } + }); + } var dateParser = $dateParser({ format: options.timeFormat, lang: lang }); angular.forEach([ 'minTime', 'maxTime' ], function(key) { - angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) { - timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue); - !isNaN(timepicker.$options[key]) && timepicker.$build(); - validateAgainstMinMaxTime(controller.$dateValue); - }); + if (angular.isDefined(attr[key])) { + attr.$observe(key, function(newValue) { + timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue); + if (!isNaN(timepicker.$options[key])) timepicker.$build(); + validateAgainstMinMaxTime(controller.$dateValue); + }); + } }); scope.$watch(attr.ngModel, function(newValue, oldValue) { timepicker.update(controller.$dateValue); @@ -1183,9 +1348,8 @@ if (!parsedTime || isNaN(parsedTime.getTime())) { controller.$setValidity('date', false); return undefined; - } else { - validateAgainstMinMaxTime(parsedTime); } + validateAgainstMinMaxTime(parsedTime); if (options.timeType === 'string') { date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true); return formatDate(date, options.modelTimeFormat || options.timeFormat); @@ -1197,9 +1361,8 @@ return date.getTime() / 1e3; } else if (options.timeType === 'iso') { return date.toISOString(); - } else { - return new Date(date); } + return new Date(date); }); controller.$formatters.push(function(modelValue) { var date; @@ -1331,7 +1494,6 @@ require: [ '^?ngModel', '^bsTabs' ], scope: true, link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsTabsCtrl = controllers[1]; element.addClass('tab-pane'); attrs.$observe('title', function(newValue, oldValue) { @@ -1380,11 +1542,11 @@ noneText: 'None', maxLength: 3, maxLengthHtml: 'selected', - iconCheckmark: 'glyphicon glyphicon-ok' + iconCheckmark: 'glyphicon glyphicon-ok', + toggle: false }; this.$get = ["$window", "$document", "$rootScope", "$bsTooltip", "$timeout", function($window, $document, $rootScope, $tooltip, $timeout) { - var bodyEl = angular.element($window.document.body); - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; function SelectFactory(element, controller, config) { var $select = {}; @@ -1438,7 +1600,11 @@ }; $select.activate = function(index) { if (options.multiple) { - $select.$isActive(index) ? scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1) : scope.$activeIndex.push(index); + if ($select.$isActive(index)) { + scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1); + } else { + scope.$activeIndex.push(index); + } if (options.sort) scope.$activeIndex.sort(function(a, b) { return a - b; }); @@ -1448,6 +1614,9 @@ return scope.$activeIndex; }; $select.select = function(index) { + if (angular.isUndefined(index) || index < 0 || index >= scope.$matches.length) { + return; + } var value = scope.$matches[index].value; scope.$apply(function() { $select.activate(index); @@ -1459,11 +1628,18 @@ return scope.$matches[index].value; })); } else { - controller.$setViewValue(value); + if (options.toggle) { + controller.$setViewValue(value === controller.$modelValue ? undefined : value); + } else { + controller.$setViewValue(value); + } $select.hide(); } }); scope.$emit(options.prefixEvent + '.select', value, index, $select); + if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) { + options.onSelect(value, index, $select); + } }; $select.$updateActiveIndex = function() { if (options.multiple) { @@ -1491,25 +1667,36 @@ $select.$isActive = function(index) { if (options.multiple) { return scope.$activeIndex.indexOf(index) !== -1; - } else { - return scope.$activeIndex === index; } + return scope.$activeIndex === index; }; $select.$getIndex = function(value) { - var l = scope.$matches.length, i = l; - if (!l) return; - for (i = l; i--; ) { - if (scope.$matches[i].value === value) break; + var index; + for (index = scope.$matches.length; index--; ) { + if (angular.equals(scope.$matches[index].value, value)) break; } - if (i < 0) return; - return i; + return index; }; $select.$onMouseDown = function(evt) { evt.preventDefault(); evt.stopPropagation(); if (isTouch) { var targetEl = angular.element(evt.target); - targetEl.triggerHandler('click'); + var anchor; + if (evt.target.nodeName !== 'A') { + var anchorCandidate = targetEl.parent(); + while (!anchor && anchorCandidate.length > 0) { + if (anchorCandidate[0].nodeName === 'A') { + anchor = anchorCandidate; + } + anchorCandidate = anchorCandidate.parent(); + } + } + if (anchor) { + angular.element(anchor).triggerHandler('click'); + } else { + targetEl.triggerHandler('click'); + } } }; $select.$onKeyDown = function(evt) { @@ -1579,16 +1766,28 @@ scope: scope, placeholder: defaults.placeholder }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent', 'toggle' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'allNoneButtons', 'sort' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } + }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); var dataMultiple = element.attr('data-multiple'); if (angular.isDefined(dataMultiple)) { - if (falseValueRegExp.test(dataMultiple)) options.multiple = false; else options.multiple = dataMultiple; + if (falseValueRegExp.test(dataMultiple)) { + options.multiple = false; + } else { + options.multiple = dataMultiple; + } } if (element[0].nodeName.toLowerCase() === 'select') { var inputEl = element; @@ -1602,22 +1801,23 @@ element[0].addEventListener('blur', select.$selectScrollFix); } var watchedOptions = parsedOptions.$match[7].replace(/\|.+/, '').trim(); - scope.$watchCollection(watchedOptions, function(newValue, oldValue) { + scope.$watch(watchedOptions, function(newValue, oldValue) { parsedOptions.valuesFn(scope, controller).then(function(values) { select.update(values); controller.$render(); }); - }); + }, true); scope.$watch(attr.ngModel, function(newValue, oldValue) { select.$updateActiveIndex(); controller.$render(); }, true); controller.$render = function() { - var selected, index; + var selected; + var index; if (options.multiple && angular.isArray(controller.$modelValue)) { selected = controller.$modelValue.map(function(value) { index = select.$getIndex(value); - return angular.isDefined(index) ? select.$scope.$matches[index].label : false; + return index !== -1 ? select.$scope.$matches[index].label : false; }).filter(angular.isDefined); if (selected.length > (options.maxLength || defaults.maxLength)) { selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml); @@ -1626,9 +1826,9 @@ } } else { index = select.$getIndex(controller.$modelValue); - selected = angular.isDefined(index) ? select.$scope.$matches[index].label : false; + selected = index !== -1 ? select.$scope.$matches[index].label : false; } - element.html((selected ? selected : options.placeholder) + (options.caretHtml ? options.caretHtml : defaults.caretHtml)); + element.html((selected || options.placeholder) + (options.caretHtml || defaults.caretHtml)); }; if (options.multiple) { controller.$isEmpty = function(value) { @@ -1668,7 +1868,8 @@ return spies[scrollId]; } var $scrollspy = {}; - var unbindViewContentLoaded, unbindIncludeContentLoaded; + var unbindViewContentLoaded; + var unbindIncludeContentLoaded; var trackedElements = $scrollspy.$trackedElements = []; var sortedElements = []; var activeTarget; @@ -1772,7 +1973,7 @@ break; } } - trackedElements = trackedElements.splice(toDelete, 1); + trackedElements.splice(toDelete, 1); }; $scrollspy.activate = function(i) { trackedElements[i].addClass('active'); @@ -1850,48 +2051,86 @@ restrict: 'EAC', scope: true, link: function postLink(scope, element, attr) { + var popover; var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent', 'bsEnabled' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'autoClose' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); var dataTarget = element.attr('data-target'); if (angular.isDefined(dataTarget)) { - if (falseValueRegExp.test(dataTarget)) options.target = false; else options.target = dataTarget; + if (falseValueRegExp.test(dataTarget)) { + options.target = false; + } else { + options.target = dataTarget; + } } angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - angular.isDefined(oldValue) && requestAnimationFrame(function() { - popover && popover.$applyPlacement(); + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + if (angular.isDefined(oldValue)) { + requestAnimationFrame(function() { + if (popover) popover.$applyPlacement(); + }); + } }); - }); - }); - attr.bsPopover && scope.$watch(attr.bsPopover, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; } - angular.isDefined(oldValue) && requestAnimationFrame(function() { - popover && popover.$applyPlacement(); - }); - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!popover || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i); - newValue === true ? popover.show() : popover.hide(); }); - attr.viewport && scope.$watch(attr.viewport, function(newValue) { - if (!popover || !angular.isDefined(newValue)) return; - popover.setViewport(newValue); - }); - var popover = $popover(element, options); + if (attr.bsPopover) { + scope.$watch(attr.bsPopover, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + if (angular.isDefined(oldValue)) { + requestAnimationFrame(function() { + if (popover) popover.$applyPlacement(); + }); + } + }, true); + } + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!popover || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i); + if (newValue === true) { + popover.show(); + } else { + popover.hide(); + } + }); + } + if (attr.bsEnabled) { + scope.$watch(attr.bsEnabled, function(newValue) { + if (!popover || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(popover),?/i); + if (newValue === false) { + popover.setEnabled(false); + } else { + popover.setEnabled(true); + } + }); + } + if (attr.viewport) { + scope.$watch(attr.viewport, function(newValue) { + if (!popover || !angular.isDefined(newValue)) return; + popover.setViewport(newValue); + }); + } + popover = $popover(element, options); scope.$on('$destroy', function() { if (popover) popover.destroy(); options = null; @@ -1945,6 +2184,7 @@ var defaults = this.defaults = { animation: 'am-fade', backdropAnimation: 'am-fade', + customClass: '', prefixClass: 'modal', prefixEvent: 'modal', placement: 'top', @@ -1956,13 +2196,21 @@ backdrop: true, keyboard: true, html: false, - show: true + show: true, + size: null, + zIndex: null }; this.$get = ["$window", "$rootScope", "$bsCompiler", "$animate", "$timeout", "$sce", "bsDimensions", function($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) { var forEach = angular.forEach; - var trim = String.prototype.trim; var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; var bodyElement = angular.element($window.document.body); + var backdropCount = 0; + var dialogBaseZindex = 1050; + var backdropBaseZindex = 1040; + var validSizes = { + lg: 'modal-lg', + sm: 'modal-sm' + }; function ModalFactory(config) { var $modal = {}; var options = $modal.$options = angular.extend({}, defaults, config); @@ -1971,6 +2219,10 @@ if (!options.element && !options.container) { options.container = 'body'; } + if (options.zIndex) { + dialogBaseZindex = parseInt(options.zIndex, 10); + backdropBaseZindex = dialogBaseZindex - 10; + } $modal.$id = options.id || options.element && options.element.attr('id') || ''; forEach([ 'title', 'content' ], function(key) { if (options[key]) scope[key] = $sce.trustAsHtml(options[key]); @@ -1991,15 +2243,16 @@ }); }; $modal.$isShown = scope.$isShown = false; - var compileData, modalElement, modalScope; + var compileData; + var modalElement; + var modalScope; var backdropElement = angular.element('
'); backdropElement.css({ position: 'fixed', top: '0px', left: '0px', bottom: '0px', - right: '0px', - 'z-index': 1038 + right: '0px' }); promise.then(function(data) { compileData = data; @@ -2022,7 +2275,8 @@ }; $modal.show = function() { if ($modal.$isShown) return; - var parent, after; + var parent; + var after; if (angular.isElement(options.container)) { parent = options.container; after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null; @@ -2038,12 +2292,30 @@ if (modalElement) destroyModalElement(); modalScope = $modal.$scope.$new(); modalElement = $modal.$element = compileData.link(modalScope, function(clonedElement, scope) {}); + if (options.backdrop) { + modalElement.css({ + 'z-index': dialogBaseZindex + backdropCount * 20 + }); + backdropElement.css({ + 'z-index': backdropBaseZindex + backdropCount * 20 + }); + backdropCount++; + } if (scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) { return; } + if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) { + options.onBeforeShow($modal); + } modalElement.css({ display: 'block' }).addClass(options.placement); + if (options.customClass) { + modalElement.addClass(options.customClass); + } + if (options.size && validSizes[options.size]) { + angular.element(findElement('.modal-dialog', modalElement[0])).addClass(validSizes[options.size]); + } if (options.animation) { if (options.backdrop) { backdropElement.addClass(options.backdropAnimation); @@ -2073,18 +2345,25 @@ }; function enterAnimateCallback() { scope.$emit(options.prefixEvent + '.show', $modal); + if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) { + options.onShow($modal); + } } $modal.hide = function() { if (!$modal.$isShown) return; if (scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) { return; } + if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) { + options.onBeforeHide($modal); + } if (angular.version.minor <= 2) { $animate.leave(modalElement, leaveAnimateCallback); } else { $animate.leave(modalElement).then(leaveAnimateCallback); } if (options.backdrop) { + backdropCount--; $animate.leave(backdropElement); } $modal.$isShown = scope.$isShown = false; @@ -2094,13 +2373,22 @@ }; function leaveAnimateCallback() { scope.$emit(options.prefixEvent + '.hide', $modal); - bodyElement.removeClass(options.prefixClass + '-open'); + if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) { + options.onHide($modal); + } + if (findElement('.modal').length <= 0) { + bodyElement.removeClass(options.prefixClass + '-open'); + } if (options.animation) { bodyElement.removeClass(options.prefixClass + '-with-' + options.animation); } } $modal.toggle = function() { - $modal.$isShown ? $modal.hide() : $modal.show(); + if ($modal.$isShown) { + $modal.hide(); + } else { + $modal.show(); + } }; $modal.focus = function() { modalElement[0].focus(); @@ -2137,7 +2425,11 @@ } function hideOnBackdropClick(evt) { if (evt.target !== evt.currentTarget) return; - options.backdrop === 'static' ? $modal.focus() : $modal.hide(); + if (options.backdrop === 'static') { + $modal.focus(); + } else { + $modal.hide(); + } } function preventEventDefault(evt) { evt.preventDefault(); @@ -2166,7 +2458,7 @@ } return ModalFactory; } ]; - }).directive('bsModal', ["$window", "$sce", "$bsModal", function($window, $sce, $modal) { + }).directive('bsModal', ["$window", "$sce", "$parse", "$bsModal", function($window, $sce, $parse, $modal) { return { restrict: 'EAC', scope: true, @@ -2176,25 +2468,38 @@ element: element, show: false }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass', 'customClass', 'modalClass', 'size', 'zIndex' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); + if (options.modalClass) { + options.customClass = options.modalClass; + } var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'backdrop', 'keyboard', 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); - attr.bsModal && scope.$watch(attr.bsModal, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + angular.forEach([ 'title', 'content' ], function(key) { + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsModal) { + scope.$watch(attr.bsModal, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var modal = $modal(options); element.on(attr.trigger || 'click', modal.toggle); scope.$on('$destroy', function() { @@ -2205,24 +2510,26 @@ } }; } ]); - angular.version.minor < 3 && angular.version.dot < 14 && angular.module('ng').factory('$$rAF', ["$window", "$timeout", function($window, $timeout) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.webkitRequestAnimationFrame || $window.mozRequestAnimationFrame; - var cancelAnimationFrame = $window.cancelAnimationFrame || $window.webkitCancelAnimationFrame || $window.mozCancelAnimationFrame || $window.webkitCancelRequestAnimationFrame; - var rafSupported = !!requestAnimationFrame; - var raf = rafSupported ? function(fn) { - var id = requestAnimationFrame(fn); - return function() { - cancelAnimationFrame(id); - }; - } : function(fn) { - var timer = $timeout(fn, 16.66, false); - return function() { - $timeout.cancel(timer); + if (angular.version.minor < 3 && angular.version.dot < 14) { + angular.module('ng').factory('$$rAF', ["$window", "$timeout", function($window, $timeout) { + var requestAnimationFrame = $window.requestAnimationFrame || $window.webkitRequestAnimationFrame || $window.mozRequestAnimationFrame; + var cancelAnimationFrame = $window.cancelAnimationFrame || $window.webkitCancelAnimationFrame || $window.mozCancelAnimationFrame || $window.webkitCancelRequestAnimationFrame; + var rafSupported = !!requestAnimationFrame; + var raf = rafSupported ? function(fn) { + var id = requestAnimationFrame(fn); + return function() { + cancelAnimationFrame(id); + }; + } : function(fn) { + var timer = $timeout(fn, 16.66, false); + return function() { + $timeout.cancel(timer); + }; }; - }; - raf.supported = rafSupported; - return raf; - } ]); + raf.supported = rafSupported; + return raf; + } ]); + } angular.module('mgcrea.ngStrap.helpers.parseOptions', []).provider('$bsParseOptions', function() { var defaults = this.defaults = { regexp: /^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/ @@ -2232,11 +2539,20 @@ var $parseOptions = {}; var options = angular.extend({}, defaults, config); $parseOptions.$values = []; - var match, displayFn, valueName, keyName, groupByFn, valueFn, valuesFn; + var match; + var displayFn; + var valueName; + var keyName; + var groupByFn; + var valueFn; + var valuesFn; $parseOptions.init = function() { $parseOptions.$match = match = attr.match(options.regexp); - displayFn = $parse(match[2] || match[1]), valueName = match[4] || match[6], keyName = match[5], - groupByFn = $parse(match[3] || ''), valueFn = $parse(match[2] ? match[1] : valueName), + displayFn = $parse(match[2] || match[1]); + valueName = match[4] || match[6]; + keyName = match[5]; + groupByFn = $parse(match[3] || ''); + valueFn = $parse(match[2] ? match[1] : valueName); valuesFn = $parse(match[7]); }; $parseOptions.valuesFn = function(scope, controller) { @@ -2255,7 +2571,9 @@ }; function parseValues(values, scope) { return values.map(function(match, index) { - var locals = {}, label, value; + var locals = {}; + var label; + var value; locals[valueName] = match; label = displayFn(scope, locals); value = valueFn(scope, locals); @@ -2272,8 +2590,7 @@ return ParseOptionsFactory; } ]; }); - angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('bsDimensions', ["$document", "$window", function($document, $window) { - var jqLite = angular.element; + angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('bsDimensions', function() { var fn = {}; var nodeName = fn.nodeName = function(element, name) { return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase(); @@ -2300,7 +2617,16 @@ }; }; fn.setOffset = function(element, options, i) { - var curPosition, curLeft, curCSSTop, curTop, curOffset, curCSSLeft, calculatePosition, position = fn.css(element, 'position'), curElem = angular.element(element), props = {}; + var curPosition; + var curLeft; + var curCSSTop; + var curTop; + var curOffset; + var curCSSLeft; + var calculatePosition; + var position = fn.css(element, 'position'); + var curElem = angular.element(element); + var props = {}; if (position === 'static') { element.style.position = 'relative'; } @@ -2338,17 +2664,19 @@ var offsetParentRect = { top: 0, left: 0 - }, offsetParentElement, offset; + }; + var offsetParentEl; + var offset; if (fn.css(element, 'position') === 'fixed') { offset = element.getBoundingClientRect(); } else { - offsetParentElement = offsetParent(element); + offsetParentEl = offsetParentElement(element); offset = fn.offset(element); - if (!nodeName(offsetParentElement, 'html')) { - offsetParentRect = fn.offset(offsetParentElement); + if (!nodeName(offsetParentEl, 'html')) { + offsetParentRect = fn.offset(offsetParentEl); } - offsetParentRect.top += fn.css(offsetParentElement, 'borderTopWidth', true); - offsetParentRect.left += fn.css(offsetParentElement, 'borderLeftWidth', true); + offsetParentRect.top += fn.css(offsetParentEl, 'borderTopWidth', true); + offsetParentRect.left += fn.css(offsetParentEl, 'borderLeftWidth', true); } return { width: element.offsetWidth, @@ -2357,7 +2685,7 @@ left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true) }; }; - var offsetParent = function offsetParentElement(element) { + function offsetParentElement(element) { var docElement = element.ownerDocument; var offsetParent = element.offsetParent || docElement; if (nodeName(offsetParent, '#document')) return docElement.documentElement; @@ -2365,7 +2693,7 @@ offsetParent = offsetParent.offsetParent; } return offsetParent || docElement.documentElement; - }; + } fn.height = function(element, outer) { var value = element.offsetHeight; if (outer) { @@ -2385,12 +2713,14 @@ return value; }; return fn; - } ]); + }); angular.module('mgcrea.ngStrap.helpers.debounce', []).factory('bsDebounce', ["$timeout", function($timeout) { return function(func, wait, immediate) { var timeout = null; return function() { - var context = this, args = arguments, callNow = immediate && !timeout; + var context = this; + var args = arguments; + var callNow = immediate && !timeout; if (timeout) { $timeout.cancel(timeout); } @@ -2409,9 +2739,10 @@ } ]).factory('bsThrottle', ["$timeout", function($timeout) { return function(func, wait, options) { var timeout = null; - options || (options = {}); + if (!options) options = {}; return function() { - var context = this, args = arguments; + var context = this; + var args = arguments; if (!timeout) { if (options.leading !== false) { func.apply(context, args); @@ -2479,7 +2810,8 @@ return !isNaN(parseFloat(n)) && isFinite(n); } function indexOfCaseInsensitive(array, value) { - var len = array.length, str = value.toString().toLowerCase(); + var len = array.length; + var str = value.toString().toLowerCase(); for (var i = 0; i < len; i++) { if (array[i].toLowerCase() === str) { return i; @@ -2556,7 +2888,8 @@ return 1 * value <= 50 && value.length === 2 ? this.setFullYear(2e3 + 1 * value) : this.setFullYear(1 * value); } }; - var regex, setMap; + var regex; + var setMap; $dateParser.init = function() { $dateParser.$format = $locale.DATETIME_FORMATS[options.format] || options.format; regex = regExpForFormat($dateParser.$format); @@ -2575,7 +2908,7 @@ if (!matches) return false; var date = baseDate && !isNaN(baseDate.getTime()) ? new ParseDate().fromDate(baseDate) : new ParseDate().fromDate(new Date(1970, 0, 1, 0)); for (var i = 0; i < matches.length - 1; i++) { - formatSetMap[i] && formatSetMap[i].call(date, matches[i + 1]); + if (formatSetMap[i]) formatSetMap[i].call(date, matches[i + 1]); } var newDate = date.toDate(); if (parseInt(date.day, 10) !== newDate.getDate()) { @@ -2592,7 +2925,7 @@ date = new Date(value.substr(1, value.length - 2)); } else if (isNumeric(value)) { date = new Date(parseInt(value, 10)); - } else if (angular.isString(value) && 0 === value.length) { + } else if (angular.isString(value) && value.length === 0) { date = key === 'minDate' ? -Infinity : +Infinity; } else { date = new Date(value); @@ -2607,7 +2940,7 @@ time = new Date(value.substr(1, value.length - 2)).setFullYear(1970, 0, 1); } else if (isNumeric(value)) { time = new Date(parseInt(value, 10)).setFullYear(1970, 0, 1); - } else if (angular.isString(value) && 0 === value.length) { + } else if (angular.isString(value) && value.length === 0) { time = key === 'minTime' ? -Infinity : +Infinity; } else { time = $dateParser.parse(value, new Date(1970, 0, 1, 0)); @@ -2631,39 +2964,66 @@ } return date; }; - function setMapForFormat(format) { - var keys = Object.keys(setFnMap), i; - var map = [], sortedMap = []; - var clonedFormat = format; - for (i = 0; i < keys.length; i++) { - if (format.split(keys[i]).length > 1) { - var index = clonedFormat.search(keys[i]); - format = format.split(keys[i]).join(''); - if (setFnMap[keys[i]]) { - map[index] = setFnMap[keys[i]]; + function regExpForFormat(format) { + var re = buildDateAbstractRegex(format); + return buildDateParseRegex(re); + } + function buildDateAbstractRegex(format) { + var escapedFormat = escapeReservedSymbols(format); + var escapedLiteralFormat = escapedFormat.replace(/''/g, '\\\''); + var literalRegex = /('(?:\\'|.)*?')/; + var formatParts = escapedLiteralFormat.split(literalRegex); + var dateElements = Object.keys(regExpMap); + var dateRegexParts = []; + angular.forEach(formatParts, function(part) { + if (isFormatStringLiteral(part)) { + part = trimLiteralEscapeChars(part); + } else { + for (var i = 0; i < dateElements.length; i++) { + part = part.split(dateElements[i]).join('${' + i + '}'); } } - } - angular.forEach(map, function(v) { - if (v) sortedMap.push(v); + dateRegexParts.push(part); }); - return sortedMap; + return dateRegexParts.join(''); } function escapeReservedSymbols(text) { - return text.replace(/\//g, '[\\/]').replace('/-/g', '[-]').replace(/\./g, '[.]').replace(/\\s/g, '[\\s]'); + return text.replace(/\\/g, '[\\\\]').replace(/-/g, '[-]').replace(/\./g, '[.]').replace(/\*/g, '[*]').replace(/\+/g, '[+]').replace(/\?/g, '[?]').replace(/\$/g, '[$]').replace(/\^/g, '[^]').replace(/\//g, '[/]').replace(/\\s/g, '[\\s]'); } - function regExpForFormat(format) { - var keys = Object.keys(regExpMap), i; - var re = format; - for (i = 0; i < keys.length; i++) { - re = re.split(keys[i]).join('${' + i + '}'); - } - for (i = 0; i < keys.length; i++) { - re = re.split('${' + i + '}').join('(' + regExpMap[keys[i]] + ')'); + function isFormatStringLiteral(text) { + return /^'.*'$/.test(text); + } + function trimLiteralEscapeChars(text) { + return text.replace(/^'(.*)'$/, '$1'); + } + function buildDateParseRegex(abstractRegex) { + var dateElements = Object.keys(regExpMap); + var re = abstractRegex; + for (var i = 0; i < dateElements.length; i++) { + re = re.split('${' + i + '}').join('(' + regExpMap[dateElements[i]] + ')'); } - format = escapeReservedSymbols(format); return new RegExp('^' + re + '$', [ 'i' ]); } + function setMapForFormat(format) { + var re = buildDateAbstractRegex(format); + return buildDateParseValuesMap(re); + } + function buildDateParseValuesMap(abstractRegex) { + var dateElements = Object.keys(regExpMap); + var valuesRegex = new RegExp('\\${(\\d+)}', 'g'); + var valuesMatch; + var keyIndex; + var valueKey; + var valueFunction; + var valuesFunctionMap = []; + while ((valuesMatch = valuesRegex.exec(abstractRegex)) !== null) { + keyIndex = valuesMatch[1]; + valueKey = dateElements[keyIndex]; + valueFunction = setFnMap[valueKey]; + valuesFunctionMap.push(valueFunction); + } + return valuesFunctionMap; + } $dateParser.init(); return $dateParser; }; @@ -2717,8 +3077,8 @@ var template = options.template || ''; var controller = options.controller; var controllerAs = options.controllerAs; - var resolve = angular.copy(options.resolve || {}); - var locals = angular.copy(options.locals || {}); + var resolve = options.resolve || {}; + var locals = options.locals || {}; var transformTemplate = options.transformTemplate || angular.identity; var bindToController = options.bindToController; angular.forEach(resolve, function(value, key) { @@ -2736,6 +3096,13 @@ } else { throw new Error('Missing `template` / `templateUrl` option.'); } + if (options.titleTemplate) { + resolve.$template = $q.all([ resolve.$template, fetchTemplate(options.titleTemplate) ]).then(function(templates) { + var templateEl = angular.element(templates[0]); + findElement('[ng-bind="title"]', templateEl[0]).removeAttr('ng-bind').html(templates[1]); + return templateEl[0].outerHTML; + }); + } if (options.contentTemplate) { resolve.$template = $q.all([ resolve.$template, fetchTemplate(options.contentTemplate) ]).then(function(templates) { var templateEl = angular.element(templates[0]); @@ -2786,7 +3153,6 @@ }); } } - bsCompilerService.$inject = ["$q", "$http", "$injector", "$compile", "$controller", "$templateCache"]; angular.module('mgcrea.ngStrap.dropdown', [ 'mgcrea.ngStrap.tooltip' ]).provider('$bsDropdown', function() { var defaults = this.defaults = { animation: 'am-fade', @@ -2806,7 +3172,7 @@ function DropdownFactory(element, config) { var $dropdown = {}; var options = angular.extend({}, defaults, config); - var scope = $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new(); + $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new(); $dropdown = $tooltip(element, options); var parentEl = element.parent(); $dropdown.$onKeyDown = function(evt) { @@ -2826,17 +3192,17 @@ $dropdown.show = function() { show(); $timeout(function() { - options.keyboard && $dropdown.$element && $dropdown.$element.on('keydown', $dropdown.$onKeyDown); + if (options.keyboard && $dropdown.$element) $dropdown.$element.on('keydown', $dropdown.$onKeyDown); bodyEl.on('click', onBodyClick); }, 0, false); - parentEl.hasClass('dropdown') && parentEl.addClass('open'); + if (parentEl.hasClass('dropdown')) parentEl.addClass('open'); }; var hide = $dropdown.hide; $dropdown.hide = function() { if (!$dropdown.$isShown) return; - options.keyboard && $dropdown.$element && $dropdown.$element.off('keydown', $dropdown.$onKeyDown); + if (options.keyboard && $dropdown.$element) $dropdown.$element.off('keydown', $dropdown.$onKeyDown); bodyEl.off('click', onBodyClick); - parentEl.hasClass('dropdown') && parentEl.removeClass('open'); + if (parentEl.hasClass('dropdown')) parentEl.removeClass('open'); hide(); }; var destroy = $dropdown.destroy; @@ -2857,36 +3223,51 @@ restrict: 'EAC', scope: true, compile: function(tElement, tAttrs) { - var options = {}; if (!tAttrs.bsDropdown) { var nextSibling = tElement[0].nextSibling; while (nextSibling && nextSibling.nodeType !== 1) { nextSibling = nextSibling.nextSibling; } - if (nextSibling.classList.contains('dropdown-menu')) { - options.template = nextSibling.outerHTML; - options.templateUrl = undefined; + if (nextSibling && nextSibling.className.split(' ').indexOf('dropdown-menu') >= 0) { + tAttrs.template = nextSibling.outerHTML; + tAttrs.templateUrl = undefined; nextSibling.parentNode.removeChild(nextSibling); } } return function postLink(scope, element, attr) { - options.scope = scope; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id' ], function(key) { + var options = { + scope: scope + }; + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id', 'autoClose' ], function(key) { if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - attr.bsDropdown && scope.$watch(attr.bsDropdown, function(newValue, oldValue) { - scope.content = newValue; - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!dropdown || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i); - newValue === true ? dropdown.show() : dropdown.hide(); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); + if (attr.bsDropdown) { + scope.$watch(attr.bsDropdown, function(newValue, oldValue) { + scope.content = newValue; + }, true); + } var dropdown = $dropdown(element, options); + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!dropdown || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i); + if (newValue === true) { + dropdown.show(); + } else { + dropdown.hide(); + } + }); + } scope.$on('$destroy', function() { if (dropdown) dropdown.destroy(); options = null; @@ -2896,28 +3277,146 @@ } }; } ]); - angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$bsDatepicker', function() { + angular.module('mgcrea.ngStrap.button', []).provider('$bsButton', function() { var defaults = this.defaults = { - animation: 'am-fade', - prefixClass: 'datepicker', - placement: 'bottom-left', - templateUrl: 'datepicker/datepicker.tpl.html', - trigger: 'focus', - container: false, - keyboard: true, - html: false, - delay: 0, - useNative: false, - dateType: 'date', - dateFormat: 'shortDate', - timezone: null, - modelDateFormat: null, - dayFormat: 'dd', - monthFormat: 'MMM', - yearFormat: 'yyyy', - monthTitleFormat: 'MMMM yyyy', - yearTitleFormat: 'yyyy', - strictFormat: false, + activeClass: 'active', + toggleEvent: 'click' + }; + this.$get = function() { + return { + defaults: defaults + }; + }; + }).directive('bsCheckboxGroup', function() { + return { + restrict: 'A', + require: 'ngModel', + compile: function postLink(element, attr) { + element.attr('data-toggle', 'buttons'); + element.removeAttr('ng-model'); + var children = element[0].querySelectorAll('input[type="checkbox"]'); + angular.forEach(children, function(child) { + var childEl = angular.element(child); + childEl.attr('bs-checkbox', ''); + childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value')); + }); + } + }; + }).directive('bsCheckbox', ["$bsButton", "$$rAF", function($button, $$rAF) { + var defaults = $button.defaults; + var constantValueRegExp = /^(true|false|\d+)$/; + return { + restrict: 'A', + require: 'ngModel', + link: function postLink(scope, element, attr, controller) { + var options = defaults; + var isInput = element[0].nodeName === 'INPUT'; + var activeElement = isInput ? element.parent() : element; + var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true; + if (constantValueRegExp.test(attr.trueValue)) { + trueValue = scope.$eval(attr.trueValue); + } + var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false; + if (constantValueRegExp.test(attr.falseValue)) { + falseValue = scope.$eval(attr.falseValue); + } + var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean'; + if (hasExoticValues) { + controller.$parsers.push(function(viewValue) { + return viewValue ? trueValue : falseValue; + }); + controller.$formatters.push(function(modelValue) { + return angular.equals(modelValue, trueValue); + }); + } + controller.$render = function() { + var isActive = !!controller.$viewValue; + $$rAF(function() { + if (isInput) element[0].checked = isActive; + activeElement.toggleClass(options.activeClass, isActive); + }); + }; + element.bind(options.toggleEvent, function() { + scope.$apply(function() { + if (!isInput) { + controller.$setViewValue(!activeElement.hasClass('active')); + } + controller.$render(); + }); + }); + } + }; + } ]).directive('bsRadioGroup', function() { + return { + restrict: 'A', + require: 'ngModel', + compile: function postLink(element, attr) { + element.attr('data-toggle', 'buttons'); + element.removeAttr('ng-model'); + var children = element[0].querySelectorAll('input[type="radio"]'); + angular.forEach(children, function(child) { + angular.element(child).attr('bs-radio', ''); + angular.element(child).attr('ng-model', attr.ngModel); + }); + } + }; + }).directive('bsRadio', ["$bsButton", "$$rAF", function($button, $$rAF) { + var defaults = $button.defaults; + var constantValueRegExp = /^(true|false|\d+)$/; + return { + restrict: 'A', + require: 'ngModel', + link: function postLink(scope, element, attr, controller) { + var options = defaults; + var isInput = element[0].nodeName === 'INPUT'; + var activeElement = isInput ? element.parent() : element; + var value; + attr.$observe('value', function(v) { + if (typeof v !== 'boolean' && constantValueRegExp.test(v)) { + value = scope.$eval(v); + } else { + value = v; + } + controller.$render(); + }); + controller.$render = function() { + var isActive = angular.equals(controller.$viewValue, value); + $$rAF(function() { + if (isInput) element[0].checked = isActive; + activeElement.toggleClass(options.activeClass, isActive); + }); + }; + element.bind(options.toggleEvent, function() { + scope.$apply(function() { + controller.$setViewValue(value); + controller.$render(); + }); + }); + } + }; + } ]); + angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$bsDatepicker', function() { + var defaults = this.defaults = { + animation: 'am-fade', + prefixClass: 'datepicker', + placement: 'bottom-left', + templateUrl: 'datepicker/datepicker.tpl.html', + trigger: 'focus', + container: false, + keyboard: true, + html: false, + delay: 0, + useNative: false, + dateType: 'date', + dateFormat: 'shortDate', + timezone: null, + modelDateFormat: null, + dayFormat: 'dd', + monthFormat: 'MMM', + yearFormat: 'yyyy', + monthTitleFormat: 'MMMM yyyy', + yearTitleFormat: 'yyyy', + strictFormat: false, autoclose: false, minDate: -Infinity, maxDate: +Infinity, @@ -2925,11 +3424,13 @@ minView: 0, startWeek: 0, daysOfWeekDisabled: '', + hasToday: false, + hasClear: false, iconLeft: 'glyphicon glyphicon-chevron-left', iconRight: 'glyphicon glyphicon-chevron-right' }; this.$get = ["$window", "$document", "$rootScope", "$sce", "$bsDateFormatter", "bsDatepickerViews", "$bsTooltip", "$timeout", function($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) { - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale(); function DatepickerFactory(element, controller, config) { @@ -2944,8 +3445,11 @@ scope.$mode = options.startView; scope.$iconLeft = options.iconLeft; scope.$iconRight = options.iconRight; + scope.$hasToday = options.hasToday; + scope.$hasClear = options.hasClear; var $picker = $datepicker.$views[scope.$mode]; - scope.$select = function(date) { + scope.$select = function(date, disabled) { + if (disabled) return; $datepicker.select(date); }; scope.$selectPane = function(value) { @@ -2954,6 +3458,22 @@ scope.$toggleMode = function() { $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length); }; + scope.$setToday = function() { + if (options.autoclose) { + $datepicker.setMode(0); + $datepicker.select(new Date()); + } else { + $datepicker.select(new Date(), true); + } + }; + scope.$clear = function() { + if (options.autoclose) { + $datepicker.setMode(0); + $datepicker.select(null); + } else { + $datepicker.select(null, true); + } + }; $datepicker.update = function(date) { if (angular.isDate(date) && !isNaN(date.getTime())) { $datepicker.$date = date; @@ -2968,7 +3488,13 @@ } }; $datepicker.select = function(date, keep) { - if (!angular.isDate(controller.$dateValue)) controller.$dateValue = new Date(date); + if (angular.isDate(date)) { + if (!angular.isDate(controller.$dateValue) || isNaN(controller.$dateValue.getTime())) { + controller.$dateValue = new Date(date); + } + } else { + controller.$dateValue = null; + } if (!scope.$mode || keep) { controller.$setViewValue(angular.copy(date)); controller.$render(); @@ -3099,8 +3625,7 @@ return DatepickerFactory; } ]; }).directive('bsDatepicker', ["$window", "$parse", "$q", "$bsDateFormatter", "$bsDateParser", "$bsDatepicker", function($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) { - var defaults = $datepicker.defaults; - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); return { restrict: 'EAC', require: 'ngModel', @@ -3108,15 +3633,21 @@ var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent', 'hasToday', 'hasClear' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; - angular.forEach([ 'html', 'container', 'autoclose', 'useNative' ], function(key) { + angular.forEach([ 'html', 'container', 'autoclose', 'useNative', 'hasToday', 'hasClear' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { options[key] = false; } }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); var datepicker = $datepicker(element, controller, options); options = datepicker.$options; if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd'; @@ -3129,21 +3660,31 @@ lang: lang, strict: options.strictFormat }); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!datepicker || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i); - newValue === true ? datepicker.show() : datepicker.hide(); - }); - angular.forEach([ 'minDate', 'maxDate' ], function(key) { - angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) { - datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue); - !isNaN(datepicker.$options[key]) && datepicker.$build(false); - validateAgainstMinMaxDate(controller.$dateValue); + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!datepicker || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i); + if (newValue === true) { + datepicker.show(); + } else { + datepicker.hide(); + } }); + } + angular.forEach([ 'minDate', 'maxDate' ], function(key) { + if (angular.isDefined(attr[key])) { + attr.$observe(key, function(newValue) { + datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue); + if (!isNaN(datepicker.$options[key])) datepicker.$build(false); + validateAgainstMinMaxDate(controller.$dateValue); + }); + } }); - angular.isDefined(attr.dateFormat) && attr.$observe('dateFormat', function(newValue) { - datepicker.$options.dateFormat = newValue; - }); + if (angular.isDefined(attr.dateFormat)) { + attr.$observe('dateFormat', function(newValue) { + datepicker.$options.dateFormat = newValue; + }); + } scope.$watch(attr.ngModel, function(newValue, oldValue) { datepicker.update(controller.$dateValue); }, true); @@ -3180,9 +3721,8 @@ if (!parsedDate || isNaN(parsedDate.getTime())) { controller.$setValidity('date', false); return; - } else { - validateAgainstMinMaxDate(parsedDate); } + validateAgainstMinMaxDate(parsedDate); if (options.dateType === 'string') { date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true); return formatDate(date, options.modelDateFormat || options.dateFormat); @@ -3194,9 +3734,8 @@ return date.getTime() / 1e3; } else if (options.dateType === 'iso') { return date.toISOString(); - } else { - return new Date(date); } + return new Date(date); }); controller.$formatters.push(function(modelValue) { var date; @@ -3228,10 +3767,6 @@ } }; } ]).provider('bsDatepickerViews', function() { - var defaults = this.defaults = { - dayFormat: 'dd', - daySplit: 7 - }; function split(arr, size) { var arrays = []; while (arr.length > 0) { @@ -3284,11 +3819,14 @@ } }, build: function() { - var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1), firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset(); - var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5), firstDateOffset = firstDate.getTimezoneOffset(); + var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1); + var firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset(); + var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5); + var firstDateOffset = firstDate.getTimezoneOffset(); var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString(); if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 6e4); - var days = [], day; + var days = []; + var day; for (var i = 0; i < 42; i++) { day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i)); days.push({ @@ -3304,6 +3842,7 @@ scope.showLabels = true; scope.labels = weekDaysLabelsHtml; scope.rows = split(days, this.split); + scope.isTodayDisabled = this.isDisabled(new Date()); this.built = true; }, isSelected: function(date) { @@ -3355,8 +3894,8 @@ } }, build: function() { - var firstMonth = new Date(viewDate.year, 0, 1); - var months = [], month; + var months = []; + var month; for (var i = 0; i < 12; i++) { month = new Date(viewDate.year, i, 1); months.push({ @@ -3413,7 +3952,8 @@ }, build: function() { var firstYear = viewDate.year - viewDate.year % (this.split * 3); - var years = [], year; + var years = []; + var year; for (var i = 0; i < 12; i++) { year = new Date(firstYear + i, 0, 1); years.push({ @@ -3439,7 +3979,8 @@ if (!picker.$date) { return; } - var actualYear = picker.$date.getFullYear(), newDate = new Date(picker.$date); + var actualYear = picker.$date.getFullYear(); + var newDate = new Date(picker.$date); if (evt.keyCode === 37) newDate.setYear(actualYear - 1); else if (evt.keyCode === 38) newDate.setYear(actualYear - 4); else if (evt.keyCode === 39) newDate.setYear(actualYear + 1); else if (evt.keyCode === 40) newDate.setYear(actualYear + 4); if (!this.isDisabled(newDate)) picker.select(newDate, true); } @@ -3499,8 +4040,8 @@ self.$setActive = $scope.$setActive = function(value) { if (angular.isArray(value)) { self.$targets.$active = value; - } else if (!self.$options.disallowToggle) { - isActive(value) ? deactivateItem(value) : activateItem(value); + } else if (!self.$options.disallowToggle && isActive(value)) { + deactivateItem(value); } else { activateItem(value); } @@ -3509,7 +4050,10 @@ }); }; self.$activeIndexes = function() { - return self.$options.allowMultiple ? self.$targets.$active : self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1; + if (self.$options.allowMultiple) { + return self.$targets.$active; + } + return self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1; }; function fixActiveItemIndexes(index) { var activeIndexes = self.$targets.$active; @@ -3524,7 +4068,7 @@ } function isActive(value) { var activeItems = self.$targets.$active; - return activeItems.indexOf(value) === -1 ? false : true; + return activeItems.indexOf(value) !== -1; } function deactivateItem(value) { var index = self.$targets.$active.indexOf(value); @@ -3548,7 +4092,6 @@ return $collapse; }; }).directive('bsCollapse', ["$window", "$animate", "$bsCollapse", function($window, $animate, $collapse) { - var defaults = $collapse.defaults; return { require: [ '?ngModel', 'bsCollapse' ], controller: [ '$scope', '$element', '$attrs', $collapse.controller ], @@ -3581,7 +4124,6 @@ return { require: [ '^?ngModel', '^bsCollapse' ], link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsCollapseCtrl = controllers[1]; element.attr('data-toggle', 'collapse'); bsCollapseCtrl.$registerToggle(element); @@ -3589,9 +4131,11 @@ bsCollapseCtrl.$unregisterToggle(element); }); element.on('click', function() { - var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element); - bsCollapseCtrl.$setActive(index * 1); - scope.$apply(); + if (!attrs.disabled) { + var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element); + bsCollapseCtrl.$setActive(index * 1); + scope.$apply(); + } }); } }; @@ -3599,7 +4143,6 @@ return { require: [ '^?ngModel', '^bsCollapse' ], link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsCollapseCtrl = controllers[1]; element.addClass('collapse'); if (bsCollapseCtrl.$options.animation) { @@ -3629,125 +4172,6 @@ } }; } ]); - angular.module('mgcrea.ngStrap.button', []).provider('$bsButton', function() { - var defaults = this.defaults = { - activeClass: 'active', - toggleEvent: 'click' - }; - this.$get = function() { - return { - defaults: defaults - }; - }; - }).directive('bsCheckboxGroup', function() { - return { - restrict: 'A', - require: 'ngModel', - compile: function postLink(element, attr) { - element.attr('data-toggle', 'buttons'); - element.removeAttr('ng-model'); - var children = element[0].querySelectorAll('input[type="checkbox"]'); - angular.forEach(children, function(child) { - var childEl = angular.element(child); - childEl.attr('bs-checkbox', ''); - childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value')); - }); - } - }; - }).directive('bsCheckbox', ["$bsButton", "$$rAF", function($button, $$rAF) { - var defaults = $button.defaults; - var constantValueRegExp = /^(true|false|\d+)$/; - return { - restrict: 'A', - require: 'ngModel', - link: function postLink(scope, element, attr, controller) { - var options = defaults; - var isInput = element[0].nodeName === 'INPUT'; - var activeElement = isInput ? element.parent() : element; - var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true; - if (constantValueRegExp.test(attr.trueValue)) { - trueValue = scope.$eval(attr.trueValue); - } - var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false; - if (constantValueRegExp.test(attr.falseValue)) { - falseValue = scope.$eval(attr.falseValue); - } - var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean'; - if (hasExoticValues) { - controller.$parsers.push(function(viewValue) { - return viewValue ? trueValue : falseValue; - }); - controller.$formatters.push(function(modelValue) { - return angular.equals(modelValue, trueValue); - }); - scope.$watch(attr.ngModel, function(newValue, oldValue) { - controller.$render(); - }); - } - controller.$render = function() { - var isActive = angular.equals(controller.$modelValue, trueValue); - $$rAF(function() { - if (isInput) element[0].checked = isActive; - activeElement.toggleClass(options.activeClass, isActive); - }); - }; - element.bind(options.toggleEvent, function() { - scope.$apply(function() { - if (!isInput) { - controller.$setViewValue(!activeElement.hasClass('active')); - } - if (!hasExoticValues) { - controller.$render(); - } - }); - }); - } - }; - } ]).directive('bsRadioGroup', function() { - return { - restrict: 'A', - require: 'ngModel', - compile: function postLink(element, attr) { - element.attr('data-toggle', 'buttons'); - element.removeAttr('ng-model'); - var children = element[0].querySelectorAll('input[type="radio"]'); - angular.forEach(children, function(child) { - angular.element(child).attr('bs-radio', ''); - angular.element(child).attr('ng-model', attr.ngModel); - }); - } - }; - }).directive('bsRadio', ["$bsButton", "$$rAF", function($button, $$rAF) { - var defaults = $button.defaults; - var constantValueRegExp = /^(true|false|\d+)$/; - return { - restrict: 'A', - require: 'ngModel', - link: function postLink(scope, element, attr, controller) { - var options = defaults; - var isInput = element[0].nodeName === 'INPUT'; - var activeElement = isInput ? element.parent() : element; - var value; - attr.$observe('value', function(v) { - value = constantValueRegExp.test(v) ? scope.$eval(v) : v; - controller.$render(); - }); - controller.$render = function() { - var isActive = angular.equals(controller.$modelValue, value); - $$rAF(function() { - if (isInput) element[0].checked = isActive; - activeElement.toggleClass(options.activeClass, isActive); - }); - }; - element.bind(options.toggleEvent, function() { - scope.$apply(function() { - controller.$setViewValue(value); - controller.$render(); - }); - }); - } - }; - } ]); angular.module('mgcrea.ngStrap.aside', [ 'mgcrea.ngStrap.modal' ]).provider('$bsAside', function() { var defaults = this.defaults = { animation: 'am-fade-and-slide-right', @@ -3773,7 +4197,6 @@ return AsideFactory; } ]; }).directive('bsAside', ["$window", "$sce", "$bsAside", function($window, $sce, $aside) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; return { restrict: 'EAC', scope: true, @@ -3790,18 +4213,28 @@ angular.forEach([ 'backdrop', 'keyboard', 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); - attr.bsAside && scope.$watch(attr.bsAside, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + angular.forEach([ 'title', 'content' ], function(key) { + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsAside) { + scope.$watch(attr.bsAside, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var aside = $aside(options); element.on(attr.trigger || 'click', aside.toggle); scope.$on('$destroy', function() { @@ -3851,7 +4284,6 @@ return AlertFactory; } ]; }).directive('bsAlert', ["$window", "$sce", "$bsAlert", function($window, $sce, $alert) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; return { restrict: 'EAC', scope: true, @@ -3868,21 +4300,31 @@ angular.forEach([ 'keyboard', 'html', 'container', 'dismissable' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); if (!scope.hasOwnProperty('title')) { scope.title = ''; } angular.forEach([ 'title', 'content', 'type' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); - }); - attr.bsAlert && scope.$watch(attr.bsAlert, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsAlert) { + scope.$watch(attr.bsAlert, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var alert = $alert(options); element.on(attr.trigger || 'click', alert.toggle); scope.$on('$destroy', function() { @@ -3896,7 +4338,8 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce' ]).provider('$bsAffix', function() { var defaults = this.defaults = { offsetTop: 'auto', - inlineStyles: true + inlineStyles: true, + setWidth: true }; this.$get = ["$window", "bsDebounce", "bsDimensions", function($window, debounce, dimensions) { var bodyEl = angular.element($window.document.body); @@ -3905,7 +4348,14 @@ var $affix = {}; var options = angular.extend({}, defaults, config); var targetEl = options.target; - var reset = 'affix affix-top affix-bottom', setWidth = false, initialAffixTop = 0, initialOffsetTop = 0, offsetTop = 0, offsetBottom = 0, affixed = null, unpin = null; + var reset = 'affix affix-top affix-bottom'; + var setWidth = false; + var initialAffixTop = 0; + var initialOffsetTop = 0; + var offsetTop = 0; + var offsetBottom = 0; + var affixed = null; + var unpin = null; var parent = element.parent(); if (options.offsetParent) { if (options.offsetParent.match(/^\d+$/)) { @@ -3919,7 +4369,7 @@ $affix.init = function() { this.$parseOffsets(); initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop; - setWidth = !element[0].style.width; + setWidth = options.setWidth && !element[0].style.width; targetEl.on('scroll', this.checkPosition); targetEl.on('click', this.checkPositionWithEventLoop); windowEl.on('resize', this.$debouncedOnResize); @@ -3981,9 +4431,11 @@ }; $affix.$debouncedOnResize = debounce($affix.$onResize, 50); $affix.$parseOffsets = function() { - var initialPosition = element.css('position'); + var initialPosition = element[0].style.position; + var initialTop = element[0].style.top; if (options.inlineStyles) { element.css('position', options.offsetParent ? '' : 'relative'); + element.css('top', ''); } if (options.offsetTop) { if (options.offsetTop === 'auto') { @@ -4009,20 +4461,20 @@ } if (options.inlineStyles) { element.css('position', initialPosition); + element.css('top', initialTop); } }; - function getRequiredAffixClass(unpin, position, elementHeight) { + function getRequiredAffixClass(_unpin, position, elementHeight) { var scrollTop = getScrollTop(); var scrollHeight = getScrollHeight(); if (scrollTop <= offsetTop) { return 'top'; - } else if (unpin !== null && scrollTop + unpin <= position.top) { - return 'middle'; + } else if (_unpin !== null) { + return scrollTop + _unpin <= position.top ? 'middle' : 'bottom'; } else if (offsetBottom !== null && position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom) { return 'bottom'; - } else { - return 'middle'; } + return 'middle'; } function getScrollTop() { return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop; @@ -4035,7 +4487,7 @@ } return AffixFactory; } ]; - }).directive('bsAffix', ["$bsAffix", "$window", function($affix, $window) { + }).directive('bsAffix', ["$bsAffix", "$window", "$timeout", function($affix, $window, $timeout) { return { restrict: 'EAC', require: '^?bsAffixTarget', @@ -4044,7 +4496,7 @@ scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window) }; - angular.forEach([ 'offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles' ], function(key) { + angular.forEach([ 'offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles', 'setWidth' ], function(key) { if (angular.isDefined(attr[key])) { var option = attr[key]; if (/true/i.test(option)) option = true; @@ -4052,9 +4504,12 @@ options[key] = option; } }); - var affix = $affix(element, options); + var affix; + $timeout(function() { + affix = $affix(element, options); + }); scope.$on('$destroy', function() { - affix && affix.destroy(); + if (affix) affix.destroy(); options = null; affix = null; }); diff --git a/dist/angular-strap.compat.min.js b/dist/angular-strap.compat.min.js index d456a5ea5..d272b53f3 100644 --- a/dist/angular-strap.compat.min.js +++ b/dist/angular-strap.compat.min.js @@ -1,3 +1,3 @@ -!function(e,t,n){'use strict';function a(e,n,a,o,i,r){function s(e,n){return angular.element((n||t).querySelectorAll(e))}function l(e){return u[e]?u[e]:u[e]=n.get(e,{cache:r}).then(function(e){return e.data})}this.compile=function(t){t.template&&/\.html$/.test(t.template)&&(console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.'),t.templateUrl=t.template,t.template='');var n=t.templateUrl,r=t.template||'',u=t.controller,c=t.controllerAs,d=angular.copy(t.resolve||{}),f=angular.copy(t.locals||{}),p=t.transformTemplate||angular.identity,g=t.bindToController;if(angular.forEach(d,function(e,t){angular.isString(e)?d[t]=a.get(e):d[t]=a.invoke(e)}),angular.extend(d,f),r)d.$template=e.when(r);else{if(!n)throw new Error('Missing `template` / `templateUrl` option.');d.$template=l(n)}return t.contentTemplate&&(d.$template=e.all([d.$template,l(t.contentTemplate)]).then(function(e){var n=angular.element(e[0]),a=s('[ng-bind="content"]',n[0]).removeAttr('ng-bind').html(e[1]);return t.templateUrl||a.next().remove(),n[0].outerHTML})),e.all(d).then(function(e){var n=p(e.$template);t.html&&(n=n.replace(/ng-bind="/gi,'ng-bind-html="'));var a=angular.element('
').html(n.trim()).contents(),r=o(a);return{locals:e,element:a,link:function(t){if(e.$scope=t,u){var n=i(u,e,!0);g&&angular.extend(n.instance,e);var o=angular.isObject(n)?n:n();a.data('$ngControllerController',o),a.children().data('$ngControllerController',o),c&&(t[c]=o)}return r.apply(null,arguments)}}})};var u={}}angular.module('mgcrea.ngStrap.typeahead',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$bsTypeahead',function(){var e=this.defaults={animation:'am-fade',prefixClass:'typeahead',prefixEvent:'$typeahead',placement:'bottom-left',templateUrl:'typeahead/typeahead.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,minLength:1,filter:'bsAsyncFilter',limit:6,autoSelect:!1,comparator:'',trimValue:!0};this.$get=['$window','$rootScope','$bsTooltip','$$rAF','$timeout',function(t,n,a,o,i){function r(t,n,r){var l={},u=angular.extend({},e,r);l=a(t,u);var c=r.scope,d=l.$scope;d.$resetMatches=function(){d.$matches=[],d.$activeIndex=u.autoSelect?0:-1},d.$resetMatches(),d.$activate=function(e){d.$$postDigest(function(){l.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){l.select(e)})},d.$isVisible=function(){return l.$isVisible()},l.update=function(e){d.$matches=e,d.$activeIndex>=e.length&&(d.$activeIndex=u.autoSelect?0:-1),s(d),o(l.$applyPlacement)},l.activate=function(e){d.$activeIndex=e},l.select=function(e){if(-1!==e){var t=d.$matches[e].value;n.$setViewValue(t),n.$render(),d.$resetMatches(),c&&c.$digest(),d.$emit(u.prefixEvent+'.select',t,e,l)}},l.$isVisible=function(){return u.minLength&&n?d.$matches.length&&angular.isString(n.$viewValue)&&n.$viewValue.length>=u.minLength:!!d.$matches.length},l.$getIndex=function(e){var t=d.$matches.length,n=t;if(t){for(n=t;n--&&d.$matches[n].value!==e;);if(!(0>n))return n}},l.$onMouseDown=function(e){e.preventDefault(),e.stopPropagation()},l.$onKeyDown=function(e){/(38|40|13)/.test(e.keyCode)&&(!l.$isVisible()||13===e.keyCode&&-1===d.$activeIndex||(e.preventDefault(),e.stopPropagation()),13===e.keyCode&&d.$matches.length?l.select(d.$activeIndex):38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:40===e.keyCode&&d.$activeIndex0?void r.$setViewValue(r.$viewValue.substring(0,r.$viewValue.length-1)):(e.length>c&&(e=e.slice(0,c)),g.update(e),void r.$render())})}),r.$formatters.push(function(e){var t=p.displayValue(e);return t?t:e&&'object'!=typeof e?e:''}),r.$render=function(){if(r.$isEmpty(r.$viewValue))return t.val('');var e=g.$getIndex(r.$modelValue),n=angular.isDefined(e)?g.$scope.$matches[e].label:r.$viewValue;n=angular.isObject(n)?p.displayValue(n):n;var a=n?n.toString().replace(/<(?:.|\n)*?>/gm,''):'';t.val(s.trimValue===!1?a:a.trim())},e.$on('$destroy',function(){g&&g.destroy(),s=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tooltip',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$bsTooltip',function(){var e=this.defaults={animation:'am-fade',customClass:'',prefixClass:'tooltip',prefixEvent:'tooltip',container:!1,target:!1,placement:'top',templateUrl:'tooltip/tooltip.tpl.html',template:'',contentTemplate:!1,trigger:'hover focus',keyboard:!1,html:!1,show:!1,title:'',type:'',delay:0,autoClose:!1,bsEnabled:!0,viewport:{selector:'body',padding:0}};this.$get=['$window','$rootScope','$bsCompiler','$q','$templateCache','$http','$animate','$sce','bsDimensions','$$rAF','$timeout',function(n,a,o,i,r,s,l,u,c,d,f){function p(i,r){function s(){P.$emit(V.prefixEvent+'.show',F)}function p(){if(P.$emit(V.prefixEvent+'.hide',F),R===j){if(z&&'focus'===V.trigger)return i[0].blur();A()}}function v(){var e=V.trigger.split(' ');angular.forEach(e,function(e){'click'===e?i.on('click',F.toggle):'manual'!==e&&(i.on('hover'===e?'mouseenter':'focus',F.enter),i.on('hover'===e?'mouseleave':'blur',F.leave),'button'===I&&'hover'!==e&&i.on($?'touchstart':'mousedown',F.$onFocusElementMouseDown))})}function w(){for(var e=V.trigger.split(' '),t=e.length;t--;){var n=e[t];'click'===n?i.off('click',F.toggle):'manual'!==n&&(i.off('hover'===n?'mouseenter':'focus',F.enter),i.off('hover'===n?'mouseleave':'blur',F.leave),'button'===I&&'hover'!==n&&i.off($?'touchstart':'mousedown',F.$onFocusElementMouseDown))}}function b(){'focus'!==V.trigger?R.on('keyup',F.$onKeyUp):i.on('keyup',F.$onFocusKeyUp)}function y(){'focus'!==V.trigger?R.off('keyup',F.$onKeyUp):i.off('keyup',F.$onFocusKeyUp)}function D(){f(function(){R.on('click',k),h.on('click',F.hide),K=!0},0,!1)}function S(){K&&(R.off('click',k),h.off('click',F.hide),K=!1)}function k(e){e.stopPropagation()}function x(e){e=e||V.target||i;var a=e[0],o='BODY'===a.tagName,r=a.getBoundingClientRect(),s={};for(var l in r)s[l]=r[l];null===s.width&&(s=angular.extend({},s,{width:r.right-r.left,height:r.bottom-r.top}));var u=o?{top:0,left:0}:c.offset(a),d={scroll:o?t.documentElement.scrollTop||t.body.scrollTop:e.prop('scrollTop')||0},f=o?{width:t.documentElement.clientWidth,height:n.innerHeight}:null;return angular.extend({},s,d,f,u)}function T(e,t,n,a){var o,i=e.split('-');switch(i[0]){case'right':o={top:t.top+t.height/2-a/2,left:t.left+t.width};break;case'bottom':o={top:t.top+t.height,left:t.left+t.width/2-n/2};break;case'left':o={top:t.top+t.height/2-a/2,left:t.left-n};break;default:o={top:t.top-a,left:t.left+t.width/2-n/2}}if(!i[1])return o;if('top'===i[0]||'bottom'===i[0])switch(i[1]){case'left':o.left=t.left;break;case'right':o.left=t.left+t.width-n}else if('left'===i[0]||'right'===i[0])switch(i[1]){case'top':o.top=t.top-a;break;case'bottom':o.top=t.top+t.height}return o}function C(e,t){var n=R[0],a=n.offsetWidth,o=n.offsetHeight,i=parseInt(c.css(n,'margin-top'),10),r=parseInt(c.css(n,'margin-left'),10);isNaN(i)&&(i=0),isNaN(r)&&(r=0),e.top=e.top+i,e.left=e.left+r,c.setOffset(n,angular.extend({using:function(e){R.css({top:Math.round(e.top)+'px',left:Math.round(e.left)+'px',right:''})}},e),0);var s=n.offsetWidth,l=n.offsetHeight;if('top'===t&&l!==o&&(e.top=e.top+o-l),!/top-left|top-right|bottom-left|bottom-right/.test(t)){var u=M(t,e,s,l);if(u.left?e.left+=u.left:e.top+=u.top,c.setOffset(n,e),/top|right|bottom|left/.test(t)){var d=/top|bottom/.test(t),f=d?2*u.left-a+s:2*u.top-o+l,p=d?'offsetWidth':'offsetHeight';E(f,n[p],d)}}}function M(e,t,n,a){var o={top:0,left:0};if(!F.$viewport)return o;var i=V.viewport&&V.viewport.padding||0,r=x(F.$viewport);if(/right|left/.test(e)){var s=t.top-i-r.scroll,l=t.top+i-r.scroll+a;sr.top+r.height&&(o.top=r.top+r.height-l)}else{var u=t.left-i,c=t.left+i+n;ur.right&&(o.left=r.left+r.width-c)}return o}function E(e,t,n){var a=m('.tooltip-arrow, .arrow',R[0]);a.css(n?'left':'top',50*(1-e/t)+'%').css(n?'top':'left','')}function A(){clearTimeout(H),F.$isShown&&null!==R&&(V.autoClose&&S(),V.keyboard&&y()),q&&(q.$destroy(),q=null),R&&(R.remove(),R=F.$element=null)}var F={},V=F.$options=angular.extend({},e,r),O=F.$promise=o.compile(V),P=F.$scope=V.scope&&V.scope.$new()||a.$new(),I=i[0].nodeName.toLowerCase();if(V.delay&&angular.isString(V.delay)){var N=V.delay.split(',').map(parseFloat);V.delay=N.length>1?{show:N[0],hide:N[1]}:N[0]}F.$id=V.id||i.attr('id')||'',V.title&&(P.title=u.trustAsHtml(V.title)),P.$setEnabled=function(e){P.$$postDigest(function(){F.setEnabled(e)})},P.$hide=function(){P.$$postDigest(function(){F.hide()})},P.$show=function(){P.$$postDigest(function(){F.show()})},P.$toggle=function(){P.$$postDigest(function(){F.toggle()})},F.$isShown=P.$isShown=!1;var H,L,U,R,Y,q;O.then(function(e){U=e,F.init()}),F.init=function(){V.delay&&angular.isNumber(V.delay)&&(V.delay={show:V.delay,hide:V.delay}),'self'===V.container?Y=i:angular.isElement(V.container)?Y=V.container:V.container&&(Y=m(V.container)),v(),V.target&&(V.target=angular.isElement(V.target)?V.target:m(V.target)),V.show&&P.$$postDigest(function(){'focus'===V.trigger?i[0].focus():F.show()})},F.destroy=function(){w(),A(),P.$destroy()},F.enter=function(){return clearTimeout(H),L='in',V.delay&&V.delay.show?void(H=setTimeout(function(){'in'===L&&F.show()},V.delay.show)):F.show()},F.show=function(){if(V.bsEnabled&&!F.$isShown){P.$emit(V.prefixEvent+'.show.before',F);var e,t;V.container?(e=Y,t=Y[0].lastChild?angular.element(Y[0].lastChild):null):(e=null,t=i),R&&A(),q=F.$scope.$new(),R=F.$element=U.link(q,function(e,t){}),R.css({top:'-9999px',left:'-9999px',right:'auto',display:'block',visibility:'hidden'}),V.animation&&R.addClass(V.animation),V.type&&R.addClass(V.prefixClass+'-'+V.type),V.customClass&&R.addClass(V.customClass),t?t.after(R):e.prepend(R),F.$isShown=P.$isShown=!0,g(P),F.$applyPlacement(),angular.version.minor<=2?l.enter(R,e,t,s):l.enter(R,e,t).then(s),g(P),d(function(){R&&R.css({visibility:'visible'}),V.keyboard&&('focus'!==V.trigger&&F.focus(),b())}),V.autoClose&&D()}},F.leave=function(){return clearTimeout(H),L='out',V.delay&&V.delay.hide?void(H=setTimeout(function(){'out'===L&&F.hide()},V.delay.hide)):F.hide()};var z,j;F.hide=function(e){F.$isShown&&(P.$emit(V.prefixEvent+'.hide.before',F),z=e,j=R,angular.version.minor<=2?l.leave(R,p):l.leave(R).then(p),F.$isShown=P.$isShown=!1,g(P),V.keyboard&&null!==R&&y(),V.autoClose&&null!==R&&S())},F.toggle=function(){F.$isShown?F.leave():F.enter()},F.focus=function(){R[0].focus()},F.setEnabled=function(e){V.bsEnabled=e},F.setViewport=function(e){V.viewport=e},F.$applyPlacement=function(){if(R){var t=V.placement,n=/\s?auto?\s?/i,a=n.test(t);a&&(t=t.replace(n,'')||e.placement),R.addClass(V.placement);var o=x(),i=R.prop('offsetWidth'),r=R.prop('offsetHeight');if(F.$viewport=V.viewport&&m(V.viewport.selector||V.viewport),a){var s=t,l=x(F.$viewport);s.indexOf('bottom')>=0&&o.bottom+r>l.bottom?t=s.replace('bottom','top'):s.indexOf('top')>=0&&o.top-rl.width?t='right'===s?'left':t.replace('left','right'):('left'===s||'bottom-right'===s||'top-right'===s)&&o.left-it?t+12:t-12),n.$setViewValue(angular.copy(n.$dateValue)),n.$render()}},f.$build=function(){var e,t,n=m.midIndex=parseInt(g.length/2,10),a=[];for(e=0;e1*g.maxTime},m.$arrowAction=function(e,t){'picker'===g.arrowBehavior?f.$setTimeByStep(e,t):f.$moveIndex(e,t)},f.$setTimeByStep=function(e,t){var n=new Date(f.$date||b),a=n.getHours(),o=n.getMinutes(),i=n.getSeconds();0===t?n.setHours(a-parseInt(g.hourStep,10)*e):1===t?n.setMinutes(o-parseInt(g.minuteStep,10)*e):2===t&&n.setSeconds(i-parseInt(g.secondStep,10)*e),f.select(n,t,!0)},f.$moveIndex=function(e,t){var n;0===t?(n=new Date(1970,0,1,y.hour+e*g.length,y.minute,y.second),angular.extend(y,{hour:n.getHours()})):1===t?(n=new Date(1970,0,1,y.hour,y.minute+e*g.length*g.minuteStep,y.second),angular.extend(y,{minute:n.getMinutes()})):2===t&&(n=new Date(1970,0,1,y.hour,y.minute,y.second+e*g.length*g.secondStep),angular.extend(y,{second:n.getSeconds()})),f.$build()},f.$onMouseDown=function(e){if('input'!==e.target.nodeName.toLowerCase()&&e.preventDefault(),e.stopPropagation(),c){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},f.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void f.hide(!0);var t=new Date(f.$date),n=t.getHours(),a=h(t,S).length,o=t.getMinutes(),i=h(t,x).length,r=t.getSeconds(),s=h(t,T).length,u=1,c=/(37|39)/.test(e.keyCode),d=2+1*C+1*M;c&&(37===e.keyCode?v=1>v?d-1:v-1:39===e.keyCode&&(v=d-1>v?v+1:0));var m=[0,a],$=0;38===e.keyCode&&($=-1),40===e.keyCode&&($=1);var w=2===v&&C,b=2===v&&!C||3===v&&C;0===v?(t.setHours(n+$*parseInt(g.hourStep,10)),a=h(t,S).length,m=[0,a]):1===v?(t.setMinutes(o+$*parseInt(g.minuteStep,10)),i=h(t,x).length,m=[a+u,i]):w?(t.setSeconds(r+$*parseInt(g.secondStep,10)),s=h(t,T).length,m=[a+u+i+u,s]):b&&(c||f.switchMeridian(),m=[a+u+i+u+(s+u)*C,2]),f.select(t,v,!0),l(m[0],m[1]),p.$digest()}};var E=f.init;f.init=function(){return u&&g.useNative?(t.prop('type','time'),void t.css('-webkit-appearance','textfield')):(c&&(t.prop('type','text'),t.attr('readonly','true'),t.on('click',d)),void E())};var A=f.destroy;f.destroy=function(){u&&g.useNative&&t.off('click',d),A()};var F=f.show;f.show=function(){!c&&t.attr('readonly')||t.attr('disabled')||(F(),s(function(){f.$element&&f.$element.on(c?'touchstart':'mousedown',f.$onMouseDown),g.keyboard&&t&&t.on('keydown',f.$onKeyDown)},0,!1))};var V=f.hide;return f.hide=function(e){f.$isShown&&(f.$element&&f.$element.off(c?'touchstart':'mousedown',f.$onMouseDown),g.keyboard&&t&&t.off('keydown',f.$onKeyDown),V(e))},f}var u=/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent),c='createTouch'in t.document&&u;return e.lang||(e.lang=i.getDefaultLocale()),l.defaults=e,l}]}).directive('bsTimepicker',['$window','$parse','$q','$bsDateFormatter','$bsDateParser','$bsTimepicker',function(e,t,a,o,i,r){var s=r.defaults,l=/(ip(a|o)d|iphone|android)/gi.test(e.navigator.userAgent);return{restrict:'EAC',require:'ngModel',link:function(e,t,a,u){function c(e){if(angular.isDate(e)){var t=isNaN(f.minTime)||new Date(e.getTime()).setFullYear(1970,0,1)>=f.minTime,n=isNaN(f.maxTime)||new Date(e.getTime()).setFullYear(1970,0,1)<=f.maxTime,a=t&&n;u.$setValidity('date',a),u.$setValidity('min',t),u.$setValidity('max',n),a&&(u.$dateValue=e)}}function d(){return!u.$dateValue||isNaN(u.$dateValue.getTime())?'':$(u.$dateValue,f.timeFormat)}var f={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','autoclose','timeType','timeFormat','timezone','modelTimeFormat','useNative','hourStep','minuteStep','secondStep','length','arrowBehavior','iconUp','iconDown','roundDisplay','id','prefixClass','prefixEvent'],function(e){angular.isDefined(a[e])&&(f[e]=a[e])});var p=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','roundDisplay'],function(e){angular.isDefined(a[e])&&p.test(a[e])&&(f[e]=!1)}),a.bsShow&&e.$watch(a.bsShow,function(e,t){g&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(timepicker),?/i)),e===!0?g.show():g.hide())}),l&&(f.useNative||s.useNative)&&(f.timeFormat='HH:mm');var g=r(t,u,f);f=g.$options;var m=f.lang,$=function(e,t,n){return o.formatDate(e,t,m,n)},h=i({format:f.timeFormat,lang:m});angular.forEach(['minTime','maxTime'],function(e){angular.isDefined(a[e])&&a.$observe(e,function(t){g.$options[e]=h.getTimeForAttribute(e,t),!isNaN(g.$options[e])&&g.$build(),c(u.$dateValue)})}),e.$watch(a.ngModel,function(e,t){g.update(u.$dateValue)},!0),u.$parsers.unshift(function(e){var t;if(!e)return u.$setValidity('date',!0),null;var a=angular.isDate(e)?e:h.parse(e,u.$dateValue);return!a||isNaN(a.getTime())?(u.$setValidity('date',!1),n):(c(a),'string'===f.timeType?(t=h.timezoneOffsetAdjust(a,f.timezone,!0),$(t,f.modelTimeFormat||f.timeFormat)):(t=h.timezoneOffsetAdjust(u.$dateValue,f.timezone,!0),'number'===f.timeType?t.getTime():'unix'===f.timeType?t.getTime()/1e3:'iso'===f.timeType?t.toISOString():new Date(t)))}),u.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===f.timeType?h.parse(e,null,f.modelTimeFormat):'unix'===f.timeType?new Date(1e3*e):new Date(e),u.$dateValue=h.timezoneOffsetAdjust(t,f.timezone),d()}),u.$render=function(){t.val(d())},e.$on('$destroy',function(){g&&g.destroy(),f=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tab',[]).provider('$bsTab',function(){var e=this.defaults={animation:'am-fade',template:'tab/tab.tpl.html',navClass:'nav-tabs',activeClass:'active'},t=this.controller=function(t,n,a){var o=this;o.$options=angular.copy(e),angular.forEach(['animation','navClass','activeClass'],function(e){angular.isDefined(a[e])&&(o.$options[e]=a[e])}),t.$navClass=o.$options.navClass,t.$activeClass=o.$options.activeClass,o.$panes=t.$panes=[],o.$activePaneChangeListeners=o.$viewChangeListeners=[],o.$push=function(e){angular.isUndefined(o.$panes.$active)&&t.$setActive(e.name||0),o.$panes.push(e)},o.$remove=function(e){var t,n=o.$panes.indexOf(e),a=o.$panes.$active;t=angular.isString(a)?o.$panes.map(function(e){return e.name}).indexOf(a):o.$panes.$active,o.$panes.splice(n,1),t>n?t--:n===t&&t===o.$panes.length&&t--,t>=0&&t',placeholder:'Choose among the following...',allText:'All',noneText:'None',maxLength:3,maxLengthHtml:'selected',iconCheckmark:'glyphicon glyphicon-ok'};this.$get=['$window','$document','$rootScope','$bsTooltip','$timeout',function(t,n,a,o,i){function r(a,r,s){var u={},c=angular.extend({},e,s);u=o(a,c);var d=u.$scope;d.$matches=[],c.multiple?d.$activeIndex=[]:d.$activeIndex=-1,d.$isMultiple=c.multiple,d.$showAllNoneButtons=c.allNoneButtons&&c.multiple,d.$iconCheckmark=c.iconCheckmark,d.$allText=c.allText,d.$noneText=c.noneText,d.$activate=function(e){d.$$postDigest(function(){u.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){u.select(e)})},d.$isVisible=function(){return u.$isVisible()},d.$isActive=function(e){return u.$isActive(e)},d.$selectAll=function(){for(var e=0;e=c.minLength:d.$matches.length},u.$isActive=function(e){return c.multiple?-1!==d.$activeIndex.indexOf(e):d.$activeIndex===e},u.$getIndex=function(e){var t=d.$matches.length,n=t;if(t){for(n=t;n--&&d.$matches[n].value!==e;);if(!(0>n))return n}},u.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),l){var t=angular.element(e.target);t.triggerHandler('click')}},u.$onKeyDown=function(e){return/(9|13|38|40)/.test(e.keyCode)?(9!==e.keyCode&&(e.preventDefault(),e.stopPropagation()),c.multiple&&9===e.keyCode?u.hide():c.multiple||13!==e.keyCode&&9!==e.keyCode?void(c.multiple||(38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:38===e.keyCode&&d.$activeIndex<0?d.$activeIndex=d.$matches.length-1:40===e.keyCode&&d.$activeIndex0||e.indexOf('Trident/')>0||e.indexOf('Edge/')>0},u.$selectScrollFix=function(e){'UL'===n[0].activeElement.tagName&&(e.preventDefault(),e.stopImmediatePropagation(),e.target.focus())};var f=u.show;u.show=function(){f(),c.multiple&&u.$element.addClass('select-multiple'),i(function(){u.$element.on(l?'touchstart':'mousedown',u.$onMouseDown),c.keyboard&&a.on('keydown',u.$onKeyDown)},0,!1)};var p=u.hide;return u.hide=function(){!c.multiple&&angular.isUndefined(r.$modelValue)&&(d.$activeIndex=-1),u.$element.off(l?'touchstart':'mousedown',u.$onMouseDown),c.keyboard&&a.off('keydown',u.$onKeyDown),p(!0)},u}var s=(angular.element(t.document.body),/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent)),l='createTouch'in t.document&&s;return r.defaults=e,r}]}).directive('bsSelect',['$window','$parse','$q','$bsSelect','$bsParseOptions',function(e,t,n,a,o){var i=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,r){var s={scope:e,placeholder:i.placeholder};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','placeholder','allNoneButtons','maxLength','maxLengthHtml','allText','noneText','iconCheckmark','autoClose','id','sort','caretHtml','prefixClass','prefixEvent'],function(e){angular.isDefined(n[e])&&(s[e]=n[e])});var l=/^(false|0|)$/i;angular.forEach(['html','container','allNoneButtons','sort'],function(e){angular.isDefined(n[e])&&l.test(n[e])&&(s[e]=!1)});var u=t.attr('data-multiple');if(angular.isDefined(u)&&(l.test(u)?s.multiple=!1:s.multiple=u),'select'===t[0].nodeName.toLowerCase()){var c=t;c.css('display','none'),t=angular.element(''),c.after(t)}var d=o(n.bsOptions),f=a(t,r,s);f.$isIE()&&t[0].addEventListener('blur',f.$selectScrollFix);var p=d.$match[7].replace(/\|.+/,'').trim();e.$watchCollection(p,function(t,n){d.valuesFn(e,r).then(function(e){f.update(e),r.$render()})}),e.$watch(n.ngModel,function(e,t){f.$updateActiveIndex(),r.$render()},!0),r.$render=function(){var e,n;s.multiple&&angular.isArray(r.$modelValue)?(e=r.$modelValue.map(function(e){ -return n=f.$getIndex(e),angular.isDefined(n)?f.$scope.$matches[n].label:!1}).filter(angular.isDefined),e=e.length>(s.maxLength||i.maxLength)?e.length+' '+(s.maxLengthHtml||i.maxLengthHtml):e.join(', ')):(n=f.$getIndex(r.$modelValue),e=angular.isDefined(n)?f.$scope.$matches[n].label:!1),t.html((e?e:s.placeholder)+(s.caretHtml?s.caretHtml:i.caretHtml))},s.multiple&&(r.$isEmpty=function(e){return!e||0===e.length}),e.$on('$destroy',function(){f&&f.destroy(),s=null,f=null})}}}]),angular.module('mgcrea.ngStrap.scrollspy',['mgcrea.ngStrap.helpers.debounce','mgcrea.ngStrap.helpers.dimensions']).provider('$bsScrollspy',function(){var e=this.$$spies={},n=this.defaults={debounce:150,throttle:100,offset:100};this.$get=['$window','$document','$rootScope','bsDimensions','bsDebounce','bsThrottle',function(a,o,i,r,s,l){function u(e,t){return e[0].nodeName&&e[0].nodeName.toLowerCase()===t.toLowerCase()}function c(o){var c=angular.extend({},n,o);c.element||(c.element=p);var g=u(c.element,'body'),m=g?d:c.element,$=g?'window':c.id;if(e[$])return e[$].$$count++,e[$];var h,v,w,b,y,D,S,k,x={},T=x.$trackedElements=[],C=[];return x.init=function(){this.$$count=1,b=s(this.checkPosition,c.debounce),y=l(this.checkPosition,c.throttle),m.on('click',this.checkPositionWithEventLoop),d.on('resize',b),m.on('scroll',y),D=s(this.checkOffsets,c.debounce),h=i.$on('$viewContentLoaded',D),v=i.$on('$includeContentLoaded',D),D(),$&&(e[$]=x)},x.destroy=function(){this.$$count--,this.$$count>0||(m.off('click',this.checkPositionWithEventLoop),d.off('resize',b),m.off('scroll',y),h(),v(),$&&delete e[$])},x.checkPosition=function(){if(C.length){if(k=(g?a.pageYOffset:m.prop('scrollTop'))||0,S=Math.max(a.innerHeight,f.prop('clientHeight')),kC[e+1].offsetTop))return x.$activateElement(C[e])}},x.checkPositionWithEventLoop=function(){setTimeout(x.checkPosition,1)},x.$activateElement=function(e){if(w){var t=x.$getTrackedElement(w);t&&(t.source.removeClass('active'),u(t.source,'li')&&u(t.source.parent().parent(),'li')&&t.source.parent().parent().removeClass('active'))}w=e.target,e.source.addClass('active'),u(e.source,'li')&&u(e.source.parent().parent(),'li')&&e.source.parent().parent().addClass('active')},x.$getTrackedElement=function(e){return T.filter(function(t){return t.target===e})[0]},x.checkOffsets=function(){angular.forEach(T,function(e){var n=t.querySelector(e.target);e.offsetTop=n?r.offset(n).top:null,c.offset&&null!==e.offsetTop&&(e.offsetTop-=1*c.offset)}),C=T.filter(function(e){return null!==e.offsetTop}).sort(function(e,t){return e.offsetTop-t.offsetTop}),b()},x.trackElement=function(e,t){T.push({target:e,source:t})},x.untrackElement=function(e,t){for(var n,a=T.length;a--;)if(T[a].target===e&&T[a].source===t){n=a;break}T=T.splice(n,1)},x.activate=function(e){T[e].addClass('active')},x.init(),x}var d=angular.element(a),f=angular.element(o.prop('documentElement')),p=angular.element(a.document.body);return c}]}).directive('bsScrollspy',['$rootScope','bsDebounce','bsDimensions','$bsScrollspy',function(e,t,n,a){return{restrict:'EAC',link:function(e,t,n){var o={scope:e};angular.forEach(['offset','target'],function(e){angular.isDefined(n[e])&&(o[e]=n[e])});var i=a(o);i.trackElement(o.target,t),e.$on('$destroy',function(){i&&(i.untrackElement(o.target,t),i.destroy()),o=null,i=null})}}}]).directive('bsScrollspyList',['$rootScope','bsDebounce','bsDimensions','$bsScrollspy',function(e,t,n,a){return{restrict:'A',compile:function(e,t){var n=e[0].querySelectorAll('li > a[href]');angular.forEach(n,function(e){var t=angular.element(e);t.parent().attr('bs-scrollspy','').attr('data-target',t.attr('href'))})}}}]),angular.module('mgcrea.ngStrap.popover',['mgcrea.ngStrap.tooltip']).provider('$bsPopover',function(){var e=this.defaults={animation:'am-fade',customClass:'',container:!1,target:!1,placement:'right',templateUrl:'popover/popover.tpl.html',contentTemplate:!1,trigger:'click',keyboard:!0,html:!1,title:'',content:'',delay:0,autoClose:!1};this.$get=['$bsTooltip',function(t){function n(n,a){var o=angular.extend({},e,a),i=t(n,o);return o.content&&(i.$scope.content=o.content),i}return n}]}).directive('bsPopover',['$window','$sce','$bsPopover',function(e,t,n){var a=e.requestAnimationFrame||e.setTimeout;return{restrict:'EAC',scope:!0,link:function(e,o,i){var r={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','container','delay','trigger','html','animation','customClass','autoClose','id','prefixClass','prefixEvent'],function(e){angular.isDefined(i[e])&&(r[e]=i[e])});var s=/^(false|0|)$/i;angular.forEach(['html','container','autoClose'],function(e){angular.isDefined(i[e])&&s.test(i[e])&&(r[e]=!1)});var l=o.attr('data-target');angular.isDefined(l)&&(s.test(l)?r.target=!1:r.target=l),angular.forEach(['title','content'],function(n){i[n]&&i.$observe(n,function(o,i){e[n]=t.trustAsHtml(o),angular.isDefined(i)&&a(function(){u&&u.$applyPlacement()})})}),i.bsPopover&&e.$watch(i.bsPopover,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t,angular.isDefined(n)&&a(function(){u&&u.$applyPlacement()})},!0),i.bsShow&&e.$watch(i.bsShow,function(e,t){u&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(popover),?/i)),e===!0?u.show():u.hide())}),i.viewport&&e.$watch(i.viewport,function(e){u&&angular.isDefined(e)&&u.setViewport(e)});var u=n(o,r);e.$on('$destroy',function(){u&&u.destroy(),r=null,u=null})}}}]),angular.module('mgcrea.ngStrap.navbar',[]).provider('$bsNavbar',function(){var e=this.defaults={activeClass:'active',routeAttr:'data-match-route',strict:!1};this.$get=function(){return{defaults:e}}}).directive('bsNavbar',['$window','$location','$bsNavbar',function(e,t,n){var a=n.defaults;return{restrict:'A',link:function(e,n,o,i){var r=angular.copy(a);angular.forEach(Object.keys(a),function(e){angular.isDefined(o[e])&&(r[e]=o[e])}),e.$watch(function(){return t.path()},function(e,t){var a=n[0].querySelectorAll('li['+r.routeAttr+']');angular.forEach(a,function(t){var n=angular.element(t),a=n.attr(r.routeAttr).replace('/','\\/');r.strict&&(a='^'+a+'$');var o=new RegExp(a,'i');o.test(e)?n.addClass(r.activeClass):n.removeClass(r.activeClass)})})}}}]),angular.module('mgcrea.ngStrap.modal',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$bsModal',function(){var e=this.defaults={animation:'am-fade',backdropAnimation:'am-fade',prefixClass:'modal',prefixEvent:'modal',placement:'top',templateUrl:'modal/modal.tpl.html',template:'',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0};this.$get=['$window','$rootScope','$bsCompiler','$animate','$timeout','$sce','bsDimensions',function(n,a,o,i,r,s,l){function u(t){function n(){S.$emit(y.prefixEvent+'.show',b)}function r(){S.$emit(y.prefixEvent+'.hide',b),g.removeClass(y.prefixClass+'-open'),y.animation&&g.removeClass(y.prefixClass+'-with-'+y.animation)}function l(){y.backdrop&&(x.on('click',h),C.on('click',h),C.on('wheel',v))}function u(){y.backdrop&&(x.off('click',h),C.off('click',h),C.off('wheel',v))}function m(){y.keyboard&&x.on('keyup',b.$onKeyUp)}function $(){y.keyboard&&x.off('keyup',b.$onKeyUp)}function h(e){e.target===e.currentTarget&&('static'===y.backdrop?b.focus():b.hide())}function v(e){e.preventDefault()}function w(){b.$isShown&&null!==x&&(u(),$()),T&&(T.$destroy(),T=null),x&&(x.remove(),x=b.$element=null)}var b={},y=b.$options=angular.extend({},e,t),D=b.$promise=o.compile(y),S=b.$scope=y.scope&&y.scope.$new()||a.$new();y.element||y.container||(y.container='body'),b.$id=y.id||y.element&&y.element.attr('id')||'',f(['title','content'],function(e){y[e]&&(S[e]=s.trustAsHtml(y[e]))}),S.$hide=function(){S.$$postDigest(function(){b.hide()})},S.$show=function(){S.$$postDigest(function(){b.show()})},S.$toggle=function(){S.$$postDigest(function(){b.toggle()})},b.$isShown=S.$isShown=!1;var k,x,T,C=angular.element('
');return C.css({position:'fixed',top:'0px',left:'0px',bottom:'0px',right:'0px','z-index':1038}),D.then(function(e){k=e,b.init()}),b.init=function(){y.show&&S.$$postDigest(function(){b.show()})},b.destroy=function(){w(),C&&(C.remove(),C=null),S.$destroy()},b.show=function(){if(!b.$isShown){var e,t;if(angular.isElement(y.container)?(e=y.container,t=y.container[0].lastChild?angular.element(y.container[0].lastChild):null):y.container?(e=d(y.container),t=e[0]&&e[0].lastChild?angular.element(e[0].lastChild):null):(e=null,t=y.element),x&&w(),T=b.$scope.$new(),x=b.$element=k.link(T,function(e,t){}),!S.$emit(y.prefixEvent+'.show.before',b).defaultPrevented){x.css({display:'block'}).addClass(y.placement),y.animation&&(y.backdrop&&C.addClass(y.backdropAnimation),x.addClass(y.animation)),y.backdrop&&i.enter(C,g,null),angular.version.minor<=2?i.enter(x,e,t,n):i.enter(x,e,t).then(n),b.$isShown=S.$isShown=!0,c(S);var a=x[0];p(function(){a.focus()}),g.addClass(y.prefixClass+'-open'),y.animation&&g.addClass(y.prefixClass+'-with-'+y.animation),l(),m()}}},b.hide=function(){b.$isShown&&(S.$emit(y.prefixEvent+'.hide.before',b).defaultPrevented||(angular.version.minor<=2?i.leave(x,r):i.leave(x).then(r),y.backdrop&&i.leave(C),b.$isShown=S.$isShown=!1,c(S),u(),$()))},b.toggle=function(){b.$isShown?b.hide():b.show()},b.focus=function(){x[0].focus()},b.$onKeyUp=function(e){27===e.which&&b.$isShown&&(b.hide(),e.stopPropagation())},b}function c(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}function d(e,n){return angular.element((n||t).querySelectorAll(e))}var f=angular.forEach,p=(String.prototype.trim,n.requestAnimationFrame||n.setTimeout),g=angular.element(n.document.body);return u}]}).directive('bsModal',['$window','$sce','$bsModal',function(e,t,n){return{restrict:'EAC',scope:!0,link:function(e,a,o,i){var r={scope:e,element:a,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation','backdropAnimation','id','prefixEvent','prefixClass'],function(e){angular.isDefined(o[e])&&(r[e]=o[e])});var s=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(e){angular.isDefined(o[e])&&s.test(o[e])&&(r[e]=!1)}),angular.forEach(['title','content'],function(n){o[n]&&o.$observe(n,function(a,o){e[n]=t.trustAsHtml(a)})}),o.bsModal&&e.$watch(o.bsModal,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t},!0);var l=n(r);a.on(o.trigger||'click',l.toggle),e.$on('$destroy',function(){l&&l.destroy(),r=null,l=null})}}}]),angular.version.minor<3&&angular.version.dot<14&&angular.module('ng').factory('$$rAF',['$window','$timeout',function(e,t){var n=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame,a=e.cancelAnimationFrame||e.webkitCancelAnimationFrame||e.mozCancelAnimationFrame||e.webkitCancelRequestAnimationFrame,o=!!n,i=o?function(e){var t=n(e);return function(){a(t)}}:function(e){var n=t(e,16.66,!1);return function(){t.cancel(n)}};return i.supported=o,i}]),angular.module('mgcrea.ngStrap.helpers.parseOptions',[]).provider('$bsParseOptions',function(){var e=this.defaults={regexp:/^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/};this.$get=['$parse','$q',function(t,n){function a(a,o){function i(e,t){return e.map(function(e,n){var a,o,i={};return i[c]=e,a=u(t,i),o=p(t,i),{label:a,value:o,index:n}})}var r={},s=angular.extend({},e,o);r.$values=[];var l,u,c,d,f,p,g;return r.init=function(){r.$match=l=a.match(s.regexp),u=t(l[2]||l[1]),c=l[4]||l[6],d=l[5],f=t(l[3]||''),p=t(l[2]?l[1]:c),g=t(l[7])},r.valuesFn=function(e,t){return n.when(g(e,t)).then(function(t){return angular.isArray(t)||(t=[]),r.$values=t.length?i(t,e):[],r.$values})},r.displayValue=function(e){var t={};return t[c]=e,u(t)},r.init(),r}return a}]}),angular.module('mgcrea.ngStrap.helpers.dimensions',[]).factory('bsDimensions',['$document','$window',function(t,n){var a=(angular.element,{}),o=a.nodeName=function(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()};a.css=function(t,n,a){var o;return o=t.currentStyle?t.currentStyle[n]:e.getComputedStyle?e.getComputedStyle(t)[n]:t.style[n],a===!0?parseFloat(o)||0:o},a.offset=function(t){var n=t.getBoundingClientRect(),a=t.ownerDocument;return{width:n.width||t.offsetWidth,height:n.height||t.offsetHeight,top:n.top+(e.pageYOffset||a.documentElement.scrollTop)-(a.documentElement.clientTop||0),left:n.left+(e.pageXOffset||a.documentElement.scrollLeft)-(a.documentElement.clientLeft||0)}},a.setOffset=function(e,t,n){var o,i,r,s,l,u,c,d=a.css(e,'position'),f=angular.element(e),p={};'static'===d&&(e.style.position='relative'),l=a.offset(e),r=a.css(e,'top'),u=a.css(e,'left'),c=('absolute'===d||'fixed'===d)&&(r+u).indexOf('auto')>-1,c?(o=a.position(e),s=o.top,i=o.left):(s=parseFloat(r)||0,i=parseFloat(u)||0),angular.isFunction(t)&&(t=t.call(e,n,l)),null!==t.top&&(p.top=t.top-l.top+s),null!==t.left&&(p.left=t.left-l.left+i),'using'in t?t.using.call(f,p):f.css({top:p.top+'px',left:p.left+'px'})},a.position=function(e){var t,n,r={top:0,left:0};return'fixed'===a.css(e,'position')?n=e.getBoundingClientRect():(t=i(e),n=a.offset(e),o(t,'html')||(r=a.offset(t)),r.top+=a.css(t,'borderTopWidth',!0),r.left+=a.css(t,'borderLeftWidth',!0)),{width:e.offsetWidth,height:e.offsetHeight,top:n.top-r.top-a.css(e,'marginTop',!0),left:n.left-r.left-a.css(e,'marginLeft',!0)}};var i=function(e){var t=e.ownerDocument,n=e.offsetParent||t;if(o(n,'#document'))return t.documentElement;for(;n&&!o(n,'html')&&'static'===a.css(n,'position');)n=n.offsetParent;return n||t.documentElement};return a.height=function(e,t){var n=e.offsetHeight;return t?n+=a.css(e,'marginTop',!0)+a.css(e,'marginBottom',!0):n-=a.css(e,'paddingTop',!0)+a.css(e,'paddingBottom',!0)+a.css(e,'borderTopWidth',!0)+a.css(e,'borderBottomWidth',!0),n},a.width=function(e,t){var n=e.offsetWidth;return t?n+=a.css(e,'marginLeft',!0)+a.css(e,'marginRight',!0):n-=a.css(e,'paddingLeft',!0)+a.css(e,'paddingRight',!0)+a.css(e,'borderLeftWidth',!0)+a.css(e,'borderRightWidth',!0),n},a}]),angular.module('mgcrea.ngStrap.helpers.debounce',[]).factory('bsDebounce',['$timeout',function(e){return function(t,n,a){var o=null;return function(){var i=this,r=arguments,s=a&&!o;return o&&e.cancel(o),o=e(function(){o=null,a||t.apply(i,r)},n,!1),s&&t.apply(i,r),o}}}]).factory('bsThrottle',['$timeout',function(e){return function(t,n,a){var o=null;return a||(a={}),function(){var i=this,r=arguments;o||(a.leading!==!1&&t.apply(i,r),o=e(function(){o=null,a.trailing!==!1&&t.apply(i,r)},n,!1))}}}]),angular.module('mgcrea.ngStrap.helpers.dateParser',[]).provider('$bsDateParser',['$localeProvider',function(e){function t(){this.year=1970,this.month=0,this.day=1,this.hours=0,this.minutes=0,this.seconds=0,this.milliseconds=0}function n(){}function a(e){return!isNaN(parseFloat(e))&&isFinite(e)}function o(e,t){for(var n=e.length,a=t.toString().toLowerCase(),o=0;n>o;o++)if(e[o].toLowerCase()===a)return o;return-1}t.prototype.setMilliseconds=function(e){this.milliseconds=e},t.prototype.setSeconds=function(e){this.seconds=e},t.prototype.setMinutes=function(e){this.minutes=e},t.prototype.setHours=function(e){this.hours=e},t.prototype.getHours=function(){return this.hours},t.prototype.setDate=function(e){this.day=e},t.prototype.setMonth=function(e){this.month=e},t.prototype.setFullYear=function(e){this.year=e},t.prototype.fromDate=function(e){return this.year=e.getFullYear(),this.month=e.getMonth(),this.day=e.getDate(),this.hours=e.getHours(),this.minutes=e.getMinutes(),this.seconds=e.getSeconds(),this.milliseconds=e.getMilliseconds(),this},t.prototype.toDate=function(){return new Date(this.year,this.month,this.day,this.hours,this.minutes,this.seconds,this.milliseconds)};var i=t.prototype,r=this.defaults={format:'shortDate',strict:!1};this.$get=['$locale','dateFilter',function(e,s){var l=function(l){function u(e){var t,n=Object.keys(h),a=[],o=[],i=e;for(t=0;t1){var r=i.search(n[t]);e=e.split(n[t]).join(''),h[n[t]]&&(a[r]=h[n[t]])}return angular.forEach(a,function(e){e&&o.push(e)}),o}function c(e){return e.replace(/\//g,'[\\/]').replace('/-/g','[-]').replace(/\./g,'[.]').replace(/\\s/g,'[\\s]')}function d(e){var t,n=Object.keys($),a=e;for(t=0;t=1*e&&2===e.length?this.setFullYear(2e3+1*e):this.setFullYear(1*e)}};return m.init=function(){m.$format=e.DATETIME_FORMATS[g.format]||g.format,f=d(m.$format),p=u(m.$format)},m.isValid=function(e){return angular.isDate(e)?!isNaN(e.getTime()):f.test(e)},m.parse=function(n,a,o,i){o&&(o=e.DATETIME_FORMATS[o]||o),angular.isDate(n)&&(n=s(n,o||m.$format,i));var r=o?d(o):f,l=o?u(o):p,c=r.exec(n);if(!c)return!1;for(var g=a&&!isNaN(a.getTime())?(new t).fromDate(a):(new t).fromDate(new Date(1970,0,1,0)),$=0;$12?e.getHours()+2:0),e):null},m.timezoneOffsetAdjust=function(e,t,n){return e?(t&&'UTC'===t&&(e=new Date(e.getTime()),e.setMinutes(e.getMinutes()+(n?-1:1)*e.getTimezoneOffset())),e):null},m.init(),m};return l}]}]),angular.module('mgcrea.ngStrap.helpers.dateFormatter',[]).service('$bsDateFormatter',['$locale','dateFilter',function(e,t){function n(e){return/(h+)([:\.])?(m+)([:\.])?(s*)[ ]?(a?)/i.exec(e).slice(1)}this.getDefaultLocale=function(){return e.id},this.getDatetimeFormat=function(t,n){return e.DATETIME_FORMATS[t]||t},this.weekdaysShort=function(t){return e.DATETIME_FORMATS.SHORTDAY},this.hoursFormat=function(e){return n(e)[0]},this.minutesFormat=function(e){return n(e)[2]},this.secondsFormat=function(e){return n(e)[4]},this.timeSeparator=function(e){return n(e)[1]},this.showSeconds=function(e){return!!n(e)[4]},this.showAM=function(e){return!!n(e)[5]},this.formatDate=function(e,n,a,o){return t(e,n,o)}}]),angular.module('mgcrea.ngStrap.core',[]).service('$bsCompiler',a),a.$inject=['$q','$http','$injector','$compile','$controller','$templateCache'],angular.module('mgcrea.ngStrap.dropdown',['mgcrea.ngStrap.tooltip']).provider('$bsDropdown',function(){var e=this.defaults={animation:'am-fade',prefixClass:'dropdown',prefixEvent:'dropdown',placement:'bottom-left',templateUrl:'dropdown/dropdown.tpl.html',trigger:'click',container:!1,keyboard:!0,html:!1,delay:0};this.$get=['$window','$rootScope','$bsTooltip','$timeout',function(t,n,a,o){function i(t,i){function l(e){return e.target!==t[0]?e.target!==t[0]&&u.hide():void 0}var u={},c=angular.extend({},e,i);u.$scope=c.scope&&c.scope.$new()||n.$new();u=a(t,c);var d=t.parent();u.$onKeyDown=function(e){if(/(38|40)/.test(e.keyCode)){e.preventDefault(),e.stopPropagation();var t=angular.element(u.$element[0].querySelectorAll('li:not(.divider) a'));if(t.length){var n;angular.forEach(t,function(e,t){s&&s.call(e,':focus')&&(n=t)}),38===e.keyCode&&n>0?n--:40===e.keyCode&&nt;t++)angular.forEach(g.rows[t],u.$setDisabledEl)},u.select=function(e,t){angular.isDate(n.$dateValue)||(n.$dateValue=new Date(e)),!g.$mode||t?(n.$setViewValue(angular.copy(e)),n.$render(),p.autoclose&&!t&&l(function(){u.hide(!0)})):(angular.extend($,{year:e.getFullYear(),month:e.getMonth(),date:e.getDate()}),u.setMode(g.$mode-1),u.$build())},u.setMode=function(e){g.$mode=e,h=u.$views[g.$mode],u.$build()},u.$build=function(e){e===!0&&h.built||(e!==!1||h.built)&&h.build.call(h)},u.$updateSelected=function(){for(var e=0,t=g.rows.length;t>e;e++)angular.forEach(g.rows[e],o)},u.$isSelected=function(e){return h.isSelected(e)},u.$setDisabledEl=function(e){e.disabled=h.isDisabled(e.date)},u.$selectPane=function(e){var t=h.steps,n=new Date(Date.UTC($.year+(t.year||0)*e,$.month+(t.month||0)*e,1));angular.extend($,{year:n.getUTCFullYear(),month:n.getUTCMonth(),date:n.getUTCDate()}),u.$build()},u.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),d){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},u.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void(g.$mode?g.$apply(function(){u.setMode(g.$mode-1)}):u.hide(!0));h.onKeyDown(e),f.$digest()}};var v=u.init;u.init=function(){return c&&p.useNative?(t.prop('type','date'),void t.css('-webkit-appearance','textfield')):(d&&(t.prop('type','text'),t.attr('readonly','true'),t.on('click',i)),void v())};var w=u.destroy;u.destroy=function(){c&&p.useNative&&t.off('click',i),w()};var b=u.show;u.show=function(){!d&&t.attr('readonly')||t.attr('disabled')||(b(),l(function(){u.$isShown&&(u.$element.on(d?'touchstart':'mousedown',u.$onMouseDown),p.keyboard&&t.on('keydown',u.$onKeyDown))},0,!1))};var y=u.hide;return u.hide=function(e){u.$isShown&&(u.$element.off(d?'touchstart':'mousedown',u.$onMouseDown),p.keyboard&&t.off('keydown',u.$onKeyDown),y(e))},u}var c=/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent),d='createTouch'in t.document&&c;return e.lang||(e.lang=i.getDefaultLocale()),u.defaults=e,u}]}).directive('bsDatepicker',['$window','$parse','$q','$bsDateFormatter','$bsDateParser','$bsDatepicker',function(e,t,n,a,o,i){var r=(i.defaults,/(ip(a|o)d|iphone|android)/gi.test(e.navigator.userAgent));return{restrict:'EAC',require:'ngModel',link:function(e,t,n,s){function l(e){return e&&e.length?e:null}function u(e){if(angular.isDate(e)){var t=isNaN(p.$options.minDate)||e.getTime()>=p.$options.minDate,n=isNaN(p.$options.maxDate)||e.getTime()<=p.$options.maxDate,a=t&&n;s.$setValidity('date',a),s.$setValidity('min',t),s.$setValidity('max',n),a&&(s.$dateValue=e)}}function c(){return!s.$dateValue||isNaN(s.$dateValue.getTime())?'':m(s.$dateValue,d.dateFormat)}var d={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','html','animation','autoclose','dateType','dateFormat','timezone','modelDateFormat','dayFormat','strictFormat','startWeek','startDate','useNative','lang','startView','minView','iconLeft','iconRight','daysOfWeekDisabled','id','prefixClass','prefixEvent'],function(e){angular.isDefined(n[e])&&(d[e]=n[e])});var f=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative'],function(e){angular.isDefined(n[e])&&f.test(n[e])&&(d[e]=!1)});var p=i(t,s,d);d=p.$options,r&&d.useNative&&(d.dateFormat='yyyy-MM-dd');var g=d.lang,m=function(e,t){return a.formatDate(e,t,g)},$=o({format:d.dateFormat,lang:g,strict:d.strictFormat});n.bsShow&&e.$watch(n.bsShow,function(e,t){p&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(datepicker),?/i)),e===!0?p.show():p.hide())}),angular.forEach(['minDate','maxDate'],function(e){angular.isDefined(n[e])&&n.$observe(e,function(t){p.$options[e]=$.getDateForAttribute(e,t),!isNaN(p.$options[e])&&p.$build(!1),u(s.$dateValue)})}),angular.isDefined(n.dateFormat)&&n.$observe('dateFormat',function(e){p.$options.dateFormat=e}),e.$watch(n.ngModel,function(e,t){p.update(s.$dateValue)},!0),angular.isDefined(n.disabledDates)&&e.$watch(n.disabledDates,function(e,t){e=l(e),t=l(t),e&&p.updateDisabledDates(e)}),s.$parsers.unshift(function(e){var t;if(!e)return s.$setValidity('date',!0),null;var n=$.parse(e,s.$dateValue);return!n||isNaN(n.getTime())?void s.$setValidity('date',!1):(u(n),'string'===d.dateType?(t=$.timezoneOffsetAdjust(n,d.timezone,!0),m(t,d.modelDateFormat||d.dateFormat)):(t=$.timezoneOffsetAdjust(s.$dateValue,d.timezone,!0),'number'===d.dateType?t.getTime():'unix'===d.dateType?t.getTime()/1e3:'iso'===d.dateType?t.toISOString():new Date(t)))}),s.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===d.dateType?$.parse(e,null,d.modelDateFormat):'unix'===d.dateType?new Date(1e3*e):new Date(e),s.$dateValue=$.timezoneOffsetAdjust(t,d.timezone),c()}),s.$render=function(){t.val(c())},e.$on('$destroy',function(){p&&p.destroy(),d=null,p=null})}}}]).provider('bsDatepickerViews',function(){function e(e,t){for(var n=[];e.length>0;)n.push(e.splice(0,t));return n}function t(e,t){return(e%t+t)%t}this.defaults={dayFormat:'dd',daySplit:7};this.$get=['$bsDateFormatter','$bsDateParser','$sce',function(n,a,o){return function(i){var r=i.$scope,s=i.$options,l=s.lang,u=function(e,t){return n.formatDate(e,t,l)},c=a({format:s.dateFormat,lang:l,strict:s.strictFormat}),d=n.weekdaysShort(l),f=d.slice(s.startWeek).concat(d.slice(0,s.startWeek)),p=o.trustAsHtml(''+f.join('')+''),g=i.$date||(s.startDate?c.getDateForAttribute('startDate',s.startDate):new Date),m={year:g.getFullYear(),month:g.getMonth(),date:g.getDate()},$=[{format:s.dayFormat,split:7,steps:{month:1},update:function(e,t){!this.built||t||e.getFullYear()!==m.year||e.getMonth()!==m.month?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):(e.getDate()!==m.date||1===e.getDate())&&(m.date=i.$date.getDate(),i.$updateSelected())},build:function(){var n=new Date(m.year,m.month,1),a=n.getTimezoneOffset(),o=new Date(+n-864e5*t(n.getDay()-s.startWeek,7)),l=o.getTimezoneOffset(),d=c.timezoneOffsetAdjust(new Date,s.timezone).toDateString();l!==a&&(o=new Date(+o+6e4*(l-a)));for(var f,g=[],$=0;42>$;$++)f=c.daylightSavingAdjust(new Date(o.getFullYear(),o.getMonth(),o.getDate()+$)),g.push({date:f,isToday:f.toDateString()===d,label:u(f,this.format),selected:i.$date&&this.isSelected(f),muted:f.getMonth()!==m.month,disabled:this.isDisabled(f)});r.title=u(n,s.monthTitleFormat),r.showLabels=!0,r.labels=p,r.rows=e(g,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()&&e.getDate()===i.$date.getDate()},isDisabled:function(e){var t=e.getTime();if(ts.maxDate)return!0;if(-1!==s.daysOfWeekDisabled.indexOf(e.getDay()))return!0;if(s.disabledDateRanges)for(var n=0;n=s.disabledDateRanges[n].start&&t<=s.disabledDateRanges[n].end)return!0;return!1},onKeyDown:function(e){if(i.$date){var t,n=i.$date.getTime();37===e.keyCode?t=new Date(n-864e5):38===e.keyCode?t=new Date(n-6048e5):39===e.keyCode?t=new Date(n+864e5):40===e.keyCode&&(t=new Date(n+6048e5)),this.isDisabled(t)||i.select(t,!0)}}},{name:'month',format:s.monthFormat,split:4,steps:{year:1 -},update:function(e,t){this.built&&e.getFullYear()===m.year?e.getMonth()!==m.month&&(angular.extend(m,{month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected()):(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build())},build:function(){for(var t,n=(new Date(m.year,0,1),[]),a=0;12>a;a++)t=new Date(m.year,a,1),n.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=u(t,s.yearTitleFormat),r.showLabels=!1,r.rows=e(n,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()},isDisabled:function(e){var t=+new Date(e.getFullYear(),e.getMonth()+1,0);return ts.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getMonth(),n=new Date(i.$date);37===e.keyCode?n.setMonth(t-1):38===e.keyCode?n.setMonth(t-4):39===e.keyCode?n.setMonth(t+1):40===e.keyCode&&n.setMonth(t+4),this.isDisabled(n)||i.select(n,!0)}}},{name:'year',format:s.yearFormat,split:4,steps:{year:12},update:function(e,t){!this.built||t||parseInt(e.getFullYear()/20,10)!==parseInt(m.year/20,10)?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):e.getFullYear()!==m.year&&(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected())},build:function(){for(var t,n=m.year-m.year%(3*this.split),a=[],o=0;12>o;o++)t=new Date(n+o,0,1),a.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=a[0].label+'-'+a[a.length-1].label,r.showLabels=!1,r.rows=e(a,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()},isDisabled:function(e){var t=+new Date(e.getFullYear()+1,0,0);return ts.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getFullYear(),n=new Date(i.$date);37===e.keyCode?n.setYear(t-1):38===e.keyCode?n.setYear(t-4):39===e.keyCode?n.setYear(t+1):40===e.keyCode&&n.setYear(t+4),this.isDisabled(n)||i.select(n,!0)}}}];return{views:s.minView?Array.prototype.slice.call($,s.minView):$,viewDate:m}}}]}),angular.module('mgcrea.ngStrap.collapse',[]).provider('$bsCollapse',function(){var e=this.defaults={animation:'am-collapse',disallowToggle:!1,activeClass:'in',startCollapsed:!1,allowMultiple:!1},t=this.controller=function(t,n,a){function o(e){for(var t=l.$targets.$active,n=0;n=a?'top':null!==e&&a+e<=t.top?'middle':null!==w&&t.top+n+$>=o-w?'bottom':'middle'}function u(){return p[0]===t?t.pageYOffset:p[0].scrollTop}function c(){return p[0]===t?t.document.body.scrollHeight:p[0].scrollHeight}var d={},f=angular.extend({},e,s),p=f.target,g='affix affix-top affix-bottom',m=!1,$=0,h=0,v=0,w=0,b=null,y=null,D=o.parent();if(f.offsetParent)if(f.offsetParent.match(/^\d+$/))for(var S=0;S<1*f.offsetParent-1;S++)D=D.parent();else D=angular.element(f.offsetParent);return d.init=function(){this.$parseOffsets(),h=a.offset(o[0]).top+$,m=!o[0].style.width,p.on('scroll',this.checkPosition),p.on('click',this.checkPositionWithEventLoop),r.on('resize',this.$debouncedOnResize),this.checkPosition(),this.checkPositionWithEventLoop()},d.destroy=function(){p.off('scroll',this.checkPosition),p.off('click',this.checkPositionWithEventLoop),r.off('resize',this.$debouncedOnResize)},d.checkPositionWithEventLoop=function(){setTimeout(d.checkPosition,1)},d.checkPosition=function(){var e=u(),t=a.offset(o[0]),n=a.height(o[0]),r=l(y,t,n);b!==r&&(b=r,'top'===r?(y=null,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',''))):'bottom'===r?(y=f.offsetUnpin?-(1*f.offsetUnpin):t.top-e,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',f.offsetParent?'':i[0].offsetHeight-w-n-h+'px'))):(y=null,m&&o.css('width',o[0].offsetWidth+'px'),f.inlineStyles&&(o.css('position','fixed'),o.css('top',$+'px'))),o.removeClass(g).addClass('affix'+('middle'!==r?'-'+r:'')))},d.$onResize=function(){d.$parseOffsets(),d.checkPosition()},d.$debouncedOnResize=n(d.$onResize,50),d.$parseOffsets=function(){var e=o.css('position');f.inlineStyles&&o.css('position',f.offsetParent?'':'relative'),f.offsetTop&&('auto'===f.offsetTop&&(f.offsetTop='+0'),f.offsetTop.match(/^[-+]\d+$/)?($=1*-f.offsetTop,v=f.offsetParent?a.offset(D[0]).top+1*f.offsetTop:a.offset(o[0]).top-a.css(o[0],'marginTop',!0)+1*f.offsetTop):v=1*f.offsetTop),f.offsetBottom&&(w=f.offsetParent&&f.offsetBottom.match(/^[-+]\d+$/)?c()-(a.offset(D[0]).top+a.height(D[0]))+1*f.offsetBottom+1:1*f.offsetBottom),f.inlineStyles&&o.css('position',e)},d.init(),d}var i=angular.element(t.document.body),r=angular.element(t);return o}]}).directive('bsAffix',['$bsAffix','$window',function(e,t){return{restrict:'EAC',require:'^?bsAffixTarget',link:function(n,a,o,i){var r={scope:n,target:i?i.$element:angular.element(t)};angular.forEach(['offsetTop','offsetBottom','offsetParent','offsetUnpin','inlineStyles'],function(e){if(angular.isDefined(o[e])){var t=o[e];/true/i.test(t)&&(t=!0),/false/i.test(t)&&(t=!1),r[e]=t}});var s=e(a,r);n.$on('$destroy',function(){s&&s.destroy(),r=null,s=null})}}}]).directive('bsAffixTarget',function(){return{controller:['$element',function(e){this.$element=e}]}}),angular.module('mgcrea.ngStrap',['mgcrea.ngStrap.modal','mgcrea.ngStrap.aside','mgcrea.ngStrap.alert','mgcrea.ngStrap.button','mgcrea.ngStrap.select','mgcrea.ngStrap.datepicker','mgcrea.ngStrap.timepicker','mgcrea.ngStrap.navbar','mgcrea.ngStrap.tooltip','mgcrea.ngStrap.popover','mgcrea.ngStrap.dropdown','mgcrea.ngStrap.typeahead','mgcrea.ngStrap.scrollspy','mgcrea.ngStrap.affix','mgcrea.ngStrap.tab','mgcrea.ngStrap.collapse'])}(window,document); \ No newline at end of file +!function(e,t,n){'use strict';function a(e,n,a,o,i,r){function s(e,n){return angular.element((n||t).querySelectorAll(e))}function l(e){return u[e]?u[e]:u[e]=n.get(e,{cache:r}).then(function(e){return e.data})}this.compile=function(t){t.template&&/\.html$/.test(t.template)&&(console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.'),t.templateUrl=t.template,t.template='');var n=t.templateUrl,r=t.template||'',u=t.controller,c=t.controllerAs,d=t.resolve||{},f=t.locals||{},p=t.transformTemplate||angular.identity,g=t.bindToController;if(angular.forEach(d,function(e,t){angular.isString(e)?d[t]=a.get(e):d[t]=a.invoke(e)}),angular.extend(d,f),r)d.$template=e.when(r);else{if(!n)throw new Error('Missing `template` / `templateUrl` option.');d.$template=l(n)}return t.titleTemplate&&(d.$template=e.all([d.$template,l(t.titleTemplate)]).then(function(e){var t=angular.element(e[0]);return s('[ng-bind="title"]',t[0]).removeAttr('ng-bind').html(e[1]),t[0].outerHTML})),t.contentTemplate&&(d.$template=e.all([d.$template,l(t.contentTemplate)]).then(function(e){var n=angular.element(e[0]),a=s('[ng-bind="content"]',n[0]).removeAttr('ng-bind').html(e[1]);return t.templateUrl||a.next().remove(),n[0].outerHTML})),e.all(d).then(function(e){var n=p(e.$template);t.html&&(n=n.replace(/ng-bind="/gi,'ng-bind-html="'));var a=angular.element('
').html(n.trim()).contents(),r=o(a);return{locals:e,element:a,link:function(t){if(e.$scope=t,u){var n=i(u,e,!0);g&&angular.extend(n.instance,e);var o=angular.isObject(n)?n:n();a.data('$ngControllerController',o),a.children().data('$ngControllerController',o),c&&(t[c]=o)}return r.apply(null,arguments)}}})};var u={}}a.$inject=['$q','$http','$injector','$compile','$controller','$templateCache'],angular.module('mgcrea.ngStrap.typeahead',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$bsTypeahead',function(){var e=this.defaults={animation:'am-fade',prefixClass:'typeahead',prefixEvent:'$typeahead',placement:'bottom-left',templateUrl:'typeahead/typeahead.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,minLength:1,filter:'bsAsyncFilter',limit:6,autoSelect:!1,comparator:'',trimValue:!0};this.$get=['$window','$rootScope','$bsTooltip','$$rAF','$timeout',function(t,n,a,o,i){function r(t,n,r){var l={},u=angular.extend({},e,r);l=a(t,u);var c=r.scope,d=l.$scope;d.$resetMatches=function(){d.$matches=[],d.$activeIndex=u.autoSelect?0:-1},d.$resetMatches(),d.$activate=function(e){d.$$postDigest(function(){l.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){l.select(e)})},d.$isVisible=function(){return l.$isVisible()},l.update=function(e){d.$matches=e,d.$activeIndex>=e.length&&(d.$activeIndex=u.autoSelect?0:-1),s(d),o(l.$applyPlacement)},l.activate=function(e){d.$activeIndex=e},l.select=function(e){if(e!==-1){var t=d.$matches[e].value;n.$setViewValue(t),n.$render(),d.$resetMatches(),c&&c.$digest(),d.$emit(u.prefixEvent+'.select',t,e,l),angular.isDefined(u.onSelect)&&angular.isFunction(u.onSelect)&&u.onSelect(t,e,l)}},l.$isVisible=function(){return u.minLength&&n?d.$matches.length&&angular.isString(n.$viewValue)&&n.$viewValue.length>=u.minLength:!!d.$matches.length},l.$getIndex=function(e){var t;for(t=d.$matches.length;t--&&!angular.equals(d.$matches[t].value,e););return t},l.$onMouseDown=function(e){e.preventDefault(),e.stopPropagation()},l.$$updateScrollTop=function(e,t){if(t>-1&&t=i&&ar&&(e.scrollTop=a)}},l.$onKeyDown=function(e){/(38|40|13)/.test(e.keyCode)&&(!l.$isVisible()||13===e.keyCode&&d.$activeIndex===-1||(e.preventDefault(),e.stopPropagation()),13===e.keyCode&&d.$matches.length?l.select(d.$activeIndex):38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:40===e.keyCode&&d.$activeIndex0?void r.$setViewValue(r.$viewValue.substring(0,r.$viewValue.length-1)):(e.length>c&&(e=e.slice(0,c)),g.update(e),void r.$render())})}),r.$formatters.push(function(e){var t=p.displayValue(e);return t?t:angular.isDefined(e)&&'object'!=typeof e?e:''}),r.$render=function(){if(r.$isEmpty(r.$viewValue))return t.val('');var e=g.$getIndex(r.$modelValue),n=e!==-1?g.$scope.$matches[e].label:r.$viewValue;n=angular.isObject(n)?p.displayValue(n):n;var a=n?n.toString().replace(/<(?:.|\n)*?>/gm,''):'',o=t[0].selectionStart,i=t[0].selectionEnd;t.val(s.trimValue===!1?a:a.trim()),t[0].setSelectionRange(o,i)},e.$on('$destroy',function(){g&&g.destroy(),s=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tooltip',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$bsTooltip',function(){var e=this.defaults={animation:'am-fade',customClass:'',prefixClass:'tooltip',prefixEvent:'tooltip',container:!1,target:!1,placement:'top',templateUrl:'tooltip/tooltip.tpl.html',template:'',titleTemplate:!1,trigger:'hover focus',keyboard:!1,html:!1,show:!1,title:'',type:'',delay:0,autoClose:!1,bsEnabled:!0,mouseDownPreventDefault:!0,mouseDownStopPropagation:!0,viewport:{selector:'body',padding:0}};this.$get=['$window','$rootScope','$bsCompiler','$q','$templateCache','$http','$animate','$sce','bsDimensions','$$rAF','$timeout',function(n,a,o,i,r,s,l,u,c,d,f){function p(i,r){function s(){I.$emit(V.prefixEvent+'.show',F),angular.isDefined(V.onShow)&&angular.isFunction(V.onShow)&&V.onShow(F)}function p(){if(I.$emit(V.prefixEvent+'.hide',F),angular.isDefined(V.onHide)&&angular.isFunction(V.onHide)&&V.onHide(F),B===q){if(Y&&'focus'===V.trigger)return i[0].blur();M()}}function $(){var e=V.trigger.split(' ');angular.forEach(e,function(e){'click'===e||'contextmenu'===e?i.on(e,F.toggle):'manual'!==e&&(i.on('hover'===e?'mouseenter':'focus',F.enter),i.on('hover'===e?'mouseleave':'blur',F.leave),'button'===O&&'hover'!==e&&i.on(h?'touchstart':'mousedown',F.$onFocusElementMouseDown))})}function w(){for(var e=V.trigger.split(' '),t=e.length;t--;){var n=e[t];'click'===n||'contextmenu'===n?i.off(n,F.toggle):'manual'!==n&&(i.off('hover'===n?'mouseenter':'focus',F.enter),i.off('hover'===n?'mouseleave':'blur',F.leave),'button'===O&&'hover'!==n&&i.off(h?'touchstart':'mousedown',F.$onFocusElementMouseDown))}}function b(){'focus'!==V.trigger?B.on('keyup',F.$onKeyUp):i.on('keyup',F.$onFocusKeyUp)}function y(){'focus'!==V.trigger?B.off('keyup',F.$onKeyUp):i.off('keyup',F.$onFocusKeyUp)}function D(){f(function(){B.on('click',T),v.on('click',F.hide),j=!0},0,!1)}function S(){j&&(B.off('click',T),v.off('click',F.hide),j=!1)}function T(e){e.stopPropagation()}function x(e){e=e||V.target||i;var a=e[0],o='BODY'===a.tagName,r=a.getBoundingClientRect(),s={};for(var l in r)s[l]=r[l];null===s.width&&(s=angular.extend({},s,{width:r.right-r.left,height:r.bottom-r.top}));var u=o?{top:0,left:0}:c.offset(a),d={scroll:o?t.documentElement.scrollTop||t.body.scrollTop:e.prop('scrollTop')||0},f=o?{width:t.documentElement.clientWidth,height:n.innerHeight}:null;return angular.extend({},s,d,f,u)}function k(e,t,n,a){var o,i=e.split('-');switch(i[0]){case'right':o={top:t.top+t.height/2-a/2,left:t.left+t.width};break;case'bottom':o={top:t.top+t.height,left:t.left+t.width/2-n/2};break;case'left':o={top:t.top+t.height/2-a/2,left:t.left-n};break;default:o={top:t.top-a,left:t.left+t.width/2-n/2}}if(!i[1])return o;if('top'===i[0]||'bottom'===i[0])switch(i[1]){case'left':o.left=t.left;break;case'right':o.left=t.left+t.width-n}else if('left'===i[0]||'right'===i[0])switch(i[1]){case'top':o.top=t.top-a+t.height;break;case'bottom':o.top=t.top}return o}function C(e,t){var n=B[0],a=n.offsetWidth,o=n.offsetHeight,i=parseInt(c.css(n,'margin-top'),10),r=parseInt(c.css(n,'margin-left'),10);isNaN(i)&&(i=0),isNaN(r)&&(r=0),e.top=e.top+i,e.left=e.left+r,c.setOffset(n,angular.extend({using:function(e){B.css({top:Math.round(e.top)+'px',left:Math.round(e.left)+'px',right:''})}},e),0);var s=n.offsetWidth,l=n.offsetHeight;if('top'===t&&l!==o&&(e.top=e.top+o-l),!/top-left|top-right|bottom-left|bottom-right/.test(t)){var u=E(t,e,s,l);if(u.left?e.left+=u.left:e.top+=u.top,c.setOffset(n,e),/top|right|bottom|left/.test(t)){var d=/top|bottom/.test(t),f=d?2*u.left-a+s:2*u.top-o+l,p=d?'offsetWidth':'offsetHeight';A(f,n[p],d)}}}function E(e,t,n,a){var o={top:0,left:0};if(!F.$viewport)return o;var i=V.viewport&&V.viewport.padding||0,r=x(F.$viewport);if(/right|left/.test(e)){var s=t.top-i-r.scroll,l=t.top+i-r.scroll+a;sr.top+r.height&&(o.top=r.top+r.height-l)}else{var u=t.left-i,c=t.left+i+n;ur.right&&(o.left=r.left+r.width-c)}return o}function A(e,t,n){var a=m('.tooltip-arrow, .arrow',B[0]);a.css(n?'left':'top',50*(1-e/t)+'%').css(n?'top':'left','')}function M(){clearTimeout(N),F.$isShown&&null!==B&&(V.autoClose&&S(),V.keyboard&&y()),z&&(z.$destroy(),z=null),B&&(B.remove(),B=F.$element=null)}var F={},V=F.$options=angular.extend({},e,r),H=F.$promise=o.compile(V),I=F.$scope=V.scope&&V.scope.$new()||a.$new(),O=i[0].nodeName.toLowerCase();if(V.delay&&angular.isString(V.delay)){var P=V.delay.split(',').map(parseFloat);V.delay=P.length>1?{show:P[0],hide:P[1]}:P[0]}F.$id=V.id||i.attr('id')||'',V.title&&(I.title=u.trustAsHtml(V.title)),I.$setEnabled=function(e){I.$$postDigest(function(){F.setEnabled(e)})},I.$hide=function(){I.$$postDigest(function(){F.hide()})},I.$show=function(){I.$$postDigest(function(){F.show()})},I.$toggle=function(){I.$$postDigest(function(){F.toggle()})},F.$isShown=I.$isShown=!1;var N,U,L,B,R,z;H.then(function(e){L=e,F.init()}),F.init=function(){V.delay&&angular.isNumber(V.delay)&&(V.delay={show:V.delay,hide:V.delay}),'self'===V.container?R=i:angular.isElement(V.container)?R=V.container:V.container&&(R=m(V.container)),$(),V.target&&(V.target=angular.isElement(V.target)?V.target:m(V.target)),V.show&&I.$$postDigest(function(){'focus'===V.trigger?i[0].focus():F.show()})},F.destroy=function(){w(),M(),I.$destroy()},F.enter=function(){return clearTimeout(N),U='in',V.delay&&V.delay.show?void(N=setTimeout(function(){'in'===U&&F.show()},V.delay.show)):F.show()},F.show=function(){if(V.bsEnabled&&!F.$isShown){I.$emit(V.prefixEvent+'.show.before',F),angular.isDefined(V.onBeforeShow)&&angular.isFunction(V.onBeforeShow)&&V.onBeforeShow(F);var e,t;V.container?(e=R,t=R[0].lastChild?angular.element(R[0].lastChild):null):(e=null,t=i),B&&M(),z=F.$scope.$new(),B=F.$element=L.link(z,function(e,t){}),B.css({top:'-9999px',left:'-9999px',right:'auto',display:'block',visibility:'hidden'}),V.animation&&B.addClass(V.animation),V.type&&B.addClass(V.prefixClass+'-'+V.type),V.customClass&&B.addClass(V.customClass),t?t.after(B):e.prepend(B),F.$isShown=I.$isShown=!0,g(I),F.$applyPlacement(),angular.version.minor<=2?l.enter(B,e,t,s):l.enter(B,e,t).then(s),g(I),d(function(){B&&B.css({visibility:'visible'}),V.keyboard&&('focus'!==V.trigger&&F.focus(),b())}),V.autoClose&&D()}},F.leave=function(){return clearTimeout(N),U='out',V.delay&&V.delay.hide?void(N=setTimeout(function(){'out'===U&&F.hide()},V.delay.hide)):F.hide()};var Y,q;F.hide=function(e){F.$isShown&&(I.$emit(V.prefixEvent+'.hide.before',F),angular.isDefined(V.onBeforeHide)&&angular.isFunction(V.onBeforeHide)&&V.onBeforeHide(F),Y=e,q=B,null!==B&&(angular.version.minor<=2?l.leave(B,p):l.leave(B).then(p)),F.$isShown=I.$isShown=!1,g(I),V.keyboard&&null!==B&&y(),V.autoClose&&null!==B&&S())},F.toggle=function(e){e&&e.preventDefault(),F.$isShown?F.leave():F.enter()},F.focus=function(){B[0].focus()},F.setEnabled=function(e){V.bsEnabled=e},F.setViewport=function(e){V.viewport=e},F.$applyPlacement=function(){if(B){var t=V.placement,n=/\s?auto?\s?/i,a=n.test(t);a&&(t=t.replace(n,'')||e.placement),B.addClass(V.placement);var o=x(),i=B.prop('offsetWidth'),r=B.prop('offsetHeight');if(F.$viewport=V.viewport&&m(V.viewport.selector||V.viewport),a){var s=t,l=x(F.$viewport);/bottom/.test(s)&&o.bottom+r>l.bottom?t=s.replace('bottom','top'):/top/.test(s)&&o.top-rl.width&&(t=t.replace('right','left')),B.removeClass(s).addClass(t)}var u=k(t,o,i,r);C(u,t)}},F.$onKeyUp=function(e){27===e.which&&F.$isShown&&(F.hide(),e.stopPropagation())},F.$onFocusKeyUp=function(e){27===e.which&&(i[0].blur(),e.stopPropagation())},F.$onFocusElementMouseDown=function(e){V.mouseDownPreventDefault&&e.preventDefault(),V.mouseDownStopPropagation&&e.stopPropagation(),F.$isShown?i[0].blur():i[0].focus()};var j=!1;return F}function g(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}function m(e,n){return angular.element((n||t).querySelectorAll(e))}var $=/(ip[ao]d|iphone|android)/gi.test(n.navigator.userAgent),h='createTouch'in n.document&&$,v=angular.element(n.document);return p}]}).directive('bsTooltip',['$window','$location','$sce','$parse','$bsTooltip','$$rAF',function(e,t,n,a,o,i){return{restrict:'EAC',scope:!0,link:function(e,t,a,r){var s,l={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','titleTemplate','placement','container','delay','trigger','html','animation','backdropAnimation','type','customClass','id'],function(e){angular.isDefined(a[e])&&(l[e]=a[e])});var u=/^(false|0|)$/i;angular.forEach(['html','container'],function(e){angular.isDefined(a[e])&&u.test(a[e])&&(l[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(a[n])&&(l[t]=e.$eval(a[n]))});var c=t.attr('data-target');angular.isDefined(c)&&(u.test(c)?l.target=!1:l.target=c),e.hasOwnProperty('title')||(e.title=''),a.$observe('title',function(t){if(angular.isDefined(t)||!e.hasOwnProperty('title')){var a=e.title;e.title=n.trustAsHtml(t),angular.isDefined(a)&&i(function(){s&&s.$applyPlacement()})}}),a.$observe('disabled',function(e){e&&s.$isShown&&s.hide()}),a.bsTooltip&&e.$watch(a.bsTooltip,function(t,n){angular.isObject(t)?angular.extend(e,t):e.title=t,angular.isDefined(n)&&i(function(){s&&s.$applyPlacement()})},!0),a.bsShow&&e.$watch(a.bsShow,function(e,t){s&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(tooltip),?/i)),e===!0?s.show():s.hide())}),a.bsEnabled&&e.$watch(a.bsEnabled,function(e,t){s&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|1|,?(tooltip),?/i)),e===!1?s.setEnabled(!1):s.setEnabled(!0))}),a.viewport&&e.$watch(a.viewport,function(e){s&&angular.isDefined(e)&&s.setViewport(e)}),s=o(t,l),e.$on('$destroy',function(){s&&s.destroy(),l=null,s=null})}}}]),angular.module('mgcrea.ngStrap.timepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$bsTimepicker',function(){var e=this.defaults={animation:'am-fade',defaultDate:'auto',prefixClass:'timepicker',placement:'bottom-left',templateUrl:'timepicker/timepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!0,timeType:'date',timeFormat:'shortTime',timezone:null,modelTimeFormat:null,autoclose:!1,minTime:-(1/0),maxTime:+(1/0),length:5,hourStep:1,minuteStep:5,secondStep:5,roundDisplay:!1,iconUp:'glyphicon glyphicon-chevron-up',iconDown:'glyphicon glyphicon-chevron-down',arrowBehavior:'pager'};this.$get=['$window','$document','$rootScope','$sce','$bsDateFormatter','$bsTooltip','$timeout',function(t,n,a,o,i,r,s){function l(t,n,a){function o(e){var t=6e4*g.minuteStep;return new Date(Math.floor(e.getTime()/t)*t)}function l(e,n){var a=e+n;if(t[0].createTextRange){var o=t[0].createTextRange();o.collapse(!0),o.moveStart('character',e),o.moveEnd('character',a),o.select()}else t[0].setSelectionRange?t[0].setSelectionRange(e,a):angular.isUndefined(t[0].selectionStart)&&(t[0].selectionStart=e,t[0].selectionEnd=a)}function d(){t[0].focus()}var f=r(t,angular.extend({},e,a)),p=a.scope,g=f.$options,m=f.$scope,$=g.lang,h=function(e,t,n){return i.formatDate(e,t,$,n)},v=0,w=g.roundDisplay?o(new Date):new Date,b=n.$dateValue||w,y={hour:b.getHours(),meridian:b.getHours()<12,minute:b.getMinutes(),second:b.getSeconds(),millisecond:b.getMilliseconds()},D=i.getDatetimeFormat(g.timeFormat,$),S=i.hoursFormat(D),T=i.timeSeparator(D),x=i.minutesFormat(D),k=i.secondsFormat(D),C=i.showSeconds(D),E=i.showAM(D);m.$iconUp=g.iconUp,m.$iconDown=g.iconDown,m.$select=function(e,t){f.select(e,t)},m.$moveIndex=function(e,t){f.$moveIndex(e,t)},m.$switchMeridian=function(e){f.switchMeridian(e)},f.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())?(f.$date=e,angular.extend(y,{hour:e.getHours(),minute:e.getMinutes(),second:e.getSeconds(),millisecond:e.getMilliseconds()}),f.$build()):f.$isBuilt||f.$build()},f.select=function(e,t,a){n.$dateValue&&!isNaN(n.$dateValue.getTime())||(n.$dateValue='today'===g.defaultDate?new Date:new Date(1970,0,1)),angular.isDate(e)||(e=new Date(e)),0===t?n.$dateValue.setHours(e.getHours()):1===t?n.$dateValue.setMinutes(e.getMinutes()):2===t&&n.$dateValue.setSeconds(e.getSeconds()),n.$setViewValue(angular.copy(n.$dateValue)),n.$render(),g.autoclose&&!a&&s(function(){f.hide(!0)})},f.switchMeridian=function(e){if(n.$dateValue&&!isNaN(n.$dateValue.getTime())){var t=(e||n.$dateValue).getHours();n.$dateValue.setHours(t<12?t+12:t-12),n.$setViewValue(angular.copy(n.$dateValue)),n.$render()}},f.$build=function(){var e,t,n=m.midIndex=parseInt(g.length/2,10),a=[];for(e=0;e1*g.maxTime},m.$arrowAction=function(e,t){'picker'===g.arrowBehavior?f.$setTimeByStep(e,t):f.$moveIndex(e,t)},f.$setTimeByStep=function(e,t){var n=new Date(f.$date||b),a=n.getHours(),o=n.getMinutes(),i=n.getSeconds();0===t?n.setHours(a-parseInt(g.hourStep,10)*e):1===t?n.setMinutes(o-parseInt(g.minuteStep,10)*e):2===t&&n.setSeconds(i-parseInt(g.secondStep,10)*e),f.select(n,t,!0)},f.$moveIndex=function(e,t){var n;0===t?(n=new Date(1970,0,1,y.hour+e*g.length,y.minute,y.second),angular.extend(y,{hour:n.getHours()})):1===t?(n=new Date(1970,0,1,y.hour,y.minute+e*g.length*g.minuteStep,y.second),angular.extend(y,{minute:n.getMinutes()})):2===t&&(n=new Date(1970,0,1,y.hour,y.minute,y.second+e*g.length*g.secondStep),angular.extend(y,{second:n.getSeconds()})),f.$build()},f.$onMouseDown=function(e){if('input'!==e.target.nodeName.toLowerCase()&&e.preventDefault(),e.stopPropagation(),c){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},f.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void f.hide(!0);var t=new Date(f.$date),n=t.getHours(),a=h(t,S).length,o=t.getMinutes(),i=h(t,x).length,r=t.getSeconds(),s=h(t,k).length,u=1,c=/(37|39)/.test(e.keyCode),d=2+1*C+1*E;c&&(37===e.keyCode?v=v<1?d-1:v-1:39===e.keyCode&&(v=v=f.minTime,n=isNaN(f.maxTime)||new Date(e.getTime()).setFullYear(1970,0,1)<=f.maxTime,a=t&&n;u.$setValidity('date',a),u.$setValidity('min',t),u.$setValidity('max',n),a&&(u.$dateValue=e)}}function d(){return!u.$dateValue||isNaN(u.$dateValue.getTime())?'':$(u.$dateValue,f.timeFormat)}var f={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','autoclose','timeType','timeFormat','timezone','modelTimeFormat','useNative','hourStep','minuteStep','secondStep','length','arrowBehavior','iconUp','iconDown','roundDisplay','id','prefixClass','prefixEvent','defaultDate'],function(e){angular.isDefined(a[e])&&(f[e]=a[e])});var p=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','roundDisplay'],function(e){angular.isDefined(a[e])&&p.test(a[e])&&(f[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(a[n])&&(f[t]=e.$eval(a[n]))}),l&&(f.useNative||s.useNative)&&(f.timeFormat='HH:mm');var g=r(t,u,f);f=g.$options;var m=f.lang,$=function(e,t,n){return o.formatDate(e,t,m,n)};a.bsShow&&e.$watch(a.bsShow,function(e,t){g&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(timepicker),?/i)),e===!0?g.show():g.hide())});var h=i({format:f.timeFormat,lang:m});angular.forEach(['minTime','maxTime'],function(e){angular.isDefined(a[e])&&a.$observe(e,function(t){g.$options[e]=h.getTimeForAttribute(e,t),isNaN(g.$options[e])||g.$build(),c(u.$dateValue)})}),e.$watch(a.ngModel,function(e,t){g.update(u.$dateValue)},!0),u.$parsers.unshift(function(e){var t;if(!e)return u.$setValidity('date',!0),null;var a=angular.isDate(e)?e:h.parse(e,u.$dateValue);return!a||isNaN(a.getTime())?(u.$setValidity('date',!1),n):(c(a),'string'===f.timeType?(t=h.timezoneOffsetAdjust(a,f.timezone,!0),$(t,f.modelTimeFormat||f.timeFormat)):(t=h.timezoneOffsetAdjust(u.$dateValue,f.timezone,!0),'number'===f.timeType?t.getTime():'unix'===f.timeType?t.getTime()/1e3:'iso'===f.timeType?t.toISOString():new Date(t)))}),u.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===f.timeType?h.parse(e,null,f.modelTimeFormat):'unix'===f.timeType?new Date(1e3*e):new Date(e),u.$dateValue=h.timezoneOffsetAdjust(t,f.timezone),d()}),u.$render=function(){t.val(d())},e.$on('$destroy',function(){g&&g.destroy(),f=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tab',[]).provider('$bsTab',function(){var e=this.defaults={animation:'am-fade',template:'tab/tab.tpl.html',navClass:'nav-tabs',activeClass:'active'},t=this.controller=function(t,n,a){var o=this;o.$options=angular.copy(e),angular.forEach(['animation','navClass','activeClass'],function(e){angular.isDefined(a[e])&&(o.$options[e]=a[e])}),t.$navClass=o.$options.navClass,t.$activeClass=o.$options.activeClass,o.$panes=t.$panes=[],o.$activePaneChangeListeners=o.$viewChangeListeners=[],o.$push=function(e){angular.isUndefined(o.$panes.$active)&&t.$setActive(e.name||0),o.$panes.push(e)},o.$remove=function(e){var t,n=o.$panes.indexOf(e),a=o.$panes.$active;t=angular.isString(a)?o.$panes.map(function(e){return e.name}).indexOf(a):o.$panes.$active,o.$panes.splice(n,1),n=0&&t',placeholder:'Choose among the following...',allText:'All',noneText:'None',maxLength:3,maxLengthHtml:'selected',iconCheckmark:'glyphicon glyphicon-ok',toggle:!1};this.$get=['$window','$document','$rootScope','$bsTooltip','$timeout',function(t,a,o,i,r){function s(o,s,l){var c={},d=angular.extend({},e,l);c=i(o,d);var f=c.$scope;f.$matches=[],d.multiple?f.$activeIndex=[]:f.$activeIndex=-1,f.$isMultiple=d.multiple,f.$showAllNoneButtons=d.allNoneButtons&&d.multiple,f.$iconCheckmark=d.iconCheckmark,f.$allText=d.allText,f.$noneText=d.noneText,f.$activate=function(e){f.$$postDigest(function(){c.activate(e)})},f.$select=function(e,t){f.$$postDigest(function(){c.select(e)})},f.$isVisible=function(){return c.$isVisible()},f.$isActive=function(e){return c.$isActive(e)},f.$selectAll=function(){for(var e=0;e=f.$matches.length)){var t=f.$matches[e].value;f.$apply(function(){c.activate(e),d.multiple?s.$setViewValue(f.$activeIndex.map(function(e){return angular.isUndefined(f.$matches[e])?null:f.$matches[e].value})):(d.toggle?s.$setViewValue(t===s.$modelValue?n:t):s.$setViewValue(t),c.hide())}),f.$emit(d.prefixEvent+'.select',t,e,c),angular.isDefined(d.onSelect)&&angular.isFunction(d.onSelect)&&d.onSelect(t,e,c)}},c.$updateActiveIndex=function(){d.multiple?angular.isArray(s.$modelValue)?f.$activeIndex=s.$modelValue.map(function(e){return c.$getIndex(e)}):f.$activeIndex=[]:angular.isDefined(s.$modelValue)&&f.$matches.length?f.$activeIndex=c.$getIndex(s.$modelValue):f.$activeIndex=-1},c.$isVisible=function(){return d.minLength&&s?f.$matches.length&&s.$viewValue.length>=d.minLength:f.$matches.length},c.$isActive=function(e){return d.multiple?f.$activeIndex.indexOf(e)!==-1:f.$activeIndex===e},c.$getIndex=function(e){var t;for(t=f.$matches.length;t--&&!angular.equals(f.$matches[t].value,e););return t},c.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),u){var t,n=angular.element(e.target);if('A'!==e.target.nodeName)for(var a=n.parent();!t&&a.length>0;)'A'===a[0].nodeName&&(t=a),a=a.parent();t?angular.element(t).triggerHandler('click'):n.triggerHandler('click')}},c.$onKeyDown=function(e){if(/(9|13|38|40)/.test(e.keyCode))return 9!==e.keyCode&&(e.preventDefault(),e.stopPropagation()),d.multiple&&9===e.keyCode?c.hide():d.multiple||13!==e.keyCode&&9!==e.keyCode?void(d.multiple||(38===e.keyCode&&f.$activeIndex>0?f.$activeIndex--:38===e.keyCode&&f.$activeIndex<0?f.$activeIndex=f.$matches.length-1:40===e.keyCode&&f.$activeIndex0||e.indexOf('Trident/')>0||e.indexOf('Edge/')>0; +},c.$selectScrollFix=function(e){'UL'===a[0].activeElement.tagName&&(e.preventDefault(),e.stopImmediatePropagation(),e.target.focus())};var p=c.show;c.show=function(){p(),d.multiple&&c.$element.addClass('select-multiple'),r(function(){c.$element.on(u?'touchstart':'mousedown',c.$onMouseDown),d.keyboard&&o.on('keydown',c.$onKeyDown)},0,!1)};var g=c.hide;return c.hide=function(){!d.multiple&&angular.isUndefined(s.$modelValue)&&(f.$activeIndex=-1),c.$element.off(u?'touchstart':'mousedown',c.$onMouseDown),d.keyboard&&o.off('keydown',c.$onKeyDown),g(!0)},c}var l=/(ip[ao]d|iphone|android)/gi.test(t.navigator.userAgent),u='createTouch'in t.document&&l;return s.defaults=e,s}]}).directive('bsSelect',['$window','$parse','$q','$bsSelect','$bsParseOptions',function(e,t,n,a,o){var i=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,r){var s={scope:e,placeholder:i.placeholder};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','placeholder','allNoneButtons','maxLength','maxLengthHtml','allText','noneText','iconCheckmark','autoClose','id','sort','caretHtml','prefixClass','prefixEvent','toggle'],function(e){angular.isDefined(n[e])&&(s[e]=n[e])});var l=/^(false|0|)$/i;angular.forEach(['html','container','allNoneButtons','sort'],function(e){angular.isDefined(n[e])&&l.test(n[e])&&(s[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide','onSelect'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(s[t]=e.$eval(n[a]))});var u=t.attr('data-multiple');if(angular.isDefined(u)&&(l.test(u)?s.multiple=!1:s.multiple=u),'select'===t[0].nodeName.toLowerCase()){var c=t;c.css('display','none'),t=angular.element(''),c.after(t)}var d=o(n.bsOptions),f=a(t,r,s);f.$isIE()&&t[0].addEventListener('blur',f.$selectScrollFix);var p=d.$match[7].replace(/\|.+/,'').trim();e.$watch(p,function(t,n){d.valuesFn(e,r).then(function(e){f.update(e),r.$render()})},!0),e.$watch(n.ngModel,function(e,t){f.$updateActiveIndex(),r.$render()},!0),r.$render=function(){var e,n;s.multiple&&angular.isArray(r.$modelValue)?(e=r.$modelValue.map(function(e){return n=f.$getIndex(e),n!==-1&&f.$scope.$matches[n].label}).filter(angular.isDefined),e=e.length>(s.maxLength||i.maxLength)?e.length+' '+(s.maxLengthHtml||i.maxLengthHtml):e.join(', ')):(n=f.$getIndex(r.$modelValue),e=n!==-1&&f.$scope.$matches[n].label),t.html((e||s.placeholder)+(s.caretHtml||i.caretHtml))},s.multiple&&(r.$isEmpty=function(e){return!e||0===e.length}),e.$on('$destroy',function(){f&&f.destroy(),s=null,f=null})}}}]),angular.module('mgcrea.ngStrap.scrollspy',['mgcrea.ngStrap.helpers.debounce','mgcrea.ngStrap.helpers.dimensions']).provider('$bsScrollspy',function(){var e=this.$$spies={},n=this.defaults={debounce:150,throttle:100,offset:100};this.$get=['$window','$document','$rootScope','bsDimensions','bsDebounce','bsThrottle',function(a,o,i,r,s,l){function u(e,t){return e[0].nodeName&&e[0].nodeName.toLowerCase()===t.toLowerCase()}function c(o){var c=angular.extend({},n,o);c.element||(c.element=p);var g=u(c.element,'body'),m=g?d:c.element,$=g?'window':c.id;if(e[$])return e[$].$$count++,e[$];var h,v,w,b,y,D,S,T,x={},k=x.$trackedElements=[],C=[];return x.init=function(){this.$$count=1,b=s(this.checkPosition,c.debounce),y=l(this.checkPosition,c.throttle),m.on('click',this.checkPositionWithEventLoop),d.on('resize',b),m.on('scroll',y),D=s(this.checkOffsets,c.debounce),h=i.$on('$viewContentLoaded',D),v=i.$on('$includeContentLoaded',D),D(),$&&(e[$]=x)},x.destroy=function(){this.$$count--,this.$$count>0||(m.off('click',this.checkPositionWithEventLoop),d.off('resize',b),m.off('scroll',y),h(),v(),$&&delete e[$])},x.checkPosition=function(){if(C.length){if(T=(g?a.pageYOffset:m.prop('scrollTop'))||0,S=Math.max(a.innerHeight,f.prop('clientHeight')),TC[e+1].offsetTop))return x.$activateElement(C[e])}},x.checkPositionWithEventLoop=function(){setTimeout(x.checkPosition,1)},x.$activateElement=function(e){if(w){var t=x.$getTrackedElement(w);t&&(t.source.removeClass('active'),u(t.source,'li')&&u(t.source.parent().parent(),'li')&&t.source.parent().parent().removeClass('active'))}w=e.target,e.source.addClass('active'),u(e.source,'li')&&u(e.source.parent().parent(),'li')&&e.source.parent().parent().addClass('active')},x.$getTrackedElement=function(e){return k.filter(function(t){return t.target===e})[0]},x.checkOffsets=function(){angular.forEach(k,function(e){var n=t.querySelector(e.target);e.offsetTop=n?r.offset(n).top:null,c.offset&&null!==e.offsetTop&&(e.offsetTop-=1*c.offset)}),C=k.filter(function(e){return null!==e.offsetTop}).sort(function(e,t){return e.offsetTop-t.offsetTop}),b()},x.trackElement=function(e,t){k.push({target:e,source:t})},x.untrackElement=function(e,t){for(var n,a=k.length;a--;)if(k[a].target===e&&k[a].source===t){n=a;break}k.splice(n,1)},x.activate=function(e){k[e].addClass('active')},x.init(),x}var d=angular.element(a),f=angular.element(o.prop('documentElement')),p=angular.element(a.document.body);return c}]}).directive('bsScrollspy',['$rootScope','bsDebounce','bsDimensions','$bsScrollspy',function(e,t,n,a){return{restrict:'EAC',link:function(e,t,n){var o={scope:e};angular.forEach(['offset','target'],function(e){angular.isDefined(n[e])&&(o[e]=n[e])});var i=a(o);i.trackElement(o.target,t),e.$on('$destroy',function(){i&&(i.untrackElement(o.target,t),i.destroy()),o=null,i=null})}}}]).directive('bsScrollspyList',['$rootScope','bsDebounce','bsDimensions','$bsScrollspy',function(e,t,n,a){return{restrict:'A',compile:function(e,t){var n=e[0].querySelectorAll('li > a[href]');angular.forEach(n,function(e){var t=angular.element(e);t.parent().attr('bs-scrollspy','').attr('data-target',t.attr('href'))})}}}]),angular.module('mgcrea.ngStrap.popover',['mgcrea.ngStrap.tooltip']).provider('$bsPopover',function(){var e=this.defaults={animation:'am-fade',customClass:'',container:!1,target:!1,placement:'right',templateUrl:'popover/popover.tpl.html',contentTemplate:!1,trigger:'click',keyboard:!0,html:!1,title:'',content:'',delay:0,autoClose:!1};this.$get=['$bsTooltip',function(t){function n(n,a){var o=angular.extend({},e,a),i=t(n,o);return o.content&&(i.$scope.content=o.content),i}return n}]}).directive('bsPopover',['$window','$sce','$bsPopover',function(e,t,n){var a=e.requestAnimationFrame||e.setTimeout;return{restrict:'EAC',scope:!0,link:function(e,o,i){var r,s={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','container','delay','trigger','html','animation','customClass','autoClose','id','prefixClass','prefixEvent','bsEnabled'],function(e){angular.isDefined(i[e])&&(s[e]=i[e])});var l=/^(false|0|)$/i;angular.forEach(['html','container','autoClose'],function(e){angular.isDefined(i[e])&&l.test(i[e])&&(s[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(i[n])&&(s[t]=e.$eval(i[n]))});var u=o.attr('data-target');angular.isDefined(u)&&(l.test(u)?s.target=!1:s.target=u),angular.forEach(['title','content'],function(n){i[n]&&i.$observe(n,function(o,i){e[n]=t.trustAsHtml(o),angular.isDefined(i)&&a(function(){r&&r.$applyPlacement()})})}),i.bsPopover&&e.$watch(i.bsPopover,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t,angular.isDefined(n)&&a(function(){r&&r.$applyPlacement()})},!0),i.bsShow&&e.$watch(i.bsShow,function(e,t){r&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(popover),?/i)),e===!0?r.show():r.hide())}),i.bsEnabled&&e.$watch(i.bsEnabled,function(e){r&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|1|,?(popover),?/i)),e===!1?r.setEnabled(!1):r.setEnabled(!0))}),i.viewport&&e.$watch(i.viewport,function(e){r&&angular.isDefined(e)&&r.setViewport(e)}),r=n(o,s),e.$on('$destroy',function(){r&&r.destroy(),s=null,r=null})}}}]),angular.module('mgcrea.ngStrap.navbar',[]).provider('$bsNavbar',function(){var e=this.defaults={activeClass:'active',routeAttr:'data-match-route',strict:!1};this.$get=function(){return{defaults:e}}}).directive('bsNavbar',['$window','$location','$bsNavbar',function(e,t,n){var a=n.defaults;return{restrict:'A',link:function(e,n,o,i){var r=angular.copy(a);angular.forEach(Object.keys(a),function(e){angular.isDefined(o[e])&&(r[e]=o[e])}),e.$watch(function(){return t.path()},function(e,t){var a=n[0].querySelectorAll('li['+r.routeAttr+']');angular.forEach(a,function(t){var n=angular.element(t),a=n.attr(r.routeAttr).replace('/','\\/');r.strict&&(a='^'+a+'$');var o=new RegExp(a,'i');o.test(e)?n.addClass(r.activeClass):n.removeClass(r.activeClass)})})}}}]),angular.module('mgcrea.ngStrap.modal',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$bsModal',function(){var e=this.defaults={animation:'am-fade',backdropAnimation:'am-fade',customClass:'',prefixClass:'modal',prefixEvent:'modal',placement:'top',templateUrl:'modal/modal.tpl.html',template:'',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0,size:null,zIndex:null};this.$get=['$window','$rootScope','$bsCompiler','$animate','$timeout','$sce','bsDimensions',function(n,a,o,i,r,s,l){function u(t){function n(){C.$emit(x.prefixEvent+'.show',T),angular.isDefined(x.onShow)&&angular.isFunction(x.onShow)&&x.onShow(T)}function r(){C.$emit(x.prefixEvent+'.hide',T),angular.isDefined(x.onHide)&&angular.isFunction(x.onHide)&&x.onHide(T),d('.modal').length<=0&&g.removeClass(x.prefixClass+'-open'),x.animation&&g.removeClass(x.prefixClass+'-with-'+x.animation)}function l(){x.backdrop&&(A.on('click',y),F.on('click',y),F.on('wheel',D))}function u(){x.backdrop&&(A.off('click',y),F.off('click',y),F.off('wheel',D))}function w(){x.keyboard&&A.on('keyup',T.$onKeyUp)}function b(){x.keyboard&&A.off('keyup',T.$onKeyUp)}function y(e){e.target===e.currentTarget&&('static'===x.backdrop?T.focus():T.hide())}function D(e){e.preventDefault()}function S(){T.$isShown&&null!==A&&(u(),b()),M&&(M.$destroy(),M=null),A&&(A.remove(),A=T.$element=null)}var T={},x=T.$options=angular.extend({},e,t),k=T.$promise=o.compile(x),C=T.$scope=x.scope&&x.scope.$new()||a.$new();x.element||x.container||(x.container='body'),x.zIndex&&($=parseInt(x.zIndex,10),h=$-10),T.$id=x.id||x.element&&x.element.attr('id')||'',f(['title','content'],function(e){x[e]&&(C[e]=s.trustAsHtml(x[e]))}),C.$hide=function(){C.$$postDigest(function(){T.hide()})},C.$show=function(){C.$$postDigest(function(){T.show()})},C.$toggle=function(){C.$$postDigest(function(){T.toggle()})},T.$isShown=C.$isShown=!1;var E,A,M,F=angular.element('
');return F.css({position:'fixed',top:'0px',left:'0px',bottom:'0px',right:'0px'}),k.then(function(e){E=e,T.init()}),T.init=function(){x.show&&C.$$postDigest(function(){T.show()})},T.destroy=function(){S(),F&&(F.remove(),F=null),C.$destroy()},T.show=function(){if(!T.$isShown){var e,t;if(angular.isElement(x.container)?(e=x.container,t=x.container[0].lastChild?angular.element(x.container[0].lastChild):null):x.container?(e=d(x.container),t=e[0]&&e[0].lastChild?angular.element(e[0].lastChild):null):(e=null,t=x.element),A&&S(),M=T.$scope.$new(),A=T.$element=E.link(M,function(e,t){}),x.backdrop&&(A.css({'z-index':$+20*m}),F.css({'z-index':h+20*m}),m++),!C.$emit(x.prefixEvent+'.show.before',T).defaultPrevented){angular.isDefined(x.onBeforeShow)&&angular.isFunction(x.onBeforeShow)&&x.onBeforeShow(T),A.css({display:'block'}).addClass(x.placement),x.customClass&&A.addClass(x.customClass),x.size&&v[x.size]&&angular.element(d('.modal-dialog',A[0])).addClass(v[x.size]),x.animation&&(x.backdrop&&F.addClass(x.backdropAnimation),A.addClass(x.animation)),x.backdrop&&i.enter(F,g,null),angular.version.minor<=2?i.enter(A,e,t,n):i.enter(A,e,t).then(n),T.$isShown=C.$isShown=!0,c(C);var a=A[0];p(function(){a.focus()}),g.addClass(x.prefixClass+'-open'),x.animation&&g.addClass(x.prefixClass+'-with-'+x.animation),l(),w()}}},T.hide=function(){T.$isShown&&(C.$emit(x.prefixEvent+'.hide.before',T).defaultPrevented||(angular.isDefined(x.onBeforeHide)&&angular.isFunction(x.onBeforeHide)&&x.onBeforeHide(T),angular.version.minor<=2?i.leave(A,r):i.leave(A).then(r),x.backdrop&&(m--,i.leave(F)),T.$isShown=C.$isShown=!1,c(C),u(),b()))},T.toggle=function(){T.$isShown?T.hide():T.show()},T.focus=function(){A[0].focus()},T.$onKeyUp=function(e){27===e.which&&T.$isShown&&(T.hide(),e.stopPropagation())},T}function c(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}function d(e,n){return angular.element((n||t).querySelectorAll(e))}var f=angular.forEach,p=n.requestAnimationFrame||n.setTimeout,g=angular.element(n.document.body),m=0,$=1050,h=1040,v={lg:'modal-lg',sm:'modal-sm'};return u}]}).directive('bsModal',['$window','$sce','$parse','$bsModal',function(e,t,n,a){return{restrict:'EAC',scope:!0,link:function(e,n,o,i){var r={scope:e,element:n,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation','backdropAnimation','id','prefixEvent','prefixClass','customClass','modalClass','size','zIndex'],function(e){angular.isDefined(o[e])&&(r[e]=o[e])}),r.modalClass&&(r.customClass=r.modalClass);var s=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(e){angular.isDefined(o[e])&&s.test(o[e])&&(r[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(o[n])&&(r[t]=e.$eval(o[n]))}),angular.forEach(['title','content'],function(n){o[n]&&o.$observe(n,function(a,o){e[n]=t.trustAsHtml(a)})}),o.bsModal&&e.$watch(o.bsModal,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t},!0);var l=a(r);n.on(o.trigger||'click',l.toggle),e.$on('$destroy',function(){l&&l.destroy(),r=null,l=null})}}}]),angular.version.minor<3&&angular.version.dot<14&&angular.module('ng').factory('$$rAF',['$window','$timeout',function(e,t){var n=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame,a=e.cancelAnimationFrame||e.webkitCancelAnimationFrame||e.mozCancelAnimationFrame||e.webkitCancelRequestAnimationFrame,o=!!n,i=o?function(e){var t=n(e);return function(){a(t)}}:function(e){var n=t(e,16.66,!1);return function(){t.cancel(n)}};return i.supported=o,i}]),angular.module('mgcrea.ngStrap.helpers.parseOptions',[]).provider('$bsParseOptions',function(){var e=this.defaults={regexp:/^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/};this.$get=['$parse','$q',function(t,n){function a(a,o){function i(e,t){return e.map(function(e,n){var a,o,i={};return i[c]=e,a=u(t,i),o=p(t,i),{label:a,value:o,index:n}})}var r={},s=angular.extend({},e,o);r.$values=[];var l,u,c,d,f,p,g;return r.init=function(){r.$match=l=a.match(s.regexp),u=t(l[2]||l[1]),c=l[4]||l[6],d=l[5],f=t(l[3]||''),p=t(l[2]?l[1]:c),g=t(l[7])},r.valuesFn=function(e,t){return n.when(g(e,t)).then(function(t){return angular.isArray(t)||(t=[]),r.$values=t.length?i(t,e):[],r.$values})},r.displayValue=function(e){var t={};return t[c]=e,u(t)},r.init(),r}return a}]}),angular.module('mgcrea.ngStrap.helpers.dimensions',[]).factory('bsDimensions',function(){function t(e){var t=e.ownerDocument,o=e.offsetParent||t;if(a(o,'#document'))return t.documentElement;for(;o&&!a(o,'html')&&'static'===n.css(o,'position');)o=o.offsetParent;return o||t.documentElement}var n={},a=n.nodeName=function(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()};return n.css=function(t,n,a){var o;return o=t.currentStyle?t.currentStyle[n]:e.getComputedStyle?e.getComputedStyle(t)[n]:t.style[n],a===!0?parseFloat(o)||0:o},n.offset=function(t){var n=t.getBoundingClientRect(),a=t.ownerDocument;return{width:n.width||t.offsetWidth,height:n.height||t.offsetHeight,top:n.top+(e.pageYOffset||a.documentElement.scrollTop)-(a.documentElement.clientTop||0),left:n.left+(e.pageXOffset||a.documentElement.scrollLeft)-(a.documentElement.clientLeft||0)}},n.setOffset=function(e,t,a){var o,i,r,s,l,u,c,d=n.css(e,'position'),f=angular.element(e),p={};'static'===d&&(e.style.position='relative'),l=n.offset(e),r=n.css(e,'top'),u=n.css(e,'left'),c=('absolute'===d||'fixed'===d)&&(r+u).indexOf('auto')>-1,c?(o=n.position(e),s=o.top,i=o.left):(s=parseFloat(r)||0,i=parseFloat(u)||0),angular.isFunction(t)&&(t=t.call(e,a,l)),null!==t.top&&(p.top=t.top-l.top+s),null!==t.left&&(p.left=t.left-l.left+i),'using'in t?t.using.call(f,p):f.css({top:p.top+'px',left:p.left+'px'})},n.position=function(e){var o,i,r={top:0,left:0};return'fixed'===n.css(e,'position')?i=e.getBoundingClientRect():(o=t(e),i=n.offset(e),a(o,'html')||(r=n.offset(o)),r.top+=n.css(o,'borderTopWidth',!0),r.left+=n.css(o,'borderLeftWidth',!0)),{width:e.offsetWidth,height:e.offsetHeight,top:i.top-r.top-n.css(e,'marginTop',!0),left:i.left-r.left-n.css(e,'marginLeft',!0)}},n.height=function(e,t){var a=e.offsetHeight;return t?a+=n.css(e,'marginTop',!0)+n.css(e,'marginBottom',!0):a-=n.css(e,'paddingTop',!0)+n.css(e,'paddingBottom',!0)+n.css(e,'borderTopWidth',!0)+n.css(e,'borderBottomWidth',!0),a},n.width=function(e,t){var a=e.offsetWidth;return t?a+=n.css(e,'marginLeft',!0)+n.css(e,'marginRight',!0):a-=n.css(e,'paddingLeft',!0)+n.css(e,'paddingRight',!0)+n.css(e,'borderLeftWidth',!0)+n.css(e,'borderRightWidth',!0),a},n}),angular.module('mgcrea.ngStrap.helpers.debounce',[]).factory('bsDebounce',['$timeout',function(e){return function(t,n,a){var o=null;return function(){var i=this,r=arguments,s=a&&!o;return o&&e.cancel(o),o=e(function(){o=null,a||t.apply(i,r)},n,!1),s&&t.apply(i,r),o}}}]).factory('bsThrottle',['$timeout',function(e){return function(t,n,a){var o=null;return a||(a={}),function(){var i=this,r=arguments;o||(a.leading!==!1&&t.apply(i,r),o=e(function(){o=null,a.trailing!==!1&&t.apply(i,r)},n,!1))}}}]),angular.module('mgcrea.ngStrap.helpers.dateParser',[]).provider('$bsDateParser',['$localeProvider',function(e){function t(){this.year=1970,this.month=0,this.day=1,this.hours=0,this.minutes=0,this.seconds=0,this.milliseconds=0}function n(){}function a(e){return!isNaN(parseFloat(e))&&isFinite(e)}function o(e,t){for(var n=e.length,a=t.toString().toLowerCase(),o=0;o12?e.getHours()+2:0),e):null},b.timezoneOffsetAdjust=function(e,t,n){return e?(t&&'UTC'===t&&(e=new Date(e.getTime()),e.setMinutes(e.getMinutes()+(n?-1:1)*e.getTimezoneOffset())),e):null},b.init(),b};return l}]}]),angular.module('mgcrea.ngStrap.helpers.dateFormatter',[]).service('$bsDateFormatter',['$locale','dateFilter',function(e,t){function n(e){return/(h+)([:\.])?(m+)([:\.])?(s*)[ ]?(a?)/i.exec(e).slice(1)}this.getDefaultLocale=function(){return e.id},this.getDatetimeFormat=function(t,n){return e.DATETIME_FORMATS[t]||t},this.weekdaysShort=function(t){return e.DATETIME_FORMATS.SHORTDAY},this.hoursFormat=function(e){return n(e)[0]},this.minutesFormat=function(e){return n(e)[2]},this.secondsFormat=function(e){return n(e)[4]},this.timeSeparator=function(e){return n(e)[1]},this.showSeconds=function(e){return!!n(e)[4]},this.showAM=function(e){return!!n(e)[5]},this.formatDate=function(e,n,a,o){return t(e,n,o)}}]),angular.module('mgcrea.ngStrap.core',[]).service('$bsCompiler',a),angular.module('mgcrea.ngStrap.dropdown',['mgcrea.ngStrap.tooltip']).provider('$bsDropdown',function(){var e=this.defaults={animation:'am-fade',prefixClass:'dropdown',prefixEvent:'dropdown',placement:'bottom-left',templateUrl:'dropdown/dropdown.tpl.html',trigger:'click',container:!1,keyboard:!0,html:!1,delay:0};this.$get=['$window','$rootScope','$bsTooltip','$timeout',function(t,n,a,o){function i(t,i){function l(e){if(e.target!==t[0])return e.target!==t[0]&&u.hide()}var u={},c=angular.extend({},e,i);u.$scope=c.scope&&c.scope.$new()||n.$new(),u=a(t,c);var d=t.parent();u.$onKeyDown=function(e){if(/(38|40)/.test(e.keyCode)){e.preventDefault(),e.stopPropagation();var t=angular.element(u.$element[0].querySelectorAll('li:not(.divider) a'));if(t.length){var n;angular.forEach(t,function(e,t){s&&s.call(e,':focus')&&(n=t)}),38===e.keyCode&&n>0?n--:40===e.keyCode&&n=0&&(t.template=o.outerHTML,t.templateUrl=n,o.parentNode.removeChild(o))}return function(e,n,o){var i={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','id','autoClose'],function(e){angular.isDefined(t[e])&&(i[e]=t[e])});var r=/^(false|0|)$/i;angular.forEach(['html','container'],function(e){angular.isDefined(o[e])&&r.test(o[e])&&(i[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(o[n])&&(i[t]=e.$eval(o[n]))}),o.bsDropdown&&e.$watch(o.bsDropdown,function(t,n){e.content=t},!0);var s=a(n,i);o.bsShow&&e.$watch(o.bsShow,function(e,t){s&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(dropdown),?/i)),e===!0?s.show():s.hide())}),e.$on('$destroy',function(){s&&s.destroy(),i=null,s=null})}}}}]),angular.module('mgcrea.ngStrap.button',[]).provider('$bsButton',function(){var e=this.defaults={activeClass:'active',toggleEvent:'click'};this.$get=function(){return{defaults:e}}}).directive('bsCheckboxGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var n=e[0].querySelectorAll('input[type="checkbox"]');angular.forEach(n,function(e){var n=angular.element(e);n.attr('bs-checkbox',''),n.attr('ng-model',t.ngModel+'.'+n.attr('value'))})}}}).directive('bsCheckbox',['$bsButton','$$rAF',function(e,t){var n=e.defaults,a=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,o,i,r){var s=n,l='INPUT'===o[0].nodeName,u=l?o.parent():o,c=!angular.isDefined(i.trueValue)||i.trueValue;a.test(i.trueValue)&&(c=e.$eval(i.trueValue));var d=!!angular.isDefined(i.falseValue)&&i.falseValue;a.test(i.falseValue)&&(d=e.$eval(i.falseValue));var f='boolean'!=typeof c||'boolean'!=typeof d;f&&(r.$parsers.push(function(e){return e?c:d}),r.$formatters.push(function(e){return angular.equals(e,c)})),r.$render=function(){var e=!!r.$viewValue;t(function(){l&&(o[0].checked=e),u.toggleClass(s.activeClass,e)})},o.bind(s.toggleEvent,function(){e.$apply(function(){l||r.$setViewValue(!u.hasClass('active')),r.$render()})})}}}]).directive('bsRadioGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var n=e[0].querySelectorAll('input[type="radio"]');angular.forEach(n,function(e){angular.element(e).attr('bs-radio',''),angular.element(e).attr('ng-model',t.ngModel)})}}}).directive('bsRadio',['$bsButton','$$rAF',function(e,t){var n=e.defaults,a=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,o,i,r){var s,l=n,u='INPUT'===o[0].nodeName,c=u?o.parent():o;i.$observe('value',function(t){s='boolean'!=typeof t&&a.test(t)?e.$eval(t):t,r.$render()}),r.$render=function(){var e=angular.equals(r.$viewValue,s);t(function(){u&&(o[0].checked=e),c.toggleClass(l.activeClass,e)})},o.bind(l.toggleEvent,function(){e.$apply(function(){r.$setViewValue(s),r.$render()})})}}}]),angular.module('mgcrea.ngStrap.datepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$bsDatepicker',function(){var e=this.defaults={animation:'am-fade',prefixClass:'datepicker',placement:'bottom-left',templateUrl:'datepicker/datepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!1,dateType:'date',dateFormat:'shortDate',timezone:null,modelDateFormat:null,dayFormat:'dd',monthFormat:'MMM',yearFormat:'yyyy',monthTitleFormat:'MMMM yyyy',yearTitleFormat:'yyyy',strictFormat:!1,autoclose:!1,minDate:-(1/0),maxDate:+(1/0),startView:0,minView:0,startWeek:0,daysOfWeekDisabled:'',hasToday:!1,hasClear:!1,iconLeft:'glyphicon glyphicon-chevron-left',iconRight:'glyphicon glyphicon-chevron-right'};this.$get=['$window','$document','$rootScope','$sce','$bsDateFormatter','bsDatepickerViews','$bsTooltip','$timeout',function(t,n,a,o,i,r,s,l){function u(t,n,a){function o(e){e.selected=u.$isSelected(e.date)}function i(){t[0].focus()}var u=s(t,angular.extend({},e,a)),f=a.scope,p=u.$options,g=u.$scope;p.startView&&(p.startView-=p.minView);var m=r(u);u.$views=m.views;var $=m.viewDate;g.$mode=p.startView,g.$iconLeft=p.iconLeft,g.$iconRight=p.iconRight,g.$hasToday=p.hasToday,g.$hasClear=p.hasClear;var h=u.$views[g.$mode];g.$select=function(e,t){t||u.select(e)},g.$selectPane=function(e){u.$selectPane(e)},g.$toggleMode=function(){u.setMode((g.$mode+1)%u.$views.length)},g.$setToday=function(){p.autoclose?(u.setMode(0),u.select(new Date)):u.select(new Date,!0)},g.$clear=function(){p.autoclose?(u.setMode(0),u.select(null)):u.select(null,!0)},u.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())&&(u.$date=e,h.update.call(h,e)),u.$build(!0)},u.updateDisabledDates=function(e){p.disabledDateRanges=e;for(var t=0,n=g.rows.length;t=p.$options.minDate,n=isNaN(p.$options.maxDate)||e.getTime()<=p.$options.maxDate,a=t&&n;s.$setValidity('date',a),s.$setValidity('min',t),s.$setValidity('max',n),a&&(s.$dateValue=e)}}function c(){return!s.$dateValue||isNaN(s.$dateValue.getTime())?'':m(s.$dateValue,d.dateFormat)}var d={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','html','animation','autoclose','dateType','dateFormat','timezone','modelDateFormat','dayFormat','strictFormat','startWeek','startDate','useNative','lang','startView','minView','iconLeft','iconRight','daysOfWeekDisabled','id','prefixClass','prefixEvent','hasToday','hasClear'],function(e){angular.isDefined(n[e])&&(d[e]=n[e])});var f=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','hasToday','hasClear'],function(e){angular.isDefined(n[e])&&f.test(n[e])&&(d[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(d[t]=e.$eval(n[a]))});var p=i(t,s,d);d=p.$options,r&&d.useNative&&(d.dateFormat='yyyy-MM-dd');var g=d.lang,m=function(e,t){return a.formatDate(e,t,g)},$=o({format:d.dateFormat,lang:g,strict:d.strictFormat});n.bsShow&&e.$watch(n.bsShow,function(e,t){p&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(datepicker),?/i)),e===!0?p.show():p.hide())}),angular.forEach(['minDate','maxDate'],function(e){angular.isDefined(n[e])&&n.$observe(e,function(t){p.$options[e]=$.getDateForAttribute(e,t),isNaN(p.$options[e])||p.$build(!1),u(s.$dateValue)})}),angular.isDefined(n.dateFormat)&&n.$observe('dateFormat',function(e){p.$options.dateFormat=e}),e.$watch(n.ngModel,function(e,t){p.update(s.$dateValue)},!0),angular.isDefined(n.disabledDates)&&e.$watch(n.disabledDates,function(e,t){e=l(e),t=l(t),e&&p.updateDisabledDates(e)}),s.$parsers.unshift(function(e){var t;if(!e)return s.$setValidity('date',!0),null;var n=$.parse(e,s.$dateValue);return!n||isNaN(n.getTime())?void s.$setValidity('date',!1):(u(n),'string'===d.dateType?(t=$.timezoneOffsetAdjust(n,d.timezone,!0),m(t,d.modelDateFormat||d.dateFormat)):(t=$.timezoneOffsetAdjust(s.$dateValue,d.timezone,!0),'number'===d.dateType?t.getTime():'unix'===d.dateType?t.getTime()/1e3:'iso'===d.dateType?t.toISOString():new Date(t)))}),s.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===d.dateType?$.parse(e,null,d.modelDateFormat):'unix'===d.dateType?new Date(1e3*e):new Date(e),s.$dateValue=$.timezoneOffsetAdjust(t,d.timezone),c()}),s.$render=function(){t.val(c())},e.$on('$destroy',function(){p&&p.destroy(),d=null,p=null})}}}]).provider('bsDatepickerViews',function(){function e(e,t){for(var n=[];e.length>0;)n.push(e.splice(0,t));return n}function t(e,t){return(e%t+t)%t}this.$get=['$bsDateFormatter','$bsDateParser','$sce',function(n,a,o){return function(i){var r=i.$scope,s=i.$options,l=s.lang,u=function(e,t){return n.formatDate(e,t,l)},c=a({format:s.dateFormat,lang:l,strict:s.strictFormat}),d=n.weekdaysShort(l),f=d.slice(s.startWeek).concat(d.slice(0,s.startWeek)),p=o.trustAsHtml(''+f.join('')+''),g=i.$date||(s.startDate?c.getDateForAttribute('startDate',s.startDate):new Date),m={year:g.getFullYear(),month:g.getMonth(),date:g.getDate()},$=[{format:s.dayFormat,split:7,steps:{month:1},update:function(e,t){!this.built||t||e.getFullYear()!==m.year||e.getMonth()!==m.month?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):e.getDate()===m.date&&1!==e.getDate()||(m.date=i.$date.getDate(),i.$updateSelected())},build:function(){var n=new Date(m.year,m.month,1),a=n.getTimezoneOffset(),o=new Date(+n-864e5*t(n.getDay()-s.startWeek,7)),l=o.getTimezoneOffset(),d=c.timezoneOffsetAdjust(new Date,s.timezone).toDateString();l!==a&&(o=new Date(+o+6e4*(l-a)));for(var f,g=[],$=0;$<42;$++)f=c.daylightSavingAdjust(new Date(o.getFullYear(),o.getMonth(),o.getDate()+$)),g.push({date:f,isToday:f.toDateString()===d,label:u(f,this.format),selected:i.$date&&this.isSelected(f),muted:f.getMonth()!==m.month,disabled:this.isDisabled(f)});r.title=u(n,s.monthTitleFormat),r.showLabels=!0,r.labels=p,r.rows=e(g,this.split),r.isTodayDisabled=this.isDisabled(new Date),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()&&e.getDate()===i.$date.getDate()},isDisabled:function(e){var t=e.getTime();if(ts.maxDate)return!0;if(s.daysOfWeekDisabled.indexOf(e.getDay())!==-1)return!0;if(s.disabledDateRanges)for(var n=0;n=s.disabledDateRanges[n].start&&t<=s.disabledDateRanges[n].end)return!0;return!1},onKeyDown:function(e){if(i.$date){var t,n=i.$date.getTime();37===e.keyCode?t=new Date(n-864e5):38===e.keyCode?t=new Date(n-6048e5):39===e.keyCode?t=new Date(n+864e5):40===e.keyCode&&(t=new Date(n+6048e5)),this.isDisabled(t)||i.select(t,!0)}}},{name:'month',format:s.monthFormat,split:4,steps:{year:1},update:function(e,t){this.built&&e.getFullYear()===m.year?e.getMonth()!==m.month&&(angular.extend(m,{month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected()):(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build())},build:function(){for(var t,n=[],a=0;a<12;a++)t=new Date(m.year,a,1),n.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=u(t,s.yearTitleFormat),r.showLabels=!1,r.rows=e(n,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()},isDisabled:function(e){var t=+new Date(e.getFullYear(),e.getMonth()+1,0);return ts.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getMonth(),n=new Date(i.$date);37===e.keyCode?n.setMonth(t-1):38===e.keyCode?n.setMonth(t-4):39===e.keyCode?n.setMonth(t+1):40===e.keyCode&&n.setMonth(t+4),this.isDisabled(n)||i.select(n,!0)}}},{name:'year',format:s.yearFormat,split:4,steps:{year:12},update:function(e,t){!this.built||t||parseInt(e.getFullYear()/20,10)!==parseInt(m.year/20,10)?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):e.getFullYear()!==m.year&&(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected())},build:function(){for(var t,n=m.year-m.year%(3*this.split),a=[],o=0;o<12;o++)t=new Date(n+o,0,1),a.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=a[0].label+'-'+a[a.length-1].label,r.showLabels=!1,r.rows=e(a,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()},isDisabled:function(e){var t=+new Date(e.getFullYear()+1,0,0);return ts.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getFullYear(),n=new Date(i.$date);37===e.keyCode?n.setYear(t-1):38===e.keyCode?n.setYear(t-4):39===e.keyCode?n.setYear(t+1):40===e.keyCode&&n.setYear(t+4),this.isDisabled(n)||i.select(n,!0)}}}];return{views:s.minView?Array.prototype.slice.call($,s.minView):$,viewDate:m}}}]}),angular.module('mgcrea.ngStrap.collapse',[]).provider('$bsCollapse',function(){var e=this.defaults={animation:'am-collapse',disallowToggle:!1,activeClass:'in',startCollapsed:!1,allowMultiple:!1},t=this.controller=function(t,n,a){function o(e){for(var t=l.$targets.$active,n=0;n=o-w?'bottom':'middle'}function u(){return p[0]===t?t.pageYOffset:p[0].scrollTop}function c(){return p[0]===t?t.document.body.scrollHeight:p[0].scrollHeight}var d={},f=angular.extend({},e,s),p=f.target,g='affix affix-top affix-bottom',m=!1,$=0,h=0,v=0,w=0,b=null,y=null,D=o.parent();if(f.offsetParent)if(f.offsetParent.match(/^\d+$/))for(var S=0;S<1*f.offsetParent-1;S++)D=D.parent();else D=angular.element(f.offsetParent);return d.init=function(){this.$parseOffsets(),h=a.offset(o[0]).top+$,m=f.setWidth&&!o[0].style.width,p.on('scroll',this.checkPosition),p.on('click',this.checkPositionWithEventLoop),r.on('resize',this.$debouncedOnResize),this.checkPosition(),this.checkPositionWithEventLoop()},d.destroy=function(){p.off('scroll',this.checkPosition),p.off('click',this.checkPositionWithEventLoop),r.off('resize',this.$debouncedOnResize)},d.checkPositionWithEventLoop=function(){setTimeout(d.checkPosition,1)},d.checkPosition=function(){var e=u(),t=a.offset(o[0]),n=a.height(o[0]),r=l(y,t,n);b!==r&&(b=r,'top'===r?(y=null,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',''))):'bottom'===r?(y=f.offsetUnpin?-(1*f.offsetUnpin):t.top-e,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',f.offsetParent?'':i[0].offsetHeight-w-n-h+'px'))):(y=null,m&&o.css('width',o[0].offsetWidth+'px'),f.inlineStyles&&(o.css('position','fixed'),o.css('top',$+'px'))),o.removeClass(g).addClass('affix'+('middle'!==r?'-'+r:'')))},d.$onResize=function(){d.$parseOffsets(),d.checkPosition()},d.$debouncedOnResize=n(d.$onResize,50),d.$parseOffsets=function(){var e=o[0].style.position,t=o[0].style.top;f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top','')),f.offsetTop&&('auto'===f.offsetTop&&(f.offsetTop='+0'),f.offsetTop.match(/^[-+]\d+$/)?($=1*-f.offsetTop,v=f.offsetParent?a.offset(D[0]).top+1*f.offsetTop:a.offset(o[0]).top-a.css(o[0],'marginTop',!0)+1*f.offsetTop):v=1*f.offsetTop),f.offsetBottom&&(w=f.offsetParent&&f.offsetBottom.match(/^[-+]\d+$/)?c()-(a.offset(D[0]).top+a.height(D[0]))+1*f.offsetBottom+1:1*f.offsetBottom),f.inlineStyles&&(o.css('position',e),o.css('top',t))},d.init(),d}var i=angular.element(t.document.body),r=angular.element(t);return o}]}).directive('bsAffix',['$bsAffix','$window','$timeout',function(e,t,n){return{restrict:'EAC',require:'^?bsAffixTarget',link:function(a,o,i,r){var s={scope:a,target:r?r.$element:angular.element(t)};angular.forEach(['offsetTop','offsetBottom','offsetParent','offsetUnpin','inlineStyles','setWidth'],function(e){if(angular.isDefined(i[e])){var t=i[e];/true/i.test(t)&&(t=!0),/false/i.test(t)&&(t=!1),s[e]=t}});var l;n(function(){l=e(o,s)}),a.$on('$destroy',function(){l&&l.destroy(),s=null,l=null})}}}]).directive('bsAffixTarget',function(){return{controller:['$element',function(e){this.$element=e}]}}),angular.module('mgcrea.ngStrap',['mgcrea.ngStrap.modal','mgcrea.ngStrap.aside','mgcrea.ngStrap.alert','mgcrea.ngStrap.button','mgcrea.ngStrap.select','mgcrea.ngStrap.datepicker','mgcrea.ngStrap.timepicker','mgcrea.ngStrap.navbar','mgcrea.ngStrap.tooltip','mgcrea.ngStrap.popover','mgcrea.ngStrap.dropdown','mgcrea.ngStrap.typeahead','mgcrea.ngStrap.scrollspy','mgcrea.ngStrap.affix','mgcrea.ngStrap.tab','mgcrea.ngStrap.collapse'])}(window,document); \ No newline at end of file diff --git a/dist/angular-strap.js b/dist/angular-strap.js index 88d922bc9..3bf29196c 100644 --- a/dist/angular-strap.js +++ b/dist/angular-strap.js @@ -1,12 +1,13 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ (function(window, document, undefined) { 'use strict'; + bsCompilerService.$inject = [ '$q', '$http', '$injector', '$compile', '$controller', '$templateCache' ]; angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions' ]).provider('$typeahead', function() { var defaults = this.defaults = { animation: 'am-fade', @@ -27,7 +28,6 @@ trimValue: true }; this.$get = [ '$window', '$rootScope', '$tooltip', '$$rAF', '$timeout', function($window, $rootScope, $tooltip, $$rAF, $timeout) { - var bodyEl = angular.element($window.document.body); function TypeaheadFactory(element, controller, config) { var $typeahead = {}; var options = angular.extend({}, defaults, config); @@ -71,6 +71,9 @@ scope.$resetMatches(); if (parentScope) parentScope.$digest(); scope.$emit(options.prefixEvent + '.select', value, index, $typeahead); + if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) { + options.onSelect(value, index, $typeahead); + } }; $typeahead.$isVisible = function() { if (!options.minLength || !controller) { @@ -79,18 +82,30 @@ return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength; }; $typeahead.$getIndex = function(value) { - var l = scope.$matches.length, i = l; - if (!l) return; - for (i = l; i--; ) { - if (scope.$matches[i].value === value) break; + var index; + for (index = scope.$matches.length; index--; ) { + if (angular.equals(scope.$matches[index].value, value)) break; } - if (i < 0) return; - return i; + return index; }; $typeahead.$onMouseDown = function(evt) { evt.preventDefault(); evt.stopPropagation(); }; + $typeahead.$$updateScrollTop = function(container, index) { + if (index > -1 && index < container.children.length) { + var active = container.children[index]; + var clientTop = active.offsetTop; + var clientBottom = active.offsetTop + active.clientHeight; + var highWatermark = container.scrollTop; + var lowWatermark = container.scrollTop + container.clientHeight; + if (clientBottom >= highWatermark && clientTop < highWatermark) { + container.scrollTop = Math.max(0, container.scrollTop - container.clientHeight); + } else if (clientBottom > lowWatermark) { + container.scrollTop = clientTop; + } + } + }; $typeahead.$onKeyDown = function(evt) { if (!/(38|40|13)/.test(evt.keyCode)) return; if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) { @@ -99,26 +114,37 @@ } if (evt.keyCode === 13 && scope.$matches.length) { $typeahead.select(scope.$activeIndex); - } else if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--; else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++; else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0; + } else if (evt.keyCode === 38 && scope.$activeIndex > 0) { + scope.$activeIndex--; + } else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) { + scope.$activeIndex++; + } else if (angular.isUndefined(scope.$activeIndex)) { + scope.$activeIndex = 0; + } + $typeahead.$$updateScrollTop($typeahead.$element[0], scope.$activeIndex); scope.$digest(); }; var show = $typeahead.show; $typeahead.show = function() { show(); $timeout(function() { - $typeahead.$element && $typeahead.$element.on('mousedown', $typeahead.$onMouseDown); - if (options.keyboard) { - element && element.on('keydown', $typeahead.$onKeyDown); + if ($typeahead.$element) { + $typeahead.$element.on('mousedown', $typeahead.$onMouseDown); + if (options.keyboard) { + if (element) element.on('keydown', $typeahead.$onKeyDown); + } } }, 0, false); }; var hide = $typeahead.hide; $typeahead.hide = function() { - $typeahead.$element && $typeahead.$element.off('mousedown', $typeahead.$onMouseDown); + if ($typeahead.$element) $typeahead.$element.off('mousedown', $typeahead.$onMouseDown); if (options.keyboard) { - element && element.off('keydown', $typeahead.$onKeyDown); + if (element) element.off('keydown', $typeahead.$onKeyDown); + } + if (!options.autoSelect) { + $typeahead.activate(-1); } - if (!options.autoSelect) $typeahead.activate(-1); hide(); }; return $typeahead; @@ -135,9 +161,8 @@ return array.then(function(results) { return $filter('filter')(results, expression, comparator); }); - } else { - return $filter('filter')(array, expression, comparator); } + return $filter('filter')(array, expression, comparator); }; } ]).directive('bsTypeahead', [ '$window', '$parse', '$q', '$typeahead', '$parseOptions', function($window, $parse, $q, $typeahead, $parseOptions) { var defaults = $typeahead.defaults; @@ -145,6 +170,7 @@ restrict: 'EAC', require: 'ngModel', link: function postLink(scope, element, attr, controller) { + element.off('change'); var options = { scope: scope }; @@ -152,16 +178,24 @@ if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; - angular.forEach([ 'html', 'container', 'trimValue' ], function(key) { + angular.forEach([ 'html', 'container', 'trimValue', 'filter' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); if (!element.attr('autocomplete')) element.attr('autocomplete', 'off'); - var filter = options.filter || defaults.filter; + var filter = angular.isDefined(options.filter) ? options.filter : defaults.filter; var limit = options.limit || defaults.limit; var comparator = options.comparator || defaults.comparator; var bsOptions = attr.bsOptions; - if (filter) bsOptions += ' | ' + filter + ':$viewValue'; - if (comparator) bsOptions += ':' + comparator; + if (filter) { + bsOptions += ' | ' + filter + ':$viewValue'; + if (comparator) bsOptions += ':' + comparator; + } if (limit) bsOptions += ' | limitTo:' + limit; var parsedOptions = $parseOptions(bsOptions); var typeahead = $typeahead(element, controller, options); @@ -191,7 +225,7 @@ if (displayValue) { return displayValue; } - if (modelValue && typeof modelValue !== 'object') { + if (angular.isDefined(modelValue) && typeof modelValue !== 'object') { return modelValue; } return ''; @@ -201,10 +235,13 @@ return element.val(''); } var index = typeahead.$getIndex(controller.$modelValue); - var selected = angular.isDefined(index) ? typeahead.$scope.$matches[index].label : controller.$viewValue; + var selected = index !== -1 ? typeahead.$scope.$matches[index].label : controller.$viewValue; selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected; var value = selected ? selected.toString().replace(/<(?:.|\n)*?>/gm, '') : ''; + var ss = element[0].selectionStart; + var sd = element[0].selectionEnd; element.val(options.trimValue === false ? value : value.trim()); + element[0].setSelectionRange(ss, sd); }; scope.$on('$destroy', function() { if (typeahead) typeahead.destroy(); @@ -225,7 +262,7 @@ placement: 'top', templateUrl: 'tooltip/tooltip.tpl.html', template: '', - contentTemplate: false, + titleTemplate: false, trigger: 'hover focus', keyboard: false, html: false, @@ -235,15 +272,16 @@ delay: 0, autoClose: false, bsEnabled: true, + mouseDownPreventDefault: true, + mouseDownStopPropagation: true, viewport: { selector: 'body', padding: 0 } }; this.$get = [ '$window', '$rootScope', '$bsCompiler', '$q', '$templateCache', '$http', '$animate', '$sce', 'dimensions', '$$rAF', '$timeout', function($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) { - var trim = String.prototype.trim; - var isTouch = 'createTouch' in $window.document; - var htmlReplaceRegExp = /ng-bind="/gi; + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); + var isTouch = 'createTouch' in $window.document && isNative; var $body = angular.element($window.document); function TooltipFactory(element, config) { var $tooltip = {}; @@ -283,8 +321,12 @@ }); }; $tooltip.$isShown = scope.$isShown = false; - var timeout, hoverState; - var compileData, tipElement, tipContainer, tipScope; + var timeout; + var hoverState; + var compileData; + var tipElement; + var tipContainer; + var tipScope; promise.then(function(data) { compileData = data; $tooltip.init(); @@ -309,7 +351,11 @@ } if (options.show) { scope.$$postDigest(function() { - options.trigger === 'focus' ? element[0].focus() : $tooltip.show(); + if (options.trigger === 'focus') { + element[0].focus(); + } else { + $tooltip.show(); + } }); } }; @@ -331,7 +377,11 @@ $tooltip.show = function() { if (!options.bsEnabled || $tooltip.$isShown) return; scope.$emit(options.prefixEvent + '.show.before', $tooltip); - var parent, after; + if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) { + options.onBeforeShow($tooltip); + } + var parent; + var after; if (options.container) { parent = tipContainer; if (tipContainer[0].lastChild) { @@ -356,7 +406,11 @@ if (options.animation) tipElement.addClass(options.animation); if (options.type) tipElement.addClass(options.prefixClass + '-' + options.type); if (options.customClass) tipElement.addClass(options.customClass); - after ? after.after(tipElement) : parent.prepend(tipElement); + if (after) { + after.after(tipElement); + } else { + parent.prepend(tipElement); + } $tooltip.$isShown = scope.$isShown = true; safeDigest(scope); $tooltip.$applyPlacement(); @@ -383,6 +437,9 @@ }; function enterAnimateCallback() { scope.$emit(options.prefixEvent + '.show', $tooltip); + if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) { + options.onShow($tooltip); + } } $tooltip.leave = function() { clearTimeout(timeout); @@ -401,12 +458,17 @@ $tooltip.hide = function(blur) { if (!$tooltip.$isShown) return; scope.$emit(options.prefixEvent + '.hide.before', $tooltip); + if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) { + options.onBeforeHide($tooltip); + } _blur = blur; _tipToHide = tipElement; - if (angular.version.minor <= 2) { - $animate.leave(tipElement, leaveAnimateCallback); - } else { - $animate.leave(tipElement).then(leaveAnimateCallback); + if (tipElement !== null) { + if (angular.version.minor <= 2) { + $animate.leave(tipElement, leaveAnimateCallback); + } else { + $animate.leave(tipElement).then(leaveAnimateCallback); + } } $tooltip.$isShown = scope.$isShown = false; safeDigest(scope); @@ -419,6 +481,9 @@ }; function leaveAnimateCallback() { scope.$emit(options.prefixEvent + '.hide', $tooltip); + if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) { + options.onHide($tooltip); + } if (tipElement === _tipToHide) { if (_blur && options.trigger === 'focus') { return element[0].blur(); @@ -426,8 +491,15 @@ destroyTipElement(); } } - $tooltip.toggle = function() { - $tooltip.$isShown ? $tooltip.leave() : $tooltip.enter(); + $tooltip.toggle = function(evt) { + if (evt) { + evt.preventDefault(); + } + if ($tooltip.$isShown) { + $tooltip.leave(); + } else { + $tooltip.enter(); + } }; $tooltip.focus = function() { tipElement[0].focus(); @@ -440,25 +512,29 @@ }; $tooltip.$applyPlacement = function() { if (!tipElement) return; - var placement = options.placement, autoToken = /\s?auto?\s?/i, autoPlace = autoToken.test(placement); + var placement = options.placement; + var autoToken = /\s?auto?\s?/i; + var autoPlace = autoToken.test(placement); if (autoPlace) { placement = placement.replace(autoToken, '') || defaults.placement; } tipElement.addClass(options.placement); - var elementPosition = getPosition(), tipWidth = tipElement.prop('offsetWidth'), tipHeight = tipElement.prop('offsetHeight'); + var elementPosition = getPosition(); + var tipWidth = tipElement.prop('offsetWidth'); + var tipHeight = tipElement.prop('offsetHeight'); $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport); if (autoPlace) { var originalPlacement = placement; var viewportPosition = getPosition($tooltip.$viewport); - if (originalPlacement.indexOf('bottom') >= 0 && elementPosition.bottom + tipHeight > viewportPosition.bottom) { + if (/bottom/.test(originalPlacement) && elementPosition.bottom + tipHeight > viewportPosition.bottom) { placement = originalPlacement.replace('bottom', 'top'); - } else if (originalPlacement.indexOf('top') >= 0 && elementPosition.top - tipHeight < viewportPosition.top) { + } else if (/top/.test(originalPlacement) && elementPosition.top - tipHeight < viewportPosition.top) { placement = originalPlacement.replace('top', 'bottom'); } - if ((originalPlacement === 'right' || originalPlacement === 'bottom-left' || originalPlacement === 'top-left') && elementPosition.right + tipWidth > viewportPosition.width) { - placement = originalPlacement === 'right' ? 'left' : placement.replace('left', 'right'); - } else if ((originalPlacement === 'left' || originalPlacement === 'bottom-right' || originalPlacement === 'top-right') && elementPosition.left - tipWidth < viewportPosition.left) { - placement = originalPlacement === 'left' ? 'right' : placement.replace('right', 'left'); + if (/left/.test(originalPlacement) && elementPosition.left - tipWidth < viewportPosition.left) { + placement = placement.replace('left', 'right'); + } else if (/right/.test(originalPlacement) && elementPosition.right + tipWidth > viewportPosition.width) { + placement = placement.replace('right', 'left'); } tipElement.removeClass(originalPlacement).addClass(placement); } @@ -478,19 +554,29 @@ } }; $tooltip.$onFocusElementMouseDown = function(evt) { - evt.preventDefault(); - evt.stopPropagation(); - $tooltip.$isShown ? element[0].blur() : element[0].focus(); + if (options.mouseDownPreventDefault) { + evt.preventDefault(); + } + if (options.mouseDownStopPropagation) { + evt.stopPropagation(); + } + if ($tooltip.$isShown) { + element[0].blur(); + } else { + element[0].focus(); + } }; function bindTriggerEvents() { var triggers = options.trigger.split(' '); angular.forEach(triggers, function(trigger) { - if (trigger === 'click') { - element.on('click', $tooltip.toggle); + if (trigger === 'click' || trigger === 'contextmenu') { + element.on(trigger, $tooltip.toggle); } else if (trigger !== 'manual') { element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter); element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave); - nodeName === 'button' && trigger !== 'hover' && element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + if (nodeName === 'button' && trigger !== 'hover') { + element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + } } }); } @@ -498,12 +584,14 @@ var triggers = options.trigger.split(' '); for (var i = triggers.length; i--; ) { var trigger = triggers[i]; - if (trigger === 'click') { - element.off('click', $tooltip.toggle); + if (trigger === 'click' || trigger === 'contextmenu') { + element.off(trigger, $tooltip.toggle); } else if (trigger !== 'manual') { element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter); element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave); - nodeName === 'button' && trigger !== 'hover' && element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + if (nodeName === 'button' && trigger !== 'hover') { + element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + } } } } @@ -541,7 +629,8 @@ } function getPosition($element) { $element = $element || (options.target || element); - var el = $element[0], isBody = el.tagName === 'BODY'; + var el = $element[0]; + var isBody = el.tagName === 'BODY'; var elRect = el.getBoundingClientRect(); var rect = {}; for (var p in elRect) { @@ -556,9 +645,11 @@ var elOffset = isBody ? { top: 0, left: 0 - } : dimensions.offset(el), scroll = { + } : dimensions.offset(el); + var scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0 - }, outerDims = isBody ? { + }; + var outerDims = isBody ? { width: document.documentElement.clientWidth, height: $window.innerHeight } : null; @@ -607,22 +698,33 @@ case 'right': offset.left = position.left + position.width - actualWidth; + break; + + default: + break; } } else if (split[0] === 'left' || split[0] === 'right') { switch (split[1]) { case 'top': - offset.top = position.top - actualHeight; + offset.top = position.top - actualHeight + position.height; break; case 'bottom': - offset.top = position.top + position.height; + offset.top = position.top; + break; + + default: + break; } } return offset; } function applyPlacement(offset, placement) { - var tip = tipElement[0], width = tip.offsetWidth, height = tip.offsetHeight; - var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10), marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10); + var tip = tipElement[0]; + var width = tip.offsetWidth; + var height = tip.offsetHeight; + var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10); + var marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10); if (isNaN(marginTop)) marginTop = 0; if (isNaN(marginLeft)) marginLeft = 0; offset.top = offset.top + marginTop; @@ -636,7 +738,8 @@ }); } }, offset), 0); - var actualWidth = tip.offsetWidth, actualHeight = tip.offsetHeight; + var actualWidth = tip.offsetWidth; + var actualHeight = tip.offsetHeight; if (placement === 'top' && actualHeight !== height) { offset.top = offset.top + height - actualHeight; } @@ -649,7 +752,9 @@ } dimensions.setOffset(tip, offset); if (/top|right|bottom|left/.test(placement)) { - var isVertical = /top|bottom/.test(placement), arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight, arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'; + var isVertical = /top|bottom/.test(placement); + var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight; + var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'; replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical); } } @@ -711,35 +816,39 @@ function findElement(query, element) { return angular.element((element || document).querySelectorAll(query)); } - var fetchPromises = {}; - function fetchTemplate(template) { - if (fetchPromises[template]) return fetchPromises[template]; - return fetchPromises[template] = $http.get(template, { - cache: $templateCache - }).then(function(res) { - return res.data; - }); - } return TooltipFactory; } ]; - }).directive('bsTooltip', [ '$window', '$location', '$sce', '$tooltip', '$$rAF', function($window, $location, $sce, $tooltip, $$rAF) { + }).directive('bsTooltip', [ '$window', '$location', '$sce', '$parse', '$tooltip', '$$rAF', function($window, $location, $sce, $parse, $tooltip, $$rAF) { return { restrict: 'EAC', scope: true, link: function postLink(scope, element, attr, transclusion) { + var tooltip; var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'titleTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } + }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); var dataTarget = element.attr('data-target'); if (angular.isDefined(dataTarget)) { - if (falseValueRegExp.test(dataTarget)) options.target = false; else options.target = dataTarget; + if (falseValueRegExp.test(dataTarget)) { + options.target = false; + } else { + options.target = dataTarget; + } } if (!scope.hasOwnProperty('title')) { scope.title = ''; @@ -748,36 +857,61 @@ if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) { var oldValue = scope.title; scope.title = $sce.trustAsHtml(newValue); - angular.isDefined(oldValue) && $$rAF(function() { - tooltip && tooltip.$applyPlacement(); - }); + if (angular.isDefined(oldValue)) { + $$rAF(function() { + if (tooltip) tooltip.$applyPlacement(); + }); + } } }); - attr.bsTooltip && scope.$watch(attr.bsTooltip, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.title = newValue; + attr.$observe('disabled', function(newValue) { + if (newValue && tooltip.$isShown) { + tooltip.hide(); } - angular.isDefined(oldValue) && $$rAF(function() { - tooltip && tooltip.$applyPlacement(); - }); - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i); - newValue === true ? tooltip.show() : tooltip.hide(); - }); - attr.bsEnabled && scope.$watch(attr.bsEnabled, function(newValue, oldValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i); - newValue === false ? tooltip.setEnabled(false) : tooltip.setEnabled(true); }); - attr.viewport && scope.$watch(attr.viewport, function(newValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - tooltip.setViewport(newValue); - }); - var tooltip = $tooltip(element, options); + if (attr.bsTooltip) { + scope.$watch(attr.bsTooltip, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.title = newValue; + } + if (angular.isDefined(oldValue)) { + $$rAF(function() { + if (tooltip) tooltip.$applyPlacement(); + }); + } + }, true); + } + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i); + if (newValue === true) { + tooltip.show(); + } else { + tooltip.hide(); + } + }); + } + if (attr.bsEnabled) { + scope.$watch(attr.bsEnabled, function(newValue, oldValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i); + if (newValue === false) { + tooltip.setEnabled(false); + } else { + tooltip.setEnabled(true); + } + }); + } + if (attr.viewport) { + scope.$watch(attr.viewport, function(newValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + tooltip.setViewport(newValue); + }); + } + tooltip = $tooltip(element, options); scope.$on('$destroy', function() { if (tooltip) tooltip.destroy(); options = null; @@ -789,6 +923,7 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$timepicker', function() { var defaults = this.defaults = { animation: 'am-fade', + defaultDate: 'auto', prefixClass: 'timepicker', placement: 'bottom-left', templateUrl: 'timepicker/timepicker.tpl.html', @@ -815,7 +950,7 @@ arrowBehavior: 'pager' }; this.$get = [ '$window', '$document', '$rootScope', '$sce', '$dateFormatter', '$tooltip', '$timeout', function($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) { - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; if (!defaults.lang) { defaults.lang = $dateFormatter.getDefaultLocale(); @@ -844,7 +979,12 @@ millisecond: startDate.getMilliseconds() }; var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang); - var hoursFormat = $dateFormatter.hoursFormat(format), timeSeparator = $dateFormatter.timeSeparator(format), minutesFormat = $dateFormatter.minutesFormat(format), secondsFormat = $dateFormatter.secondsFormat(format), showSeconds = $dateFormatter.showSeconds(format), showAM = $dateFormatter.showAM(format); + var hoursFormat = $dateFormatter.hoursFormat(format); + var timeSeparator = $dateFormatter.timeSeparator(format); + var minutesFormat = $dateFormatter.minutesFormat(format); + var secondsFormat = $dateFormatter.secondsFormat(format); + var showSeconds = $dateFormatter.showSeconds(format); + var showAM = $dateFormatter.showAM(format); scope.$iconUp = options.iconUp; scope.$iconDown = options.iconDown; scope.$select = function(date, index) { @@ -871,7 +1011,9 @@ } }; $timepicker.select = function(date, index, keep) { - if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) controller.$dateValue = new Date(1970, 0, 1); + if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) { + controller.$dateValue = options.defaultDate === 'today' ? new Date() : new Date(1970, 0, 1); + } if (!angular.isDate(date)) date = new Date(date); if (index === 0) controller.$dateValue.setHours(date.getHours()); else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes()); else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds()); controller.$setViewValue(angular.copy(controller.$dateValue)); @@ -892,8 +1034,10 @@ controller.$render(); }; $timepicker.$build = function() { - var i, midIndex = scope.midIndex = parseInt(options.length / 2, 10); - var hours = [], hour; + var i; + var midIndex = scope.midIndex = parseInt(options.length / 2, 10); + var hours = []; + var hour; for (i = 0; i < options.length; i++) { hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep); hours.push({ @@ -903,7 +1047,8 @@ disabled: $timepicker.$isDisabled(hour, 0) }); } - var minutes = [], minute; + var minutes = []; + var minute; for (i = 0; i < options.length; i++) { minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep); minutes.push({ @@ -913,7 +1058,8 @@ disabled: $timepicker.$isDisabled(minute, 1) }); } - var seconds = [], second; + var seconds = []; + var second; for (i = 0; i < options.length; i++) { second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep); seconds.push({ @@ -1019,9 +1165,12 @@ return; } var newDate = new Date($timepicker.$date); - var hours = newDate.getHours(), hoursLength = formatDate(newDate, hoursFormat).length; - var minutes = newDate.getMinutes(), minutesLength = formatDate(newDate, minutesFormat).length; - var seconds = newDate.getSeconds(), secondsLength = formatDate(newDate, secondsFormat).length; + var hours = newDate.getHours(); + var hoursLength = formatDate(newDate, hoursFormat).length; + var minutes = newDate.getMinutes(); + var minutesLength = formatDate(newDate, minutesFormat).length; + var seconds = newDate.getSeconds(); + var secondsLength = formatDate(newDate, secondsFormat).length; var sepLength = 1; var lateralMove = /(37|39)/.test(evt.keyCode); var count = 2 + showSeconds * 1 + showAM * 1; @@ -1097,18 +1246,18 @@ if (!isTouch && element.attr('readonly') || element.attr('disabled')) return; _show(); $timeout(function() { - $timepicker.$element && $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); + if ($timepicker.$element) $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); if (options.keyboard) { - element && element.on('keydown', $timepicker.$onKeyDown); + if (element) element.on('keydown', $timepicker.$onKeyDown); } }, 0, false); }; var _hide = $timepicker.hide; $timepicker.hide = function(blur) { if (!$timepicker.$isShown) return; - $timepicker.$element && $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); + if ($timepicker.$element) $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); if (options.keyboard) { - element && element.off('keydown', $timepicker.$onKeyDown); + if (element) element.off('keydown', $timepicker.$onKeyDown); } _hide(blur); }; @@ -1119,7 +1268,7 @@ } ]; }).directive('bsTimepicker', [ '$window', '$parse', '$q', '$dateFormatter', '$dateParser', '$timepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) { var defaults = $timepicker.defaults; - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); return { restrict: 'EAC', require: 'ngModel', @@ -1127,17 +1276,20 @@ var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent', 'defaultDate' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'autoclose', 'useNative', 'roundDisplay' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } }); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!timepicker || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i); - newValue === true ? timepicker.show() : timepicker.hide(); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm'; var timepicker = $timepicker(element, controller, options); @@ -1146,16 +1298,29 @@ var formatDate = function(date, format, timezone) { return $dateFormatter.formatDate(date, format, lang, timezone); }; + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!timepicker || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i); + if (newValue === true) { + timepicker.show(); + } else { + timepicker.hide(); + } + }); + } var dateParser = $dateParser({ format: options.timeFormat, lang: lang }); angular.forEach([ 'minTime', 'maxTime' ], function(key) { - angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) { - timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue); - !isNaN(timepicker.$options[key]) && timepicker.$build(); - validateAgainstMinMaxTime(controller.$dateValue); - }); + if (angular.isDefined(attr[key])) { + attr.$observe(key, function(newValue) { + timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue); + if (!isNaN(timepicker.$options[key])) timepicker.$build(); + validateAgainstMinMaxTime(controller.$dateValue); + }); + } }); scope.$watch(attr.ngModel, function(newValue, oldValue) { timepicker.update(controller.$dateValue); @@ -1183,9 +1348,8 @@ if (!parsedTime || isNaN(parsedTime.getTime())) { controller.$setValidity('date', false); return undefined; - } else { - validateAgainstMinMaxTime(parsedTime); } + validateAgainstMinMaxTime(parsedTime); if (options.timeType === 'string') { date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true); return formatDate(date, options.modelTimeFormat || options.timeFormat); @@ -1197,9 +1361,8 @@ return date.getTime() / 1e3; } else if (options.timeType === 'iso') { return date.toISOString(); - } else { - return new Date(date); } + return new Date(date); }); controller.$formatters.push(function(modelValue) { var date; @@ -1331,7 +1494,6 @@ require: [ '^?ngModel', '^bsTabs' ], scope: true, link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsTabsCtrl = controllers[1]; element.addClass('tab-pane'); attrs.$observe('title', function(newValue, oldValue) { @@ -1380,11 +1542,11 @@ noneText: 'None', maxLength: 3, maxLengthHtml: 'selected', - iconCheckmark: 'glyphicon glyphicon-ok' + iconCheckmark: 'glyphicon glyphicon-ok', + toggle: false }; this.$get = [ '$window', '$document', '$rootScope', '$tooltip', '$timeout', function($window, $document, $rootScope, $tooltip, $timeout) { - var bodyEl = angular.element($window.document.body); - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; function SelectFactory(element, controller, config) { var $select = {}; @@ -1438,7 +1600,11 @@ }; $select.activate = function(index) { if (options.multiple) { - $select.$isActive(index) ? scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1) : scope.$activeIndex.push(index); + if ($select.$isActive(index)) { + scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1); + } else { + scope.$activeIndex.push(index); + } if (options.sort) scope.$activeIndex.sort(function(a, b) { return a - b; }); @@ -1448,6 +1614,9 @@ return scope.$activeIndex; }; $select.select = function(index) { + if (angular.isUndefined(index) || index < 0 || index >= scope.$matches.length) { + return; + } var value = scope.$matches[index].value; scope.$apply(function() { $select.activate(index); @@ -1459,11 +1628,18 @@ return scope.$matches[index].value; })); } else { - controller.$setViewValue(value); + if (options.toggle) { + controller.$setViewValue(value === controller.$modelValue ? undefined : value); + } else { + controller.$setViewValue(value); + } $select.hide(); } }); scope.$emit(options.prefixEvent + '.select', value, index, $select); + if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) { + options.onSelect(value, index, $select); + } }; $select.$updateActiveIndex = function() { if (options.multiple) { @@ -1491,25 +1667,36 @@ $select.$isActive = function(index) { if (options.multiple) { return scope.$activeIndex.indexOf(index) !== -1; - } else { - return scope.$activeIndex === index; } + return scope.$activeIndex === index; }; $select.$getIndex = function(value) { - var l = scope.$matches.length, i = l; - if (!l) return; - for (i = l; i--; ) { - if (scope.$matches[i].value === value) break; + var index; + for (index = scope.$matches.length; index--; ) { + if (angular.equals(scope.$matches[index].value, value)) break; } - if (i < 0) return; - return i; + return index; }; $select.$onMouseDown = function(evt) { evt.preventDefault(); evt.stopPropagation(); if (isTouch) { var targetEl = angular.element(evt.target); - targetEl.triggerHandler('click'); + var anchor; + if (evt.target.nodeName !== 'A') { + var anchorCandidate = targetEl.parent(); + while (!anchor && anchorCandidate.length > 0) { + if (anchorCandidate[0].nodeName === 'A') { + anchor = anchorCandidate; + } + anchorCandidate = anchorCandidate.parent(); + } + } + if (anchor) { + angular.element(anchor).triggerHandler('click'); + } else { + targetEl.triggerHandler('click'); + } } }; $select.$onKeyDown = function(evt) { @@ -1579,16 +1766,28 @@ scope: scope, placeholder: defaults.placeholder }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent', 'toggle' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'allNoneButtons', 'sort' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } + }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); var dataMultiple = element.attr('data-multiple'); if (angular.isDefined(dataMultiple)) { - if (falseValueRegExp.test(dataMultiple)) options.multiple = false; else options.multiple = dataMultiple; + if (falseValueRegExp.test(dataMultiple)) { + options.multiple = false; + } else { + options.multiple = dataMultiple; + } } if (element[0].nodeName.toLowerCase() === 'select') { var inputEl = element; @@ -1602,22 +1801,23 @@ element[0].addEventListener('blur', select.$selectScrollFix); } var watchedOptions = parsedOptions.$match[7].replace(/\|.+/, '').trim(); - scope.$watchCollection(watchedOptions, function(newValue, oldValue) { + scope.$watch(watchedOptions, function(newValue, oldValue) { parsedOptions.valuesFn(scope, controller).then(function(values) { select.update(values); controller.$render(); }); - }); + }, true); scope.$watch(attr.ngModel, function(newValue, oldValue) { select.$updateActiveIndex(); controller.$render(); }, true); controller.$render = function() { - var selected, index; + var selected; + var index; if (options.multiple && angular.isArray(controller.$modelValue)) { selected = controller.$modelValue.map(function(value) { index = select.$getIndex(value); - return angular.isDefined(index) ? select.$scope.$matches[index].label : false; + return index !== -1 ? select.$scope.$matches[index].label : false; }).filter(angular.isDefined); if (selected.length > (options.maxLength || defaults.maxLength)) { selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml); @@ -1626,9 +1826,9 @@ } } else { index = select.$getIndex(controller.$modelValue); - selected = angular.isDefined(index) ? select.$scope.$matches[index].label : false; + selected = index !== -1 ? select.$scope.$matches[index].label : false; } - element.html((selected ? selected : options.placeholder) + (options.caretHtml ? options.caretHtml : defaults.caretHtml)); + element.html((selected || options.placeholder) + (options.caretHtml || defaults.caretHtml)); }; if (options.multiple) { controller.$isEmpty = function(value) { @@ -1668,7 +1868,8 @@ return spies[scrollId]; } var $scrollspy = {}; - var unbindViewContentLoaded, unbindIncludeContentLoaded; + var unbindViewContentLoaded; + var unbindIncludeContentLoaded; var trackedElements = $scrollspy.$trackedElements = []; var sortedElements = []; var activeTarget; @@ -1772,7 +1973,7 @@ break; } } - trackedElements = trackedElements.splice(toDelete, 1); + trackedElements.splice(toDelete, 1); }; $scrollspy.activate = function(i) { trackedElements[i].addClass('active'); @@ -1850,48 +2051,86 @@ restrict: 'EAC', scope: true, link: function postLink(scope, element, attr) { + var popover; var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent', 'bsEnabled' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'autoClose' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); var dataTarget = element.attr('data-target'); if (angular.isDefined(dataTarget)) { - if (falseValueRegExp.test(dataTarget)) options.target = false; else options.target = dataTarget; + if (falseValueRegExp.test(dataTarget)) { + options.target = false; + } else { + options.target = dataTarget; + } } angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - angular.isDefined(oldValue) && requestAnimationFrame(function() { - popover && popover.$applyPlacement(); + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + if (angular.isDefined(oldValue)) { + requestAnimationFrame(function() { + if (popover) popover.$applyPlacement(); + }); + } }); - }); - }); - attr.bsPopover && scope.$watch(attr.bsPopover, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; } - angular.isDefined(oldValue) && requestAnimationFrame(function() { - popover && popover.$applyPlacement(); - }); - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!popover || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i); - newValue === true ? popover.show() : popover.hide(); }); - attr.viewport && scope.$watch(attr.viewport, function(newValue) { - if (!popover || !angular.isDefined(newValue)) return; - popover.setViewport(newValue); - }); - var popover = $popover(element, options); + if (attr.bsPopover) { + scope.$watch(attr.bsPopover, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + if (angular.isDefined(oldValue)) { + requestAnimationFrame(function() { + if (popover) popover.$applyPlacement(); + }); + } + }, true); + } + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!popover || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i); + if (newValue === true) { + popover.show(); + } else { + popover.hide(); + } + }); + } + if (attr.bsEnabled) { + scope.$watch(attr.bsEnabled, function(newValue) { + if (!popover || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(popover),?/i); + if (newValue === false) { + popover.setEnabled(false); + } else { + popover.setEnabled(true); + } + }); + } + if (attr.viewport) { + scope.$watch(attr.viewport, function(newValue) { + if (!popover || !angular.isDefined(newValue)) return; + popover.setViewport(newValue); + }); + } + popover = $popover(element, options); scope.$on('$destroy', function() { if (popover) popover.destroy(); options = null; @@ -1945,6 +2184,7 @@ var defaults = this.defaults = { animation: 'am-fade', backdropAnimation: 'am-fade', + customClass: '', prefixClass: 'modal', prefixEvent: 'modal', placement: 'top', @@ -1956,13 +2196,21 @@ backdrop: true, keyboard: true, html: false, - show: true + show: true, + size: null, + zIndex: null }; this.$get = [ '$window', '$rootScope', '$bsCompiler', '$animate', '$timeout', '$sce', 'dimensions', function($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) { var forEach = angular.forEach; - var trim = String.prototype.trim; var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; var bodyElement = angular.element($window.document.body); + var backdropCount = 0; + var dialogBaseZindex = 1050; + var backdropBaseZindex = 1040; + var validSizes = { + lg: 'modal-lg', + sm: 'modal-sm' + }; function ModalFactory(config) { var $modal = {}; var options = $modal.$options = angular.extend({}, defaults, config); @@ -1971,6 +2219,10 @@ if (!options.element && !options.container) { options.container = 'body'; } + if (options.zIndex) { + dialogBaseZindex = parseInt(options.zIndex, 10); + backdropBaseZindex = dialogBaseZindex - 10; + } $modal.$id = options.id || options.element && options.element.attr('id') || ''; forEach([ 'title', 'content' ], function(key) { if (options[key]) scope[key] = $sce.trustAsHtml(options[key]); @@ -1991,15 +2243,16 @@ }); }; $modal.$isShown = scope.$isShown = false; - var compileData, modalElement, modalScope; + var compileData; + var modalElement; + var modalScope; var backdropElement = angular.element('
'); backdropElement.css({ position: 'fixed', top: '0px', left: '0px', bottom: '0px', - right: '0px', - 'z-index': 1038 + right: '0px' }); promise.then(function(data) { compileData = data; @@ -2022,7 +2275,8 @@ }; $modal.show = function() { if ($modal.$isShown) return; - var parent, after; + var parent; + var after; if (angular.isElement(options.container)) { parent = options.container; after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null; @@ -2038,12 +2292,30 @@ if (modalElement) destroyModalElement(); modalScope = $modal.$scope.$new(); modalElement = $modal.$element = compileData.link(modalScope, function(clonedElement, scope) {}); + if (options.backdrop) { + modalElement.css({ + 'z-index': dialogBaseZindex + backdropCount * 20 + }); + backdropElement.css({ + 'z-index': backdropBaseZindex + backdropCount * 20 + }); + backdropCount++; + } if (scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) { return; } + if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) { + options.onBeforeShow($modal); + } modalElement.css({ display: 'block' }).addClass(options.placement); + if (options.customClass) { + modalElement.addClass(options.customClass); + } + if (options.size && validSizes[options.size]) { + angular.element(findElement('.modal-dialog', modalElement[0])).addClass(validSizes[options.size]); + } if (options.animation) { if (options.backdrop) { backdropElement.addClass(options.backdropAnimation); @@ -2073,18 +2345,25 @@ }; function enterAnimateCallback() { scope.$emit(options.prefixEvent + '.show', $modal); + if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) { + options.onShow($modal); + } } $modal.hide = function() { if (!$modal.$isShown) return; if (scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) { return; } + if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) { + options.onBeforeHide($modal); + } if (angular.version.minor <= 2) { $animate.leave(modalElement, leaveAnimateCallback); } else { $animate.leave(modalElement).then(leaveAnimateCallback); } if (options.backdrop) { + backdropCount--; $animate.leave(backdropElement); } $modal.$isShown = scope.$isShown = false; @@ -2094,13 +2373,22 @@ }; function leaveAnimateCallback() { scope.$emit(options.prefixEvent + '.hide', $modal); - bodyElement.removeClass(options.prefixClass + '-open'); + if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) { + options.onHide($modal); + } + if (findElement('.modal').length <= 0) { + bodyElement.removeClass(options.prefixClass + '-open'); + } if (options.animation) { bodyElement.removeClass(options.prefixClass + '-with-' + options.animation); } } $modal.toggle = function() { - $modal.$isShown ? $modal.hide() : $modal.show(); + if ($modal.$isShown) { + $modal.hide(); + } else { + $modal.show(); + } }; $modal.focus = function() { modalElement[0].focus(); @@ -2137,7 +2425,11 @@ } function hideOnBackdropClick(evt) { if (evt.target !== evt.currentTarget) return; - options.backdrop === 'static' ? $modal.focus() : $modal.hide(); + if (options.backdrop === 'static') { + $modal.focus(); + } else { + $modal.hide(); + } } function preventEventDefault(evt) { evt.preventDefault(); @@ -2166,7 +2458,7 @@ } return ModalFactory; } ]; - }).directive('bsModal', [ '$window', '$sce', '$modal', function($window, $sce, $modal) { + }).directive('bsModal', [ '$window', '$sce', '$parse', '$modal', function($window, $sce, $parse, $modal) { return { restrict: 'EAC', scope: true, @@ -2176,25 +2468,38 @@ element: element, show: false }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass', 'customClass', 'modalClass', 'size', 'zIndex' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); + if (options.modalClass) { + options.customClass = options.modalClass; + } var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'backdrop', 'keyboard', 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); - attr.bsModal && scope.$watch(attr.bsModal, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + angular.forEach([ 'title', 'content' ], function(key) { + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsModal) { + scope.$watch(attr.bsModal, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var modal = $modal(options); element.on(attr.trigger || 'click', modal.toggle); scope.$on('$destroy', function() { @@ -2205,24 +2510,26 @@ } }; } ]); - angular.version.minor < 3 && angular.version.dot < 14 && angular.module('ng').factory('$$rAF', [ '$window', '$timeout', function($window, $timeout) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.webkitRequestAnimationFrame || $window.mozRequestAnimationFrame; - var cancelAnimationFrame = $window.cancelAnimationFrame || $window.webkitCancelAnimationFrame || $window.mozCancelAnimationFrame || $window.webkitCancelRequestAnimationFrame; - var rafSupported = !!requestAnimationFrame; - var raf = rafSupported ? function(fn) { - var id = requestAnimationFrame(fn); - return function() { - cancelAnimationFrame(id); - }; - } : function(fn) { - var timer = $timeout(fn, 16.66, false); - return function() { - $timeout.cancel(timer); + if (angular.version.minor < 3 && angular.version.dot < 14) { + angular.module('ng').factory('$$rAF', [ '$window', '$timeout', function($window, $timeout) { + var requestAnimationFrame = $window.requestAnimationFrame || $window.webkitRequestAnimationFrame || $window.mozRequestAnimationFrame; + var cancelAnimationFrame = $window.cancelAnimationFrame || $window.webkitCancelAnimationFrame || $window.mozCancelAnimationFrame || $window.webkitCancelRequestAnimationFrame; + var rafSupported = !!requestAnimationFrame; + var raf = rafSupported ? function(fn) { + var id = requestAnimationFrame(fn); + return function() { + cancelAnimationFrame(id); + }; + } : function(fn) { + var timer = $timeout(fn, 16.66, false); + return function() { + $timeout.cancel(timer); + }; }; - }; - raf.supported = rafSupported; - return raf; - } ]); + raf.supported = rafSupported; + return raf; + } ]); + } angular.module('mgcrea.ngStrap.helpers.parseOptions', []).provider('$parseOptions', function() { var defaults = this.defaults = { regexp: /^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/ @@ -2232,11 +2539,20 @@ var $parseOptions = {}; var options = angular.extend({}, defaults, config); $parseOptions.$values = []; - var match, displayFn, valueName, keyName, groupByFn, valueFn, valuesFn; + var match; + var displayFn; + var valueName; + var keyName; + var groupByFn; + var valueFn; + var valuesFn; $parseOptions.init = function() { $parseOptions.$match = match = attr.match(options.regexp); - displayFn = $parse(match[2] || match[1]), valueName = match[4] || match[6], keyName = match[5], - groupByFn = $parse(match[3] || ''), valueFn = $parse(match[2] ? match[1] : valueName), + displayFn = $parse(match[2] || match[1]); + valueName = match[4] || match[6]; + keyName = match[5]; + groupByFn = $parse(match[3] || ''); + valueFn = $parse(match[2] ? match[1] : valueName); valuesFn = $parse(match[7]); }; $parseOptions.valuesFn = function(scope, controller) { @@ -2255,7 +2571,9 @@ }; function parseValues(values, scope) { return values.map(function(match, index) { - var locals = {}, label, value; + var locals = {}; + var label; + var value; locals[valueName] = match; label = displayFn(scope, locals); value = valueFn(scope, locals); @@ -2272,8 +2590,7 @@ return ParseOptionsFactory; } ]; }); - angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ '$document', '$window', function($document, $window) { - var jqLite = angular.element; + angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', function() { var fn = {}; var nodeName = fn.nodeName = function(element, name) { return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase(); @@ -2300,7 +2617,16 @@ }; }; fn.setOffset = function(element, options, i) { - var curPosition, curLeft, curCSSTop, curTop, curOffset, curCSSLeft, calculatePosition, position = fn.css(element, 'position'), curElem = angular.element(element), props = {}; + var curPosition; + var curLeft; + var curCSSTop; + var curTop; + var curOffset; + var curCSSLeft; + var calculatePosition; + var position = fn.css(element, 'position'); + var curElem = angular.element(element); + var props = {}; if (position === 'static') { element.style.position = 'relative'; } @@ -2338,17 +2664,19 @@ var offsetParentRect = { top: 0, left: 0 - }, offsetParentElement, offset; + }; + var offsetParentEl; + var offset; if (fn.css(element, 'position') === 'fixed') { offset = element.getBoundingClientRect(); } else { - offsetParentElement = offsetParent(element); + offsetParentEl = offsetParentElement(element); offset = fn.offset(element); - if (!nodeName(offsetParentElement, 'html')) { - offsetParentRect = fn.offset(offsetParentElement); + if (!nodeName(offsetParentEl, 'html')) { + offsetParentRect = fn.offset(offsetParentEl); } - offsetParentRect.top += fn.css(offsetParentElement, 'borderTopWidth', true); - offsetParentRect.left += fn.css(offsetParentElement, 'borderLeftWidth', true); + offsetParentRect.top += fn.css(offsetParentEl, 'borderTopWidth', true); + offsetParentRect.left += fn.css(offsetParentEl, 'borderLeftWidth', true); } return { width: element.offsetWidth, @@ -2357,7 +2685,7 @@ left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true) }; }; - var offsetParent = function offsetParentElement(element) { + function offsetParentElement(element) { var docElement = element.ownerDocument; var offsetParent = element.offsetParent || docElement; if (nodeName(offsetParent, '#document')) return docElement.documentElement; @@ -2365,7 +2693,7 @@ offsetParent = offsetParent.offsetParent; } return offsetParent || docElement.documentElement; - }; + } fn.height = function(element, outer) { var value = element.offsetHeight; if (outer) { @@ -2385,12 +2713,14 @@ return value; }; return fn; - } ]); + }); angular.module('mgcrea.ngStrap.helpers.debounce', []).factory('debounce', [ '$timeout', function($timeout) { return function(func, wait, immediate) { var timeout = null; return function() { - var context = this, args = arguments, callNow = immediate && !timeout; + var context = this; + var args = arguments; + var callNow = immediate && !timeout; if (timeout) { $timeout.cancel(timeout); } @@ -2409,9 +2739,10 @@ } ]).factory('throttle', [ '$timeout', function($timeout) { return function(func, wait, options) { var timeout = null; - options || (options = {}); + if (!options) options = {}; return function() { - var context = this, args = arguments; + var context = this; + var args = arguments; if (!timeout) { if (options.leading !== false) { func.apply(context, args); @@ -2479,7 +2810,8 @@ return !isNaN(parseFloat(n)) && isFinite(n); } function indexOfCaseInsensitive(array, value) { - var len = array.length, str = value.toString().toLowerCase(); + var len = array.length; + var str = value.toString().toLowerCase(); for (var i = 0; i < len; i++) { if (array[i].toLowerCase() === str) { return i; @@ -2556,7 +2888,8 @@ return 1 * value <= 50 && value.length === 2 ? this.setFullYear(2e3 + 1 * value) : this.setFullYear(1 * value); } }; - var regex, setMap; + var regex; + var setMap; $dateParser.init = function() { $dateParser.$format = $locale.DATETIME_FORMATS[options.format] || options.format; regex = regExpForFormat($dateParser.$format); @@ -2575,7 +2908,7 @@ if (!matches) return false; var date = baseDate && !isNaN(baseDate.getTime()) ? new ParseDate().fromDate(baseDate) : new ParseDate().fromDate(new Date(1970, 0, 1, 0)); for (var i = 0; i < matches.length - 1; i++) { - formatSetMap[i] && formatSetMap[i].call(date, matches[i + 1]); + if (formatSetMap[i]) formatSetMap[i].call(date, matches[i + 1]); } var newDate = date.toDate(); if (parseInt(date.day, 10) !== newDate.getDate()) { @@ -2592,7 +2925,7 @@ date = new Date(value.substr(1, value.length - 2)); } else if (isNumeric(value)) { date = new Date(parseInt(value, 10)); - } else if (angular.isString(value) && 0 === value.length) { + } else if (angular.isString(value) && value.length === 0) { date = key === 'minDate' ? -Infinity : +Infinity; } else { date = new Date(value); @@ -2607,7 +2940,7 @@ time = new Date(value.substr(1, value.length - 2)).setFullYear(1970, 0, 1); } else if (isNumeric(value)) { time = new Date(parseInt(value, 10)).setFullYear(1970, 0, 1); - } else if (angular.isString(value) && 0 === value.length) { + } else if (angular.isString(value) && value.length === 0) { time = key === 'minTime' ? -Infinity : +Infinity; } else { time = $dateParser.parse(value, new Date(1970, 0, 1, 0)); @@ -2631,39 +2964,66 @@ } return date; }; - function setMapForFormat(format) { - var keys = Object.keys(setFnMap), i; - var map = [], sortedMap = []; - var clonedFormat = format; - for (i = 0; i < keys.length; i++) { - if (format.split(keys[i]).length > 1) { - var index = clonedFormat.search(keys[i]); - format = format.split(keys[i]).join(''); - if (setFnMap[keys[i]]) { - map[index] = setFnMap[keys[i]]; + function regExpForFormat(format) { + var re = buildDateAbstractRegex(format); + return buildDateParseRegex(re); + } + function buildDateAbstractRegex(format) { + var escapedFormat = escapeReservedSymbols(format); + var escapedLiteralFormat = escapedFormat.replace(/''/g, '\\\''); + var literalRegex = /('(?:\\'|.)*?')/; + var formatParts = escapedLiteralFormat.split(literalRegex); + var dateElements = Object.keys(regExpMap); + var dateRegexParts = []; + angular.forEach(formatParts, function(part) { + if (isFormatStringLiteral(part)) { + part = trimLiteralEscapeChars(part); + } else { + for (var i = 0; i < dateElements.length; i++) { + part = part.split(dateElements[i]).join('${' + i + '}'); } } - } - angular.forEach(map, function(v) { - if (v) sortedMap.push(v); + dateRegexParts.push(part); }); - return sortedMap; + return dateRegexParts.join(''); } function escapeReservedSymbols(text) { - return text.replace(/\//g, '[\\/]').replace('/-/g', '[-]').replace(/\./g, '[.]').replace(/\\s/g, '[\\s]'); + return text.replace(/\\/g, '[\\\\]').replace(/-/g, '[-]').replace(/\./g, '[.]').replace(/\*/g, '[*]').replace(/\+/g, '[+]').replace(/\?/g, '[?]').replace(/\$/g, '[$]').replace(/\^/g, '[^]').replace(/\//g, '[/]').replace(/\\s/g, '[\\s]'); } - function regExpForFormat(format) { - var keys = Object.keys(regExpMap), i; - var re = format; - for (i = 0; i < keys.length; i++) { - re = re.split(keys[i]).join('${' + i + '}'); - } - for (i = 0; i < keys.length; i++) { - re = re.split('${' + i + '}').join('(' + regExpMap[keys[i]] + ')'); + function isFormatStringLiteral(text) { + return /^'.*'$/.test(text); + } + function trimLiteralEscapeChars(text) { + return text.replace(/^'(.*)'$/, '$1'); + } + function buildDateParseRegex(abstractRegex) { + var dateElements = Object.keys(regExpMap); + var re = abstractRegex; + for (var i = 0; i < dateElements.length; i++) { + re = re.split('${' + i + '}').join('(' + regExpMap[dateElements[i]] + ')'); } - format = escapeReservedSymbols(format); return new RegExp('^' + re + '$', [ 'i' ]); } + function setMapForFormat(format) { + var re = buildDateAbstractRegex(format); + return buildDateParseValuesMap(re); + } + function buildDateParseValuesMap(abstractRegex) { + var dateElements = Object.keys(regExpMap); + var valuesRegex = new RegExp('\\${(\\d+)}', 'g'); + var valuesMatch; + var keyIndex; + var valueKey; + var valueFunction; + var valuesFunctionMap = []; + while ((valuesMatch = valuesRegex.exec(abstractRegex)) !== null) { + keyIndex = valuesMatch[1]; + valueKey = dateElements[keyIndex]; + valueFunction = setFnMap[valueKey]; + valuesFunctionMap.push(valueFunction); + } + return valuesFunctionMap; + } $dateParser.init(); return $dateParser; }; @@ -2717,8 +3077,8 @@ var template = options.template || ''; var controller = options.controller; var controllerAs = options.controllerAs; - var resolve = angular.copy(options.resolve || {}); - var locals = angular.copy(options.locals || {}); + var resolve = options.resolve || {}; + var locals = options.locals || {}; var transformTemplate = options.transformTemplate || angular.identity; var bindToController = options.bindToController; angular.forEach(resolve, function(value, key) { @@ -2736,6 +3096,13 @@ } else { throw new Error('Missing `template` / `templateUrl` option.'); } + if (options.titleTemplate) { + resolve.$template = $q.all([ resolve.$template, fetchTemplate(options.titleTemplate) ]).then(function(templates) { + var templateEl = angular.element(templates[0]); + findElement('[ng-bind="title"]', templateEl[0]).removeAttr('ng-bind').html(templates[1]); + return templateEl[0].outerHTML; + }); + } if (options.contentTemplate) { resolve.$template = $q.all([ resolve.$template, fetchTemplate(options.contentTemplate) ]).then(function(templates) { var templateEl = angular.element(templates[0]); @@ -2786,7 +3153,6 @@ }); } } - bsCompilerService.$inject = [ '$q', '$http', '$injector', '$compile', '$controller', '$templateCache' ]; angular.module('mgcrea.ngStrap.dropdown', [ 'mgcrea.ngStrap.tooltip' ]).provider('$dropdown', function() { var defaults = this.defaults = { animation: 'am-fade', @@ -2806,7 +3172,7 @@ function DropdownFactory(element, config) { var $dropdown = {}; var options = angular.extend({}, defaults, config); - var scope = $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new(); + $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new(); $dropdown = $tooltip(element, options); var parentEl = element.parent(); $dropdown.$onKeyDown = function(evt) { @@ -2826,17 +3192,17 @@ $dropdown.show = function() { show(); $timeout(function() { - options.keyboard && $dropdown.$element && $dropdown.$element.on('keydown', $dropdown.$onKeyDown); + if (options.keyboard && $dropdown.$element) $dropdown.$element.on('keydown', $dropdown.$onKeyDown); bodyEl.on('click', onBodyClick); }, 0, false); - parentEl.hasClass('dropdown') && parentEl.addClass('open'); + if (parentEl.hasClass('dropdown')) parentEl.addClass('open'); }; var hide = $dropdown.hide; $dropdown.hide = function() { if (!$dropdown.$isShown) return; - options.keyboard && $dropdown.$element && $dropdown.$element.off('keydown', $dropdown.$onKeyDown); + if (options.keyboard && $dropdown.$element) $dropdown.$element.off('keydown', $dropdown.$onKeyDown); bodyEl.off('click', onBodyClick); - parentEl.hasClass('dropdown') && parentEl.removeClass('open'); + if (parentEl.hasClass('dropdown')) parentEl.removeClass('open'); hide(); }; var destroy = $dropdown.destroy; @@ -2857,36 +3223,51 @@ restrict: 'EAC', scope: true, compile: function(tElement, tAttrs) { - var options = {}; if (!tAttrs.bsDropdown) { var nextSibling = tElement[0].nextSibling; while (nextSibling && nextSibling.nodeType !== 1) { nextSibling = nextSibling.nextSibling; } - if (nextSibling.classList.contains('dropdown-menu')) { - options.template = nextSibling.outerHTML; - options.templateUrl = undefined; + if (nextSibling && nextSibling.className.split(' ').indexOf('dropdown-menu') >= 0) { + tAttrs.template = nextSibling.outerHTML; + tAttrs.templateUrl = undefined; nextSibling.parentNode.removeChild(nextSibling); } } return function postLink(scope, element, attr) { - options.scope = scope; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id' ], function(key) { + var options = { + scope: scope + }; + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id', 'autoClose' ], function(key) { if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - attr.bsDropdown && scope.$watch(attr.bsDropdown, function(newValue, oldValue) { - scope.content = newValue; - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!dropdown || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i); - newValue === true ? dropdown.show() : dropdown.hide(); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); + if (attr.bsDropdown) { + scope.$watch(attr.bsDropdown, function(newValue, oldValue) { + scope.content = newValue; + }, true); + } var dropdown = $dropdown(element, options); + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!dropdown || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i); + if (newValue === true) { + dropdown.show(); + } else { + dropdown.hide(); + } + }); + } scope.$on('$destroy', function() { if (dropdown) dropdown.destroy(); options = null; @@ -2896,28 +3277,146 @@ } }; } ]); - angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$datepicker', function() { + angular.module('mgcrea.ngStrap.button', []).provider('$button', function() { var defaults = this.defaults = { - animation: 'am-fade', - prefixClass: 'datepicker', - placement: 'bottom-left', - templateUrl: 'datepicker/datepicker.tpl.html', - trigger: 'focus', - container: false, - keyboard: true, - html: false, - delay: 0, - useNative: false, - dateType: 'date', - dateFormat: 'shortDate', - timezone: null, - modelDateFormat: null, - dayFormat: 'dd', - monthFormat: 'MMM', - yearFormat: 'yyyy', - monthTitleFormat: 'MMMM yyyy', - yearTitleFormat: 'yyyy', - strictFormat: false, + activeClass: 'active', + toggleEvent: 'click' + }; + this.$get = function() { + return { + defaults: defaults + }; + }; + }).directive('bsCheckboxGroup', function() { + return { + restrict: 'A', + require: 'ngModel', + compile: function postLink(element, attr) { + element.attr('data-toggle', 'buttons'); + element.removeAttr('ng-model'); + var children = element[0].querySelectorAll('input[type="checkbox"]'); + angular.forEach(children, function(child) { + var childEl = angular.element(child); + childEl.attr('bs-checkbox', ''); + childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value')); + }); + } + }; + }).directive('bsCheckbox', [ '$button', '$$rAF', function($button, $$rAF) { + var defaults = $button.defaults; + var constantValueRegExp = /^(true|false|\d+)$/; + return { + restrict: 'A', + require: 'ngModel', + link: function postLink(scope, element, attr, controller) { + var options = defaults; + var isInput = element[0].nodeName === 'INPUT'; + var activeElement = isInput ? element.parent() : element; + var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true; + if (constantValueRegExp.test(attr.trueValue)) { + trueValue = scope.$eval(attr.trueValue); + } + var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false; + if (constantValueRegExp.test(attr.falseValue)) { + falseValue = scope.$eval(attr.falseValue); + } + var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean'; + if (hasExoticValues) { + controller.$parsers.push(function(viewValue) { + return viewValue ? trueValue : falseValue; + }); + controller.$formatters.push(function(modelValue) { + return angular.equals(modelValue, trueValue); + }); + } + controller.$render = function() { + var isActive = !!controller.$viewValue; + $$rAF(function() { + if (isInput) element[0].checked = isActive; + activeElement.toggleClass(options.activeClass, isActive); + }); + }; + element.bind(options.toggleEvent, function() { + scope.$apply(function() { + if (!isInput) { + controller.$setViewValue(!activeElement.hasClass('active')); + } + controller.$render(); + }); + }); + } + }; + } ]).directive('bsRadioGroup', function() { + return { + restrict: 'A', + require: 'ngModel', + compile: function postLink(element, attr) { + element.attr('data-toggle', 'buttons'); + element.removeAttr('ng-model'); + var children = element[0].querySelectorAll('input[type="radio"]'); + angular.forEach(children, function(child) { + angular.element(child).attr('bs-radio', ''); + angular.element(child).attr('ng-model', attr.ngModel); + }); + } + }; + }).directive('bsRadio', [ '$button', '$$rAF', function($button, $$rAF) { + var defaults = $button.defaults; + var constantValueRegExp = /^(true|false|\d+)$/; + return { + restrict: 'A', + require: 'ngModel', + link: function postLink(scope, element, attr, controller) { + var options = defaults; + var isInput = element[0].nodeName === 'INPUT'; + var activeElement = isInput ? element.parent() : element; + var value; + attr.$observe('value', function(v) { + if (typeof v !== 'boolean' && constantValueRegExp.test(v)) { + value = scope.$eval(v); + } else { + value = v; + } + controller.$render(); + }); + controller.$render = function() { + var isActive = angular.equals(controller.$viewValue, value); + $$rAF(function() { + if (isInput) element[0].checked = isActive; + activeElement.toggleClass(options.activeClass, isActive); + }); + }; + element.bind(options.toggleEvent, function() { + scope.$apply(function() { + controller.$setViewValue(value); + controller.$render(); + }); + }); + } + }; + } ]); + angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$datepicker', function() { + var defaults = this.defaults = { + animation: 'am-fade', + prefixClass: 'datepicker', + placement: 'bottom-left', + templateUrl: 'datepicker/datepicker.tpl.html', + trigger: 'focus', + container: false, + keyboard: true, + html: false, + delay: 0, + useNative: false, + dateType: 'date', + dateFormat: 'shortDate', + timezone: null, + modelDateFormat: null, + dayFormat: 'dd', + monthFormat: 'MMM', + yearFormat: 'yyyy', + monthTitleFormat: 'MMMM yyyy', + yearTitleFormat: 'yyyy', + strictFormat: false, autoclose: false, minDate: -Infinity, maxDate: +Infinity, @@ -2925,11 +3424,13 @@ minView: 0, startWeek: 0, daysOfWeekDisabled: '', + hasToday: false, + hasClear: false, iconLeft: 'glyphicon glyphicon-chevron-left', iconRight: 'glyphicon glyphicon-chevron-right' }; this.$get = [ '$window', '$document', '$rootScope', '$sce', '$dateFormatter', 'datepickerViews', '$tooltip', '$timeout', function($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) { - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale(); function DatepickerFactory(element, controller, config) { @@ -2944,8 +3445,11 @@ scope.$mode = options.startView; scope.$iconLeft = options.iconLeft; scope.$iconRight = options.iconRight; + scope.$hasToday = options.hasToday; + scope.$hasClear = options.hasClear; var $picker = $datepicker.$views[scope.$mode]; - scope.$select = function(date) { + scope.$select = function(date, disabled) { + if (disabled) return; $datepicker.select(date); }; scope.$selectPane = function(value) { @@ -2954,6 +3458,22 @@ scope.$toggleMode = function() { $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length); }; + scope.$setToday = function() { + if (options.autoclose) { + $datepicker.setMode(0); + $datepicker.select(new Date()); + } else { + $datepicker.select(new Date(), true); + } + }; + scope.$clear = function() { + if (options.autoclose) { + $datepicker.setMode(0); + $datepicker.select(null); + } else { + $datepicker.select(null, true); + } + }; $datepicker.update = function(date) { if (angular.isDate(date) && !isNaN(date.getTime())) { $datepicker.$date = date; @@ -2968,7 +3488,13 @@ } }; $datepicker.select = function(date, keep) { - if (!angular.isDate(controller.$dateValue)) controller.$dateValue = new Date(date); + if (angular.isDate(date)) { + if (!angular.isDate(controller.$dateValue) || isNaN(controller.$dateValue.getTime())) { + controller.$dateValue = new Date(date); + } + } else { + controller.$dateValue = null; + } if (!scope.$mode || keep) { controller.$setViewValue(angular.copy(date)); controller.$render(); @@ -3099,8 +3625,7 @@ return DatepickerFactory; } ]; }).directive('bsDatepicker', [ '$window', '$parse', '$q', '$dateFormatter', '$dateParser', '$datepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) { - var defaults = $datepicker.defaults; - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); return { restrict: 'EAC', require: 'ngModel', @@ -3108,15 +3633,21 @@ var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent', 'hasToday', 'hasClear' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; - angular.forEach([ 'html', 'container', 'autoclose', 'useNative' ], function(key) { + angular.forEach([ 'html', 'container', 'autoclose', 'useNative', 'hasToday', 'hasClear' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { options[key] = false; } }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); var datepicker = $datepicker(element, controller, options); options = datepicker.$options; if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd'; @@ -3129,21 +3660,31 @@ lang: lang, strict: options.strictFormat }); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!datepicker || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i); - newValue === true ? datepicker.show() : datepicker.hide(); - }); - angular.forEach([ 'minDate', 'maxDate' ], function(key) { - angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) { - datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue); - !isNaN(datepicker.$options[key]) && datepicker.$build(false); - validateAgainstMinMaxDate(controller.$dateValue); + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!datepicker || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i); + if (newValue === true) { + datepicker.show(); + } else { + datepicker.hide(); + } }); + } + angular.forEach([ 'minDate', 'maxDate' ], function(key) { + if (angular.isDefined(attr[key])) { + attr.$observe(key, function(newValue) { + datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue); + if (!isNaN(datepicker.$options[key])) datepicker.$build(false); + validateAgainstMinMaxDate(controller.$dateValue); + }); + } }); - angular.isDefined(attr.dateFormat) && attr.$observe('dateFormat', function(newValue) { - datepicker.$options.dateFormat = newValue; - }); + if (angular.isDefined(attr.dateFormat)) { + attr.$observe('dateFormat', function(newValue) { + datepicker.$options.dateFormat = newValue; + }); + } scope.$watch(attr.ngModel, function(newValue, oldValue) { datepicker.update(controller.$dateValue); }, true); @@ -3180,9 +3721,8 @@ if (!parsedDate || isNaN(parsedDate.getTime())) { controller.$setValidity('date', false); return; - } else { - validateAgainstMinMaxDate(parsedDate); } + validateAgainstMinMaxDate(parsedDate); if (options.dateType === 'string') { date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true); return formatDate(date, options.modelDateFormat || options.dateFormat); @@ -3194,9 +3734,8 @@ return date.getTime() / 1e3; } else if (options.dateType === 'iso') { return date.toISOString(); - } else { - return new Date(date); } + return new Date(date); }); controller.$formatters.push(function(modelValue) { var date; @@ -3228,10 +3767,6 @@ } }; } ]).provider('datepickerViews', function() { - var defaults = this.defaults = { - dayFormat: 'dd', - daySplit: 7 - }; function split(arr, size) { var arrays = []; while (arr.length > 0) { @@ -3284,11 +3819,14 @@ } }, build: function() { - var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1), firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset(); - var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5), firstDateOffset = firstDate.getTimezoneOffset(); + var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1); + var firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset(); + var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5); + var firstDateOffset = firstDate.getTimezoneOffset(); var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString(); if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 6e4); - var days = [], day; + var days = []; + var day; for (var i = 0; i < 42; i++) { day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i)); days.push({ @@ -3304,6 +3842,7 @@ scope.showLabels = true; scope.labels = weekDaysLabelsHtml; scope.rows = split(days, this.split); + scope.isTodayDisabled = this.isDisabled(new Date()); this.built = true; }, isSelected: function(date) { @@ -3355,8 +3894,8 @@ } }, build: function() { - var firstMonth = new Date(viewDate.year, 0, 1); - var months = [], month; + var months = []; + var month; for (var i = 0; i < 12; i++) { month = new Date(viewDate.year, i, 1); months.push({ @@ -3413,7 +3952,8 @@ }, build: function() { var firstYear = viewDate.year - viewDate.year % (this.split * 3); - var years = [], year; + var years = []; + var year; for (var i = 0; i < 12; i++) { year = new Date(firstYear + i, 0, 1); years.push({ @@ -3439,7 +3979,8 @@ if (!picker.$date) { return; } - var actualYear = picker.$date.getFullYear(), newDate = new Date(picker.$date); + var actualYear = picker.$date.getFullYear(); + var newDate = new Date(picker.$date); if (evt.keyCode === 37) newDate.setYear(actualYear - 1); else if (evt.keyCode === 38) newDate.setYear(actualYear - 4); else if (evt.keyCode === 39) newDate.setYear(actualYear + 1); else if (evt.keyCode === 40) newDate.setYear(actualYear + 4); if (!this.isDisabled(newDate)) picker.select(newDate, true); } @@ -3499,8 +4040,8 @@ self.$setActive = $scope.$setActive = function(value) { if (angular.isArray(value)) { self.$targets.$active = value; - } else if (!self.$options.disallowToggle) { - isActive(value) ? deactivateItem(value) : activateItem(value); + } else if (!self.$options.disallowToggle && isActive(value)) { + deactivateItem(value); } else { activateItem(value); } @@ -3509,7 +4050,10 @@ }); }; self.$activeIndexes = function() { - return self.$options.allowMultiple ? self.$targets.$active : self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1; + if (self.$options.allowMultiple) { + return self.$targets.$active; + } + return self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1; }; function fixActiveItemIndexes(index) { var activeIndexes = self.$targets.$active; @@ -3524,7 +4068,7 @@ } function isActive(value) { var activeItems = self.$targets.$active; - return activeItems.indexOf(value) === -1 ? false : true; + return activeItems.indexOf(value) !== -1; } function deactivateItem(value) { var index = self.$targets.$active.indexOf(value); @@ -3548,7 +4092,6 @@ return $collapse; }; }).directive('bsCollapse', [ '$window', '$animate', '$collapse', function($window, $animate, $collapse) { - var defaults = $collapse.defaults; return { require: [ '?ngModel', 'bsCollapse' ], controller: [ '$scope', '$element', '$attrs', $collapse.controller ], @@ -3581,7 +4124,6 @@ return { require: [ '^?ngModel', '^bsCollapse' ], link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsCollapseCtrl = controllers[1]; element.attr('data-toggle', 'collapse'); bsCollapseCtrl.$registerToggle(element); @@ -3589,9 +4131,11 @@ bsCollapseCtrl.$unregisterToggle(element); }); element.on('click', function() { - var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element); - bsCollapseCtrl.$setActive(index * 1); - scope.$apply(); + if (!attrs.disabled) { + var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element); + bsCollapseCtrl.$setActive(index * 1); + scope.$apply(); + } }); } }; @@ -3599,7 +4143,6 @@ return { require: [ '^?ngModel', '^bsCollapse' ], link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsCollapseCtrl = controllers[1]; element.addClass('collapse'); if (bsCollapseCtrl.$options.animation) { @@ -3629,125 +4172,6 @@ } }; } ]); - angular.module('mgcrea.ngStrap.button', []).provider('$button', function() { - var defaults = this.defaults = { - activeClass: 'active', - toggleEvent: 'click' - }; - this.$get = function() { - return { - defaults: defaults - }; - }; - }).directive('bsCheckboxGroup', function() { - return { - restrict: 'A', - require: 'ngModel', - compile: function postLink(element, attr) { - element.attr('data-toggle', 'buttons'); - element.removeAttr('ng-model'); - var children = element[0].querySelectorAll('input[type="checkbox"]'); - angular.forEach(children, function(child) { - var childEl = angular.element(child); - childEl.attr('bs-checkbox', ''); - childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value')); - }); - } - }; - }).directive('bsCheckbox', [ '$button', '$$rAF', function($button, $$rAF) { - var defaults = $button.defaults; - var constantValueRegExp = /^(true|false|\d+)$/; - return { - restrict: 'A', - require: 'ngModel', - link: function postLink(scope, element, attr, controller) { - var options = defaults; - var isInput = element[0].nodeName === 'INPUT'; - var activeElement = isInput ? element.parent() : element; - var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true; - if (constantValueRegExp.test(attr.trueValue)) { - trueValue = scope.$eval(attr.trueValue); - } - var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false; - if (constantValueRegExp.test(attr.falseValue)) { - falseValue = scope.$eval(attr.falseValue); - } - var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean'; - if (hasExoticValues) { - controller.$parsers.push(function(viewValue) { - return viewValue ? trueValue : falseValue; - }); - controller.$formatters.push(function(modelValue) { - return angular.equals(modelValue, trueValue); - }); - scope.$watch(attr.ngModel, function(newValue, oldValue) { - controller.$render(); - }); - } - controller.$render = function() { - var isActive = angular.equals(controller.$modelValue, trueValue); - $$rAF(function() { - if (isInput) element[0].checked = isActive; - activeElement.toggleClass(options.activeClass, isActive); - }); - }; - element.bind(options.toggleEvent, function() { - scope.$apply(function() { - if (!isInput) { - controller.$setViewValue(!activeElement.hasClass('active')); - } - if (!hasExoticValues) { - controller.$render(); - } - }); - }); - } - }; - } ]).directive('bsRadioGroup', function() { - return { - restrict: 'A', - require: 'ngModel', - compile: function postLink(element, attr) { - element.attr('data-toggle', 'buttons'); - element.removeAttr('ng-model'); - var children = element[0].querySelectorAll('input[type="radio"]'); - angular.forEach(children, function(child) { - angular.element(child).attr('bs-radio', ''); - angular.element(child).attr('ng-model', attr.ngModel); - }); - } - }; - }).directive('bsRadio', [ '$button', '$$rAF', function($button, $$rAF) { - var defaults = $button.defaults; - var constantValueRegExp = /^(true|false|\d+)$/; - return { - restrict: 'A', - require: 'ngModel', - link: function postLink(scope, element, attr, controller) { - var options = defaults; - var isInput = element[0].nodeName === 'INPUT'; - var activeElement = isInput ? element.parent() : element; - var value; - attr.$observe('value', function(v) { - value = constantValueRegExp.test(v) ? scope.$eval(v) : v; - controller.$render(); - }); - controller.$render = function() { - var isActive = angular.equals(controller.$modelValue, value); - $$rAF(function() { - if (isInput) element[0].checked = isActive; - activeElement.toggleClass(options.activeClass, isActive); - }); - }; - element.bind(options.toggleEvent, function() { - scope.$apply(function() { - controller.$setViewValue(value); - controller.$render(); - }); - }); - } - }; - } ]); angular.module('mgcrea.ngStrap.aside', [ 'mgcrea.ngStrap.modal' ]).provider('$aside', function() { var defaults = this.defaults = { animation: 'am-fade-and-slide-right', @@ -3773,7 +4197,6 @@ return AsideFactory; } ]; }).directive('bsAside', [ '$window', '$sce', '$aside', function($window, $sce, $aside) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; return { restrict: 'EAC', scope: true, @@ -3790,18 +4213,28 @@ angular.forEach([ 'backdrop', 'keyboard', 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); - attr.bsAside && scope.$watch(attr.bsAside, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + angular.forEach([ 'title', 'content' ], function(key) { + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsAside) { + scope.$watch(attr.bsAside, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var aside = $aside(options); element.on(attr.trigger || 'click', aside.toggle); scope.$on('$destroy', function() { @@ -3851,7 +4284,6 @@ return AlertFactory; } ]; }).directive('bsAlert', [ '$window', '$sce', '$alert', function($window, $sce, $alert) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; return { restrict: 'EAC', scope: true, @@ -3868,21 +4300,31 @@ angular.forEach([ 'keyboard', 'html', 'container', 'dismissable' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); if (!scope.hasOwnProperty('title')) { scope.title = ''; } angular.forEach([ 'title', 'content', 'type' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); - }); - attr.bsAlert && scope.$watch(attr.bsAlert, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsAlert) { + scope.$watch(attr.bsAlert, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var alert = $alert(options); element.on(attr.trigger || 'click', alert.toggle); scope.$on('$destroy', function() { @@ -3896,7 +4338,8 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce' ]).provider('$affix', function() { var defaults = this.defaults = { offsetTop: 'auto', - inlineStyles: true + inlineStyles: true, + setWidth: true }; this.$get = [ '$window', 'debounce', 'dimensions', function($window, debounce, dimensions) { var bodyEl = angular.element($window.document.body); @@ -3905,7 +4348,14 @@ var $affix = {}; var options = angular.extend({}, defaults, config); var targetEl = options.target; - var reset = 'affix affix-top affix-bottom', setWidth = false, initialAffixTop = 0, initialOffsetTop = 0, offsetTop = 0, offsetBottom = 0, affixed = null, unpin = null; + var reset = 'affix affix-top affix-bottom'; + var setWidth = false; + var initialAffixTop = 0; + var initialOffsetTop = 0; + var offsetTop = 0; + var offsetBottom = 0; + var affixed = null; + var unpin = null; var parent = element.parent(); if (options.offsetParent) { if (options.offsetParent.match(/^\d+$/)) { @@ -3919,7 +4369,7 @@ $affix.init = function() { this.$parseOffsets(); initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop; - setWidth = !element[0].style.width; + setWidth = options.setWidth && !element[0].style.width; targetEl.on('scroll', this.checkPosition); targetEl.on('click', this.checkPositionWithEventLoop); windowEl.on('resize', this.$debouncedOnResize); @@ -3981,9 +4431,11 @@ }; $affix.$debouncedOnResize = debounce($affix.$onResize, 50); $affix.$parseOffsets = function() { - var initialPosition = element.css('position'); + var initialPosition = element[0].style.position; + var initialTop = element[0].style.top; if (options.inlineStyles) { element.css('position', options.offsetParent ? '' : 'relative'); + element.css('top', ''); } if (options.offsetTop) { if (options.offsetTop === 'auto') { @@ -4009,20 +4461,20 @@ } if (options.inlineStyles) { element.css('position', initialPosition); + element.css('top', initialTop); } }; - function getRequiredAffixClass(unpin, position, elementHeight) { + function getRequiredAffixClass(_unpin, position, elementHeight) { var scrollTop = getScrollTop(); var scrollHeight = getScrollHeight(); if (scrollTop <= offsetTop) { return 'top'; - } else if (unpin !== null && scrollTop + unpin <= position.top) { - return 'middle'; + } else if (_unpin !== null) { + return scrollTop + _unpin <= position.top ? 'middle' : 'bottom'; } else if (offsetBottom !== null && position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom) { return 'bottom'; - } else { - return 'middle'; } + return 'middle'; } function getScrollTop() { return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop; @@ -4035,7 +4487,7 @@ } return AffixFactory; } ]; - }).directive('bsAffix', [ '$affix', '$window', function($affix, $window) { + }).directive('bsAffix', [ '$affix', '$window', '$timeout', function($affix, $window, $timeout) { return { restrict: 'EAC', require: '^?bsAffixTarget', @@ -4044,7 +4496,7 @@ scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window) }; - angular.forEach([ 'offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles' ], function(key) { + angular.forEach([ 'offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles', 'setWidth' ], function(key) { if (angular.isDefined(attr[key])) { var option = attr[key]; if (/true/i.test(option)) option = true; @@ -4052,9 +4504,12 @@ options[key] = option; } }); - var affix = $affix(element, options); + var affix; + $timeout(function() { + affix = $affix(element, options); + }); scope.$on('$destroy', function() { - affix && affix.destroy(); + if (affix) affix.destroy(); options = null; affix = null; }); diff --git a/dist/angular-strap.min.js b/dist/angular-strap.min.js index b2152425d..32e18dc27 100644 --- a/dist/angular-strap.min.js +++ b/dist/angular-strap.min.js @@ -1,11 +1,11 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -!function(e,t,n){'use strict';function a(e,n,a,o,i,r){function s(e,n){return angular.element((n||t).querySelectorAll(e))}function l(e){return u[e]?u[e]:u[e]=n.get(e,{cache:r}).then(function(e){return e.data})}this.compile=function(t){t.template&&/\.html$/.test(t.template)&&(console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.'),t.templateUrl=t.template,t.template='');var n=t.templateUrl,r=t.template||'',u=t.controller,c=t.controllerAs,d=angular.copy(t.resolve||{}),f=angular.copy(t.locals||{}),p=t.transformTemplate||angular.identity,g=t.bindToController;if(angular.forEach(d,function(e,t){angular.isString(e)?d[t]=a.get(e):d[t]=a.invoke(e)}),angular.extend(d,f),r)d.$template=e.when(r);else{if(!n)throw new Error('Missing `template` / `templateUrl` option.');d.$template=l(n)}return t.contentTemplate&&(d.$template=e.all([d.$template,l(t.contentTemplate)]).then(function(e){var n=angular.element(e[0]),a=s('[ng-bind="content"]',n[0]).removeAttr('ng-bind').html(e[1]);return t.templateUrl||a.next().remove(),n[0].outerHTML})),e.all(d).then(function(e){var n=p(e.$template);t.html&&(n=n.replace(/ng-bind="/gi,'ng-bind-html="'));var a=angular.element('
').html(n.trim()).contents(),r=o(a);return{locals:e,element:a,link:function(t){if(e.$scope=t,u){var n=i(u,e,!0);g&&angular.extend(n.instance,e);var o=angular.isObject(n)?n:n();a.data('$ngControllerController',o),a.children().data('$ngControllerController',o),c&&(t[c]=o)}return r.apply(null,arguments)}}})};var u={}}angular.module('mgcrea.ngStrap.typeahead',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$typeahead',function(){var e=this.defaults={animation:'am-fade',prefixClass:'typeahead',prefixEvent:'$typeahead',placement:'bottom-left',templateUrl:'typeahead/typeahead.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,minLength:1,filter:'bsAsyncFilter',limit:6,autoSelect:!1,comparator:'',trimValue:!0};this.$get=['$window','$rootScope','$tooltip','$$rAF','$timeout',function(t,n,a,o,i){function r(t,n,r){var l={},u=angular.extend({},e,r);l=a(t,u);var c=r.scope,d=l.$scope;d.$resetMatches=function(){d.$matches=[],d.$activeIndex=u.autoSelect?0:-1},d.$resetMatches(),d.$activate=function(e){d.$$postDigest(function(){l.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){l.select(e)})},d.$isVisible=function(){return l.$isVisible()},l.update=function(e){d.$matches=e,d.$activeIndex>=e.length&&(d.$activeIndex=u.autoSelect?0:-1),s(d),o(l.$applyPlacement)},l.activate=function(e){d.$activeIndex=e},l.select=function(e){if(-1!==e){var t=d.$matches[e].value;n.$setViewValue(t),n.$render(),d.$resetMatches(),c&&c.$digest(),d.$emit(u.prefixEvent+'.select',t,e,l)}},l.$isVisible=function(){return u.minLength&&n?d.$matches.length&&angular.isString(n.$viewValue)&&n.$viewValue.length>=u.minLength:!!d.$matches.length},l.$getIndex=function(e){var t=d.$matches.length,n=t;if(t){for(n=t;n--&&d.$matches[n].value!==e;);if(!(0>n))return n}},l.$onMouseDown=function(e){e.preventDefault(),e.stopPropagation()},l.$onKeyDown=function(e){/(38|40|13)/.test(e.keyCode)&&(!l.$isVisible()||13===e.keyCode&&-1===d.$activeIndex||(e.preventDefault(),e.stopPropagation()),13===e.keyCode&&d.$matches.length?l.select(d.$activeIndex):38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:40===e.keyCode&&d.$activeIndex0?void r.$setViewValue(r.$viewValue.substring(0,r.$viewValue.length-1)):(e.length>c&&(e=e.slice(0,c)),g.update(e),void r.$render())})}),r.$formatters.push(function(e){var t=p.displayValue(e);return t?t:e&&'object'!=typeof e?e:''}),r.$render=function(){if(r.$isEmpty(r.$viewValue))return t.val('');var e=g.$getIndex(r.$modelValue),n=angular.isDefined(e)?g.$scope.$matches[e].label:r.$viewValue;n=angular.isObject(n)?p.displayValue(n):n;var a=n?n.toString().replace(/<(?:.|\n)*?>/gm,''):'';t.val(s.trimValue===!1?a:a.trim())},e.$on('$destroy',function(){g&&g.destroy(),s=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tooltip',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$tooltip',function(){var e=this.defaults={animation:'am-fade',customClass:'',prefixClass:'tooltip',prefixEvent:'tooltip',container:!1,target:!1,placement:'top',templateUrl:'tooltip/tooltip.tpl.html',template:'',contentTemplate:!1,trigger:'hover focus',keyboard:!1,html:!1,show:!1,title:'',type:'',delay:0,autoClose:!1,bsEnabled:!0,viewport:{selector:'body',padding:0}};this.$get=['$window','$rootScope','$bsCompiler','$q','$templateCache','$http','$animate','$sce','dimensions','$$rAF','$timeout',function(n,a,o,i,r,s,l,u,c,d,f){function p(i,r){function s(){I.$emit(V.prefixEvent+'.show',F)}function p(){if(I.$emit(V.prefixEvent+'.hide',F),R===j){if(z&&'focus'===V.trigger)return i[0].blur();A()}}function v(){var e=V.trigger.split(' ');angular.forEach(e,function(e){'click'===e?i.on('click',F.toggle):'manual'!==e&&(i.on('hover'===e?'mouseenter':'focus',F.enter),i.on('hover'===e?'mouseleave':'blur',F.leave),'button'===N&&'hover'!==e&&i.on($?'touchstart':'mousedown',F.$onFocusElementMouseDown))})}function w(){for(var e=V.trigger.split(' '),t=e.length;t--;){var n=e[t];'click'===n?i.off('click',F.toggle):'manual'!==n&&(i.off('hover'===n?'mouseenter':'focus',F.enter),i.off('hover'===n?'mouseleave':'blur',F.leave),'button'===N&&'hover'!==n&&i.off($?'touchstart':'mousedown',F.$onFocusElementMouseDown))}}function y(){'focus'!==V.trigger?R.on('keyup',F.$onKeyUp):i.on('keyup',F.$onFocusKeyUp)}function b(){'focus'!==V.trigger?R.off('keyup',F.$onKeyUp):i.off('keyup',F.$onFocusKeyUp)}function D(){f(function(){R.on('click',S),h.on('click',F.hide),K=!0},0,!1)}function k(){K&&(R.off('click',S),h.off('click',F.hide),K=!1)}function S(e){e.stopPropagation()}function x(e){e=e||V.target||i;var a=e[0],o='BODY'===a.tagName,r=a.getBoundingClientRect(),s={};for(var l in r)s[l]=r[l];null===s.width&&(s=angular.extend({},s,{width:r.right-r.left,height:r.bottom-r.top}));var u=o?{top:0,left:0}:c.offset(a),d={scroll:o?t.documentElement.scrollTop||t.body.scrollTop:e.prop('scrollTop')||0},f=o?{width:t.documentElement.clientWidth,height:n.innerHeight}:null;return angular.extend({},s,d,f,u)}function T(e,t,n,a){var o,i=e.split('-');switch(i[0]){case'right':o={top:t.top+t.height/2-a/2,left:t.left+t.width};break;case'bottom':o={top:t.top+t.height,left:t.left+t.width/2-n/2};break;case'left':o={top:t.top+t.height/2-a/2,left:t.left-n};break;default:o={top:t.top-a,left:t.left+t.width/2-n/2}}if(!i[1])return o;if('top'===i[0]||'bottom'===i[0])switch(i[1]){case'left':o.left=t.left;break;case'right':o.left=t.left+t.width-n}else if('left'===i[0]||'right'===i[0])switch(i[1]){case'top':o.top=t.top-a;break;case'bottom':o.top=t.top+t.height}return o}function C(e,t){var n=R[0],a=n.offsetWidth,o=n.offsetHeight,i=parseInt(c.css(n,'margin-top'),10),r=parseInt(c.css(n,'margin-left'),10);isNaN(i)&&(i=0),isNaN(r)&&(r=0),e.top=e.top+i,e.left=e.left+r,c.setOffset(n,angular.extend({using:function(e){R.css({top:Math.round(e.top)+'px',left:Math.round(e.left)+'px',right:''})}},e),0);var s=n.offsetWidth,l=n.offsetHeight;if('top'===t&&l!==o&&(e.top=e.top+o-l),!/top-left|top-right|bottom-left|bottom-right/.test(t)){var u=E(t,e,s,l);if(u.left?e.left+=u.left:e.top+=u.top,c.setOffset(n,e),/top|right|bottom|left/.test(t)){var d=/top|bottom/.test(t),f=d?2*u.left-a+s:2*u.top-o+l,p=d?'offsetWidth':'offsetHeight';M(f,n[p],d)}}}function E(e,t,n,a){var o={top:0,left:0};if(!F.$viewport)return o;var i=V.viewport&&V.viewport.padding||0,r=x(F.$viewport);if(/right|left/.test(e)){var s=t.top-i-r.scroll,l=t.top+i-r.scroll+a;sr.top+r.height&&(o.top=r.top+r.height-l)}else{var u=t.left-i,c=t.left+i+n;ur.right&&(o.left=r.left+r.width-c)}return o}function M(e,t,n){var a=m('.tooltip-arrow, .arrow',R[0]);a.css(n?'left':'top',50*(1-e/t)+'%').css(n?'top':'left','')}function A(){clearTimeout(H),F.$isShown&&null!==R&&(V.autoClose&&k(),V.keyboard&&b()),q&&(q.$destroy(),q=null),R&&(R.remove(),R=F.$element=null)}var F={},V=F.$options=angular.extend({},e,r),O=F.$promise=o.compile(V),I=F.$scope=V.scope&&V.scope.$new()||a.$new(),N=i[0].nodeName.toLowerCase();if(V.delay&&angular.isString(V.delay)){var P=V.delay.split(',').map(parseFloat);V.delay=P.length>1?{show:P[0],hide:P[1]}:P[0]}F.$id=V.id||i.attr('id')||'',V.title&&(I.title=u.trustAsHtml(V.title)),I.$setEnabled=function(e){I.$$postDigest(function(){F.setEnabled(e)})},I.$hide=function(){I.$$postDigest(function(){F.hide()})},I.$show=function(){I.$$postDigest(function(){F.show()})},I.$toggle=function(){I.$$postDigest(function(){F.toggle()})},F.$isShown=I.$isShown=!1;var H,L,U,R,Y,q;O.then(function(e){U=e,F.init()}),F.init=function(){V.delay&&angular.isNumber(V.delay)&&(V.delay={show:V.delay,hide:V.delay}),'self'===V.container?Y=i:angular.isElement(V.container)?Y=V.container:V.container&&(Y=m(V.container)),v(),V.target&&(V.target=angular.isElement(V.target)?V.target:m(V.target)),V.show&&I.$$postDigest(function(){'focus'===V.trigger?i[0].focus():F.show()})},F.destroy=function(){w(),A(),I.$destroy()},F.enter=function(){return clearTimeout(H),L='in',V.delay&&V.delay.show?void(H=setTimeout(function(){'in'===L&&F.show()},V.delay.show)):F.show()},F.show=function(){if(V.bsEnabled&&!F.$isShown){I.$emit(V.prefixEvent+'.show.before',F);var e,t;V.container?(e=Y,t=Y[0].lastChild?angular.element(Y[0].lastChild):null):(e=null,t=i),R&&A(),q=F.$scope.$new(),R=F.$element=U.link(q,function(e,t){}),R.css({top:'-9999px',left:'-9999px',right:'auto',display:'block',visibility:'hidden'}),V.animation&&R.addClass(V.animation),V.type&&R.addClass(V.prefixClass+'-'+V.type),V.customClass&&R.addClass(V.customClass),t?t.after(R):e.prepend(R),F.$isShown=I.$isShown=!0,g(I),F.$applyPlacement(),angular.version.minor<=2?l.enter(R,e,t,s):l.enter(R,e,t).then(s),g(I),d(function(){R&&R.css({visibility:'visible'}),V.keyboard&&('focus'!==V.trigger&&F.focus(),y())}),V.autoClose&&D()}},F.leave=function(){return clearTimeout(H),L='out',V.delay&&V.delay.hide?void(H=setTimeout(function(){'out'===L&&F.hide()},V.delay.hide)):F.hide()};var z,j;F.hide=function(e){F.$isShown&&(I.$emit(V.prefixEvent+'.hide.before',F),z=e,j=R,angular.version.minor<=2?l.leave(R,p):l.leave(R).then(p),F.$isShown=I.$isShown=!1,g(I),V.keyboard&&null!==R&&b(),V.autoClose&&null!==R&&k())},F.toggle=function(){F.$isShown?F.leave():F.enter()},F.focus=function(){R[0].focus()},F.setEnabled=function(e){V.bsEnabled=e},F.setViewport=function(e){V.viewport=e},F.$applyPlacement=function(){if(R){var t=V.placement,n=/\s?auto?\s?/i,a=n.test(t);a&&(t=t.replace(n,'')||e.placement),R.addClass(V.placement);var o=x(),i=R.prop('offsetWidth'),r=R.prop('offsetHeight');if(F.$viewport=V.viewport&&m(V.viewport.selector||V.viewport),a){var s=t,l=x(F.$viewport);s.indexOf('bottom')>=0&&o.bottom+r>l.bottom?t=s.replace('bottom','top'):s.indexOf('top')>=0&&o.top-rl.width?t='right'===s?'left':t.replace('left','right'):('left'===s||'bottom-right'===s||'top-right'===s)&&o.left-it?t+12:t-12),n.$setViewValue(angular.copy(n.$dateValue)),n.$render()}},f.$build=function(){var e,t,n=m.midIndex=parseInt(g.length/2,10),a=[];for(e=0;e1*g.maxTime},m.$arrowAction=function(e,t){'picker'===g.arrowBehavior?f.$setTimeByStep(e,t):f.$moveIndex(e,t)},f.$setTimeByStep=function(e,t){var n=new Date(f.$date||y),a=n.getHours(),o=n.getMinutes(),i=n.getSeconds();0===t?n.setHours(a-parseInt(g.hourStep,10)*e):1===t?n.setMinutes(o-parseInt(g.minuteStep,10)*e):2===t&&n.setSeconds(i-parseInt(g.secondStep,10)*e),f.select(n,t,!0)},f.$moveIndex=function(e,t){var n;0===t?(n=new Date(1970,0,1,b.hour+e*g.length,b.minute,b.second),angular.extend(b,{hour:n.getHours()})):1===t?(n=new Date(1970,0,1,b.hour,b.minute+e*g.length*g.minuteStep,b.second),angular.extend(b,{minute:n.getMinutes()})):2===t&&(n=new Date(1970,0,1,b.hour,b.minute,b.second+e*g.length*g.secondStep),angular.extend(b,{second:n.getSeconds()})),f.$build()},f.$onMouseDown=function(e){if('input'!==e.target.nodeName.toLowerCase()&&e.preventDefault(),e.stopPropagation(),c){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},f.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void f.hide(!0);var t=new Date(f.$date),n=t.getHours(),a=h(t,k).length,o=t.getMinutes(),i=h(t,x).length,r=t.getSeconds(),s=h(t,T).length,u=1,c=/(37|39)/.test(e.keyCode),d=2+1*C+1*E;c&&(37===e.keyCode?v=1>v?d-1:v-1:39===e.keyCode&&(v=d-1>v?v+1:0));var m=[0,a],$=0;38===e.keyCode&&($=-1),40===e.keyCode&&($=1);var w=2===v&&C,y=2===v&&!C||3===v&&C;0===v?(t.setHours(n+$*parseInt(g.hourStep,10)),a=h(t,k).length,m=[0,a]):1===v?(t.setMinutes(o+$*parseInt(g.minuteStep,10)),i=h(t,x).length,m=[a+u,i]):w?(t.setSeconds(r+$*parseInt(g.secondStep,10)),s=h(t,T).length,m=[a+u+i+u,s]):y&&(c||f.switchMeridian(),m=[a+u+i+u+(s+u)*C,2]),f.select(t,v,!0),l(m[0],m[1]),p.$digest()}};var M=f.init;f.init=function(){return u&&g.useNative?(t.prop('type','time'),void t.css('-webkit-appearance','textfield')):(c&&(t.prop('type','text'),t.attr('readonly','true'),t.on('click',d)),void M())};var A=f.destroy;f.destroy=function(){u&&g.useNative&&t.off('click',d),A()};var F=f.show;f.show=function(){!c&&t.attr('readonly')||t.attr('disabled')||(F(),s(function(){f.$element&&f.$element.on(c?'touchstart':'mousedown',f.$onMouseDown),g.keyboard&&t&&t.on('keydown',f.$onKeyDown)},0,!1))};var V=f.hide;return f.hide=function(e){f.$isShown&&(f.$element&&f.$element.off(c?'touchstart':'mousedown',f.$onMouseDown),g.keyboard&&t&&t.off('keydown',f.$onKeyDown),V(e))},f}var u=/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent),c='createTouch'in t.document&&u;return e.lang||(e.lang=i.getDefaultLocale()),l.defaults=e,l}]}).directive('bsTimepicker',['$window','$parse','$q','$dateFormatter','$dateParser','$timepicker',function(e,t,a,o,i,r){var s=r.defaults,l=/(ip(a|o)d|iphone|android)/gi.test(e.navigator.userAgent);return{restrict:'EAC',require:'ngModel',link:function(e,t,a,u){function c(e){if(angular.isDate(e)){var t=isNaN(f.minTime)||new Date(e.getTime()).setFullYear(1970,0,1)>=f.minTime,n=isNaN(f.maxTime)||new Date(e.getTime()).setFullYear(1970,0,1)<=f.maxTime,a=t&&n;u.$setValidity('date',a),u.$setValidity('min',t),u.$setValidity('max',n),a&&(u.$dateValue=e)}}function d(){return!u.$dateValue||isNaN(u.$dateValue.getTime())?'':$(u.$dateValue,f.timeFormat)}var f={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','autoclose','timeType','timeFormat','timezone','modelTimeFormat','useNative','hourStep','minuteStep','secondStep','length','arrowBehavior','iconUp','iconDown','roundDisplay','id','prefixClass','prefixEvent'],function(e){angular.isDefined(a[e])&&(f[e]=a[e])});var p=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','roundDisplay'],function(e){angular.isDefined(a[e])&&p.test(a[e])&&(f[e]=!1)}),a.bsShow&&e.$watch(a.bsShow,function(e,t){g&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(timepicker),?/i)),e===!0?g.show():g.hide())}),l&&(f.useNative||s.useNative)&&(f.timeFormat='HH:mm');var g=r(t,u,f);f=g.$options;var m=f.lang,$=function(e,t,n){return o.formatDate(e,t,m,n)},h=i({format:f.timeFormat,lang:m});angular.forEach(['minTime','maxTime'],function(e){angular.isDefined(a[e])&&a.$observe(e,function(t){g.$options[e]=h.getTimeForAttribute(e,t),!isNaN(g.$options[e])&&g.$build(),c(u.$dateValue)})}),e.$watch(a.ngModel,function(e,t){g.update(u.$dateValue)},!0),u.$parsers.unshift(function(e){var t;if(!e)return u.$setValidity('date',!0),null;var a=angular.isDate(e)?e:h.parse(e,u.$dateValue);return!a||isNaN(a.getTime())?(u.$setValidity('date',!1),n):(c(a),'string'===f.timeType?(t=h.timezoneOffsetAdjust(a,f.timezone,!0),$(t,f.modelTimeFormat||f.timeFormat)):(t=h.timezoneOffsetAdjust(u.$dateValue,f.timezone,!0),'number'===f.timeType?t.getTime():'unix'===f.timeType?t.getTime()/1e3:'iso'===f.timeType?t.toISOString():new Date(t)))}),u.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===f.timeType?h.parse(e,null,f.modelTimeFormat):'unix'===f.timeType?new Date(1e3*e):new Date(e),u.$dateValue=h.timezoneOffsetAdjust(t,f.timezone),d()}),u.$render=function(){t.val(d())},e.$on('$destroy',function(){g&&g.destroy(),f=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tab',[]).provider('$tab',function(){var e=this.defaults={animation:'am-fade',template:'tab/tab.tpl.html',navClass:'nav-tabs',activeClass:'active'},t=this.controller=function(t,n,a){var o=this;o.$options=angular.copy(e),angular.forEach(['animation','navClass','activeClass'],function(e){angular.isDefined(a[e])&&(o.$options[e]=a[e])}),t.$navClass=o.$options.navClass,t.$activeClass=o.$options.activeClass,o.$panes=t.$panes=[],o.$activePaneChangeListeners=o.$viewChangeListeners=[],o.$push=function(e){angular.isUndefined(o.$panes.$active)&&t.$setActive(e.name||0),o.$panes.push(e)},o.$remove=function(e){var t,n=o.$panes.indexOf(e),a=o.$panes.$active;t=angular.isString(a)?o.$panes.map(function(e){return e.name}).indexOf(a):o.$panes.$active,o.$panes.splice(n,1),t>n?t--:n===t&&t===o.$panes.length&&t--,t>=0&&t',placeholder:'Choose among the following...',allText:'All',noneText:'None',maxLength:3,maxLengthHtml:'selected',iconCheckmark:'glyphicon glyphicon-ok'};this.$get=['$window','$document','$rootScope','$tooltip','$timeout',function(t,n,a,o,i){function r(a,r,s){var u={},c=angular.extend({},e,s);u=o(a,c);var d=u.$scope;d.$matches=[],c.multiple?d.$activeIndex=[]:d.$activeIndex=-1,d.$isMultiple=c.multiple,d.$showAllNoneButtons=c.allNoneButtons&&c.multiple,d.$iconCheckmark=c.iconCheckmark,d.$allText=c.allText,d.$noneText=c.noneText,d.$activate=function(e){d.$$postDigest(function(){u.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){u.select(e)})},d.$isVisible=function(){return u.$isVisible()},d.$isActive=function(e){return u.$isActive(e)},d.$selectAll=function(){for(var e=0;e=c.minLength:d.$matches.length},u.$isActive=function(e){return c.multiple?-1!==d.$activeIndex.indexOf(e):d.$activeIndex===e},u.$getIndex=function(e){var t=d.$matches.length,n=t;if(t){for(n=t;n--&&d.$matches[n].value!==e;);if(!(0>n))return n}},u.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),l){var t=angular.element(e.target);t.triggerHandler('click')}},u.$onKeyDown=function(e){return/(9|13|38|40)/.test(e.keyCode)?(9!==e.keyCode&&(e.preventDefault(),e.stopPropagation()),c.multiple&&9===e.keyCode?u.hide():c.multiple||13!==e.keyCode&&9!==e.keyCode?void(c.multiple||(38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:38===e.keyCode&&d.$activeIndex<0?d.$activeIndex=d.$matches.length-1:40===e.keyCode&&d.$activeIndex0||e.indexOf('Trident/')>0||e.indexOf('Edge/')>0},u.$selectScrollFix=function(e){'UL'===n[0].activeElement.tagName&&(e.preventDefault(),e.stopImmediatePropagation(),e.target.focus())};var f=u.show;u.show=function(){f(),c.multiple&&u.$element.addClass('select-multiple'),i(function(){u.$element.on(l?'touchstart':'mousedown',u.$onMouseDown),c.keyboard&&a.on('keydown',u.$onKeyDown)},0,!1)};var p=u.hide;return u.hide=function(){!c.multiple&&angular.isUndefined(r.$modelValue)&&(d.$activeIndex=-1),u.$element.off(l?'touchstart':'mousedown',u.$onMouseDown),c.keyboard&&a.off('keydown',u.$onKeyDown),p(!0)},u}var s=(angular.element(t.document.body),/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent)),l='createTouch'in t.document&&s;return r.defaults=e,r}]}).directive('bsSelect',['$window','$parse','$q','$select','$parseOptions',function(e,t,n,a,o){var i=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,r){var s={scope:e,placeholder:i.placeholder};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','placeholder','allNoneButtons','maxLength','maxLengthHtml','allText','noneText','iconCheckmark','autoClose','id','sort','caretHtml','prefixClass','prefixEvent'],function(e){angular.isDefined(n[e])&&(s[e]=n[e])});var l=/^(false|0|)$/i;angular.forEach(['html','container','allNoneButtons','sort'],function(e){angular.isDefined(n[e])&&l.test(n[e])&&(s[e]=!1)});var u=t.attr('data-multiple');if(angular.isDefined(u)&&(l.test(u)?s.multiple=!1:s.multiple=u),'select'===t[0].nodeName.toLowerCase()){var c=t;c.css('display','none'),t=angular.element(''),c.after(t)}var d=o(n.bsOptions),f=a(t,r,s);f.$isIE()&&t[0].addEventListener('blur',f.$selectScrollFix);var p=d.$match[7].replace(/\|.+/,'').trim();e.$watchCollection(p,function(t,n){d.valuesFn(e,r).then(function(e){f.update(e),r.$render()})}),e.$watch(n.ngModel,function(e,t){f.$updateActiveIndex(),r.$render()},!0),r.$render=function(){var e,n;s.multiple&&angular.isArray(r.$modelValue)?(e=r.$modelValue.map(function(e){return n=f.$getIndex(e), -angular.isDefined(n)?f.$scope.$matches[n].label:!1}).filter(angular.isDefined),e=e.length>(s.maxLength||i.maxLength)?e.length+' '+(s.maxLengthHtml||i.maxLengthHtml):e.join(', ')):(n=f.$getIndex(r.$modelValue),e=angular.isDefined(n)?f.$scope.$matches[n].label:!1),t.html((e?e:s.placeholder)+(s.caretHtml?s.caretHtml:i.caretHtml))},s.multiple&&(r.$isEmpty=function(e){return!e||0===e.length}),e.$on('$destroy',function(){f&&f.destroy(),s=null,f=null})}}}]),angular.module('mgcrea.ngStrap.scrollspy',['mgcrea.ngStrap.helpers.debounce','mgcrea.ngStrap.helpers.dimensions']).provider('$scrollspy',function(){var e=this.$$spies={},n=this.defaults={debounce:150,throttle:100,offset:100};this.$get=['$window','$document','$rootScope','dimensions','debounce','throttle',function(a,o,i,r,s,l){function u(e,t){return e[0].nodeName&&e[0].nodeName.toLowerCase()===t.toLowerCase()}function c(o){var c=angular.extend({},n,o);c.element||(c.element=p);var g=u(c.element,'body'),m=g?d:c.element,$=g?'window':c.id;if(e[$])return e[$].$$count++,e[$];var h,v,w,y,b,D,k,S,x={},T=x.$trackedElements=[],C=[];return x.init=function(){this.$$count=1,y=s(this.checkPosition,c.debounce),b=l(this.checkPosition,c.throttle),m.on('click',this.checkPositionWithEventLoop),d.on('resize',y),m.on('scroll',b),D=s(this.checkOffsets,c.debounce),h=i.$on('$viewContentLoaded',D),v=i.$on('$includeContentLoaded',D),D(),$&&(e[$]=x)},x.destroy=function(){this.$$count--,this.$$count>0||(m.off('click',this.checkPositionWithEventLoop),d.off('resize',y),m.off('scroll',b),h(),v(),$&&delete e[$])},x.checkPosition=function(){if(C.length){if(S=(g?a.pageYOffset:m.prop('scrollTop'))||0,k=Math.max(a.innerHeight,f.prop('clientHeight')),SC[e+1].offsetTop))return x.$activateElement(C[e])}},x.checkPositionWithEventLoop=function(){setTimeout(x.checkPosition,1)},x.$activateElement=function(e){if(w){var t=x.$getTrackedElement(w);t&&(t.source.removeClass('active'),u(t.source,'li')&&u(t.source.parent().parent(),'li')&&t.source.parent().parent().removeClass('active'))}w=e.target,e.source.addClass('active'),u(e.source,'li')&&u(e.source.parent().parent(),'li')&&e.source.parent().parent().addClass('active')},x.$getTrackedElement=function(e){return T.filter(function(t){return t.target===e})[0]},x.checkOffsets=function(){angular.forEach(T,function(e){var n=t.querySelector(e.target);e.offsetTop=n?r.offset(n).top:null,c.offset&&null!==e.offsetTop&&(e.offsetTop-=1*c.offset)}),C=T.filter(function(e){return null!==e.offsetTop}).sort(function(e,t){return e.offsetTop-t.offsetTop}),y()},x.trackElement=function(e,t){T.push({target:e,source:t})},x.untrackElement=function(e,t){for(var n,a=T.length;a--;)if(T[a].target===e&&T[a].source===t){n=a;break}T=T.splice(n,1)},x.activate=function(e){T[e].addClass('active')},x.init(),x}var d=angular.element(a),f=angular.element(o.prop('documentElement')),p=angular.element(a.document.body);return c}]}).directive('bsScrollspy',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,a){return{restrict:'EAC',link:function(e,t,n){var o={scope:e};angular.forEach(['offset','target'],function(e){angular.isDefined(n[e])&&(o[e]=n[e])});var i=a(o);i.trackElement(o.target,t),e.$on('$destroy',function(){i&&(i.untrackElement(o.target,t),i.destroy()),o=null,i=null})}}}]).directive('bsScrollspyList',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,a){return{restrict:'A',compile:function(e,t){var n=e[0].querySelectorAll('li > a[href]');angular.forEach(n,function(e){var t=angular.element(e);t.parent().attr('bs-scrollspy','').attr('data-target',t.attr('href'))})}}}]),angular.module('mgcrea.ngStrap.popover',['mgcrea.ngStrap.tooltip']).provider('$popover',function(){var e=this.defaults={animation:'am-fade',customClass:'',container:!1,target:!1,placement:'right',templateUrl:'popover/popover.tpl.html',contentTemplate:!1,trigger:'click',keyboard:!0,html:!1,title:'',content:'',delay:0,autoClose:!1};this.$get=['$tooltip',function(t){function n(n,a){var o=angular.extend({},e,a),i=t(n,o);return o.content&&(i.$scope.content=o.content),i}return n}]}).directive('bsPopover',['$window','$sce','$popover',function(e,t,n){var a=e.requestAnimationFrame||e.setTimeout;return{restrict:'EAC',scope:!0,link:function(e,o,i){var r={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','container','delay','trigger','html','animation','customClass','autoClose','id','prefixClass','prefixEvent'],function(e){angular.isDefined(i[e])&&(r[e]=i[e])});var s=/^(false|0|)$/i;angular.forEach(['html','container','autoClose'],function(e){angular.isDefined(i[e])&&s.test(i[e])&&(r[e]=!1)});var l=o.attr('data-target');angular.isDefined(l)&&(s.test(l)?r.target=!1:r.target=l),angular.forEach(['title','content'],function(n){i[n]&&i.$observe(n,function(o,i){e[n]=t.trustAsHtml(o),angular.isDefined(i)&&a(function(){u&&u.$applyPlacement()})})}),i.bsPopover&&e.$watch(i.bsPopover,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t,angular.isDefined(n)&&a(function(){u&&u.$applyPlacement()})},!0),i.bsShow&&e.$watch(i.bsShow,function(e,t){u&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(popover),?/i)),e===!0?u.show():u.hide())}),i.viewport&&e.$watch(i.viewport,function(e){u&&angular.isDefined(e)&&u.setViewport(e)});var u=n(o,r);e.$on('$destroy',function(){u&&u.destroy(),r=null,u=null})}}}]),angular.module('mgcrea.ngStrap.navbar',[]).provider('$navbar',function(){var e=this.defaults={activeClass:'active',routeAttr:'data-match-route',strict:!1};this.$get=function(){return{defaults:e}}}).directive('bsNavbar',['$window','$location','$navbar',function(e,t,n){var a=n.defaults;return{restrict:'A',link:function(e,n,o,i){var r=angular.copy(a);angular.forEach(Object.keys(a),function(e){angular.isDefined(o[e])&&(r[e]=o[e])}),e.$watch(function(){return t.path()},function(e,t){var a=n[0].querySelectorAll('li['+r.routeAttr+']');angular.forEach(a,function(t){var n=angular.element(t),a=n.attr(r.routeAttr).replace('/','\\/');r.strict&&(a='^'+a+'$');var o=new RegExp(a,'i');o.test(e)?n.addClass(r.activeClass):n.removeClass(r.activeClass)})})}}}]),angular.module('mgcrea.ngStrap.modal',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$modal',function(){var e=this.defaults={animation:'am-fade',backdropAnimation:'am-fade',prefixClass:'modal',prefixEvent:'modal',placement:'top',templateUrl:'modal/modal.tpl.html',template:'',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0};this.$get=['$window','$rootScope','$bsCompiler','$animate','$timeout','$sce','dimensions',function(n,a,o,i,r,s,l){function u(t){function n(){k.$emit(b.prefixEvent+'.show',y)}function r(){k.$emit(b.prefixEvent+'.hide',y),g.removeClass(b.prefixClass+'-open'),b.animation&&g.removeClass(b.prefixClass+'-with-'+b.animation)}function l(){b.backdrop&&(x.on('click',h),C.on('click',h),C.on('wheel',v))}function u(){b.backdrop&&(x.off('click',h),C.off('click',h),C.off('wheel',v))}function m(){b.keyboard&&x.on('keyup',y.$onKeyUp)}function $(){b.keyboard&&x.off('keyup',y.$onKeyUp)}function h(e){e.target===e.currentTarget&&('static'===b.backdrop?y.focus():y.hide())}function v(e){e.preventDefault()}function w(){y.$isShown&&null!==x&&(u(),$()),T&&(T.$destroy(),T=null),x&&(x.remove(),x=y.$element=null)}var y={},b=y.$options=angular.extend({},e,t),D=y.$promise=o.compile(b),k=y.$scope=b.scope&&b.scope.$new()||a.$new();b.element||b.container||(b.container='body'),y.$id=b.id||b.element&&b.element.attr('id')||'',f(['title','content'],function(e){b[e]&&(k[e]=s.trustAsHtml(b[e]))}),k.$hide=function(){k.$$postDigest(function(){y.hide()})},k.$show=function(){k.$$postDigest(function(){y.show()})},k.$toggle=function(){k.$$postDigest(function(){y.toggle()})},y.$isShown=k.$isShown=!1;var S,x,T,C=angular.element('
');return C.css({position:'fixed',top:'0px',left:'0px',bottom:'0px',right:'0px','z-index':1038}),D.then(function(e){S=e,y.init()}),y.init=function(){b.show&&k.$$postDigest(function(){y.show()})},y.destroy=function(){w(),C&&(C.remove(),C=null),k.$destroy()},y.show=function(){if(!y.$isShown){var e,t;if(angular.isElement(b.container)?(e=b.container,t=b.container[0].lastChild?angular.element(b.container[0].lastChild):null):b.container?(e=d(b.container),t=e[0]&&e[0].lastChild?angular.element(e[0].lastChild):null):(e=null,t=b.element),x&&w(),T=y.$scope.$new(),x=y.$element=S.link(T,function(e,t){}),!k.$emit(b.prefixEvent+'.show.before',y).defaultPrevented){x.css({display:'block'}).addClass(b.placement),b.animation&&(b.backdrop&&C.addClass(b.backdropAnimation),x.addClass(b.animation)),b.backdrop&&i.enter(C,g,null),angular.version.minor<=2?i.enter(x,e,t,n):i.enter(x,e,t).then(n),y.$isShown=k.$isShown=!0,c(k);var a=x[0];p(function(){a.focus()}),g.addClass(b.prefixClass+'-open'),b.animation&&g.addClass(b.prefixClass+'-with-'+b.animation),l(),m()}}},y.hide=function(){y.$isShown&&(k.$emit(b.prefixEvent+'.hide.before',y).defaultPrevented||(angular.version.minor<=2?i.leave(x,r):i.leave(x).then(r),b.backdrop&&i.leave(C),y.$isShown=k.$isShown=!1,c(k),u(),$()))},y.toggle=function(){y.$isShown?y.hide():y.show()},y.focus=function(){x[0].focus()},y.$onKeyUp=function(e){27===e.which&&y.$isShown&&(y.hide(),e.stopPropagation())},y}function c(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}function d(e,n){return angular.element((n||t).querySelectorAll(e))}var f=angular.forEach,p=(String.prototype.trim,n.requestAnimationFrame||n.setTimeout),g=angular.element(n.document.body);return u}]}).directive('bsModal',['$window','$sce','$modal',function(e,t,n){return{restrict:'EAC',scope:!0,link:function(e,a,o,i){var r={scope:e,element:a,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation','backdropAnimation','id','prefixEvent','prefixClass'],function(e){angular.isDefined(o[e])&&(r[e]=o[e])});var s=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(e){angular.isDefined(o[e])&&s.test(o[e])&&(r[e]=!1)}),angular.forEach(['title','content'],function(n){o[n]&&o.$observe(n,function(a,o){e[n]=t.trustAsHtml(a)})}),o.bsModal&&e.$watch(o.bsModal,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t},!0);var l=n(r);a.on(o.trigger||'click',l.toggle),e.$on('$destroy',function(){l&&l.destroy(),r=null,l=null})}}}]),angular.version.minor<3&&angular.version.dot<14&&angular.module('ng').factory('$$rAF',['$window','$timeout',function(e,t){var n=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame,a=e.cancelAnimationFrame||e.webkitCancelAnimationFrame||e.mozCancelAnimationFrame||e.webkitCancelRequestAnimationFrame,o=!!n,i=o?function(e){var t=n(e);return function(){a(t)}}:function(e){var n=t(e,16.66,!1);return function(){t.cancel(n)}};return i.supported=o,i}]),angular.module('mgcrea.ngStrap.helpers.parseOptions',[]).provider('$parseOptions',function(){var e=this.defaults={regexp:/^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/};this.$get=['$parse','$q',function(t,n){function a(a,o){function i(e,t){return e.map(function(e,n){var a,o,i={};return i[c]=e,a=u(t,i),o=p(t,i),{label:a,value:o,index:n}})}var r={},s=angular.extend({},e,o);r.$values=[];var l,u,c,d,f,p,g;return r.init=function(){r.$match=l=a.match(s.regexp),u=t(l[2]||l[1]),c=l[4]||l[6],d=l[5],f=t(l[3]||''),p=t(l[2]?l[1]:c),g=t(l[7])},r.valuesFn=function(e,t){return n.when(g(e,t)).then(function(t){return angular.isArray(t)||(t=[]),r.$values=t.length?i(t,e):[],r.$values})},r.displayValue=function(e){var t={};return t[c]=e,u(t)},r.init(),r}return a}]}),angular.module('mgcrea.ngStrap.helpers.dimensions',[]).factory('dimensions',['$document','$window',function(t,n){var a=(angular.element,{}),o=a.nodeName=function(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()};a.css=function(t,n,a){var o;return o=t.currentStyle?t.currentStyle[n]:e.getComputedStyle?e.getComputedStyle(t)[n]:t.style[n],a===!0?parseFloat(o)||0:o},a.offset=function(t){var n=t.getBoundingClientRect(),a=t.ownerDocument;return{width:n.width||t.offsetWidth,height:n.height||t.offsetHeight,top:n.top+(e.pageYOffset||a.documentElement.scrollTop)-(a.documentElement.clientTop||0),left:n.left+(e.pageXOffset||a.documentElement.scrollLeft)-(a.documentElement.clientLeft||0)}},a.setOffset=function(e,t,n){var o,i,r,s,l,u,c,d=a.css(e,'position'),f=angular.element(e),p={};'static'===d&&(e.style.position='relative'),l=a.offset(e),r=a.css(e,'top'),u=a.css(e,'left'),c=('absolute'===d||'fixed'===d)&&(r+u).indexOf('auto')>-1,c?(o=a.position(e),s=o.top,i=o.left):(s=parseFloat(r)||0,i=parseFloat(u)||0),angular.isFunction(t)&&(t=t.call(e,n,l)),null!==t.top&&(p.top=t.top-l.top+s),null!==t.left&&(p.left=t.left-l.left+i),'using'in t?t.using.call(f,p):f.css({top:p.top+'px',left:p.left+'px'})},a.position=function(e){var t,n,r={top:0,left:0};return'fixed'===a.css(e,'position')?n=e.getBoundingClientRect():(t=i(e),n=a.offset(e),o(t,'html')||(r=a.offset(t)),r.top+=a.css(t,'borderTopWidth',!0),r.left+=a.css(t,'borderLeftWidth',!0)),{width:e.offsetWidth,height:e.offsetHeight,top:n.top-r.top-a.css(e,'marginTop',!0),left:n.left-r.left-a.css(e,'marginLeft',!0)}};var i=function(e){var t=e.ownerDocument,n=e.offsetParent||t;if(o(n,'#document'))return t.documentElement;for(;n&&!o(n,'html')&&'static'===a.css(n,'position');)n=n.offsetParent;return n||t.documentElement};return a.height=function(e,t){var n=e.offsetHeight;return t?n+=a.css(e,'marginTop',!0)+a.css(e,'marginBottom',!0):n-=a.css(e,'paddingTop',!0)+a.css(e,'paddingBottom',!0)+a.css(e,'borderTopWidth',!0)+a.css(e,'borderBottomWidth',!0),n},a.width=function(e,t){var n=e.offsetWidth;return t?n+=a.css(e,'marginLeft',!0)+a.css(e,'marginRight',!0):n-=a.css(e,'paddingLeft',!0)+a.css(e,'paddingRight',!0)+a.css(e,'borderLeftWidth',!0)+a.css(e,'borderRightWidth',!0),n},a}]),angular.module('mgcrea.ngStrap.helpers.debounce',[]).factory('debounce',['$timeout',function(e){return function(t,n,a){var o=null;return function(){var i=this,r=arguments,s=a&&!o;return o&&e.cancel(o),o=e(function(){o=null,a||t.apply(i,r)},n,!1),s&&t.apply(i,r),o}}}]).factory('throttle',['$timeout',function(e){return function(t,n,a){var o=null;return a||(a={}),function(){var i=this,r=arguments;o||(a.leading!==!1&&t.apply(i,r),o=e(function(){o=null,a.trailing!==!1&&t.apply(i,r)},n,!1))}}}]),angular.module('mgcrea.ngStrap.helpers.dateParser',[]).provider('$dateParser',['$localeProvider',function(e){function t(){this.year=1970,this.month=0,this.day=1,this.hours=0,this.minutes=0,this.seconds=0,this.milliseconds=0}function n(){}function a(e){return!isNaN(parseFloat(e))&&isFinite(e)}function o(e,t){for(var n=e.length,a=t.toString().toLowerCase(),o=0;n>o;o++)if(e[o].toLowerCase()===a)return o;return-1}t.prototype.setMilliseconds=function(e){this.milliseconds=e},t.prototype.setSeconds=function(e){this.seconds=e},t.prototype.setMinutes=function(e){this.minutes=e},t.prototype.setHours=function(e){this.hours=e},t.prototype.getHours=function(){return this.hours},t.prototype.setDate=function(e){this.day=e},t.prototype.setMonth=function(e){this.month=e},t.prototype.setFullYear=function(e){this.year=e},t.prototype.fromDate=function(e){return this.year=e.getFullYear(),this.month=e.getMonth(),this.day=e.getDate(),this.hours=e.getHours(),this.minutes=e.getMinutes(),this.seconds=e.getSeconds(),this.milliseconds=e.getMilliseconds(),this},t.prototype.toDate=function(){return new Date(this.year,this.month,this.day,this.hours,this.minutes,this.seconds,this.milliseconds)};var i=t.prototype,r=this.defaults={format:'shortDate',strict:!1};this.$get=['$locale','dateFilter',function(e,s){var l=function(l){function u(e){var t,n=Object.keys(h),a=[],o=[],i=e;for(t=0;t1){var r=i.search(n[t]);e=e.split(n[t]).join(''),h[n[t]]&&(a[r]=h[n[t]])}return angular.forEach(a,function(e){e&&o.push(e)}),o}function c(e){return e.replace(/\//g,'[\\/]').replace('/-/g','[-]').replace(/\./g,'[.]').replace(/\\s/g,'[\\s]')}function d(e){var t,n=Object.keys($),a=e;for(t=0;t=1*e&&2===e.length?this.setFullYear(2e3+1*e):this.setFullYear(1*e)}};return m.init=function(){m.$format=e.DATETIME_FORMATS[g.format]||g.format,f=d(m.$format),p=u(m.$format)},m.isValid=function(e){return angular.isDate(e)?!isNaN(e.getTime()):f.test(e)},m.parse=function(n,a,o,i){o&&(o=e.DATETIME_FORMATS[o]||o),angular.isDate(n)&&(n=s(n,o||m.$format,i));var r=o?d(o):f,l=o?u(o):p,c=r.exec(n);if(!c)return!1;for(var g=a&&!isNaN(a.getTime())?(new t).fromDate(a):(new t).fromDate(new Date(1970,0,1,0)),$=0;$12?e.getHours()+2:0),e):null},m.timezoneOffsetAdjust=function(e,t,n){return e?(t&&'UTC'===t&&(e=new Date(e.getTime()),e.setMinutes(e.getMinutes()+(n?-1:1)*e.getTimezoneOffset())),e):null},m.init(),m};return l}]}]),angular.module('mgcrea.ngStrap.helpers.dateFormatter',[]).service('$dateFormatter',['$locale','dateFilter',function(e,t){function n(e){return/(h+)([:\.])?(m+)([:\.])?(s*)[ ]?(a?)/i.exec(e).slice(1)}this.getDefaultLocale=function(){return e.id},this.getDatetimeFormat=function(t,n){return e.DATETIME_FORMATS[t]||t},this.weekdaysShort=function(t){return e.DATETIME_FORMATS.SHORTDAY},this.hoursFormat=function(e){return n(e)[0]},this.minutesFormat=function(e){return n(e)[2]},this.secondsFormat=function(e){return n(e)[4]},this.timeSeparator=function(e){return n(e)[1]},this.showSeconds=function(e){return!!n(e)[4]},this.showAM=function(e){return!!n(e)[5]},this.formatDate=function(e,n,a,o){return t(e,n,o)}}]),angular.module('mgcrea.ngStrap.core',[]).service('$bsCompiler',a),a.$inject=['$q','$http','$injector','$compile','$controller','$templateCache'],angular.module('mgcrea.ngStrap.dropdown',['mgcrea.ngStrap.tooltip']).provider('$dropdown',function(){var e=this.defaults={animation:'am-fade',prefixClass:'dropdown',prefixEvent:'dropdown',placement:'bottom-left',templateUrl:'dropdown/dropdown.tpl.html',trigger:'click',container:!1,keyboard:!0,html:!1,delay:0};this.$get=['$window','$rootScope','$tooltip','$timeout',function(t,n,a,o){function i(t,i){function l(e){return e.target!==t[0]?e.target!==t[0]&&u.hide():void 0}var u={},c=angular.extend({},e,i);u.$scope=c.scope&&c.scope.$new()||n.$new();u=a(t,c);var d=t.parent();u.$onKeyDown=function(e){if(/(38|40)/.test(e.keyCode)){e.preventDefault(),e.stopPropagation();var t=angular.element(u.$element[0].querySelectorAll('li:not(.divider) a'));if(t.length){var n;angular.forEach(t,function(e,t){s&&s.call(e,':focus')&&(n=t)}),38===e.keyCode&&n>0?n--:40===e.keyCode&&nt;t++)angular.forEach(g.rows[t],u.$setDisabledEl)},u.select=function(e,t){angular.isDate(n.$dateValue)||(n.$dateValue=new Date(e)),!g.$mode||t?(n.$setViewValue(angular.copy(e)),n.$render(),p.autoclose&&!t&&l(function(){u.hide(!0)})):(angular.extend($,{year:e.getFullYear(),month:e.getMonth(),date:e.getDate()}),u.setMode(g.$mode-1),u.$build())},u.setMode=function(e){g.$mode=e,h=u.$views[g.$mode],u.$build()},u.$build=function(e){e===!0&&h.built||(e!==!1||h.built)&&h.build.call(h)},u.$updateSelected=function(){for(var e=0,t=g.rows.length;t>e;e++)angular.forEach(g.rows[e],o)},u.$isSelected=function(e){return h.isSelected(e)},u.$setDisabledEl=function(e){e.disabled=h.isDisabled(e.date)},u.$selectPane=function(e){var t=h.steps,n=new Date(Date.UTC($.year+(t.year||0)*e,$.month+(t.month||0)*e,1));angular.extend($,{year:n.getUTCFullYear(),month:n.getUTCMonth(),date:n.getUTCDate()}),u.$build()},u.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),d){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},u.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void(g.$mode?g.$apply(function(){u.setMode(g.$mode-1)}):u.hide(!0));h.onKeyDown(e),f.$digest()}};var v=u.init;u.init=function(){return c&&p.useNative?(t.prop('type','date'),void t.css('-webkit-appearance','textfield')):(d&&(t.prop('type','text'),t.attr('readonly','true'),t.on('click',i)),void v())};var w=u.destroy;u.destroy=function(){c&&p.useNative&&t.off('click',i),w()};var y=u.show;u.show=function(){!d&&t.attr('readonly')||t.attr('disabled')||(y(),l(function(){u.$isShown&&(u.$element.on(d?'touchstart':'mousedown',u.$onMouseDown),p.keyboard&&t.on('keydown',u.$onKeyDown))},0,!1))};var b=u.hide;return u.hide=function(e){u.$isShown&&(u.$element.off(d?'touchstart':'mousedown',u.$onMouseDown),p.keyboard&&t.off('keydown',u.$onKeyDown),b(e))},u}var c=/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent),d='createTouch'in t.document&&c;return e.lang||(e.lang=i.getDefaultLocale()),u.defaults=e,u}]}).directive('bsDatepicker',['$window','$parse','$q','$dateFormatter','$dateParser','$datepicker',function(e,t,n,a,o,i){var r=(i.defaults,/(ip(a|o)d|iphone|android)/gi.test(e.navigator.userAgent));return{restrict:'EAC',require:'ngModel',link:function(e,t,n,s){function l(e){return e&&e.length?e:null}function u(e){if(angular.isDate(e)){var t=isNaN(p.$options.minDate)||e.getTime()>=p.$options.minDate,n=isNaN(p.$options.maxDate)||e.getTime()<=p.$options.maxDate,a=t&&n;s.$setValidity('date',a),s.$setValidity('min',t),s.$setValidity('max',n),a&&(s.$dateValue=e)}}function c(){return!s.$dateValue||isNaN(s.$dateValue.getTime())?'':m(s.$dateValue,d.dateFormat)}var d={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','html','animation','autoclose','dateType','dateFormat','timezone','modelDateFormat','dayFormat','strictFormat','startWeek','startDate','useNative','lang','startView','minView','iconLeft','iconRight','daysOfWeekDisabled','id','prefixClass','prefixEvent'],function(e){angular.isDefined(n[e])&&(d[e]=n[e])});var f=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative'],function(e){angular.isDefined(n[e])&&f.test(n[e])&&(d[e]=!1)});var p=i(t,s,d);d=p.$options,r&&d.useNative&&(d.dateFormat='yyyy-MM-dd');var g=d.lang,m=function(e,t){return a.formatDate(e,t,g)},$=o({format:d.dateFormat,lang:g,strict:d.strictFormat});n.bsShow&&e.$watch(n.bsShow,function(e,t){p&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(datepicker),?/i)),e===!0?p.show():p.hide())}),angular.forEach(['minDate','maxDate'],function(e){angular.isDefined(n[e])&&n.$observe(e,function(t){p.$options[e]=$.getDateForAttribute(e,t),!isNaN(p.$options[e])&&p.$build(!1),u(s.$dateValue)})}),angular.isDefined(n.dateFormat)&&n.$observe('dateFormat',function(e){p.$options.dateFormat=e}),e.$watch(n.ngModel,function(e,t){p.update(s.$dateValue)},!0),angular.isDefined(n.disabledDates)&&e.$watch(n.disabledDates,function(e,t){e=l(e),t=l(t),e&&p.updateDisabledDates(e)}),s.$parsers.unshift(function(e){var t;if(!e)return s.$setValidity('date',!0),null;var n=$.parse(e,s.$dateValue);return!n||isNaN(n.getTime())?void s.$setValidity('date',!1):(u(n),'string'===d.dateType?(t=$.timezoneOffsetAdjust(n,d.timezone,!0),m(t,d.modelDateFormat||d.dateFormat)):(t=$.timezoneOffsetAdjust(s.$dateValue,d.timezone,!0),'number'===d.dateType?t.getTime():'unix'===d.dateType?t.getTime()/1e3:'iso'===d.dateType?t.toISOString():new Date(t)))}),s.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===d.dateType?$.parse(e,null,d.modelDateFormat):'unix'===d.dateType?new Date(1e3*e):new Date(e),s.$dateValue=$.timezoneOffsetAdjust(t,d.timezone),c()}),s.$render=function(){t.val(c())},e.$on('$destroy',function(){p&&p.destroy(),d=null,p=null})}}}]).provider('datepickerViews',function(){function e(e,t){for(var n=[];e.length>0;)n.push(e.splice(0,t));return n}function t(e,t){return(e%t+t)%t}this.defaults={dayFormat:'dd',daySplit:7};this.$get=['$dateFormatter','$dateParser','$sce',function(n,a,o){return function(i){var r=i.$scope,s=i.$options,l=s.lang,u=function(e,t){return n.formatDate(e,t,l)},c=a({format:s.dateFormat,lang:l,strict:s.strictFormat}),d=n.weekdaysShort(l),f=d.slice(s.startWeek).concat(d.slice(0,s.startWeek)),p=o.trustAsHtml(''+f.join('')+''),g=i.$date||(s.startDate?c.getDateForAttribute('startDate',s.startDate):new Date),m={year:g.getFullYear(),month:g.getMonth(),date:g.getDate()},$=[{format:s.dayFormat,split:7,steps:{month:1},update:function(e,t){!this.built||t||e.getFullYear()!==m.year||e.getMonth()!==m.month?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):(e.getDate()!==m.date||1===e.getDate())&&(m.date=i.$date.getDate(),i.$updateSelected())},build:function(){var n=new Date(m.year,m.month,1),a=n.getTimezoneOffset(),o=new Date(+n-864e5*t(n.getDay()-s.startWeek,7)),l=o.getTimezoneOffset(),d=c.timezoneOffsetAdjust(new Date,s.timezone).toDateString();l!==a&&(o=new Date(+o+6e4*(l-a)));for(var f,g=[],$=0;42>$;$++)f=c.daylightSavingAdjust(new Date(o.getFullYear(),o.getMonth(),o.getDate()+$)),g.push({date:f,isToday:f.toDateString()===d,label:u(f,this.format),selected:i.$date&&this.isSelected(f),muted:f.getMonth()!==m.month,disabled:this.isDisabled(f)});r.title=u(n,s.monthTitleFormat),r.showLabels=!0,r.labels=p,r.rows=e(g,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()&&e.getDate()===i.$date.getDate()},isDisabled:function(e){var t=e.getTime();if(ts.maxDate)return!0;if(-1!==s.daysOfWeekDisabled.indexOf(e.getDay()))return!0;if(s.disabledDateRanges)for(var n=0;n=s.disabledDateRanges[n].start&&t<=s.disabledDateRanges[n].end)return!0;return!1},onKeyDown:function(e){if(i.$date){var t,n=i.$date.getTime();37===e.keyCode?t=new Date(n-864e5):38===e.keyCode?t=new Date(n-6048e5):39===e.keyCode?t=new Date(n+864e5):40===e.keyCode&&(t=new Date(n+6048e5)),this.isDisabled(t)||i.select(t,!0)}}},{name:'month',format:s.monthFormat,split:4,steps:{year:1},update:function(e,t){this.built&&e.getFullYear()===m.year?e.getMonth()!==m.month&&(angular.extend(m,{ -month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected()):(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build())},build:function(){for(var t,n=(new Date(m.year,0,1),[]),a=0;12>a;a++)t=new Date(m.year,a,1),n.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=u(t,s.yearTitleFormat),r.showLabels=!1,r.rows=e(n,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()},isDisabled:function(e){var t=+new Date(e.getFullYear(),e.getMonth()+1,0);return ts.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getMonth(),n=new Date(i.$date);37===e.keyCode?n.setMonth(t-1):38===e.keyCode?n.setMonth(t-4):39===e.keyCode?n.setMonth(t+1):40===e.keyCode&&n.setMonth(t+4),this.isDisabled(n)||i.select(n,!0)}}},{name:'year',format:s.yearFormat,split:4,steps:{year:12},update:function(e,t){!this.built||t||parseInt(e.getFullYear()/20,10)!==parseInt(m.year/20,10)?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):e.getFullYear()!==m.year&&(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected())},build:function(){for(var t,n=m.year-m.year%(3*this.split),a=[],o=0;12>o;o++)t=new Date(n+o,0,1),a.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=a[0].label+'-'+a[a.length-1].label,r.showLabels=!1,r.rows=e(a,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()},isDisabled:function(e){var t=+new Date(e.getFullYear()+1,0,0);return ts.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getFullYear(),n=new Date(i.$date);37===e.keyCode?n.setYear(t-1):38===e.keyCode?n.setYear(t-4):39===e.keyCode?n.setYear(t+1):40===e.keyCode&&n.setYear(t+4),this.isDisabled(n)||i.select(n,!0)}}}];return{views:s.minView?Array.prototype.slice.call($,s.minView):$,viewDate:m}}}]}),angular.module('mgcrea.ngStrap.collapse',[]).provider('$collapse',function(){var e=this.defaults={animation:'am-collapse',disallowToggle:!1,activeClass:'in',startCollapsed:!1,allowMultiple:!1},t=this.controller=function(t,n,a){function o(e){for(var t=l.$targets.$active,n=0;n=a?'top':null!==e&&a+e<=t.top?'middle':null!==w&&t.top+n+$>=o-w?'bottom':'middle'}function u(){return p[0]===t?t.pageYOffset:p[0].scrollTop}function c(){return p[0]===t?t.document.body.scrollHeight:p[0].scrollHeight}var d={},f=angular.extend({},e,s),p=f.target,g='affix affix-top affix-bottom',m=!1,$=0,h=0,v=0,w=0,y=null,b=null,D=o.parent();if(f.offsetParent)if(f.offsetParent.match(/^\d+$/))for(var k=0;k<1*f.offsetParent-1;k++)D=D.parent();else D=angular.element(f.offsetParent);return d.init=function(){this.$parseOffsets(),h=a.offset(o[0]).top+$,m=!o[0].style.width,p.on('scroll',this.checkPosition),p.on('click',this.checkPositionWithEventLoop),r.on('resize',this.$debouncedOnResize),this.checkPosition(),this.checkPositionWithEventLoop()},d.destroy=function(){p.off('scroll',this.checkPosition),p.off('click',this.checkPositionWithEventLoop),r.off('resize',this.$debouncedOnResize)},d.checkPositionWithEventLoop=function(){setTimeout(d.checkPosition,1)},d.checkPosition=function(){var e=u(),t=a.offset(o[0]),n=a.height(o[0]),r=l(b,t,n);y!==r&&(y=r,'top'===r?(b=null,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',''))):'bottom'===r?(b=f.offsetUnpin?-(1*f.offsetUnpin):t.top-e,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',f.offsetParent?'':i[0].offsetHeight-w-n-h+'px'))):(b=null,m&&o.css('width',o[0].offsetWidth+'px'),f.inlineStyles&&(o.css('position','fixed'),o.css('top',$+'px'))),o.removeClass(g).addClass('affix'+('middle'!==r?'-'+r:'')))},d.$onResize=function(){d.$parseOffsets(),d.checkPosition()},d.$debouncedOnResize=n(d.$onResize,50),d.$parseOffsets=function(){var e=o.css('position');f.inlineStyles&&o.css('position',f.offsetParent?'':'relative'),f.offsetTop&&('auto'===f.offsetTop&&(f.offsetTop='+0'),f.offsetTop.match(/^[-+]\d+$/)?($=1*-f.offsetTop,v=f.offsetParent?a.offset(D[0]).top+1*f.offsetTop:a.offset(o[0]).top-a.css(o[0],'marginTop',!0)+1*f.offsetTop):v=1*f.offsetTop),f.offsetBottom&&(w=f.offsetParent&&f.offsetBottom.match(/^[-+]\d+$/)?c()-(a.offset(D[0]).top+a.height(D[0]))+1*f.offsetBottom+1:1*f.offsetBottom),f.inlineStyles&&o.css('position',e)},d.init(),d}var i=angular.element(t.document.body),r=angular.element(t);return o}]}).directive('bsAffix',['$affix','$window',function(e,t){return{restrict:'EAC',require:'^?bsAffixTarget',link:function(n,a,o,i){var r={scope:n,target:i?i.$element:angular.element(t)};angular.forEach(['offsetTop','offsetBottom','offsetParent','offsetUnpin','inlineStyles'],function(e){if(angular.isDefined(o[e])){var t=o[e];/true/i.test(t)&&(t=!0),/false/i.test(t)&&(t=!1),r[e]=t}});var s=e(a,r);n.$on('$destroy',function(){s&&s.destroy(),r=null,s=null})}}}]).directive('bsAffixTarget',function(){return{controller:['$element',function(e){this.$element=e}]}}),angular.module('mgcrea.ngStrap',['mgcrea.ngStrap.modal','mgcrea.ngStrap.aside','mgcrea.ngStrap.alert','mgcrea.ngStrap.button','mgcrea.ngStrap.select','mgcrea.ngStrap.datepicker','mgcrea.ngStrap.timepicker','mgcrea.ngStrap.navbar','mgcrea.ngStrap.tooltip','mgcrea.ngStrap.popover','mgcrea.ngStrap.dropdown','mgcrea.ngStrap.typeahead','mgcrea.ngStrap.scrollspy','mgcrea.ngStrap.affix','mgcrea.ngStrap.tab','mgcrea.ngStrap.collapse'])}(window,document); +!function(e,t,n){'use strict';function a(e,n,a,o,i,r){function l(e,n){return angular.element((n||t).querySelectorAll(e))}function s(e){return u[e]?u[e]:u[e]=n.get(e,{cache:r}).then(function(e){return e.data})}this.compile=function(t){t.template&&/\.html$/.test(t.template)&&(console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.'),t.templateUrl=t.template,t.template='');var n=t.templateUrl,r=t.template||'',u=t.controller,c=t.controllerAs,d=t.resolve||{},f=t.locals||{},p=t.transformTemplate||angular.identity,g=t.bindToController;if(angular.forEach(d,function(e,t){angular.isString(e)?d[t]=a.get(e):d[t]=a.invoke(e)}),angular.extend(d,f),r)d.$template=e.when(r);else{if(!n)throw new Error('Missing `template` / `templateUrl` option.');d.$template=s(n)}return t.titleTemplate&&(d.$template=e.all([d.$template,s(t.titleTemplate)]).then(function(e){var t=angular.element(e[0]);return l('[ng-bind="title"]',t[0]).removeAttr('ng-bind').html(e[1]),t[0].outerHTML})),t.contentTemplate&&(d.$template=e.all([d.$template,s(t.contentTemplate)]).then(function(e){var n=angular.element(e[0]),a=l('[ng-bind="content"]',n[0]).removeAttr('ng-bind').html(e[1]);return t.templateUrl||a.next().remove(),n[0].outerHTML})),e.all(d).then(function(e){var n=p(e.$template);t.html&&(n=n.replace(/ng-bind="/gi,'ng-bind-html="'));var a=angular.element('
').html(n.trim()).contents(),r=o(a);return{locals:e,element:a,link:function(t){if(e.$scope=t,u){var n=i(u,e,!0);g&&angular.extend(n.instance,e);var o=angular.isObject(n)?n:n();a.data('$ngControllerController',o),a.children().data('$ngControllerController',o),c&&(t[c]=o)}return r.apply(null,arguments)}}})};var u={}}a.$inject=['$q','$http','$injector','$compile','$controller','$templateCache'],angular.module('mgcrea.ngStrap.typeahead',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$typeahead',function(){var e=this.defaults={animation:'am-fade',prefixClass:'typeahead',prefixEvent:'$typeahead',placement:'bottom-left',templateUrl:'typeahead/typeahead.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,minLength:1,filter:'bsAsyncFilter',limit:6,autoSelect:!1,comparator:'',trimValue:!0};this.$get=['$window','$rootScope','$tooltip','$$rAF','$timeout',function(t,n,a,o,i){function r(t,n,r){var s={},u=angular.extend({},e,r);s=a(t,u);var c=r.scope,d=s.$scope;d.$resetMatches=function(){d.$matches=[],d.$activeIndex=u.autoSelect?0:-1},d.$resetMatches(),d.$activate=function(e){d.$$postDigest(function(){s.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){s.select(e)})},d.$isVisible=function(){return s.$isVisible()},s.update=function(e){d.$matches=e,d.$activeIndex>=e.length&&(d.$activeIndex=u.autoSelect?0:-1),l(d),o(s.$applyPlacement)},s.activate=function(e){d.$activeIndex=e},s.select=function(e){if(e!==-1){var t=d.$matches[e].value;n.$setViewValue(t),n.$render(),d.$resetMatches(),c&&c.$digest(),d.$emit(u.prefixEvent+'.select',t,e,s),angular.isDefined(u.onSelect)&&angular.isFunction(u.onSelect)&&u.onSelect(t,e,s)}},s.$isVisible=function(){return u.minLength&&n?d.$matches.length&&angular.isString(n.$viewValue)&&n.$viewValue.length>=u.minLength:!!d.$matches.length},s.$getIndex=function(e){var t;for(t=d.$matches.length;t--&&!angular.equals(d.$matches[t].value,e););return t},s.$onMouseDown=function(e){e.preventDefault(),e.stopPropagation()},s.$$updateScrollTop=function(e,t){if(t>-1&&t=i&&ar&&(e.scrollTop=a)}},s.$onKeyDown=function(e){/(38|40|13)/.test(e.keyCode)&&(!s.$isVisible()||13===e.keyCode&&d.$activeIndex===-1||(e.preventDefault(),e.stopPropagation()),13===e.keyCode&&d.$matches.length?s.select(d.$activeIndex):38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:40===e.keyCode&&d.$activeIndex0?void r.$setViewValue(r.$viewValue.substring(0,r.$viewValue.length-1)):(e.length>c&&(e=e.slice(0,c)),g.update(e),void r.$render())})}),r.$formatters.push(function(e){var t=p.displayValue(e);return t?t:angular.isDefined(e)&&'object'!=typeof e?e:''}),r.$render=function(){if(r.$isEmpty(r.$viewValue))return t.val('');var e=g.$getIndex(r.$modelValue),n=e!==-1?g.$scope.$matches[e].label:r.$viewValue;n=angular.isObject(n)?p.displayValue(n):n;var a=n?n.toString().replace(/<(?:.|\n)*?>/gm,''):'',o=t[0].selectionStart,i=t[0].selectionEnd;t.val(l.trimValue===!1?a:a.trim()),t[0].setSelectionRange(o,i)},e.$on('$destroy',function(){g&&g.destroy(),l=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tooltip',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$tooltip',function(){var e=this.defaults={animation:'am-fade',customClass:'',prefixClass:'tooltip',prefixEvent:'tooltip',container:!1,target:!1,placement:'top',templateUrl:'tooltip/tooltip.tpl.html',template:'',titleTemplate:!1,trigger:'hover focus',keyboard:!1,html:!1,show:!1,title:'',type:'',delay:0,autoClose:!1,bsEnabled:!0,mouseDownPreventDefault:!0,mouseDownStopPropagation:!0,viewport:{selector:'body',padding:0}};this.$get=['$window','$rootScope','$bsCompiler','$q','$templateCache','$http','$animate','$sce','dimensions','$$rAF','$timeout',function(n,a,o,i,r,l,s,u,c,d,f){function p(i,r){function l(){I.$emit(V.prefixEvent+'.show',F),angular.isDefined(V.onShow)&&angular.isFunction(V.onShow)&&V.onShow(F)}function p(){if(I.$emit(V.prefixEvent+'.hide',F),angular.isDefined(V.onHide)&&angular.isFunction(V.onHide)&&V.onHide(F),B===q){if(Y&&'focus'===V.trigger)return i[0].blur();M()}}function $(){var e=V.trigger.split(' ');angular.forEach(e,function(e){'click'===e||'contextmenu'===e?i.on(e,F.toggle):'manual'!==e&&(i.on('hover'===e?'mouseenter':'focus',F.enter),i.on('hover'===e?'mouseleave':'blur',F.leave),'button'===O&&'hover'!==e&&i.on(h?'touchstart':'mousedown',F.$onFocusElementMouseDown))})}function w(){for(var e=V.trigger.split(' '),t=e.length;t--;){var n=e[t];'click'===n||'contextmenu'===n?i.off(n,F.toggle):'manual'!==n&&(i.off('hover'===n?'mouseenter':'focus',F.enter),i.off('hover'===n?'mouseleave':'blur',F.leave),'button'===O&&'hover'!==n&&i.off(h?'touchstart':'mousedown',F.$onFocusElementMouseDown))}}function y(){'focus'!==V.trigger?B.on('keyup',F.$onKeyUp):i.on('keyup',F.$onFocusKeyUp)}function b(){'focus'!==V.trigger?B.off('keyup',F.$onKeyUp):i.off('keyup',F.$onFocusKeyUp)}function D(){f(function(){B.on('click',x),v.on('click',F.hide),j=!0},0,!1)}function S(){j&&(B.off('click',x),v.off('click',F.hide),j=!1)}function x(e){e.stopPropagation()}function k(e){e=e||V.target||i;var a=e[0],o='BODY'===a.tagName,r=a.getBoundingClientRect(),l={};for(var s in r)l[s]=r[s];null===l.width&&(l=angular.extend({},l,{width:r.right-r.left,height:r.bottom-r.top}));var u=o?{top:0,left:0}:c.offset(a),d={scroll:o?t.documentElement.scrollTop||t.body.scrollTop:e.prop('scrollTop')||0},f=o?{width:t.documentElement.clientWidth,height:n.innerHeight}:null;return angular.extend({},l,d,f,u)}function C(e,t,n,a){var o,i=e.split('-');switch(i[0]){case'right':o={top:t.top+t.height/2-a/2,left:t.left+t.width};break;case'bottom':o={top:t.top+t.height,left:t.left+t.width/2-n/2};break;case'left':o={top:t.top+t.height/2-a/2,left:t.left-n};break;default:o={top:t.top-a,left:t.left+t.width/2-n/2}}if(!i[1])return o;if('top'===i[0]||'bottom'===i[0])switch(i[1]){case'left':o.left=t.left;break;case'right':o.left=t.left+t.width-n}else if('left'===i[0]||'right'===i[0])switch(i[1]){case'top':o.top=t.top-a+t.height;break;case'bottom':o.top=t.top}return o}function T(e,t){var n=B[0],a=n.offsetWidth,o=n.offsetHeight,i=parseInt(c.css(n,'margin-top'),10),r=parseInt(c.css(n,'margin-left'),10);isNaN(i)&&(i=0),isNaN(r)&&(r=0),e.top=e.top+i,e.left=e.left+r,c.setOffset(n,angular.extend({using:function(e){B.css({top:Math.round(e.top)+'px',left:Math.round(e.left)+'px',right:''})}},e),0);var l=n.offsetWidth,s=n.offsetHeight;if('top'===t&&s!==o&&(e.top=e.top+o-s),!/top-left|top-right|bottom-left|bottom-right/.test(t)){var u=E(t,e,l,s);if(u.left?e.left+=u.left:e.top+=u.top,c.setOffset(n,e),/top|right|bottom|left/.test(t)){var d=/top|bottom/.test(t),f=d?2*u.left-a+l:2*u.top-o+s,p=d?'offsetWidth':'offsetHeight';A(f,n[p],d)}}}function E(e,t,n,a){var o={top:0,left:0};if(!F.$viewport)return o;var i=V.viewport&&V.viewport.padding||0,r=k(F.$viewport);if(/right|left/.test(e)){var l=t.top-i-r.scroll,s=t.top+i-r.scroll+a;lr.top+r.height&&(o.top=r.top+r.height-s)}else{var u=t.left-i,c=t.left+i+n;ur.right&&(o.left=r.left+r.width-c)}return o}function A(e,t,n){var a=m('.tooltip-arrow, .arrow',B[0]);a.css(n?'left':'top',50*(1-e/t)+'%').css(n?'top':'left','')}function M(){clearTimeout(P),F.$isShown&&null!==B&&(V.autoClose&&S(),V.keyboard&&b()),z&&(z.$destroy(),z=null),B&&(B.remove(),B=F.$element=null)}var F={},V=F.$options=angular.extend({},e,r),H=F.$promise=o.compile(V),I=F.$scope=V.scope&&V.scope.$new()||a.$new(),O=i[0].nodeName.toLowerCase();if(V.delay&&angular.isString(V.delay)){var N=V.delay.split(',').map(parseFloat);V.delay=N.length>1?{show:N[0],hide:N[1]}:N[0]}F.$id=V.id||i.attr('id')||'',V.title&&(I.title=u.trustAsHtml(V.title)),I.$setEnabled=function(e){I.$$postDigest(function(){F.setEnabled(e)})},I.$hide=function(){I.$$postDigest(function(){F.hide()})},I.$show=function(){I.$$postDigest(function(){F.show()})},I.$toggle=function(){I.$$postDigest(function(){F.toggle()})},F.$isShown=I.$isShown=!1;var P,U,L,B,R,z;H.then(function(e){L=e,F.init()}),F.init=function(){V.delay&&angular.isNumber(V.delay)&&(V.delay={show:V.delay,hide:V.delay}),'self'===V.container?R=i:angular.isElement(V.container)?R=V.container:V.container&&(R=m(V.container)),$(),V.target&&(V.target=angular.isElement(V.target)?V.target:m(V.target)),V.show&&I.$$postDigest(function(){'focus'===V.trigger?i[0].focus():F.show()})},F.destroy=function(){w(),M(),I.$destroy()},F.enter=function(){return clearTimeout(P),U='in',V.delay&&V.delay.show?void(P=setTimeout(function(){'in'===U&&F.show()},V.delay.show)):F.show()},F.show=function(){if(V.bsEnabled&&!F.$isShown){I.$emit(V.prefixEvent+'.show.before',F),angular.isDefined(V.onBeforeShow)&&angular.isFunction(V.onBeforeShow)&&V.onBeforeShow(F);var e,t;V.container?(e=R,t=R[0].lastChild?angular.element(R[0].lastChild):null):(e=null,t=i),B&&M(),z=F.$scope.$new(),B=F.$element=L.link(z,function(e,t){}),B.css({top:'-9999px',left:'-9999px',right:'auto',display:'block',visibility:'hidden'}),V.animation&&B.addClass(V.animation),V.type&&B.addClass(V.prefixClass+'-'+V.type),V.customClass&&B.addClass(V.customClass),t?t.after(B):e.prepend(B),F.$isShown=I.$isShown=!0,g(I),F.$applyPlacement(),angular.version.minor<=2?s.enter(B,e,t,l):s.enter(B,e,t).then(l),g(I),d(function(){B&&B.css({visibility:'visible'}),V.keyboard&&('focus'!==V.trigger&&F.focus(),y())}),V.autoClose&&D()}},F.leave=function(){return clearTimeout(P),U='out',V.delay&&V.delay.hide?void(P=setTimeout(function(){'out'===U&&F.hide()},V.delay.hide)):F.hide()};var Y,q;F.hide=function(e){F.$isShown&&(I.$emit(V.prefixEvent+'.hide.before',F),angular.isDefined(V.onBeforeHide)&&angular.isFunction(V.onBeforeHide)&&V.onBeforeHide(F),Y=e,q=B,null!==B&&(angular.version.minor<=2?s.leave(B,p):s.leave(B).then(p)),F.$isShown=I.$isShown=!1,g(I),V.keyboard&&null!==B&&b(),V.autoClose&&null!==B&&S())},F.toggle=function(e){e&&e.preventDefault(),F.$isShown?F.leave():F.enter()},F.focus=function(){B[0].focus()},F.setEnabled=function(e){V.bsEnabled=e},F.setViewport=function(e){V.viewport=e},F.$applyPlacement=function(){if(B){var t=V.placement,n=/\s?auto?\s?/i,a=n.test(t);a&&(t=t.replace(n,'')||e.placement),B.addClass(V.placement);var o=k(),i=B.prop('offsetWidth'),r=B.prop('offsetHeight');if(F.$viewport=V.viewport&&m(V.viewport.selector||V.viewport),a){var l=t,s=k(F.$viewport);/bottom/.test(l)&&o.bottom+r>s.bottom?t=l.replace('bottom','top'):/top/.test(l)&&o.top-rs.width&&(t=t.replace('right','left')),B.removeClass(l).addClass(t)}var u=C(t,o,i,r);T(u,t)}},F.$onKeyUp=function(e){27===e.which&&F.$isShown&&(F.hide(),e.stopPropagation())},F.$onFocusKeyUp=function(e){27===e.which&&(i[0].blur(),e.stopPropagation())},F.$onFocusElementMouseDown=function(e){V.mouseDownPreventDefault&&e.preventDefault(),V.mouseDownStopPropagation&&e.stopPropagation(),F.$isShown?i[0].blur():i[0].focus()};var j=!1;return F}function g(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}function m(e,n){return angular.element((n||t).querySelectorAll(e))}var $=/(ip[ao]d|iphone|android)/gi.test(n.navigator.userAgent),h='createTouch'in n.document&&$,v=angular.element(n.document);return p}]}).directive('bsTooltip',['$window','$location','$sce','$parse','$tooltip','$$rAF',function(e,t,n,a,o,i){return{restrict:'EAC',scope:!0,link:function(e,t,a,r){var l,s={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','titleTemplate','placement','container','delay','trigger','html','animation','backdropAnimation','type','customClass','id'],function(e){angular.isDefined(a[e])&&(s[e]=a[e])});var u=/^(false|0|)$/i;angular.forEach(['html','container'],function(e){angular.isDefined(a[e])&&u.test(a[e])&&(s[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(a[n])&&(s[t]=e.$eval(a[n]))});var c=t.attr('data-target');angular.isDefined(c)&&(u.test(c)?s.target=!1:s.target=c),e.hasOwnProperty('title')||(e.title=''),a.$observe('title',function(t){if(angular.isDefined(t)||!e.hasOwnProperty('title')){var a=e.title;e.title=n.trustAsHtml(t),angular.isDefined(a)&&i(function(){l&&l.$applyPlacement()})}}),a.$observe('disabled',function(e){e&&l.$isShown&&l.hide()}),a.bsTooltip&&e.$watch(a.bsTooltip,function(t,n){angular.isObject(t)?angular.extend(e,t):e.title=t,angular.isDefined(n)&&i(function(){l&&l.$applyPlacement()})},!0),a.bsShow&&e.$watch(a.bsShow,function(e,t){l&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(tooltip),?/i)),e===!0?l.show():l.hide())}),a.bsEnabled&&e.$watch(a.bsEnabled,function(e,t){l&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|1|,?(tooltip),?/i)),e===!1?l.setEnabled(!1):l.setEnabled(!0))}),a.viewport&&e.$watch(a.viewport,function(e){l&&angular.isDefined(e)&&l.setViewport(e)}),l=o(t,s),e.$on('$destroy',function(){l&&l.destroy(),s=null,l=null})}}}]),angular.module('mgcrea.ngStrap.timepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$timepicker',function(){var e=this.defaults={animation:'am-fade',defaultDate:'auto',prefixClass:'timepicker',placement:'bottom-left',templateUrl:'timepicker/timepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!0,timeType:'date',timeFormat:'shortTime',timezone:null,modelTimeFormat:null,autoclose:!1,minTime:-(1/0),maxTime:+(1/0),length:5,hourStep:1,minuteStep:5,secondStep:5,roundDisplay:!1,iconUp:'glyphicon glyphicon-chevron-up',iconDown:'glyphicon glyphicon-chevron-down',arrowBehavior:'pager'};this.$get=['$window','$document','$rootScope','$sce','$dateFormatter','$tooltip','$timeout',function(t,n,a,o,i,r,l){function s(t,n,a){function o(e){var t=6e4*g.minuteStep;return new Date(Math.floor(e.getTime()/t)*t)}function s(e,n){var a=e+n;if(t[0].createTextRange){var o=t[0].createTextRange();o.collapse(!0),o.moveStart('character',e),o.moveEnd('character',a),o.select()}else t[0].setSelectionRange?t[0].setSelectionRange(e,a):angular.isUndefined(t[0].selectionStart)&&(t[0].selectionStart=e,t[0].selectionEnd=a)}function d(){t[0].focus()}var f=r(t,angular.extend({},e,a)),p=a.scope,g=f.$options,m=f.$scope,$=g.lang,h=function(e,t,n){return i.formatDate(e,t,$,n)},v=0,w=g.roundDisplay?o(new Date):new Date,y=n.$dateValue||w,b={hour:y.getHours(),meridian:y.getHours()<12,minute:y.getMinutes(),second:y.getSeconds(),millisecond:y.getMilliseconds()},D=i.getDatetimeFormat(g.timeFormat,$),S=i.hoursFormat(D),x=i.timeSeparator(D),k=i.minutesFormat(D),C=i.secondsFormat(D),T=i.showSeconds(D),E=i.showAM(D);m.$iconUp=g.iconUp,m.$iconDown=g.iconDown,m.$select=function(e,t){f.select(e,t)},m.$moveIndex=function(e,t){f.$moveIndex(e,t)},m.$switchMeridian=function(e){f.switchMeridian(e)},f.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())?(f.$date=e,angular.extend(b,{hour:e.getHours(),minute:e.getMinutes(),second:e.getSeconds(),millisecond:e.getMilliseconds()}),f.$build()):f.$isBuilt||f.$build()},f.select=function(e,t,a){n.$dateValue&&!isNaN(n.$dateValue.getTime())||(n.$dateValue='today'===g.defaultDate?new Date:new Date(1970,0,1)),angular.isDate(e)||(e=new Date(e)),0===t?n.$dateValue.setHours(e.getHours()):1===t?n.$dateValue.setMinutes(e.getMinutes()):2===t&&n.$dateValue.setSeconds(e.getSeconds()),n.$setViewValue(angular.copy(n.$dateValue)),n.$render(),g.autoclose&&!a&&l(function(){f.hide(!0)})},f.switchMeridian=function(e){if(n.$dateValue&&!isNaN(n.$dateValue.getTime())){var t=(e||n.$dateValue).getHours();n.$dateValue.setHours(t<12?t+12:t-12),n.$setViewValue(angular.copy(n.$dateValue)),n.$render()}},f.$build=function(){var e,t,n=m.midIndex=parseInt(g.length/2,10),a=[];for(e=0;e1*g.maxTime},m.$arrowAction=function(e,t){'picker'===g.arrowBehavior?f.$setTimeByStep(e,t):f.$moveIndex(e,t)},f.$setTimeByStep=function(e,t){var n=new Date(f.$date||y),a=n.getHours(),o=n.getMinutes(),i=n.getSeconds();0===t?n.setHours(a-parseInt(g.hourStep,10)*e):1===t?n.setMinutes(o-parseInt(g.minuteStep,10)*e):2===t&&n.setSeconds(i-parseInt(g.secondStep,10)*e),f.select(n,t,!0)},f.$moveIndex=function(e,t){var n;0===t?(n=new Date(1970,0,1,b.hour+e*g.length,b.minute,b.second),angular.extend(b,{hour:n.getHours()})):1===t?(n=new Date(1970,0,1,b.hour,b.minute+e*g.length*g.minuteStep,b.second),angular.extend(b,{minute:n.getMinutes()})):2===t&&(n=new Date(1970,0,1,b.hour,b.minute,b.second+e*g.length*g.secondStep),angular.extend(b,{second:n.getSeconds()})),f.$build()},f.$onMouseDown=function(e){if('input'!==e.target.nodeName.toLowerCase()&&e.preventDefault(),e.stopPropagation(),c){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},f.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void f.hide(!0);var t=new Date(f.$date),n=t.getHours(),a=h(t,S).length,o=t.getMinutes(),i=h(t,k).length,r=t.getSeconds(),l=h(t,C).length,u=1,c=/(37|39)/.test(e.keyCode),d=2+1*T+1*E;c&&(37===e.keyCode?v=v<1?d-1:v-1:39===e.keyCode&&(v=v=f.minTime,n=isNaN(f.maxTime)||new Date(e.getTime()).setFullYear(1970,0,1)<=f.maxTime,a=t&&n;u.$setValidity('date',a),u.$setValidity('min',t),u.$setValidity('max',n),a&&(u.$dateValue=e)}}function d(){return!u.$dateValue||isNaN(u.$dateValue.getTime())?'':$(u.$dateValue,f.timeFormat)}var f={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','autoclose','timeType','timeFormat','timezone','modelTimeFormat','useNative','hourStep','minuteStep','secondStep','length','arrowBehavior','iconUp','iconDown','roundDisplay','id','prefixClass','prefixEvent','defaultDate'],function(e){angular.isDefined(a[e])&&(f[e]=a[e])});var p=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','roundDisplay'],function(e){angular.isDefined(a[e])&&p.test(a[e])&&(f[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(a[n])&&(f[t]=e.$eval(a[n]))}),s&&(f.useNative||l.useNative)&&(f.timeFormat='HH:mm');var g=r(t,u,f);f=g.$options;var m=f.lang,$=function(e,t,n){return o.formatDate(e,t,m,n)};a.bsShow&&e.$watch(a.bsShow,function(e,t){g&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(timepicker),?/i)),e===!0?g.show():g.hide())});var h=i({format:f.timeFormat,lang:m});angular.forEach(['minTime','maxTime'],function(e){angular.isDefined(a[e])&&a.$observe(e,function(t){g.$options[e]=h.getTimeForAttribute(e,t),isNaN(g.$options[e])||g.$build(),c(u.$dateValue)})}),e.$watch(a.ngModel,function(e,t){g.update(u.$dateValue)},!0),u.$parsers.unshift(function(e){var t;if(!e)return u.$setValidity('date',!0),null;var a=angular.isDate(e)?e:h.parse(e,u.$dateValue);return!a||isNaN(a.getTime())?(u.$setValidity('date',!1),n):(c(a),'string'===f.timeType?(t=h.timezoneOffsetAdjust(a,f.timezone,!0),$(t,f.modelTimeFormat||f.timeFormat)):(t=h.timezoneOffsetAdjust(u.$dateValue,f.timezone,!0),'number'===f.timeType?t.getTime():'unix'===f.timeType?t.getTime()/1e3:'iso'===f.timeType?t.toISOString():new Date(t)))}),u.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===f.timeType?h.parse(e,null,f.modelTimeFormat):'unix'===f.timeType?new Date(1e3*e):new Date(e),u.$dateValue=h.timezoneOffsetAdjust(t,f.timezone),d()}),u.$render=function(){t.val(d())},e.$on('$destroy',function(){g&&g.destroy(),f=null,g=null})}}}]),angular.module('mgcrea.ngStrap.tab',[]).provider('$tab',function(){var e=this.defaults={animation:'am-fade',template:'tab/tab.tpl.html',navClass:'nav-tabs',activeClass:'active'},t=this.controller=function(t,n,a){var o=this;o.$options=angular.copy(e),angular.forEach(['animation','navClass','activeClass'],function(e){angular.isDefined(a[e])&&(o.$options[e]=a[e])}),t.$navClass=o.$options.navClass,t.$activeClass=o.$options.activeClass,o.$panes=t.$panes=[],o.$activePaneChangeListeners=o.$viewChangeListeners=[],o.$push=function(e){angular.isUndefined(o.$panes.$active)&&t.$setActive(e.name||0),o.$panes.push(e)},o.$remove=function(e){var t,n=o.$panes.indexOf(e),a=o.$panes.$active;t=angular.isString(a)?o.$panes.map(function(e){return e.name}).indexOf(a):o.$panes.$active,o.$panes.splice(n,1),n=0&&t',placeholder:'Choose among the following...',allText:'All',noneText:'None',maxLength:3,maxLengthHtml:'selected',iconCheckmark:'glyphicon glyphicon-ok',toggle:!1};this.$get=['$window','$document','$rootScope','$tooltip','$timeout',function(t,a,o,i,r){function l(o,l,s){var c={},d=angular.extend({},e,s);c=i(o,d);var f=c.$scope;f.$matches=[],d.multiple?f.$activeIndex=[]:f.$activeIndex=-1,f.$isMultiple=d.multiple,f.$showAllNoneButtons=d.allNoneButtons&&d.multiple,f.$iconCheckmark=d.iconCheckmark,f.$allText=d.allText,f.$noneText=d.noneText,f.$activate=function(e){f.$$postDigest(function(){c.activate(e)})},f.$select=function(e,t){f.$$postDigest(function(){c.select(e)})},f.$isVisible=function(){return c.$isVisible()},f.$isActive=function(e){return c.$isActive(e)},f.$selectAll=function(){for(var e=0;e=f.$matches.length)){var t=f.$matches[e].value;f.$apply(function(){c.activate(e),d.multiple?l.$setViewValue(f.$activeIndex.map(function(e){return angular.isUndefined(f.$matches[e])?null:f.$matches[e].value})):(d.toggle?l.$setViewValue(t===l.$modelValue?n:t):l.$setViewValue(t),c.hide())}),f.$emit(d.prefixEvent+'.select',t,e,c),angular.isDefined(d.onSelect)&&angular.isFunction(d.onSelect)&&d.onSelect(t,e,c)}},c.$updateActiveIndex=function(){d.multiple?angular.isArray(l.$modelValue)?f.$activeIndex=l.$modelValue.map(function(e){return c.$getIndex(e)}):f.$activeIndex=[]:angular.isDefined(l.$modelValue)&&f.$matches.length?f.$activeIndex=c.$getIndex(l.$modelValue):f.$activeIndex=-1},c.$isVisible=function(){return d.minLength&&l?f.$matches.length&&l.$viewValue.length>=d.minLength:f.$matches.length},c.$isActive=function(e){return d.multiple?f.$activeIndex.indexOf(e)!==-1:f.$activeIndex===e},c.$getIndex=function(e){var t;for(t=f.$matches.length;t--&&!angular.equals(f.$matches[t].value,e););return t},c.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),u){var t,n=angular.element(e.target);if('A'!==e.target.nodeName)for(var a=n.parent();!t&&a.length>0;)'A'===a[0].nodeName&&(t=a),a=a.parent();t?angular.element(t).triggerHandler('click'):n.triggerHandler('click')}},c.$onKeyDown=function(e){if(/(9|13|38|40)/.test(e.keyCode))return 9!==e.keyCode&&(e.preventDefault(),e.stopPropagation()),d.multiple&&9===e.keyCode?c.hide():d.multiple||13!==e.keyCode&&9!==e.keyCode?void(d.multiple||(38===e.keyCode&&f.$activeIndex>0?f.$activeIndex--:38===e.keyCode&&f.$activeIndex<0?f.$activeIndex=f.$matches.length-1:40===e.keyCode&&f.$activeIndex0||e.indexOf('Trident/')>0||e.indexOf('Edge/')>0; +},c.$selectScrollFix=function(e){'UL'===a[0].activeElement.tagName&&(e.preventDefault(),e.stopImmediatePropagation(),e.target.focus())};var p=c.show;c.show=function(){p(),d.multiple&&c.$element.addClass('select-multiple'),r(function(){c.$element.on(u?'touchstart':'mousedown',c.$onMouseDown),d.keyboard&&o.on('keydown',c.$onKeyDown)},0,!1)};var g=c.hide;return c.hide=function(){!d.multiple&&angular.isUndefined(l.$modelValue)&&(f.$activeIndex=-1),c.$element.off(u?'touchstart':'mousedown',c.$onMouseDown),d.keyboard&&o.off('keydown',c.$onKeyDown),g(!0)},c}var s=/(ip[ao]d|iphone|android)/gi.test(t.navigator.userAgent),u='createTouch'in t.document&&s;return l.defaults=e,l}]}).directive('bsSelect',['$window','$parse','$q','$select','$parseOptions',function(e,t,n,a,o){var i=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,r){var l={scope:e,placeholder:i.placeholder};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','placeholder','allNoneButtons','maxLength','maxLengthHtml','allText','noneText','iconCheckmark','autoClose','id','sort','caretHtml','prefixClass','prefixEvent','toggle'],function(e){angular.isDefined(n[e])&&(l[e]=n[e])});var s=/^(false|0|)$/i;angular.forEach(['html','container','allNoneButtons','sort'],function(e){angular.isDefined(n[e])&&s.test(n[e])&&(l[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide','onSelect'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(l[t]=e.$eval(n[a]))});var u=t.attr('data-multiple');if(angular.isDefined(u)&&(s.test(u)?l.multiple=!1:l.multiple=u),'select'===t[0].nodeName.toLowerCase()){var c=t;c.css('display','none'),t=angular.element(''),c.after(t)}var d=o(n.bsOptions),f=a(t,r,l);f.$isIE()&&t[0].addEventListener('blur',f.$selectScrollFix);var p=d.$match[7].replace(/\|.+/,'').trim();e.$watch(p,function(t,n){d.valuesFn(e,r).then(function(e){f.update(e),r.$render()})},!0),e.$watch(n.ngModel,function(e,t){f.$updateActiveIndex(),r.$render()},!0),r.$render=function(){var e,n;l.multiple&&angular.isArray(r.$modelValue)?(e=r.$modelValue.map(function(e){return n=f.$getIndex(e),n!==-1&&f.$scope.$matches[n].label}).filter(angular.isDefined),e=e.length>(l.maxLength||i.maxLength)?e.length+' '+(l.maxLengthHtml||i.maxLengthHtml):e.join(', ')):(n=f.$getIndex(r.$modelValue),e=n!==-1&&f.$scope.$matches[n].label),t.html((e||l.placeholder)+(l.caretHtml||i.caretHtml))},l.multiple&&(r.$isEmpty=function(e){return!e||0===e.length}),e.$on('$destroy',function(){f&&f.destroy(),l=null,f=null})}}}]),angular.module('mgcrea.ngStrap.scrollspy',['mgcrea.ngStrap.helpers.debounce','mgcrea.ngStrap.helpers.dimensions']).provider('$scrollspy',function(){var e=this.$$spies={},n=this.defaults={debounce:150,throttle:100,offset:100};this.$get=['$window','$document','$rootScope','dimensions','debounce','throttle',function(a,o,i,r,l,s){function u(e,t){return e[0].nodeName&&e[0].nodeName.toLowerCase()===t.toLowerCase()}function c(o){var c=angular.extend({},n,o);c.element||(c.element=p);var g=u(c.element,'body'),m=g?d:c.element,$=g?'window':c.id;if(e[$])return e[$].$$count++,e[$];var h,v,w,y,b,D,S,x,k={},C=k.$trackedElements=[],T=[];return k.init=function(){this.$$count=1,y=l(this.checkPosition,c.debounce),b=s(this.checkPosition,c.throttle),m.on('click',this.checkPositionWithEventLoop),d.on('resize',y),m.on('scroll',b),D=l(this.checkOffsets,c.debounce),h=i.$on('$viewContentLoaded',D),v=i.$on('$includeContentLoaded',D),D(),$&&(e[$]=k)},k.destroy=function(){this.$$count--,this.$$count>0||(m.off('click',this.checkPositionWithEventLoop),d.off('resize',y),m.off('scroll',b),h(),v(),$&&delete e[$])},k.checkPosition=function(){if(T.length){if(x=(g?a.pageYOffset:m.prop('scrollTop'))||0,S=Math.max(a.innerHeight,f.prop('clientHeight')),xT[e+1].offsetTop))return k.$activateElement(T[e])}},k.checkPositionWithEventLoop=function(){setTimeout(k.checkPosition,1)},k.$activateElement=function(e){if(w){var t=k.$getTrackedElement(w);t&&(t.source.removeClass('active'),u(t.source,'li')&&u(t.source.parent().parent(),'li')&&t.source.parent().parent().removeClass('active'))}w=e.target,e.source.addClass('active'),u(e.source,'li')&&u(e.source.parent().parent(),'li')&&e.source.parent().parent().addClass('active')},k.$getTrackedElement=function(e){return C.filter(function(t){return t.target===e})[0]},k.checkOffsets=function(){angular.forEach(C,function(e){var n=t.querySelector(e.target);e.offsetTop=n?r.offset(n).top:null,c.offset&&null!==e.offsetTop&&(e.offsetTop-=1*c.offset)}),T=C.filter(function(e){return null!==e.offsetTop}).sort(function(e,t){return e.offsetTop-t.offsetTop}),y()},k.trackElement=function(e,t){C.push({target:e,source:t})},k.untrackElement=function(e,t){for(var n,a=C.length;a--;)if(C[a].target===e&&C[a].source===t){n=a;break}C.splice(n,1)},k.activate=function(e){C[e].addClass('active')},k.init(),k}var d=angular.element(a),f=angular.element(o.prop('documentElement')),p=angular.element(a.document.body);return c}]}).directive('bsScrollspy',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,a){return{restrict:'EAC',link:function(e,t,n){var o={scope:e};angular.forEach(['offset','target'],function(e){angular.isDefined(n[e])&&(o[e]=n[e])});var i=a(o);i.trackElement(o.target,t),e.$on('$destroy',function(){i&&(i.untrackElement(o.target,t),i.destroy()),o=null,i=null})}}}]).directive('bsScrollspyList',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,a){return{restrict:'A',compile:function(e,t){var n=e[0].querySelectorAll('li > a[href]');angular.forEach(n,function(e){var t=angular.element(e);t.parent().attr('bs-scrollspy','').attr('data-target',t.attr('href'))})}}}]),angular.module('mgcrea.ngStrap.popover',['mgcrea.ngStrap.tooltip']).provider('$popover',function(){var e=this.defaults={animation:'am-fade',customClass:'',container:!1,target:!1,placement:'right',templateUrl:'popover/popover.tpl.html',contentTemplate:!1,trigger:'click',keyboard:!0,html:!1,title:'',content:'',delay:0,autoClose:!1};this.$get=['$tooltip',function(t){function n(n,a){var o=angular.extend({},e,a),i=t(n,o);return o.content&&(i.$scope.content=o.content),i}return n}]}).directive('bsPopover',['$window','$sce','$popover',function(e,t,n){var a=e.requestAnimationFrame||e.setTimeout;return{restrict:'EAC',scope:!0,link:function(e,o,i){var r,l={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','container','delay','trigger','html','animation','customClass','autoClose','id','prefixClass','prefixEvent','bsEnabled'],function(e){angular.isDefined(i[e])&&(l[e]=i[e])});var s=/^(false|0|)$/i;angular.forEach(['html','container','autoClose'],function(e){angular.isDefined(i[e])&&s.test(i[e])&&(l[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(i[n])&&(l[t]=e.$eval(i[n]))});var u=o.attr('data-target');angular.isDefined(u)&&(s.test(u)?l.target=!1:l.target=u),angular.forEach(['title','content'],function(n){i[n]&&i.$observe(n,function(o,i){e[n]=t.trustAsHtml(o),angular.isDefined(i)&&a(function(){r&&r.$applyPlacement()})})}),i.bsPopover&&e.$watch(i.bsPopover,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t,angular.isDefined(n)&&a(function(){r&&r.$applyPlacement()})},!0),i.bsShow&&e.$watch(i.bsShow,function(e,t){r&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(popover),?/i)),e===!0?r.show():r.hide())}),i.bsEnabled&&e.$watch(i.bsEnabled,function(e){r&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|1|,?(popover),?/i)),e===!1?r.setEnabled(!1):r.setEnabled(!0))}),i.viewport&&e.$watch(i.viewport,function(e){r&&angular.isDefined(e)&&r.setViewport(e)}),r=n(o,l),e.$on('$destroy',function(){r&&r.destroy(),l=null,r=null})}}}]),angular.module('mgcrea.ngStrap.navbar',[]).provider('$navbar',function(){var e=this.defaults={activeClass:'active',routeAttr:'data-match-route',strict:!1};this.$get=function(){return{defaults:e}}}).directive('bsNavbar',['$window','$location','$navbar',function(e,t,n){var a=n.defaults;return{restrict:'A',link:function(e,n,o,i){var r=angular.copy(a);angular.forEach(Object.keys(a),function(e){angular.isDefined(o[e])&&(r[e]=o[e])}),e.$watch(function(){return t.path()},function(e,t){var a=n[0].querySelectorAll('li['+r.routeAttr+']');angular.forEach(a,function(t){var n=angular.element(t),a=n.attr(r.routeAttr).replace('/','\\/');r.strict&&(a='^'+a+'$');var o=new RegExp(a,'i');o.test(e)?n.addClass(r.activeClass):n.removeClass(r.activeClass)})})}}}]),angular.module('mgcrea.ngStrap.modal',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$modal',function(){var e=this.defaults={animation:'am-fade',backdropAnimation:'am-fade',customClass:'',prefixClass:'modal',prefixEvent:'modal',placement:'top',templateUrl:'modal/modal.tpl.html',template:'',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0,size:null,zIndex:null};this.$get=['$window','$rootScope','$bsCompiler','$animate','$timeout','$sce','dimensions',function(n,a,o,i,r,l,s){function u(t){function n(){T.$emit(k.prefixEvent+'.show',x),angular.isDefined(k.onShow)&&angular.isFunction(k.onShow)&&k.onShow(x)}function r(){T.$emit(k.prefixEvent+'.hide',x),angular.isDefined(k.onHide)&&angular.isFunction(k.onHide)&&k.onHide(x),d('.modal').length<=0&&g.removeClass(k.prefixClass+'-open'),k.animation&&g.removeClass(k.prefixClass+'-with-'+k.animation)}function s(){k.backdrop&&(A.on('click',b),F.on('click',b),F.on('wheel',D))}function u(){k.backdrop&&(A.off('click',b),F.off('click',b),F.off('wheel',D))}function w(){k.keyboard&&A.on('keyup',x.$onKeyUp)}function y(){k.keyboard&&A.off('keyup',x.$onKeyUp)}function b(e){e.target===e.currentTarget&&('static'===k.backdrop?x.focus():x.hide())}function D(e){e.preventDefault()}function S(){x.$isShown&&null!==A&&(u(),y()),M&&(M.$destroy(),M=null),A&&(A.remove(),A=x.$element=null)}var x={},k=x.$options=angular.extend({},e,t),C=x.$promise=o.compile(k),T=x.$scope=k.scope&&k.scope.$new()||a.$new();k.element||k.container||(k.container='body'),k.zIndex&&($=parseInt(k.zIndex,10),h=$-10),x.$id=k.id||k.element&&k.element.attr('id')||'',f(['title','content'],function(e){k[e]&&(T[e]=l.trustAsHtml(k[e]))}),T.$hide=function(){T.$$postDigest(function(){x.hide()})},T.$show=function(){T.$$postDigest(function(){x.show()})},T.$toggle=function(){T.$$postDigest(function(){x.toggle()})},x.$isShown=T.$isShown=!1;var E,A,M,F=angular.element('
');return F.css({position:'fixed',top:'0px',left:'0px',bottom:'0px',right:'0px'}),C.then(function(e){E=e,x.init()}),x.init=function(){k.show&&T.$$postDigest(function(){x.show()})},x.destroy=function(){S(),F&&(F.remove(),F=null),T.$destroy()},x.show=function(){if(!x.$isShown){var e,t;if(angular.isElement(k.container)?(e=k.container,t=k.container[0].lastChild?angular.element(k.container[0].lastChild):null):k.container?(e=d(k.container),t=e[0]&&e[0].lastChild?angular.element(e[0].lastChild):null):(e=null,t=k.element),A&&S(),M=x.$scope.$new(),A=x.$element=E.link(M,function(e,t){}),k.backdrop&&(A.css({'z-index':$+20*m}),F.css({'z-index':h+20*m}),m++),!T.$emit(k.prefixEvent+'.show.before',x).defaultPrevented){angular.isDefined(k.onBeforeShow)&&angular.isFunction(k.onBeforeShow)&&k.onBeforeShow(x),A.css({display:'block'}).addClass(k.placement),k.customClass&&A.addClass(k.customClass),k.size&&v[k.size]&&angular.element(d('.modal-dialog',A[0])).addClass(v[k.size]),k.animation&&(k.backdrop&&F.addClass(k.backdropAnimation),A.addClass(k.animation)),k.backdrop&&i.enter(F,g,null),angular.version.minor<=2?i.enter(A,e,t,n):i.enter(A,e,t).then(n),x.$isShown=T.$isShown=!0,c(T);var a=A[0];p(function(){a.focus()}),g.addClass(k.prefixClass+'-open'),k.animation&&g.addClass(k.prefixClass+'-with-'+k.animation),s(),w()}}},x.hide=function(){x.$isShown&&(T.$emit(k.prefixEvent+'.hide.before',x).defaultPrevented||(angular.isDefined(k.onBeforeHide)&&angular.isFunction(k.onBeforeHide)&&k.onBeforeHide(x),angular.version.minor<=2?i.leave(A,r):i.leave(A).then(r),k.backdrop&&(m--,i.leave(F)),x.$isShown=T.$isShown=!1,c(T),u(),y()))},x.toggle=function(){x.$isShown?x.hide():x.show()},x.focus=function(){A[0].focus()},x.$onKeyUp=function(e){27===e.which&&x.$isShown&&(x.hide(),e.stopPropagation())},x}function c(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}function d(e,n){return angular.element((n||t).querySelectorAll(e))}var f=angular.forEach,p=n.requestAnimationFrame||n.setTimeout,g=angular.element(n.document.body),m=0,$=1050,h=1040,v={lg:'modal-lg',sm:'modal-sm'};return u}]}).directive('bsModal',['$window','$sce','$parse','$modal',function(e,t,n,a){return{restrict:'EAC',scope:!0,link:function(e,n,o,i){var r={scope:e,element:n,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation','backdropAnimation','id','prefixEvent','prefixClass','customClass','modalClass','size','zIndex'],function(e){angular.isDefined(o[e])&&(r[e]=o[e])}),r.modalClass&&(r.customClass=r.modalClass);var l=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(e){angular.isDefined(o[e])&&l.test(o[e])&&(r[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(o[n])&&(r[t]=e.$eval(o[n]))}),angular.forEach(['title','content'],function(n){o[n]&&o.$observe(n,function(a,o){e[n]=t.trustAsHtml(a)})}),o.bsModal&&e.$watch(o.bsModal,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t},!0);var s=a(r);n.on(o.trigger||'click',s.toggle),e.$on('$destroy',function(){s&&s.destroy(),r=null,s=null})}}}]),angular.version.minor<3&&angular.version.dot<14&&angular.module('ng').factory('$$rAF',['$window','$timeout',function(e,t){var n=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame,a=e.cancelAnimationFrame||e.webkitCancelAnimationFrame||e.mozCancelAnimationFrame||e.webkitCancelRequestAnimationFrame,o=!!n,i=o?function(e){var t=n(e);return function(){a(t)}}:function(e){var n=t(e,16.66,!1);return function(){t.cancel(n)}};return i.supported=o,i}]),angular.module('mgcrea.ngStrap.helpers.parseOptions',[]).provider('$parseOptions',function(){var e=this.defaults={regexp:/^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/};this.$get=['$parse','$q',function(t,n){function a(a,o){function i(e,t){return e.map(function(e,n){var a,o,i={};return i[c]=e,a=u(t,i),o=p(t,i),{label:a,value:o,index:n}})}var r={},l=angular.extend({},e,o);r.$values=[];var s,u,c,d,f,p,g;return r.init=function(){r.$match=s=a.match(l.regexp),u=t(s[2]||s[1]),c=s[4]||s[6],d=s[5],f=t(s[3]||''),p=t(s[2]?s[1]:c),g=t(s[7])},r.valuesFn=function(e,t){return n.when(g(e,t)).then(function(t){return angular.isArray(t)||(t=[]),r.$values=t.length?i(t,e):[],r.$values})},r.displayValue=function(e){var t={};return t[c]=e,u(t)},r.init(),r}return a}]}),angular.module('mgcrea.ngStrap.helpers.dimensions',[]).factory('dimensions',function(){function t(e){var t=e.ownerDocument,o=e.offsetParent||t;if(a(o,'#document'))return t.documentElement;for(;o&&!a(o,'html')&&'static'===n.css(o,'position');)o=o.offsetParent;return o||t.documentElement}var n={},a=n.nodeName=function(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()};return n.css=function(t,n,a){var o;return o=t.currentStyle?t.currentStyle[n]:e.getComputedStyle?e.getComputedStyle(t)[n]:t.style[n],a===!0?parseFloat(o)||0:o},n.offset=function(t){var n=t.getBoundingClientRect(),a=t.ownerDocument;return{width:n.width||t.offsetWidth,height:n.height||t.offsetHeight,top:n.top+(e.pageYOffset||a.documentElement.scrollTop)-(a.documentElement.clientTop||0),left:n.left+(e.pageXOffset||a.documentElement.scrollLeft)-(a.documentElement.clientLeft||0)}},n.setOffset=function(e,t,a){var o,i,r,l,s,u,c,d=n.css(e,'position'),f=angular.element(e),p={};'static'===d&&(e.style.position='relative'),s=n.offset(e),r=n.css(e,'top'),u=n.css(e,'left'),c=('absolute'===d||'fixed'===d)&&(r+u).indexOf('auto')>-1,c?(o=n.position(e),l=o.top,i=o.left):(l=parseFloat(r)||0,i=parseFloat(u)||0),angular.isFunction(t)&&(t=t.call(e,a,s)),null!==t.top&&(p.top=t.top-s.top+l),null!==t.left&&(p.left=t.left-s.left+i),'using'in t?t.using.call(f,p):f.css({top:p.top+'px',left:p.left+'px'})},n.position=function(e){var o,i,r={top:0,left:0};return'fixed'===n.css(e,'position')?i=e.getBoundingClientRect():(o=t(e),i=n.offset(e),a(o,'html')||(r=n.offset(o)),r.top+=n.css(o,'borderTopWidth',!0),r.left+=n.css(o,'borderLeftWidth',!0)),{width:e.offsetWidth,height:e.offsetHeight,top:i.top-r.top-n.css(e,'marginTop',!0),left:i.left-r.left-n.css(e,'marginLeft',!0)}},n.height=function(e,t){var a=e.offsetHeight;return t?a+=n.css(e,'marginTop',!0)+n.css(e,'marginBottom',!0):a-=n.css(e,'paddingTop',!0)+n.css(e,'paddingBottom',!0)+n.css(e,'borderTopWidth',!0)+n.css(e,'borderBottomWidth',!0),a},n.width=function(e,t){var a=e.offsetWidth;return t?a+=n.css(e,'marginLeft',!0)+n.css(e,'marginRight',!0):a-=n.css(e,'paddingLeft',!0)+n.css(e,'paddingRight',!0)+n.css(e,'borderLeftWidth',!0)+n.css(e,'borderRightWidth',!0),a},n}),angular.module('mgcrea.ngStrap.helpers.debounce',[]).factory('debounce',['$timeout',function(e){return function(t,n,a){var o=null;return function(){var i=this,r=arguments,l=a&&!o;return o&&e.cancel(o),o=e(function(){o=null,a||t.apply(i,r)},n,!1),l&&t.apply(i,r),o}}}]).factory('throttle',['$timeout',function(e){return function(t,n,a){var o=null;return a||(a={}),function(){var i=this,r=arguments;o||(a.leading!==!1&&t.apply(i,r),o=e(function(){o=null,a.trailing!==!1&&t.apply(i,r)},n,!1))}}}]),angular.module('mgcrea.ngStrap.helpers.dateParser',[]).provider('$dateParser',['$localeProvider',function(e){function t(){this.year=1970,this.month=0,this.day=1,this.hours=0,this.minutes=0,this.seconds=0,this.milliseconds=0}function n(){}function a(e){return!isNaN(parseFloat(e))&&isFinite(e)}function o(e,t){for(var n=e.length,a=t.toString().toLowerCase(),o=0;o12?e.getHours()+2:0),e):null},y.timezoneOffsetAdjust=function(e,t,n){return e?(t&&'UTC'===t&&(e=new Date(e.getTime()),e.setMinutes(e.getMinutes()+(n?-1:1)*e.getTimezoneOffset())),e):null},y.init(),y};return s}]}]),angular.module('mgcrea.ngStrap.helpers.dateFormatter',[]).service('$dateFormatter',['$locale','dateFilter',function(e,t){function n(e){return/(h+)([:\.])?(m+)([:\.])?(s*)[ ]?(a?)/i.exec(e).slice(1)}this.getDefaultLocale=function(){return e.id},this.getDatetimeFormat=function(t,n){return e.DATETIME_FORMATS[t]||t},this.weekdaysShort=function(t){return e.DATETIME_FORMATS.SHORTDAY},this.hoursFormat=function(e){return n(e)[0]},this.minutesFormat=function(e){return n(e)[2]},this.secondsFormat=function(e){return n(e)[4]},this.timeSeparator=function(e){return n(e)[1]},this.showSeconds=function(e){return!!n(e)[4]},this.showAM=function(e){return!!n(e)[5]},this.formatDate=function(e,n,a,o){return t(e,n,o)}}]),angular.module('mgcrea.ngStrap.core',[]).service('$bsCompiler',a),angular.module('mgcrea.ngStrap.dropdown',['mgcrea.ngStrap.tooltip']).provider('$dropdown',function(){var e=this.defaults={animation:'am-fade',prefixClass:'dropdown',prefixEvent:'dropdown',placement:'bottom-left',templateUrl:'dropdown/dropdown.tpl.html',trigger:'click',container:!1,keyboard:!0,html:!1,delay:0};this.$get=['$window','$rootScope','$tooltip','$timeout',function(t,n,a,o){function i(t,i){function s(e){if(e.target!==t[0])return e.target!==t[0]&&u.hide()}var u={},c=angular.extend({},e,i);u.$scope=c.scope&&c.scope.$new()||n.$new(),u=a(t,c);var d=t.parent();u.$onKeyDown=function(e){if(/(38|40)/.test(e.keyCode)){e.preventDefault(),e.stopPropagation();var t=angular.element(u.$element[0].querySelectorAll('li:not(.divider) a'));if(t.length){var n;angular.forEach(t,function(e,t){l&&l.call(e,':focus')&&(n=t)}),38===e.keyCode&&n>0?n--:40===e.keyCode&&n=0&&(t.template=o.outerHTML,t.templateUrl=n,o.parentNode.removeChild(o))}return function(e,n,o){var i={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','id','autoClose'],function(e){angular.isDefined(t[e])&&(i[e]=t[e])});var r=/^(false|0|)$/i;angular.forEach(['html','container'],function(e){angular.isDefined(o[e])&&r.test(o[e])&&(i[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(o[n])&&(i[t]=e.$eval(o[n]))}),o.bsDropdown&&e.$watch(o.bsDropdown,function(t,n){e.content=t},!0);var l=a(n,i);o.bsShow&&e.$watch(o.bsShow,function(e,t){l&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(dropdown),?/i)),e===!0?l.show():l.hide())}),e.$on('$destroy',function(){l&&l.destroy(),i=null,l=null})}}}}]),angular.module('mgcrea.ngStrap.button',[]).provider('$button',function(){var e=this.defaults={activeClass:'active',toggleEvent:'click'};this.$get=function(){return{defaults:e}}}).directive('bsCheckboxGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var n=e[0].querySelectorAll('input[type="checkbox"]');angular.forEach(n,function(e){var n=angular.element(e);n.attr('bs-checkbox',''),n.attr('ng-model',t.ngModel+'.'+n.attr('value'))})}}}).directive('bsCheckbox',['$button','$$rAF',function(e,t){var n=e.defaults,a=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,o,i,r){var l=n,s='INPUT'===o[0].nodeName,u=s?o.parent():o,c=!angular.isDefined(i.trueValue)||i.trueValue;a.test(i.trueValue)&&(c=e.$eval(i.trueValue));var d=!!angular.isDefined(i.falseValue)&&i.falseValue;a.test(i.falseValue)&&(d=e.$eval(i.falseValue));var f='boolean'!=typeof c||'boolean'!=typeof d;f&&(r.$parsers.push(function(e){return e?c:d}),r.$formatters.push(function(e){return angular.equals(e,c)})),r.$render=function(){var e=!!r.$viewValue;t(function(){s&&(o[0].checked=e),u.toggleClass(l.activeClass,e)})},o.bind(l.toggleEvent,function(){e.$apply(function(){s||r.$setViewValue(!u.hasClass('active')),r.$render()})})}}}]).directive('bsRadioGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var n=e[0].querySelectorAll('input[type="radio"]');angular.forEach(n,function(e){angular.element(e).attr('bs-radio',''),angular.element(e).attr('ng-model',t.ngModel)})}}}).directive('bsRadio',['$button','$$rAF',function(e,t){var n=e.defaults,a=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,o,i,r){var l,s=n,u='INPUT'===o[0].nodeName,c=u?o.parent():o;i.$observe('value',function(t){l='boolean'!=typeof t&&a.test(t)?e.$eval(t):t,r.$render()}),r.$render=function(){var e=angular.equals(r.$viewValue,l);t(function(){u&&(o[0].checked=e),c.toggleClass(s.activeClass,e)})},o.bind(s.toggleEvent,function(){e.$apply(function(){r.$setViewValue(l),r.$render()})})}}}]),angular.module('mgcrea.ngStrap.datepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$datepicker',function(){var e=this.defaults={animation:'am-fade',prefixClass:'datepicker',placement:'bottom-left',templateUrl:'datepicker/datepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!1,dateType:'date',dateFormat:'shortDate',timezone:null,modelDateFormat:null,dayFormat:'dd',monthFormat:'MMM',yearFormat:'yyyy',monthTitleFormat:'MMMM yyyy',yearTitleFormat:'yyyy',strictFormat:!1,autoclose:!1,minDate:-(1/0),maxDate:+(1/0),startView:0,minView:0,startWeek:0,daysOfWeekDisabled:'',hasToday:!1,hasClear:!1,iconLeft:'glyphicon glyphicon-chevron-left',iconRight:'glyphicon glyphicon-chevron-right'};this.$get=['$window','$document','$rootScope','$sce','$dateFormatter','datepickerViews','$tooltip','$timeout',function(t,n,a,o,i,r,l,s){function u(t,n,a){function o(e){e.selected=u.$isSelected(e.date)}function i(){t[0].focus()}var u=l(t,angular.extend({},e,a)),f=a.scope,p=u.$options,g=u.$scope;p.startView&&(p.startView-=p.minView);var m=r(u);u.$views=m.views;var $=m.viewDate;g.$mode=p.startView,g.$iconLeft=p.iconLeft,g.$iconRight=p.iconRight,g.$hasToday=p.hasToday,g.$hasClear=p.hasClear;var h=u.$views[g.$mode];g.$select=function(e,t){t||u.select(e)},g.$selectPane=function(e){u.$selectPane(e)},g.$toggleMode=function(){u.setMode((g.$mode+1)%u.$views.length)},g.$setToday=function(){p.autoclose?(u.setMode(0),u.select(new Date)):u.select(new Date,!0)},g.$clear=function(){p.autoclose?(u.setMode(0),u.select(null)):u.select(null,!0)},u.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())&&(u.$date=e,h.update.call(h,e)),u.$build(!0)},u.updateDisabledDates=function(e){p.disabledDateRanges=e;for(var t=0,n=g.rows.length;t=p.$options.minDate,n=isNaN(p.$options.maxDate)||e.getTime()<=p.$options.maxDate,a=t&&n;l.$setValidity('date',a),l.$setValidity('min',t),l.$setValidity('max',n),a&&(l.$dateValue=e)}}function c(){return!l.$dateValue||isNaN(l.$dateValue.getTime())?'':m(l.$dateValue,d.dateFormat)}var d={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','html','animation','autoclose','dateType','dateFormat','timezone','modelDateFormat','dayFormat','strictFormat','startWeek','startDate','useNative','lang','startView','minView','iconLeft','iconRight','daysOfWeekDisabled','id','prefixClass','prefixEvent','hasToday','hasClear'],function(e){angular.isDefined(n[e])&&(d[e]=n[e])});var f=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','hasToday','hasClear'],function(e){angular.isDefined(n[e])&&f.test(n[e])&&(d[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(d[t]=e.$eval(n[a]))});var p=i(t,l,d);d=p.$options,r&&d.useNative&&(d.dateFormat='yyyy-MM-dd');var g=d.lang,m=function(e,t){return a.formatDate(e,t,g)},$=o({format:d.dateFormat,lang:g,strict:d.strictFormat});n.bsShow&&e.$watch(n.bsShow,function(e,t){p&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(datepicker),?/i)),e===!0?p.show():p.hide())}),angular.forEach(['minDate','maxDate'],function(e){angular.isDefined(n[e])&&n.$observe(e,function(t){p.$options[e]=$.getDateForAttribute(e,t),isNaN(p.$options[e])||p.$build(!1),u(l.$dateValue)})}),angular.isDefined(n.dateFormat)&&n.$observe('dateFormat',function(e){p.$options.dateFormat=e}),e.$watch(n.ngModel,function(e,t){p.update(l.$dateValue)},!0),angular.isDefined(n.disabledDates)&&e.$watch(n.disabledDates,function(e,t){e=s(e),t=s(t),e&&p.updateDisabledDates(e)}),l.$parsers.unshift(function(e){var t;if(!e)return l.$setValidity('date',!0),null;var n=$.parse(e,l.$dateValue);return!n||isNaN(n.getTime())?void l.$setValidity('date',!1):(u(n),'string'===d.dateType?(t=$.timezoneOffsetAdjust(n,d.timezone,!0),m(t,d.modelDateFormat||d.dateFormat)):(t=$.timezoneOffsetAdjust(l.$dateValue,d.timezone,!0),'number'===d.dateType?t.getTime():'unix'===d.dateType?t.getTime()/1e3:'iso'===d.dateType?t.toISOString():new Date(t)))}),l.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===d.dateType?$.parse(e,null,d.modelDateFormat):'unix'===d.dateType?new Date(1e3*e):new Date(e),l.$dateValue=$.timezoneOffsetAdjust(t,d.timezone),c()}),l.$render=function(){t.val(c())},e.$on('$destroy',function(){p&&p.destroy(),d=null,p=null})}}}]).provider('datepickerViews',function(){function e(e,t){for(var n=[];e.length>0;)n.push(e.splice(0,t));return n}function t(e,t){return(e%t+t)%t}this.$get=['$dateFormatter','$dateParser','$sce',function(n,a,o){return function(i){var r=i.$scope,l=i.$options,s=l.lang,u=function(e,t){return n.formatDate(e,t,s)},c=a({format:l.dateFormat,lang:s,strict:l.strictFormat}),d=n.weekdaysShort(s),f=d.slice(l.startWeek).concat(d.slice(0,l.startWeek)),p=o.trustAsHtml(''+f.join('')+''),g=i.$date||(l.startDate?c.getDateForAttribute('startDate',l.startDate):new Date),m={year:g.getFullYear(),month:g.getMonth(),date:g.getDate()},$=[{format:l.dayFormat,split:7,steps:{month:1},update:function(e,t){!this.built||t||e.getFullYear()!==m.year||e.getMonth()!==m.month?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):e.getDate()===m.date&&1!==e.getDate()||(m.date=i.$date.getDate(),i.$updateSelected())},build:function(){var n=new Date(m.year,m.month,1),a=n.getTimezoneOffset(),o=new Date(+n-864e5*t(n.getDay()-l.startWeek,7)),s=o.getTimezoneOffset(),d=c.timezoneOffsetAdjust(new Date,l.timezone).toDateString();s!==a&&(o=new Date(+o+6e4*(s-a)));for(var f,g=[],$=0;$<42;$++)f=c.daylightSavingAdjust(new Date(o.getFullYear(),o.getMonth(),o.getDate()+$)),g.push({date:f,isToday:f.toDateString()===d,label:u(f,this.format),selected:i.$date&&this.isSelected(f),muted:f.getMonth()!==m.month,disabled:this.isDisabled(f)});r.title=u(n,l.monthTitleFormat),r.showLabels=!0,r.labels=p,r.rows=e(g,this.split),r.isTodayDisabled=this.isDisabled(new Date),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()&&e.getDate()===i.$date.getDate()},isDisabled:function(e){var t=e.getTime();if(tl.maxDate)return!0;if(l.daysOfWeekDisabled.indexOf(e.getDay())!==-1)return!0;if(l.disabledDateRanges)for(var n=0;n=l.disabledDateRanges[n].start&&t<=l.disabledDateRanges[n].end)return!0;return!1},onKeyDown:function(e){if(i.$date){var t,n=i.$date.getTime();37===e.keyCode?t=new Date(n-864e5):38===e.keyCode?t=new Date(n-6048e5):39===e.keyCode?t=new Date(n+864e5):40===e.keyCode&&(t=new Date(n+6048e5)),this.isDisabled(t)||i.select(t,!0)}}},{name:'month',format:l.monthFormat,split:4,steps:{year:1},update:function(e,t){this.built&&e.getFullYear()===m.year?e.getMonth()!==m.month&&(angular.extend(m,{month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected()):(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build())},build:function(){for(var t,n=[],a=0;a<12;a++)t=new Date(m.year,a,1),n.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=u(t,l.yearTitleFormat),r.showLabels=!1,r.rows=e(n,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()&&e.getMonth()===i.$date.getMonth()},isDisabled:function(e){var t=+new Date(e.getFullYear(),e.getMonth()+1,0);return tl.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getMonth(),n=new Date(i.$date);37===e.keyCode?n.setMonth(t-1):38===e.keyCode?n.setMonth(t-4):39===e.keyCode?n.setMonth(t+1):40===e.keyCode&&n.setMonth(t+4),this.isDisabled(n)||i.select(n,!0)}}},{name:'year',format:l.yearFormat,split:4,steps:{year:12},update:function(e,t){!this.built||t||parseInt(e.getFullYear()/20,10)!==parseInt(m.year/20,10)?(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$build()):e.getFullYear()!==m.year&&(angular.extend(m,{year:i.$date.getFullYear(),month:i.$date.getMonth(),date:i.$date.getDate()}),i.$updateSelected())},build:function(){for(var t,n=m.year-m.year%(3*this.split),a=[],o=0;o<12;o++)t=new Date(n+o,0,1),a.push({date:t,label:u(t,this.format),selected:i.$isSelected(t),disabled:this.isDisabled(t)});r.title=a[0].label+'-'+a[a.length-1].label,r.showLabels=!1,r.rows=e(a,this.split),this.built=!0},isSelected:function(e){return i.$date&&e.getFullYear()===i.$date.getFullYear()},isDisabled:function(e){var t=+new Date(e.getFullYear()+1,0,0);return tl.maxDate},onKeyDown:function(e){if(i.$date){var t=i.$date.getFullYear(),n=new Date(i.$date);37===e.keyCode?n.setYear(t-1):38===e.keyCode?n.setYear(t-4):39===e.keyCode?n.setYear(t+1):40===e.keyCode&&n.setYear(t+4),this.isDisabled(n)||i.select(n,!0)}}}];return{views:l.minView?Array.prototype.slice.call($,l.minView):$,viewDate:m}}}]}),angular.module('mgcrea.ngStrap.collapse',[]).provider('$collapse',function(){var e=this.defaults={animation:'am-collapse',disallowToggle:!1,activeClass:'in',startCollapsed:!1,allowMultiple:!1},t=this.controller=function(t,n,a){function o(e){for(var t=s.$targets.$active,n=0;n=o-w?'bottom':'middle'}function u(){return p[0]===t?t.pageYOffset:p[0].scrollTop}function c(){return p[0]===t?t.document.body.scrollHeight:p[0].scrollHeight}var d={},f=angular.extend({},e,l),p=f.target,g='affix affix-top affix-bottom',m=!1,$=0,h=0,v=0,w=0,y=null,b=null,D=o.parent();if(f.offsetParent)if(f.offsetParent.match(/^\d+$/))for(var S=0;S<1*f.offsetParent-1;S++)D=D.parent();else D=angular.element(f.offsetParent);return d.init=function(){this.$parseOffsets(),h=a.offset(o[0]).top+$,m=f.setWidth&&!o[0].style.width,p.on('scroll',this.checkPosition),p.on('click',this.checkPositionWithEventLoop),r.on('resize',this.$debouncedOnResize),this.checkPosition(),this.checkPositionWithEventLoop()},d.destroy=function(){p.off('scroll',this.checkPosition),p.off('click',this.checkPositionWithEventLoop),r.off('resize',this.$debouncedOnResize)},d.checkPositionWithEventLoop=function(){setTimeout(d.checkPosition,1)},d.checkPosition=function(){var e=u(),t=a.offset(o[0]),n=a.height(o[0]),r=s(b,t,n);y!==r&&(y=r,'top'===r?(b=null,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',''))):'bottom'===r?(b=f.offsetUnpin?-(1*f.offsetUnpin):t.top-e,m&&o.css('width',''),f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top',f.offsetParent?'':i[0].offsetHeight-w-n-h+'px'))):(b=null,m&&o.css('width',o[0].offsetWidth+'px'),f.inlineStyles&&(o.css('position','fixed'),o.css('top',$+'px'))),o.removeClass(g).addClass('affix'+('middle'!==r?'-'+r:'')))},d.$onResize=function(){d.$parseOffsets(),d.checkPosition()},d.$debouncedOnResize=n(d.$onResize,50),d.$parseOffsets=function(){var e=o[0].style.position,t=o[0].style.top;f.inlineStyles&&(o.css('position',f.offsetParent?'':'relative'),o.css('top','')),f.offsetTop&&('auto'===f.offsetTop&&(f.offsetTop='+0'),f.offsetTop.match(/^[-+]\d+$/)?($=1*-f.offsetTop,v=f.offsetParent?a.offset(D[0]).top+1*f.offsetTop:a.offset(o[0]).top-a.css(o[0],'marginTop',!0)+1*f.offsetTop):v=1*f.offsetTop),f.offsetBottom&&(w=f.offsetParent&&f.offsetBottom.match(/^[-+]\d+$/)?c()-(a.offset(D[0]).top+a.height(D[0]))+1*f.offsetBottom+1:1*f.offsetBottom),f.inlineStyles&&(o.css('position',e),o.css('top',t))},d.init(),d}var i=angular.element(t.document.body),r=angular.element(t);return o}]}).directive('bsAffix',['$affix','$window','$timeout',function(e,t,n){return{restrict:'EAC',require:'^?bsAffixTarget',link:function(a,o,i,r){var l={scope:a,target:r?r.$element:angular.element(t)};angular.forEach(['offsetTop','offsetBottom','offsetParent','offsetUnpin','inlineStyles','setWidth'],function(e){if(angular.isDefined(i[e])){var t=i[e];/true/i.test(t)&&(t=!0),/false/i.test(t)&&(t=!1),l[e]=t}});var s;n(function(){s=e(o,l)}),a.$on('$destroy',function(){s&&s.destroy(),l=null,s=null})}}}]).directive('bsAffixTarget',function(){return{controller:['$element',function(e){this.$element=e}]}}),angular.module('mgcrea.ngStrap',['mgcrea.ngStrap.modal','mgcrea.ngStrap.aside','mgcrea.ngStrap.alert','mgcrea.ngStrap.button','mgcrea.ngStrap.select','mgcrea.ngStrap.datepicker','mgcrea.ngStrap.timepicker','mgcrea.ngStrap.navbar','mgcrea.ngStrap.tooltip','mgcrea.ngStrap.popover','mgcrea.ngStrap.dropdown','mgcrea.ngStrap.typeahead','mgcrea.ngStrap.scrollspy','mgcrea.ngStrap.affix','mgcrea.ngStrap.tab','mgcrea.ngStrap.collapse'])}(window,document); //# sourceMappingURL=angular-strap.min.js.map diff --git a/dist/angular-strap.min.js.map b/dist/angular-strap.min.js.map index a23cba617..7989a6e9d 100644 --- a/dist/angular-strap.min.js.map +++ b/dist/angular-strap.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["angular-strap.js","typeahead/typeahead.js","helpers/compiler.js","dropdown/dropdown.js","tooltip/tooltip.js","timepicker/timepicker.js","tab/tab.js","select/select.js","scrollspy/scrollspy.js","popover/popover.js","navbar/navbar.js","modal/modal.js","helpers/raf.js","helpers/parse-options.js","helpers/dimensions.js","helpers/debounce.js","helpers/date-parser.js","helpers/date-formatter.js","datepicker/datepicker.js","collapse/collapse.js","button/button.js","aside/aside.js","alert/alert.js","affix/affix.js","module.js"],"names":["placement","document","undefined","templateUrl","options","cache","$templateCache","then","element","res","fetchTemplate","template","fetchPromises","bsCompilerService","$inject","$http","get","angular","module","defaults","this","animation","data","compile","controller","console","controllerAs","resolve","copy","locals","bindToController","forEach","value","isString","$injector","key","invoke","transformTemplate","identity","extend","$template","when","$q","all","templateEl","contentEl","findElement","contentTemplate","removeAttr","html","templates","replace","next","remove","outerHTML","link","scope","trim","contents","linkFn","invokeCtrl","children","instance","ctrl","isObject","arguments","apply","trigger","provider","container","keyboard","delay","minLength","filter","limit","autoSelect","comparator","trimValue","$get","bodyEl","$typeahead","$scope","$matches","config","$resetMatches","$$postDigest","activate","$activeIndex","select","index","$select","evt","matches","$isVisible","safeDigest","update","$$rAF","$render","$emit","prefixEvent","parentScope","length","l","$viewValue","i","preventDefault","stopPropagation","keyCode","$digest","show","$timeout","$element","$onMouseDown","hide","on","$onKeyDown","TypeaheadFactory","array","isFunction","$$phase","$window","body","$filter","expression","directive","results","restrict","require","falseValueRegExp","attr","bsOptions","test","parsedOptions","$parseOptions","typeahead","watchOptions","watchedOptions","$watchCollection","values","$match","$watch","ngModel","newValue","oldValue","$modelValue","valuesFn","selectMode","$setViewValue","substring","displayValue","slice","$formatters","push","modelValue","selected","val","destroy","isDefined","label","toString","$on","target","title","type","autoClose","bsEnabled","viewport","selector","padding","String","htmlReplaceRegExp","$body","$tooltip","split","enterAnimateCallback","leaveAnimateCallback","_tipToHide","leave","$isShown","blur","tipElement","nodeName","triggers","toggle","unbindTriggerEvents","enter","off","bindKeyboardEvents","$onKeyUp","isTouch","$onFocusElementMouseDown","unbindKeyboardEvents","_autoCloseEventsBinded","bindAutoCloseEvents","unbindAutoCloseEvents","stopEventPropagation","event","getPosition","rect","width","elRect","height","p","top","left","dimensions","offset","el","scroll","isBody","documentElement","getCalculatedOffset","position","actualWidth","actualHeight","outerDims","clientWidth","innerHeight","tip","marginTop","parseInt","using","props","css","isNaN","right","marginLeft","setOffset","delta","getViewportAdjustedDelta","isVertical","replaceArrow","arrowDelta","arrowOffsetPosition","viewportDimensions","$viewport","topEdgeOffset","bottomEdgeOffset","viewportPadding","leftEdgeOffset","rightEdgeOffset","dimension","isHorizontal","$arrow","clearTimeout","timeout","tipScope","$destroy","$options","$promise","$bsCompiler","$new","$rootScope","toLowerCase","$id","map","parseFloat","$sce","trustAsHtml","$setEnabled","id","$hide","setEnabled","isEnabled","compileData","promise","init","tipContainer","bindTriggerEvents","isElement","destroyTipElement","hoverState","parent","after","lastChild","display","visibility","clonedElement","version","minor","addClass","$animate","customClass","$applyPlacement","focus","_blur","elementPosition","autoPlace","autoToken","viewportPosition","originalPlacement","indexOf","tipHeight","removeClass","tipPosition","applyPlacement","tipWidth","$location","transclusion","dataTarget","hasOwnProperty","tooltip","$observe","bsTooltip","bsShow","match","setViewport","useNative","timeType","timeFormat","timezone","modelTimeFormat","autoclose","minTime","maxTime","Infinity","hourStep","minuteStep","secondStep","roundDisplay","iconUp","iconDown","arrowBehavior","isNative","$timepicker","timepickerFactory","formatDate","format","viewDate","hour","startDate","getHours","meridian","coeff","selRange","end","start","setSelectionRange","isUndefined","collapse","selectionStart","moveStart","selectionEnd","moveEnd","focusElement","_init","floorMinutes","time","floor","lang","selectedIndex","date","defaultDate","second","getSeconds","millisecond","getMilliseconds","$dateValue","hoursFormat","$dateFormatter","$iconUp","$iconDown","$moveIndex","$switchMeridian","switchMeridian","isDate","getMinutes","getTime","$build","minute","keep","Date","setHours","setMinutes","setSeconds","hours","midIndex","$isSelected","disabled","minutes","seconds","minutesFormat","rows","showSeconds","secondsFormat","$isDisabled","showAM","timeSeparator","$date","isAM","selectedTime","$arrowAction","$setTimeByStep","newDate","targetDate","targetEl","triggerHandler","sepLength","lateralMove","count","minutesLength","selectRange","hoursLength","incr","isSeconds","isMeridian","secondsLength","createSelection","createTextRange","prop","_destroy","_show","_hide","navigator","userAgent","isMaxValid","isValid","parsedTime","isMinValid","setFullYear","$setValidity","$parsers","unshift","viewValue","getTimeFormattedString","timepicker","dateParser","$dateParser","validateAgainstMinMaxTime","getTimeForAttribute","parse","timezoneOffsetAdjust","NaN","self","navClass","activeClass","$activeClass","$panes","$activePaneChangeListeners","$push","pane","$active","$attrs","$navClass","$remove","active","activeIndex","splice","$setActive","name","fn","$pane","$tab","transclude","postLink","ngModelCtrl","bsTabsCtrl","attrs","bsActivePane","parsedBsActivePane","assign","$parse","render","$isActive","prefixClass","multiple","allNoneButtons","sort","caretHtml","placeholder","allText","noneText","maxLength","maxLengthHtml","iconCheckmark","SelectFactory","$isMultiple","$showAllNoneButtons","$iconCheckmark","$allText","$activate","$selectNone","$updateActiveIndex","b","a","$apply","isArray","$getIndex","$selectScrollFix","$isIE","stopImmediatePropagation","ua","activeElement","tagName","e","dataMultiple","inputEl","addEventListener","join","$isEmpty","spies","$document","windowEl","debounce","throttle","ScrollSpyFactory","scrollEl","isWindowSpy","scrollId","$$count","$scrollspy","unbindViewContentLoaded","unbindIncludeContentLoaded","trackedElements","$trackedElements","sortedElements","activeTarget","debouncedCheckPosition","viewportHeight","throttledCheckPosition","debouncedCheckOffsets","scrollTop","checkPositionWithEventLoop","checkOffsets","checkPosition","docEl","$activateElement","offsetTop","setTimeout","source","$getTrackedElement","targetElement","querySelector","trackedElement","trackElement","toDelete","untrackElement","scrollspy","childEl","querySelectorAll","child","content","$popover","PopoverFactory","requestAnimationFrame","popover","bsPopover","routeAttr","$navbar","liElements","li","liElement","pattern","path","RegExp","regexp","backdrop","bodyElement","ModalFactory","$modal","modalElement","unbindBackdropEvents","hideOnBackdropClick","backdropElement","preventEventDefault","destroyModalElement","modalScope","$show","bottom","z-index","defaultPrevented","backdropAnimation","bindBackdropEvents","which","$root","query","bsModal","modal","cancelAnimationFrame","webkitRequestAnimationFrame","mozRequestAnimationFrame","rafSupported","raf","timer","$values","displayFn","valueName","valueFn","ParseOptionsFactory","groupByFn","keyName","jqLite","currentStyle","window","getComputedStyle","extra","boxRect","getBoundingClientRect","style","offsetHeight","docElement","ownerDocument","curPosition","curLeft","curCSSTop","pageYOffset","clientTop","pageXOffset","scrollLeft","clientLeft","curCSSLeft","calculatePosition","curTop","curOffset","curElem","call","offsetParentRect","offsetParentElement","offsetParent","offsetWidth","outer","func","immediate","factory","context","args","cancel","callNow","leading","trailing","wait","year","$localeProvider","milliseconds","ParseDate","prototype","isNumeric","n","isFinite","indexOfCaseInsensitive","len","str","strict","DateParserFactory","day","month","getFullYear","getMonth","proto","noop","toDate","regExpMap","sss","$locale","dateFilter","mm","keys","setFnMap","clonedFormat","search","v","sortedMap","regExpForFormat","re","text","Object","escapeReservedSymbols","regex","m","HH","H","hh","h","EEEE","EEE","DATETIME_FORMATS","SHORTDAY","dd","d","MMMM","MMM","SHORTMONTH","MM","M","DAY","yyyy","yy","y","MONTH","ss","s","setDate","setMonth","setMap","setMapForFormat","$format","baseDate","formatRegex","formatSetMap","exec","fromDate","getDate","today","getDateForAttribute","substr","daylightSavingAdjust","undo","getDefaultLocale","getDatetimeFormat","weekdaysShort","splitTimeFormat","service","matchesSelector","DropdownFactory","$dropdown","onBodyClick","items","parentEl","hasClass","nextSibling","tElement","classList","parentNode","removeChild","nodeType","tAttrs","bsDropdown","dropdown","dateType","dateFormat","modelDateFormat","dayFormat","monthFormat","yearFormat","monthTitleFormat","yearTitleFormat","strictFormat","minDate","maxDate","startView","minView","startWeek","daysOfWeekDisabled","iconLeft","iconRight","DatepickerFactory","$datepicker","pickerViews","views","$iconLeft","$iconRight","$picker","$views","$mode","datepickerViews","$selectPane","$toggleMode","setMode","updateDisabledDates","disabledDateRanges","dateRanges","mode","pristine","$updateSelected","built","$setDisabledEl","isDisabled","steps","getUTCFullYear","getUTCMonth","UTC","getUTCDate","shiftKey","altKey","onKeyDown","previousValue","normalizeDateRanges","ranges","disabledRanges","datepicker","parsedDate","getDateFormattedString","validateAgainstMinMaxDate","disabledDates","daySplit","arr","mod","arrays","size","weekDaysMin","weekDaysLabelsHtml","picker","weekDaysLabels","concat","firstDayOfMonth","firstDate","getDay","firstDateOffset","build","days","isToday","toDateString","muted","showLabels","labels","isSelected","firstMonth","months","lastDate","actualMonth","firstYear","years","actualYear","setYear","startCollapsed","allowMultiple","activeIndexes","$targets","activeItems","activateItem","$collapse","$viewChangeListeners","$registerToggle","$toggles","$unregisterToggle","$unregisterTarget","deactivateItem","fixActiveItemIndexes","disallowToggle","isActive","$activeIndexes","bsCollapseCtrl","controllers","bsCollapseToggle","$registerTarget","action","toggleEvent","$button","constantValueRegExp","isInput","trueValue","falseValue","hasExoticValues","equals","checked","bind","toggleClass","AsideFactory","$aside","bsAside","aside","duration","dismissable","AlertFactory","$alert","bsAlert","alert","AffixFactory","$affix","inlineStyles","reset","setWidth","initialAffixTop","getRequiredAffixClass","unpin","getScrollTop","scrollHeight","getScrollHeight","initialOffsetTop","offsetBottom","affixed","$parseOffsets","affix","elementHeight","offsetUnpin","$onResize","$debouncedOnResize","initialPosition","affixTarget","option"],"mappings":"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","file":"angular-strap.min.js","sourcesContent":["(function(window, document, undefined) {\n'use strict';\n\n// Source: typeahead/typeahead.js\nangular.module('mgcrea.ngStrap.typeahead', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$typeahead', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'typeahead',\n prefixEvent: '$typeahead',\n placement: 'bottom-left',\n templateUrl: 'typeahead/typeahead.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n minLength: 1,\n filter: 'bsAsyncFilter',\n limit: 6,\n autoSelect: false,\n comparator: '',\n trimValue: true\n };\n\n this.$get = function($window, $rootScope, $tooltip, $$rAF, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n\n function TypeaheadFactory(element, controller, config) {\n\n var $typeahead = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $typeahead = $tooltip(element, options);\n var parentScope = config.scope;\n var scope = $typeahead.$scope;\n\n scope.$resetMatches = function() {\n scope.$matches = [];\n scope.$activeIndex = options.autoSelect ? 0 : -1; // If set to 0, the first match will be highlighted\n };\n scope.$resetMatches();\n\n scope.$activate = function(index) {\n scope.$$postDigest(function() {\n $typeahead.activate(index);\n });\n };\n\n scope.$select = function(index, evt) {\n scope.$$postDigest(function() {\n $typeahead.select(index);\n });\n };\n\n scope.$isVisible = function() {\n return $typeahead.$isVisible();\n };\n\n // Public methods\n\n $typeahead.update = function(matches) {\n scope.$matches = matches;\n if (scope.$activeIndex >= matches.length) {\n scope.$activeIndex = options.autoSelect ? 0 : -1;\n }\n\n // wrap in a $timeout so the results are updated\n // before repositioning\n safeDigest(scope);\n $$rAF($typeahead.$applyPlacement);\n };\n\n $typeahead.activate = function(index) {\n scope.$activeIndex = index;\n };\n\n $typeahead.select = function(index) {\n if (index === -1) return;\n var value = scope.$matches[index].value;\n // console.log('$setViewValue', value);\n controller.$setViewValue(value);\n controller.$render();\n scope.$resetMatches();\n if (parentScope) parentScope.$digest();\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $typeahead);\n };\n\n // Protected methods\n\n $typeahead.$isVisible = function() {\n if (!options.minLength || !controller) {\n return !!scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength;\n };\n\n $typeahead.$getIndex = function(value) {\n var l = scope.$matches.length,\n i = l;\n if (!l) return;\n for (i = l; i--;) {\n if (scope.$matches[i].value === value) break;\n }\n if (i < 0) return;\n return i;\n };\n\n $typeahead.$onMouseDown = function(evt) {\n // Prevent blur on mousedown\n evt.preventDefault();\n evt.stopPropagation();\n };\n\n $typeahead.$onKeyDown = function(evt) {\n if (!/(38|40|13)/.test(evt.keyCode)) return;\n\n // Let ngSubmit pass if the typeahead tip is hidden or no option is selected\n if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // Select with enter\n if (evt.keyCode === 13 && scope.$matches.length) {\n $typeahead.select(scope.$activeIndex);\n }\n\n // Navigate with keyboard\n else if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n };\n\n // Overrides\n\n var show = $typeahead.show;\n $typeahead.show = function() {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed immediately.\n $timeout(function() {\n $typeahead.$element && $typeahead.$element.on('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n element && element.on('keydown', $typeahead.$onKeyDown);\n }\n }, 0, false);\n };\n\n var hide = $typeahead.hide;\n $typeahead.hide = function() {\n $typeahead.$element && $typeahead.$element.off('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n element && element.off('keydown', $typeahead.$onKeyDown);\n }\n if (!options.autoSelect)\n $typeahead.activate(-1);\n hide();\n };\n\n return $typeahead;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n TypeaheadFactory.defaults = defaults;\n return TypeaheadFactory;\n\n };\n\n })\n\n .filter('bsAsyncFilter', function($filter) {\n return function(array, expression, comparator) {\n if (array && angular.isFunction(array.then)) {\n return array.then(function(results) {\n return $filter('filter')(results, expression, comparator);\n });\n } else {\n return $filter('filter')(array, expression, comparator);\n }\n };\n })\n\n .directive('bsTypeahead', function($window, $parse, $q, $typeahead, $parseOptions) {\n\n var defaults = $typeahead.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'filter', 'limit', 'minLength', 'watchOptions', 'selectMode', 'autoSelect', 'comparator', 'id', 'prefixEvent', 'prefixClass'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'trimValue'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // Disable browser autocompletion\n if (!element.attr('autocomplete')) element.attr('autocomplete', 'off');\n\n // Build proper bsOptions\n var filter = options.filter || defaults.filter;\n var limit = options.limit || defaults.limit;\n var comparator = options.comparator || defaults.comparator;\n\n var bsOptions = attr.bsOptions;\n if (filter) bsOptions += ' | ' + filter + ':$viewValue';\n if (comparator) bsOptions += ':' + comparator;\n if (limit) bsOptions += ' | limitTo:' + limit;\n var parsedOptions = $parseOptions(bsOptions);\n\n // Initialize typeahead\n var typeahead = $typeahead(element, controller, options);\n\n // Watch options on demand\n if (options.watchOptions) {\n // Watch bsOptions values before filtering for changes, drop function calls\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').replace(/\\(.*\\)/g, '').trim();\n scope.$watchCollection(watchedOptions, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller).then(function(values) {\n typeahead.update(values);\n controller.$render();\n });\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('$watch', element.attr('ng-model'), newValue);\n scope.$modelValue = newValue; // Publish modelValue on scope for custom templates\n parsedOptions.valuesFn(scope, controller)\n .then(function(values) {\n // Prevent input with no future prospect if selectMode is truthy\n // @TODO test selectMode\n if (options.selectMode && !values.length && newValue.length > 0) {\n controller.$setViewValue(controller.$viewValue.substring(0, controller.$viewValue.length - 1));\n return;\n }\n if (values.length > limit) values = values.slice(0, limit);\n typeahead.update(values);\n // Queue a new rendering that will leverage collection loading\n controller.$render();\n });\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var displayValue = parsedOptions.displayValue(modelValue);\n\n // If we can determine the displayValue, use that\n if (displayValue) {\n return displayValue;\n }\n\n // If there's no display value, attempt to use the modelValue.\n // If the model is an object not much we can do\n if (modelValue && typeof modelValue !== 'object') {\n return modelValue;\n }\n return '';\n });\n\n // Model rendering in view\n controller.$render = function() {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n if (controller.$isEmpty(controller.$viewValue)) {\n return element.val('');\n }\n var index = typeahead.$getIndex(controller.$modelValue);\n var selected = angular.isDefined(index) ? typeahead.$scope.$matches[index].label : controller.$viewValue;\n selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected;\n var value = selected ? selected.toString().replace(/<(?:.|\\n)*?>/gm, '') : '';\n element.val(options.trimValue === false ? value : value.trim());\n };\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (typeahead) typeahead.destroy();\n options = null;\n typeahead = null;\n });\n\n }\n };\n\n });\n\n// Source: tooltip/tooltip.js\nangular.module('mgcrea.ngStrap.tooltip', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$tooltip', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n prefixClass: 'tooltip',\n prefixEvent: 'tooltip',\n container: false,\n target: false,\n placement: 'top',\n templateUrl: 'tooltip/tooltip.tpl.html',\n template: '',\n contentTemplate: false,\n trigger: 'hover focus',\n keyboard: false,\n html: false,\n show: false,\n title: '',\n type: '',\n delay: 0,\n autoClose: false,\n bsEnabled: true,\n viewport: {\n selector: 'body',\n padding: 0\n }\n };\n\n this.$get = function($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) {\n\n var trim = String.prototype.trim;\n var isTouch = 'createTouch' in $window.document;\n var htmlReplaceRegExp = /ng-bind=\"/ig;\n var $body = angular.element($window.document);\n\n function TooltipFactory(element, config) {\n\n var $tooltip = {};\n\n // Common vars\n var options = $tooltip.$options = angular.extend({}, defaults, config);\n var promise = $tooltip.$promise = $bsCompiler.compile(options);\n var scope = $tooltip.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n var nodeName = element[0].nodeName.toLowerCase();\n if(options.delay && angular.isString(options.delay)) {\n var split = options.delay.split(',').map(parseFloat);\n options.delay = split.length > 1 ? {show: split[0], hide: split[1]} : split[0];\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $tooltip.$id = options.id || element.attr('id') || '';\n\n // Support scope as string options\n if(options.title) {\n scope.title = $sce.trustAsHtml(options.title);\n }\n\n // Provide scope helpers\n scope.$setEnabled = function(isEnabled) {\n scope.$$postDigest(function() {\n $tooltip.setEnabled(isEnabled);\n });\n };\n scope.$hide = function() {\n scope.$$postDigest(function() {\n $tooltip.hide();\n });\n };\n scope.$show = function() {\n scope.$$postDigest(function() {\n $tooltip.show();\n });\n };\n scope.$toggle = function() {\n scope.$$postDigest(function() {\n $tooltip.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $tooltip.$isShown = scope.$isShown = false;\n\n // Private vars\n var timeout, hoverState;\n\n // Fetch, compile then initialize tooltip\n var compileData, tipElement, tipContainer, tipScope;\n promise.then(function(data) {\n compileData = data;\n $tooltip.init();\n });\n\n $tooltip.init = function() {\n\n // Options: delay\n if (options.delay && angular.isNumber(options.delay)) {\n options.delay = {\n show: options.delay,\n hide: options.delay\n };\n }\n\n // Replace trigger on touch devices ?\n // if(isTouch && options.trigger === defaults.trigger) {\n // options.trigger.replace(/hover/g, 'click');\n // }\n\n // Options : container\n if(options.container === 'self') {\n tipContainer = element;\n } else if(angular.isElement(options.container)) {\n tipContainer = options.container;\n } else if(options.container) {\n tipContainer = findElement(options.container);\n }\n\n // Options: trigger\n bindTriggerEvents();\n\n // Options: target\n if(options.target) {\n options.target = angular.isElement(options.target) ? options.target : findElement(options.target);\n }\n\n // Options: show\n if(options.show) {\n scope.$$postDigest(function() {\n options.trigger === 'focus' ? element[0].focus() : $tooltip.show();\n });\n }\n\n };\n\n $tooltip.destroy = function() {\n\n // Unbind events\n unbindTriggerEvents();\n\n // Remove element\n destroyTipElement();\n\n // Destroy scope\n scope.$destroy();\n\n };\n\n $tooltip.enter = function() {\n\n clearTimeout(timeout);\n hoverState = 'in';\n if (!options.delay || !options.delay.show) {\n return $tooltip.show();\n }\n\n timeout = setTimeout(function() {\n if (hoverState ==='in') $tooltip.show();\n }, options.delay.show);\n\n };\n\n $tooltip.show = function() {\n if (!options.bsEnabled || $tooltip.$isShown) return;\n\n scope.$emit(options.prefixEvent + '.show.before', $tooltip);\n var parent, after;\n if (options.container) {\n parent = tipContainer;\n if (tipContainer[0].lastChild) {\n after = angular.element(tipContainer[0].lastChild);\n } else {\n after = null;\n }\n } else {\n parent = null;\n after = element;\n }\n\n\n // Hide any existing tipElement\n if(tipElement) destroyTipElement();\n // Fetch a cloned element linked from template\n tipScope = $tooltip.$scope.$new();\n tipElement = $tooltip.$element = compileData.link(tipScope, function(clonedElement, scope) {});\n\n // Set the initial positioning. Make the tooltip invisible\n // so IE doesn't try to focus on it off screen.\n tipElement.css({top: '-9999px', left: '-9999px', right: 'auto', display: 'block', visibility: 'hidden'});\n\n // Options: animation\n if(options.animation) tipElement.addClass(options.animation);\n // Options: type\n if(options.type) tipElement.addClass(options.prefixClass + '-' + options.type);\n // Options: custom classes\n if(options.customClass) tipElement.addClass(options.customClass);\n\n // Append the element, without any animations. If we append\n // using $animate.enter, some of the animations cause the placement\n // to be off due to the transforms.\n after ? after.after(tipElement) : parent.prepend(tipElement);\n\n $tooltip.$isShown = scope.$isShown = true;\n safeDigest(scope);\n\n // Now, apply placement\n $tooltip.$applyPlacement();\n\n // Once placed, animate it.\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.enter(tipElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(tipElement, parent, after).then(enterAnimateCallback);\n }\n safeDigest(scope);\n\n $$rAF(function () {\n // Once the tooltip is placed and the animation starts, make the tooltip visible\n if(tipElement) tipElement.css({visibility: 'visible'});\n\n // Bind events\n if(options.keyboard) {\n if(options.trigger !== 'focus') {\n $tooltip.focus();\n }\n bindKeyboardEvents();\n }\n });\n\n if(options.autoClose) {\n bindAutoCloseEvents();\n }\n\n };\n\n function enterAnimateCallback() {\n scope.$emit(options.prefixEvent + '.show', $tooltip);\n }\n\n $tooltip.leave = function() {\n\n clearTimeout(timeout);\n hoverState = 'out';\n if (!options.delay || !options.delay.hide) {\n return $tooltip.hide();\n }\n timeout = setTimeout(function () {\n if (hoverState === 'out') {\n $tooltip.hide();\n }\n }, options.delay.hide);\n\n };\n\n var _blur;\n var _tipToHide;\n $tooltip.hide = function(blur) {\n\n if(!$tooltip.$isShown) return;\n scope.$emit(options.prefixEvent + '.hide.before', $tooltip);\n\n // store blur value for leaveAnimateCallback to use\n _blur = blur;\n\n // store current tipElement reference to use\n // in leaveAnimateCallback\n _tipToHide = tipElement;\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.leave(tipElement, leaveAnimateCallback);\n } else {\n $animate.leave(tipElement).then(leaveAnimateCallback);\n }\n\n $tooltip.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n if(options.keyboard && tipElement !== null) {\n unbindKeyboardEvents();\n }\n\n if(options.autoClose && tipElement !== null) {\n unbindAutoCloseEvents();\n }\n };\n\n function leaveAnimateCallback() {\n scope.$emit(options.prefixEvent + '.hide', $tooltip);\n\n // check if current tipElement still references\n // the same element when hide was called\n if (tipElement === _tipToHide) {\n // Allow to blur the input when hidden, like when pressing enter key\n if(_blur && options.trigger === 'focus') {\n return element[0].blur();\n }\n\n // clean up child scopes\n destroyTipElement();\n }\n }\n\n $tooltip.toggle = function() {\n $tooltip.$isShown ? $tooltip.leave() : $tooltip.enter();\n };\n\n $tooltip.focus = function() {\n tipElement[0].focus();\n };\n\n $tooltip.setEnabled = function(isEnabled) {\n options.bsEnabled = isEnabled;\n };\n\n $tooltip.setViewport = function(viewport) {\n options.viewport = viewport;\n };\n\n // Protected methods\n\n $tooltip.$applyPlacement = function() {\n if(!tipElement) return;\n\n // Determine if we're doing an auto or normal placement\n var placement = options.placement,\n autoToken = /\\s?auto?\\s?/i,\n autoPlace = autoToken.test(placement);\n\n if (autoPlace) {\n placement = placement.replace(autoToken, '') || defaults.placement;\n }\n\n // Need to add the position class before we get\n // the offsets\n tipElement.addClass(options.placement);\n\n // Get the position of the target element\n // and the height and width of the tooltip so we can center it.\n var elementPosition = getPosition(),\n tipWidth = tipElement.prop('offsetWidth'),\n tipHeight = tipElement.prop('offsetHeight');\n\n // Refresh viewport position\n $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport);\n\n // If we're auto placing, we need to check the positioning\n if (autoPlace) {\n var originalPlacement = placement;\n var viewportPosition = getPosition($tooltip.$viewport);\n\n // Determine if the vertical placement\n if (originalPlacement.indexOf('bottom') >= 0 && elementPosition.bottom + tipHeight > viewportPosition.bottom) {\n placement = originalPlacement.replace('bottom', 'top');\n } else if (originalPlacement.indexOf('top') >= 0 && elementPosition.top - tipHeight < viewportPosition.top) {\n placement = originalPlacement.replace('top', 'bottom');\n }\n\n // Determine the horizontal placement\n // The exotic placements of left and right are opposite of the standard placements. Their arrows are put on the left/right\n // and flow in the opposite direction of their placement.\n if ((originalPlacement === 'right' || originalPlacement === 'bottom-left' || originalPlacement === 'top-left') &&\n elementPosition.right + tipWidth > viewportPosition.width) {\n\n placement = originalPlacement === 'right' ? 'left' : placement.replace('left', 'right');\n } else if ((originalPlacement === 'left' || originalPlacement === 'bottom-right' || originalPlacement === 'top-right') &&\n elementPosition.left - tipWidth < viewportPosition.left) {\n\n placement = originalPlacement === 'left' ? 'right' : placement.replace('right', 'left');\n }\n\n tipElement.removeClass(originalPlacement).addClass(placement);\n }\n\n // Get the tooltip's top and left coordinates to center it with this directive.\n var tipPosition = getCalculatedOffset(placement, elementPosition, tipWidth, tipHeight);\n applyPlacement(tipPosition, placement);\n };\n\n $tooltip.$onKeyUp = function(evt) {\n if (evt.which === 27 && $tooltip.$isShown) {\n $tooltip.hide();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusKeyUp = function(evt) {\n if (evt.which === 27) {\n element[0].blur();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusElementMouseDown = function(evt) {\n evt.preventDefault();\n evt.stopPropagation();\n // Some browsers do not auto-focus buttons (eg. Safari)\n $tooltip.$isShown ? element[0].blur() : element[0].focus();\n };\n\n // bind/unbind events\n function bindTriggerEvents() {\n var triggers = options.trigger.split(' ');\n angular.forEach(triggers, function(trigger) {\n if(trigger === 'click') {\n element.on('click', $tooltip.toggle);\n } else if(trigger !== 'manual') {\n element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n nodeName === 'button' && trigger !== 'hover' && element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n });\n }\n\n function unbindTriggerEvents() {\n var triggers = options.trigger.split(' ');\n for (var i = triggers.length; i--;) {\n var trigger = triggers[i];\n if(trigger === 'click') {\n element.off('click', $tooltip.toggle);\n } else if(trigger !== 'manual') {\n element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n nodeName === 'button' && trigger !== 'hover' && element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n }\n\n function bindKeyboardEvents() {\n if(options.trigger !== 'focus') {\n tipElement.on('keyup', $tooltip.$onKeyUp);\n } else {\n element.on('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n function unbindKeyboardEvents() {\n if(options.trigger !== 'focus') {\n tipElement.off('keyup', $tooltip.$onKeyUp);\n } else {\n element.off('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n var _autoCloseEventsBinded = false;\n function bindAutoCloseEvents() {\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n // Stop propagation when clicking inside tooltip\n tipElement.on('click', stopEventPropagation);\n\n // Hide when clicking outside tooltip\n $body.on('click', $tooltip.hide);\n\n _autoCloseEventsBinded = true;\n }, 0, false);\n }\n\n function unbindAutoCloseEvents() {\n if (_autoCloseEventsBinded) {\n tipElement.off('click', stopEventPropagation);\n $body.off('click', $tooltip.hide);\n _autoCloseEventsBinded = false;\n }\n }\n\n function stopEventPropagation(event) {\n event.stopPropagation();\n }\n\n // Private methods\n\n function getPosition($element) {\n $element = $element || (options.target || element);\n\n var el = $element[0],\n isBody = el.tagName === 'BODY';\n\n var elRect = el.getBoundingClientRect();\n var rect = {};\n\n // IE8 has issues with angular.extend and using elRect directly.\n // By coping the values of elRect into a new object, we can continue to use extend\n for (var p in elRect) {\n // DO NOT use hasOwnProperty when inspecting the return of getBoundingClientRect.\n rect[p] = elRect[p];\n }\n\n if (rect.width === null) {\n // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093\n rect = angular.extend({}, rect, { width: elRect.right - elRect.left, height: elRect.bottom - elRect.top });\n }\n var elOffset = isBody ? { top: 0, left: 0 } : dimensions.offset(el),\n scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0 },\n outerDims = isBody ? { width: document.documentElement.clientWidth, height: $window.innerHeight } : null;\n\n return angular.extend({}, rect, scroll, outerDims, elOffset);\n }\n\n function getCalculatedOffset(placement, position, actualWidth, actualHeight) {\n var offset;\n var split = placement.split('-');\n\n switch (split[0]) {\n case 'right':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left + position.width\n };\n break;\n case 'bottom':\n offset = {\n top: position.top + position.height,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n case 'left':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left - actualWidth\n };\n break;\n default:\n offset = {\n top: position.top - actualHeight,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n }\n\n if(!split[1]) {\n return offset;\n }\n\n // Add support for corners @todo css\n if(split[0] === 'top' || split[0] === 'bottom') {\n switch (split[1]) {\n case 'left':\n offset.left = position.left;\n break;\n case 'right':\n offset.left = position.left + position.width - actualWidth;\n }\n } else if(split[0] === 'left' || split[0] === 'right') {\n switch (split[1]) {\n case 'top':\n offset.top = position.top - actualHeight;\n break;\n case 'bottom':\n offset.top = position.top + position.height;\n }\n }\n\n return offset;\n }\n\n function applyPlacement(offset, placement) {\n var tip = tipElement[0],\n width = tip.offsetWidth,\n height = tip.offsetHeight;\n\n // manually read margins because getBoundingClientRect includes difference\n var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10),\n marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10);\n\n // we must check for NaN for ie 8/9\n if (isNaN(marginTop)) marginTop = 0;\n if (isNaN(marginLeft)) marginLeft = 0;\n\n offset.top = offset.top + marginTop;\n offset.left = offset.left + marginLeft;\n\n // dimensions setOffset doesn't round pixel values\n // so we use setOffset directly with our own function\n dimensions.setOffset(tip, angular.extend({\n using: function (props) {\n tipElement.css({\n top: Math.round(props.top) + 'px',\n left: Math.round(props.left) + 'px',\n right: ''\n });\n }\n }, offset), 0);\n\n // check to see if placing tip in new offset caused the tip to resize itself\n var actualWidth = tip.offsetWidth,\n actualHeight = tip.offsetHeight;\n\n if (placement === 'top' && actualHeight !== height) {\n offset.top = offset.top + height - actualHeight;\n }\n\n // If it's an exotic placement, exit now instead of\n // applying a delta and changing the arrow\n if (/top-left|top-right|bottom-left|bottom-right/.test(placement)) return;\n\n var delta = getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight);\n\n if (delta.left) {\n offset.left += delta.left;\n } else {\n offset.top += delta.top;\n }\n\n dimensions.setOffset(tip, offset);\n\n if (/top|right|bottom|left/.test(placement)) {\n var isVertical = /top|bottom/.test(placement),\n arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight,\n arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight';\n\n replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical);\n }\n }\n\n // @source https://github.com/twbs/bootstrap/blob/v3.3.5/js/tooltip.js#L380\n function getViewportAdjustedDelta(placement, position, actualWidth, actualHeight) {\n var delta = {top: 0, left: 0};\n if (!$tooltip.$viewport) return delta;\n\n var viewportPadding = options.viewport && options.viewport.padding || 0;\n var viewportDimensions = getPosition($tooltip.$viewport);\n\n if (/right|left/.test(placement)) {\n var topEdgeOffset = position.top - viewportPadding - viewportDimensions.scroll;\n var bottomEdgeOffset = position.top + viewportPadding - viewportDimensions.scroll + actualHeight;\n if (topEdgeOffset < viewportDimensions.top) { // top overflow\n delta.top = viewportDimensions.top - topEdgeOffset;\n } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow\n delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset;\n }\n } else {\n var leftEdgeOffset = position.left - viewportPadding;\n var rightEdgeOffset = position.left + viewportPadding + actualWidth;\n if (leftEdgeOffset < viewportDimensions.left) { // left overflow\n delta.left = viewportDimensions.left - leftEdgeOffset;\n } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow\n delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset;\n }\n }\n\n return delta;\n }\n\n function replaceArrow(delta, dimension, isHorizontal) {\n var $arrow = findElement('.tooltip-arrow, .arrow', tipElement[0]);\n\n $arrow.css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')\n .css(isHorizontal ? 'top' : 'left', '');\n }\n\n function destroyTipElement() {\n // Cancel pending callbacks\n clearTimeout(timeout);\n\n if($tooltip.$isShown && tipElement !== null) {\n if(options.autoClose) {\n unbindAutoCloseEvents();\n }\n\n if(options.keyboard) {\n unbindKeyboardEvents();\n }\n }\n\n if(tipScope) {\n tipScope.$destroy();\n tipScope = null;\n }\n\n if(tipElement) {\n tipElement.remove();\n tipElement = $tooltip.$element = null;\n }\n }\n\n return $tooltip;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate(template) {\n if(fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache}).then(function(res) {\n return res.data;\n }));\n }\n\n return TooltipFactory;\n\n };\n\n })\n\n .directive('bsTooltip', function($window, $location, $sce, $tooltip, $$rAF) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if(angular.isDefined(dataTarget)) {\n if(falseValueRegExp.test(dataTarget))\n options.target = false;\n else\n options.target = dataTarget;\n }\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')){\n scope.title = '';\n }\n\n // Observe scope attributes for change\n attr.$observe('title', function(newValue) {\n if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) {\n var oldValue = scope.title;\n scope.title = $sce.trustAsHtml(newValue);\n angular.isDefined(oldValue) && $$rAF(function() {\n tooltip && tooltip.$applyPlacement();\n });\n }\n });\n\n // Support scope as an object\n attr.bsTooltip && scope.$watch(attr.bsTooltip, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.title = newValue;\n }\n angular.isDefined(oldValue) && $$rAF(function() {\n tooltip && tooltip.$applyPlacement();\n });\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if(!tooltip || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i);\n newValue === true ? tooltip.show() : tooltip.hide();\n });\n\n // Enabled binding support\n attr.bsEnabled && scope.$watch(attr.bsEnabled, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.bsEnabled, newValue, oldValue);\n if(!tooltip || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i);\n newValue === false ? tooltip.setEnabled(false) : tooltip.setEnabled(true);\n });\n\n // Viewport support\n attr.viewport && scope.$watch(attr.viewport, function (newValue) {\n if(!tooltip || !angular.isDefined(newValue)) return;\n tooltip.setViewport(newValue);\n });\n\n // Initialize popover\n var tooltip = $tooltip(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if(tooltip) tooltip.destroy();\n options = null;\n tooltip = null;\n });\n\n }\n };\n\n });\n\n// Source: timepicker/timepicker.js\nangular.module('mgcrea.ngStrap.timepicker', ['mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip'])\n\n .provider('$timepicker', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n //uncommenting the following line will break backwards compatability\n // prefixEvent: 'timepicker',\n prefixClass: 'timepicker',\n placement: 'bottom-left',\n templateUrl: 'timepicker/timepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: true,\n timeType: 'date',\n timeFormat: 'shortTime',\n timezone: null,\n modelTimeFormat: null,\n autoclose: false,\n minTime: -Infinity,\n maxTime: +Infinity,\n length: 5,\n hourStep: 1,\n minuteStep: 5,\n secondStep: 5,\n roundDisplay: false,\n iconUp: 'glyphicon glyphicon-chevron-up',\n iconDown: 'glyphicon glyphicon-chevron-down',\n arrowBehavior: 'pager'\n };\n\n this.$get = function($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) {\n\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) {\n defaults.lang = $dateFormatter.getDefaultLocale();\n }\n\n function timepickerFactory(element, controller, config) {\n\n var $timepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $timepicker.$options;\n var scope = $timepicker.$scope;\n\n var lang = options.lang;\n var formatDate = function(date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n function floorMinutes(time) {\n // coeff used to floor current time to nearest minuteStep interval\n var coeff = 1000 * 60 * options.minuteStep;\n return new Date(Math.floor(time.getTime() / coeff) * coeff);\n }\n\n // View vars\n\n var selectedIndex = 0;\n var defaultDate = options.roundDisplay ? floorMinutes(new Date()) : new Date();\n var startDate = controller.$dateValue || defaultDate;\n var viewDate = {\n hour: startDate.getHours(),\n meridian: startDate.getHours() < 12,\n minute: startDate.getMinutes(),\n second: startDate.getSeconds(),\n millisecond: startDate.getMilliseconds()\n };\n\n var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang);\n\n var hoursFormat = $dateFormatter.hoursFormat(format),\n timeSeparator = $dateFormatter.timeSeparator(format),\n minutesFormat = $dateFormatter.minutesFormat(format),\n secondsFormat = $dateFormatter.secondsFormat(format),\n showSeconds = $dateFormatter.showSeconds(format),\n showAM = $dateFormatter.showAM(format);\n\n scope.$iconUp = options.iconUp;\n scope.$iconDown = options.iconDown;\n\n // Scope methods\n\n scope.$select = function(date, index) {\n $timepicker.select(date, index);\n };\n scope.$moveIndex = function(value, index) {\n $timepicker.$moveIndex(value, index);\n };\n scope.$switchMeridian = function(date) {\n $timepicker.switchMeridian(date);\n };\n\n // Public methods\n\n $timepicker.update = function(date) {\n // console.warn('$timepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $timepicker.$date = date;\n angular.extend(viewDate, {\n hour: date.getHours(),\n minute: date.getMinutes(),\n second: date.getSeconds(),\n millisecond: date.getMilliseconds()\n });\n $timepicker.$build();\n } else if (!$timepicker.$isBuilt) {\n $timepicker.$build();\n }\n };\n\n $timepicker.select = function(date, index, keep) {\n // console.warn('$timepicker.select', date, scope.$mode);\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) controller.$dateValue = new Date(1970, 0, 1);\n if (!angular.isDate(date)) date = new Date(date);\n if (index === 0) controller.$dateValue.setHours(date.getHours());\n else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes());\n else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds());\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function() {\n $timepicker.hide(true);\n });\n }\n };\n\n $timepicker.switchMeridian = function(date) {\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n return;\n }\n var hours = (date || controller.$dateValue).getHours();\n controller.$dateValue.setHours(hours < 12 ? hours + 12 : hours - 12);\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n };\n\n // Protected methods\n\n $timepicker.$build = function() {\n // console.warn('$timepicker.$build() viewDate=%o', viewDate);\n var i, midIndex = scope.midIndex = parseInt(options.length / 2, 10);\n var hours = [],\n hour;\n for (i = 0; i < options.length; i++) {\n hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep);\n hours.push({\n date: hour,\n label: formatDate(hour, hoursFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(hour, 0),\n disabled: $timepicker.$isDisabled(hour, 0)\n });\n }\n var minutes = [],\n minute;\n for (i = 0; i < options.length; i++) {\n minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep);\n minutes.push({\n date: minute,\n label: formatDate(minute, minutesFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(minute, 1),\n disabled: $timepicker.$isDisabled(minute, 1)\n });\n }\n var seconds = [],\n second;\n for (i = 0; i < options.length; i++) {\n second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep);\n seconds.push({\n date: second,\n label: formatDate(second, secondsFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(second, 2),\n disabled: $timepicker.$isDisabled(second, 2)\n });\n }\n\n var rows = [];\n for (i = 0; i < options.length; i++) {\n if (showSeconds) {\n rows.push([hours[i], minutes[i], seconds[i]]);\n } else {\n rows.push([hours[i], minutes[i]]);\n }\n }\n scope.rows = rows;\n scope.showSeconds = showSeconds;\n scope.showAM = showAM;\n scope.isAM = ($timepicker.$date || hours[midIndex].date).getHours() < 12;\n scope.timeSeparator = timeSeparator;\n $timepicker.$isBuilt = true;\n };\n\n $timepicker.$isSelected = function(date, index) {\n if (!$timepicker.$date) return false;\n else if (index === 0) {\n return date.getHours() === $timepicker.$date.getHours();\n } else if (index === 1) {\n return date.getMinutes() === $timepicker.$date.getMinutes();\n } else if (index === 2) {\n return date.getSeconds() === $timepicker.$date.getSeconds();\n }\n };\n\n $timepicker.$isDisabled = function(date, index) {\n var selectedTime;\n if (index === 0) {\n selectedTime = date.getTime() + viewDate.minute * 6e4 + viewDate.second * 1e3;\n } else if (index === 1) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.second * 1e3;\n } else if (index === 2) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.minute * 6e4;\n }\n return selectedTime < options.minTime * 1 || selectedTime > options.maxTime * 1;\n };\n\n scope.$arrowAction = function(value, index) {\n if (options.arrowBehavior === 'picker') {\n $timepicker.$setTimeByStep(value, index);\n } else {\n $timepicker.$moveIndex(value, index);\n }\n };\n\n $timepicker.$setTimeByStep = function(value, index) {\n var newDate = new Date($timepicker.$date || startDate);\n var hours = newDate.getHours();\n var minutes = newDate.getMinutes();\n var seconds = newDate.getSeconds();\n if (index === 0) {\n newDate.setHours(hours - (parseInt(options.hourStep, 10) * value));\n } else if (index === 1) {\n newDate.setMinutes(minutes - (parseInt(options.minuteStep, 10) * value));\n } else if (index === 2) {\n newDate.setSeconds(seconds - (parseInt(options.secondStep, 10) * value));\n }\n $timepicker.select(newDate, index, true);\n };\n\n $timepicker.$moveIndex = function(value, index) {\n var targetDate;\n if (index === 0) {\n targetDate = new Date(1970, 0, 1, viewDate.hour + (value * options.length), viewDate.minute, viewDate.second);\n angular.extend(viewDate, {\n hour: targetDate.getHours()\n });\n } else if (index === 1) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute + (value * options.length * options.minuteStep), viewDate.second);\n angular.extend(viewDate, {\n minute: targetDate.getMinutes()\n });\n } else if (index === 2) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute, viewDate.second + (value * options.length * options.secondStep));\n angular.extend(viewDate, {\n second: targetDate.getSeconds()\n });\n }\n $timepicker.$build();\n };\n\n $timepicker.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n if (evt.target.nodeName.toLowerCase() !== 'input') evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $timepicker.$onKeyDown = function(evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Close on enter\n if (evt.keyCode === 13) {\n $timepicker.hide(true);\n return;\n }\n\n // Navigate with keyboard\n var newDate = new Date($timepicker.$date);\n var hours = newDate.getHours(),\n hoursLength = formatDate(newDate, hoursFormat).length;\n var minutes = newDate.getMinutes(),\n minutesLength = formatDate(newDate, minutesFormat).length;\n var seconds = newDate.getSeconds(),\n secondsLength = formatDate(newDate, secondsFormat).length;\n var sepLength = 1;\n var lateralMove = /(37|39)/.test(evt.keyCode);\n var count = 2 + showSeconds * 1 + showAM * 1;\n\n // Navigate indexes (left, right)\n if (lateralMove) {\n if (evt.keyCode === 37) selectedIndex = selectedIndex < 1 ? count - 1 : selectedIndex - 1;\n else if (evt.keyCode === 39) selectedIndex = selectedIndex < count - 1 ? selectedIndex + 1 : 0;\n }\n\n // Update values (up, down)\n var selectRange = [0, hoursLength];\n var incr = 0;\n if (evt.keyCode === 38) incr = -1;\n if (evt.keyCode === 40) incr = +1;\n var isSeconds = selectedIndex === 2 && showSeconds;\n var isMeridian = selectedIndex === 2 && !showSeconds || selectedIndex === 3 && showSeconds;\n if (selectedIndex === 0) {\n newDate.setHours(hours + incr * parseInt(options.hourStep, 10));\n // re-calculate hours length because we have changed hours value\n hoursLength = formatDate(newDate, hoursFormat).length;\n selectRange = [0, hoursLength];\n } else if (selectedIndex === 1) {\n newDate.setMinutes(minutes + incr * parseInt(options.minuteStep, 10));\n // re-calculate minutes length because we have changes minutes value\n minutesLength = formatDate(newDate, minutesFormat).length;\n selectRange = [hoursLength + sepLength, minutesLength];\n } else if (isSeconds) {\n newDate.setSeconds(seconds + incr * parseInt(options.secondStep, 10));\n // re-calculate seconds length because we have changes seconds value\n secondsLength = formatDate(newDate, secondsFormat).length;\n selectRange = [hoursLength + sepLength + minutesLength + sepLength, secondsLength];\n } else if (isMeridian) {\n if (!lateralMove) $timepicker.switchMeridian();\n selectRange = [hoursLength + sepLength + minutesLength + sepLength + (secondsLength + sepLength) * showSeconds, 2];\n }\n $timepicker.select(newDate, selectedIndex, true);\n createSelection(selectRange[0], selectRange[1]);\n parentScope.$digest();\n };\n\n // Private\n\n function createSelection(start, length) {\n var end = start + length;\n if (element[0].createTextRange) {\n var selRange = element[0].createTextRange();\n selRange.collapse(true);\n selRange.moveStart('character', start);\n selRange.moveEnd('character', end);\n selRange.select();\n } else if (element[0].setSelectionRange) {\n element[0].setSelectionRange(start, end);\n } else if (angular.isUndefined(element[0].selectionStart)) {\n element[0].selectionStart = start;\n element[0].selectionEnd = end;\n }\n }\n\n function focusElement() {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $timepicker.init;\n $timepicker.init = function() {\n if (isNative && options.useNative) {\n element.prop('type', 'time');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $timepicker.destroy;\n $timepicker.destroy = function() {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $timepicker.show;\n $timepicker.show = function() {\n if((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n $timepicker.$element && $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n element && element.on('keydown', $timepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $timepicker.hide;\n $timepicker.hide = function(blur) {\n if (!$timepicker.$isShown) return;\n $timepicker.$element && $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n element && element.off('keydown', $timepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $timepicker;\n\n }\n\n timepickerFactory.defaults = defaults;\n return timepickerFactory;\n\n };\n\n })\n\n\n .directive('bsTimepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) {\n\n var defaults = $timepicker.defaults;\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'roundDisplay'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!timepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i);\n newValue === true ? timepicker.show() : timepicker.hide();\n });\n\n // Initialize timepicker\n if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm';\n var timepicker = $timepicker(element, controller, options);\n options = timepicker.$options;\n\n var lang = options.lang;\n var formatDate = function(date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n // Initialize parser\n var dateParser = $dateParser({\n format: options.timeFormat,\n lang: lang\n });\n\n // Observe attributes for changes\n angular.forEach(['minTime', 'maxTime'], function(key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) {\n timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue);\n !isNaN(timepicker.$options[key]) && timepicker.$build();\n validateAgainstMinMaxTime(controller.$dateValue);\n });\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue, controller.$dateValue);\n timepicker.update(controller.$dateValue);\n }, true);\n\n function validateAgainstMinMaxTime(parsedTime) {\n if (!angular.isDate(parsedTime)) return;\n var isMinValid = isNaN(options.minTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) >= options.minTime;\n var isMaxValid = isNaN(options.maxTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) <= options.maxTime;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (!isValid) {\n return;\n }\n controller.$dateValue = parsedTime;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function(viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n controller.$setValidity('date', true);\n return null;\n }\n var parsedTime = angular.isDate(viewValue) ? viewValue : dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedTime || isNaN(parsedTime.getTime())) {\n controller.$setValidity('date', false);\n // Return undefined, causes ngModelController to\n // invalidate model value\n return undefined;\n } else {\n validateAgainstMinMaxTime(parsedTime);\n }\n\n if (options.timeType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true);\n return formatDate(date, options.modelTimeFormat || options.timeFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.timeType === 'number') {\n return date.getTime();\n } else if (options.timeType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.timeType === 'iso') {\n return date.toISOString();\n } else {\n return new Date(date);\n }\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.timeType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelTimeFormat);\n } else if (options.timeType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if(isNaN(date.getTime())) date = new Date(new Date().setMinutes(0) + 36e5);\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getTimeFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function() {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getTimeFormattedString());\n };\n\n function getTimeFormattedString() {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.timeFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (timepicker) timepicker.destroy();\n options = null;\n timepicker = null;\n });\n\n }\n };\n\n });\n\n// Source: tab/tab.js\nangular.module('mgcrea.ngStrap.tab', [])\n\n .provider('$tab', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n template: 'tab/tab.tpl.html',\n navClass: 'nav-tabs',\n activeClass: 'active'\n };\n\n var controller = this.controller = function($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'navClass', 'activeClass'], function(key) {\n if(angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // Publish options on scope\n $scope.$navClass = self.$options.navClass;\n $scope.$activeClass = self.$options.activeClass;\n\n self.$panes = $scope.$panes = [];\n\n // Please use $activePaneChangeListeners if you use `bsActivePane`\n // Because we removed `ngModel` as default, we rename viewChangeListeners to\n // activePaneChangeListeners to make more sense.\n self.$activePaneChangeListeners = self.$viewChangeListeners = [];\n\n self.$push = function(pane) {\n if(angular.isUndefined(self.$panes.$active)) {\n $scope.$setActive(pane.name || 0);\n }\n self.$panes.push(pane);\n };\n\n self.$remove = function(pane) {\n var index = self.$panes.indexOf(pane);\n var active = self.$panes.$active;\n var activeIndex;\n if(angular.isString(active)) {\n activeIndex = self.$panes.map(function(pane) {\n return pane.name;\n }).indexOf(active);\n } else {\n activeIndex = self.$panes.$active;\n }\n\n // remove pane from $panes array\n self.$panes.splice(index, 1);\n\n if (index < activeIndex) {\n // we removed a pane before the active pane, so we need to\n // decrement the active pane index\n activeIndex--;\n }\n else if (index === activeIndex && activeIndex === self.$panes.length) {\n // we remove the active pane and it was the one at the end,\n // so select the previous one\n activeIndex--;\n }\n if(activeIndex >= 0 && activeIndex < self.$panes.length) {\n self.$setActive(self.$panes[activeIndex].name || activeIndex);\n } else {\n self.$setActive();\n }\n };\n\n self.$setActive = $scope.$setActive = function(value) {\n self.$panes.$active = value;\n self.$activePaneChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n self.$isActive = $scope.$isActive = function($pane, $index) {\n return self.$panes.$active === $pane.name || self.$panes.$active === $index;\n };\n\n };\n\n this.$get = function() {\n var $tab = {};\n $tab.defaults = defaults;\n $tab.controller = controller;\n return $tab;\n };\n\n })\n\n .directive('bsTabs', function($window, $animate, $tab, $parse) {\n\n var defaults = $tab.defaults;\n\n return {\n require: ['?ngModel', 'bsTabs'],\n transclude: true,\n scope: true,\n controller: ['$scope', '$element', '$attrs', $tab.controller],\n templateUrl: function(element, attr) {\n return attr.template || defaults.template;\n },\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // 'ngModel' does interfere with form validation\n // and status, use `bsActivePane` instead to avoid it\n if(ngModelCtrl) {\n\n // Update the modelValue following\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n ngModelCtrl.$setViewValue(bsTabsCtrl.$panes.$active);\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n bsTabsCtrl.$setActive(modelValue);\n return modelValue;\n });\n\n }\n\n if (attrs.bsActivePane) {\n // adapted from angularjs ngModelController bindings\n // https://github.com/angular/angular.js/blob/v1.3.1/src%2Fng%2Fdirective%2Finput.js#L1730\n var parsedBsActivePane = $parse(attrs.bsActivePane);\n\n // Update bsActivePane value with change\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n parsedBsActivePane.assign(scope, bsTabsCtrl.$panes.$active);\n });\n\n // watch bsActivePane for value changes\n scope.$watch(attrs.bsActivePane, function(newValue, oldValue) {\n bsTabsCtrl.$setActive(newValue);\n }, true);\n }\n }\n };\n\n })\n\n .directive('bsPane', function($window, $animate, $sce) {\n\n return {\n require: ['^?ngModel', '^bsTabs'],\n scope: true,\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // Add base class\n element.addClass('tab-pane');\n\n // Observe title attribute for change\n attrs.$observe('title', function(newValue, oldValue) {\n scope.title = $sce.trustAsHtml(newValue);\n });\n\n // Save tab name into scope\n scope.name = attrs.name;\n\n // Add animation class\n if(bsTabsCtrl.$options.animation) {\n element.addClass(bsTabsCtrl.$options.animation);\n }\n\n attrs.$observe('disabled', function(newValue, oldValue) {\n scope.disabled = scope.$eval(newValue);\n });\n\n // Push pane to parent bsTabs controller\n bsTabsCtrl.$push(scope);\n\n // remove pane from tab controller when pane is destroyed\n scope.$on('$destroy', function() {\n bsTabsCtrl.$remove(scope);\n });\n\n function render() {\n var index = bsTabsCtrl.$panes.indexOf(scope);\n $animate[bsTabsCtrl.$isActive(scope, index) ? 'addClass' : 'removeClass'](element, bsTabsCtrl.$options.activeClass);\n }\n\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n render();\n });\n render();\n\n }\n };\n\n });\n\n// Source: select/select.js\nangular.module('mgcrea.ngStrap.select', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$select', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'select',\n prefixEvent: '$select',\n placement: 'bottom-left',\n templateUrl: 'select/select.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n multiple: false,\n allNoneButtons: false,\n sort: true,\n caretHtml: ' ',\n placeholder: 'Choose among the following...',\n allText: 'All',\n noneText: 'None',\n maxLength: 3,\n maxLengthHtml: 'selected',\n iconCheckmark: 'glyphicon glyphicon-ok'\n };\n\n this.$get = function($window, $document, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n\n function SelectFactory(element, controller, config) {\n\n var $select = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $select = $tooltip(element, options);\n var scope = $select.$scope;\n\n scope.$matches = [];\n if (options.multiple) {\n scope.$activeIndex = [];\n }\n else {\n scope.$activeIndex = -1;\n }\n scope.$isMultiple = options.multiple;\n scope.$showAllNoneButtons = options.allNoneButtons && options.multiple;\n scope.$iconCheckmark = options.iconCheckmark;\n scope.$allText = options.allText;\n scope.$noneText = options.noneText;\n\n scope.$activate = function(index) {\n scope.$$postDigest(function() {\n $select.activate(index);\n });\n };\n\n scope.$select = function(index, evt) {\n scope.$$postDigest(function() {\n $select.select(index);\n });\n };\n\n scope.$isVisible = function() {\n return $select.$isVisible();\n };\n\n scope.$isActive = function(index) {\n return $select.$isActive(index);\n };\n\n scope.$selectAll = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (!scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n scope.$selectNone = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n // Public methods\n\n $select.update = function(matches) {\n scope.$matches = matches;\n $select.$updateActiveIndex();\n };\n\n $select.activate = function(index) {\n if(options.multiple) {\n $select.$isActive(index) ? scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1) : scope.$activeIndex.push(index);\n if(options.sort) scope.$activeIndex.sort(function(a, b) { return a - b; }); // use numeric sort instead of default sort\n } else {\n scope.$activeIndex = index;\n }\n return scope.$activeIndex;\n };\n\n $select.select = function(index) {\n var value = scope.$matches[index].value;\n scope.$apply(function() {\n $select.activate(index);\n if(options.multiple) {\n controller.$setViewValue(scope.$activeIndex.map(function(index) {\n if (angular.isUndefined(scope.$matches[index])) {\n return null;\n }\n return scope.$matches[index].value;\n }));\n } else {\n controller.$setViewValue(value);\n // Hide if single select\n $select.hide();\n }\n });\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $select);\n };\n\n // Protected methods\n\n $select.$updateActiveIndex = function() {\n if(options.multiple) {\n if(angular.isArray(controller.$modelValue)) {\n scope.$activeIndex = controller.$modelValue.map(function(value) {\n return $select.$getIndex(value);\n });\n } else {\n scope.$activeIndex = [];\n }\n } else {\n if(angular.isDefined(controller.$modelValue) && scope.$matches.length) {\n scope.$activeIndex = $select.$getIndex(controller.$modelValue);\n } else {\n scope.$activeIndex = -1;\n }\n }\n };\n\n $select.$isVisible = function() {\n if(!options.minLength || !controller) {\n return scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && controller.$viewValue.length >= options.minLength;\n };\n\n $select.$isActive = function(index) {\n if(options.multiple) {\n return scope.$activeIndex.indexOf(index) !== -1;\n } else {\n return scope.$activeIndex === index;\n }\n };\n\n $select.$getIndex = function(value) {\n var l = scope.$matches.length, i = l;\n if(!l) return;\n for(i = l; i--;) {\n if(scope.$matches[i].value === value) break;\n }\n if(i < 0) return;\n return i;\n };\n\n $select.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if(isTouch) {\n var targetEl = angular.element(evt.target);\n targetEl.triggerHandler('click');\n }\n };\n\n $select.$onKeyDown = function(evt) {\n if (!/(9|13|38|40)/.test(evt.keyCode)) return;\n // Let tab propagate\n if (evt.keyCode !== 9) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // release focus on tab\n if (options.multiple && evt.keyCode === 9) {\n return $select.hide();\n }\n\n // Select with enter\n if(!options.multiple && (evt.keyCode === 13 || evt.keyCode === 9)) {\n return $select.select(scope.$activeIndex);\n }\n\n if (!options.multiple) {\n // Navigate with keyboard\n if(evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if(evt.keyCode === 38 && scope.$activeIndex < 0) scope.$activeIndex = scope.$matches.length - 1;\n else if(evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if(angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n }\n };\n\n $select.$isIE = function() {\n var ua = $window.navigator.userAgent;\n return ua.indexOf('MSIE ') > 0 || ua.indexOf('Trident/') > 0 || ua.indexOf('Edge/') > 0;\n };\n\n $select.$selectScrollFix = function(e) {\n if ($document[0].activeElement.tagName === 'UL') {\n e.preventDefault();\n e.stopImmediatePropagation();\n e.target.focus();\n }\n };\n\n // Overrides\n\n var _show = $select.show;\n $select.show = function() {\n _show();\n if(options.multiple) {\n $select.$element.addClass('select-multiple');\n }\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n $select.$element.on(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if(options.keyboard) {\n element.on('keydown', $select.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $select.hide;\n $select.hide = function() {\n if(!options.multiple && angular.isUndefined(controller.$modelValue)) {\n scope.$activeIndex = -1;\n }\n $select.$element.off(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if(options.keyboard) {\n element.off('keydown', $select.$onKeyDown);\n }\n _hide(true);\n };\n\n return $select;\n\n }\n\n SelectFactory.defaults = defaults;\n return SelectFactory;\n\n };\n\n })\n\n .directive('bsSelect', function($window, $parse, $q, $select, $parseOptions) {\n\n var defaults = $select.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope, placeholder: defaults.placeholder};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'allNoneButtons', 'sort'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Only parse data-multiple. Angular sets existence attributes to true (multiple/required/etc), they apply this\n // to data-multiple as well for some reason, so we'll parse this ourselves and disregard multiple\n var dataMultiple = element.attr('data-multiple');\n if(angular.isDefined(dataMultiple)) {\n if(falseValueRegExp.test(dataMultiple))\n options.multiple = false;\n else\n options.multiple = dataMultiple;\n }\n\n // Add support for select markup\n if(element[0].nodeName.toLowerCase() === 'select') {\n var inputEl = element;\n inputEl.css('display', 'none');\n element = angular.element('');\n inputEl.after(element);\n }\n\n // Build proper bsOptions\n var parsedOptions = $parseOptions(attr.bsOptions);\n\n // Initialize select\n var select = $select(element, controller, options);\n\n if (select.$isIE()) {\n element[0].addEventListener('blur', select.$selectScrollFix);\n }\n\n // Watch bsOptions values before filtering for changes\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').trim();\n scope.$watchCollection(watchedOptions, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller)\n .then(function(values) {\n select.update(values);\n controller.$render();\n });\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue);\n select.$updateActiveIndex();\n controller.$render();\n }, true);\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var selected, index;\n if(options.multiple && angular.isArray(controller.$modelValue)) {\n selected = controller.$modelValue.map(function(value) {\n index = select.$getIndex(value);\n return angular.isDefined(index) ? select.$scope.$matches[index].label : false;\n }).filter(angular.isDefined);\n if(selected.length > (options.maxLength || defaults.maxLength)) {\n selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml);\n } else {\n selected = selected.join(', ');\n }\n } else {\n index = select.$getIndex(controller.$modelValue);\n selected = angular.isDefined(index) ? select.$scope.$matches[index].label : false;\n }\n element.html((selected ? selected : options.placeholder) + (options.caretHtml ? options.caretHtml : defaults.caretHtml));\n };\n\n if(options.multiple){\n controller.$isEmpty = function(value){\n return !value || value.length === 0;\n };\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (select) select.destroy();\n options = null;\n select = null;\n });\n\n }\n };\n\n });\n\n// Source: scrollspy/scrollspy.js\nangular.module('mgcrea.ngStrap.scrollspy', ['mgcrea.ngStrap.helpers.debounce', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$scrollspy', function() {\n\n // Pool of registered spies\n var spies = this.$$spies = {};\n\n var defaults = this.defaults = {\n debounce: 150,\n throttle: 100,\n offset: 100\n };\n\n this.$get = function($window, $document, $rootScope, dimensions, debounce, throttle) {\n\n var windowEl = angular.element($window);\n var docEl = angular.element($document.prop('documentElement'));\n var bodyEl = angular.element($window.document.body);\n\n // Helper functions\n\n function nodeName(element, name) {\n return element[0].nodeName && element[0].nodeName.toLowerCase() === name.toLowerCase();\n }\n\n function ScrollSpyFactory(config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n if(!options.element) options.element = bodyEl;\n var isWindowSpy = nodeName(options.element, 'body');\n var scrollEl = isWindowSpy ? windowEl : options.element;\n var scrollId = isWindowSpy ? 'window' : options.id;\n\n // Use existing spy\n if(spies[scrollId]) {\n spies[scrollId].$$count++;\n return spies[scrollId];\n }\n\n var $scrollspy = {};\n\n // Private vars\n var unbindViewContentLoaded, unbindIncludeContentLoaded;\n var trackedElements = $scrollspy.$trackedElements = [];\n var sortedElements = [];\n var activeTarget;\n var debouncedCheckPosition;\n var throttledCheckPosition;\n var debouncedCheckOffsets;\n var viewportHeight;\n var scrollTop;\n\n $scrollspy.init = function() {\n\n // Setup internal ref counter\n this.$$count = 1;\n\n // Bind events\n debouncedCheckPosition = debounce(this.checkPosition, options.debounce);\n throttledCheckPosition = throttle(this.checkPosition, options.throttle);\n scrollEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', debouncedCheckPosition);\n scrollEl.on('scroll', throttledCheckPosition);\n\n debouncedCheckOffsets = debounce(this.checkOffsets, options.debounce);\n unbindViewContentLoaded = $rootScope.$on('$viewContentLoaded', debouncedCheckOffsets);\n unbindIncludeContentLoaded = $rootScope.$on('$includeContentLoaded', debouncedCheckOffsets);\n debouncedCheckOffsets();\n\n // Register spy for reuse\n if(scrollId) {\n spies[scrollId] = $scrollspy;\n }\n\n };\n\n $scrollspy.destroy = function() {\n\n // Check internal ref counter\n this.$$count--;\n if(this.$$count > 0) {\n return;\n }\n\n // Unbind events\n scrollEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', debouncedCheckPosition);\n scrollEl.off('scroll', throttledCheckPosition);\n unbindViewContentLoaded();\n unbindIncludeContentLoaded();\n if (scrollId) {\n delete spies[scrollId];\n }\n };\n\n $scrollspy.checkPosition = function() {\n\n // Not ready yet\n if(!sortedElements.length) return;\n\n // Calculate the scroll position\n scrollTop = (isWindowSpy ? $window.pageYOffset : scrollEl.prop('scrollTop')) || 0;\n\n // Calculate the viewport height for use by the components\n viewportHeight = Math.max($window.innerHeight, docEl.prop('clientHeight'));\n\n // Activate first element if scroll is smaller\n if(scrollTop < sortedElements[0].offsetTop && activeTarget !== sortedElements[0].target) {\n return $scrollspy.$activateElement(sortedElements[0]);\n }\n\n // Activate proper element\n for (var i = sortedElements.length; i--;) {\n if(angular.isUndefined(sortedElements[i].offsetTop) || sortedElements[i].offsetTop === null) continue;\n if(activeTarget === sortedElements[i].target) continue;\n if(scrollTop < sortedElements[i].offsetTop) continue;\n if(sortedElements[i + 1] && scrollTop > sortedElements[i + 1].offsetTop) continue;\n return $scrollspy.$activateElement(sortedElements[i]);\n }\n\n };\n\n $scrollspy.checkPositionWithEventLoop = function() {\n // IE 9 throws an error if we use 'this' instead of '$scrollspy'\n // in this setTimeout call\n setTimeout($scrollspy.checkPosition, 1);\n };\n\n // Protected methods\n\n $scrollspy.$activateElement = function(element) {\n if(activeTarget) {\n var activeElement = $scrollspy.$getTrackedElement(activeTarget);\n if(activeElement) {\n activeElement.source.removeClass('active');\n if(nodeName(activeElement.source, 'li') && nodeName(activeElement.source.parent().parent(), 'li')) {\n activeElement.source.parent().parent().removeClass('active');\n }\n }\n }\n activeTarget = element.target;\n element.source.addClass('active');\n if(nodeName(element.source, 'li') && nodeName(element.source.parent().parent(), 'li')) {\n element.source.parent().parent().addClass('active');\n }\n };\n\n $scrollspy.$getTrackedElement = function(target) {\n return trackedElements.filter(function(obj) {\n return obj.target === target;\n })[0];\n };\n\n // Track offsets behavior\n\n $scrollspy.checkOffsets = function() {\n\n angular.forEach(trackedElements, function(trackedElement) {\n var targetElement = document.querySelector(trackedElement.target);\n trackedElement.offsetTop = targetElement ? dimensions.offset(targetElement).top : null;\n if(options.offset && trackedElement.offsetTop !== null) trackedElement.offsetTop -= options.offset * 1;\n });\n\n sortedElements = trackedElements\n .filter(function(el) {\n return el.offsetTop !== null;\n })\n .sort(function(a, b) {\n return a.offsetTop - b.offsetTop;\n });\n\n debouncedCheckPosition();\n\n };\n\n $scrollspy.trackElement = function(target, source) {\n trackedElements.push({target: target, source: source});\n };\n\n $scrollspy.untrackElement = function(target, source) {\n var toDelete;\n for (var i = trackedElements.length; i--;) {\n if(trackedElements[i].target === target && trackedElements[i].source === source) {\n toDelete = i;\n break;\n }\n }\n trackedElements = trackedElements.splice(toDelete, 1);\n };\n\n $scrollspy.activate = function(i) {\n trackedElements[i].addClass('active');\n };\n\n // Initialize plugin\n\n $scrollspy.init();\n return $scrollspy;\n\n }\n\n return ScrollSpyFactory;\n\n };\n\n })\n\n .directive('bsScrollspy', function($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'EAC',\n link: function postLink(scope, element, attr) {\n\n var options = {scope: scope};\n angular.forEach(['offset', 'target'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n var scrollspy = $scrollspy(options);\n scrollspy.trackElement(options.target, element);\n\n scope.$on('$destroy', function() {\n if (scrollspy) {\n scrollspy.untrackElement(options.target, element);\n scrollspy.destroy();\n }\n options = null;\n scrollspy = null;\n });\n\n }\n };\n\n })\n\n\n .directive('bsScrollspyList', function($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'A',\n compile: function postLink(element, attr) {\n var children = element[0].querySelectorAll('li > a[href]');\n angular.forEach(children, function(child) {\n var childEl = angular.element(child);\n childEl.parent().attr('bs-scrollspy', '').attr('data-target', childEl.attr('href'));\n });\n }\n\n };\n\n });\n\n// Source: popover/popover.js\nangular.module('mgcrea.ngStrap.popover', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$popover', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n // uncommenting the next two lines will break backwards compatability\n // prefixClass: 'popover',\n // prefixEvent: 'popover',\n container: false,\n target: false,\n placement: 'right',\n templateUrl: 'popover/popover.tpl.html',\n contentTemplate: false,\n trigger: 'click',\n keyboard: true,\n html: false,\n title: '',\n content: '',\n delay: 0,\n autoClose: false\n };\n\n this.$get = function($tooltip) {\n\n function PopoverFactory(element, config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n var $popover = $tooltip(element, options);\n\n // Support scope as string options [/*title, */content]\n if(options.content) {\n $popover.$scope.content = options.content;\n }\n\n return $popover;\n\n }\n\n return PopoverFactory;\n\n };\n\n })\n\n .directive('bsPopover', function($window, $sce, $popover) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoClose'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if(angular.isDefined(dataTarget)) {\n if(falseValueRegExp.test(dataTarget))\n options.target = false;\n else\n options.target = dataTarget;\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n angular.isDefined(oldValue) && requestAnimationFrame(function() {\n popover && popover.$applyPlacement();\n });\n });\n });\n\n // Support scope as an object\n attr.bsPopover && scope.$watch(attr.bsPopover, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n angular.isDefined(oldValue) && requestAnimationFrame(function() {\n popover && popover.$applyPlacement();\n });\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if(!popover || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i);\n newValue === true ? popover.show() : popover.hide();\n });\n\n // Viewport support\n attr.viewport && scope.$watch(attr.viewport, function (newValue) {\n if(!popover || !angular.isDefined(newValue)) return;\n popover.setViewport(newValue);\n });\n\n // Initialize popover\n var popover = $popover(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (popover) popover.destroy();\n options = null;\n popover = null;\n });\n\n }\n };\n\n });\n\n// Source: navbar/navbar.js\nangular.module('mgcrea.ngStrap.navbar', [])\n\n .provider('$navbar', function() {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n routeAttr: 'data-match-route',\n strict: false\n };\n\n this.$get = function() {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsNavbar', function($window, $location, $navbar) {\n\n var defaults = $navbar.defaults;\n\n return {\n restrict: 'A',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = angular.copy(defaults);\n angular.forEach(Object.keys(defaults), function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Watch for the $location\n scope.$watch(function() {\n\n return $location.path();\n\n }, function(newValue, oldValue) {\n\n var liElements = element[0].querySelectorAll('li[' + options.routeAttr + ']');\n\n angular.forEach(liElements, function(li) {\n\n var liElement = angular.element(li);\n var pattern = liElement.attr(options.routeAttr).replace('/', '\\\\/');\n if(options.strict) {\n pattern = '^' + pattern + '$';\n }\n var regexp = new RegExp(pattern, 'i');\n\n if(regexp.test(newValue)) {\n liElement.addClass(options.activeClass);\n } else {\n liElement.removeClass(options.activeClass);\n }\n\n });\n\n });\n\n }\n\n };\n\n });\n\n// Source: modal/modal.js\nangular.module('mgcrea.ngStrap.modal', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$modal', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n backdropAnimation: 'am-fade',\n prefixClass: 'modal',\n prefixEvent: 'modal',\n placement: 'top',\n templateUrl: 'modal/modal.tpl.html',\n template: '',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) {\n\n var forEach = angular.forEach;\n var trim = String.prototype.trim;\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n var bodyElement = angular.element($window.document.body);\n\n function ModalFactory(config) {\n\n var $modal = {};\n\n // Common vars\n var options = $modal.$options = angular.extend({}, defaults, config);\n var promise = $modal.$promise = $bsCompiler.compile(options);\n var scope = $modal.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n if(!options.element && !options.container) {\n options.container = 'body';\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $modal.$id = options.id || options.element && options.element.attr('id') || '';\n\n // Support scope as string options\n forEach(['title', 'content'], function(key) {\n if(options[key]) scope[key] = $sce.trustAsHtml(options[key]);\n });\n\n // Provide scope helpers\n scope.$hide = function() {\n scope.$$postDigest(function() {\n $modal.hide();\n });\n };\n scope.$show = function() {\n scope.$$postDigest(function() {\n $modal.show();\n });\n };\n scope.$toggle = function() {\n scope.$$postDigest(function() {\n $modal.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $modal.$isShown = scope.$isShown = false;\n\n // Fetch, compile then initialize modal\n var compileData, modalElement, modalScope;\n var backdropElement = angular.element('
');\n backdropElement.css({position:'fixed', top:'0px', left:'0px', bottom:'0px', right:'0px', 'z-index': 1038});\n promise.then(function(data) {\n compileData = data;\n $modal.init();\n });\n\n $modal.init = function() {\n\n // Options: show\n if(options.show) {\n scope.$$postDigest(function() {\n $modal.show();\n });\n }\n\n };\n\n $modal.destroy = function() {\n\n // Remove element\n destroyModalElement();\n\n // remove backdrop element\n if(backdropElement) {\n backdropElement.remove();\n backdropElement = null;\n }\n\n // Destroy scope\n scope.$destroy();\n };\n\n $modal.show = function() {\n if($modal.$isShown) return;\n\n var parent, after;\n if(angular.isElement(options.container)) {\n parent = options.container;\n after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null;\n } else {\n if (options.container) {\n parent = findElement(options.container);\n after = parent[0] && parent[0].lastChild ? angular.element(parent[0].lastChild) : null;\n } else {\n parent = null;\n after = options.element;\n }\n }\n\n // destroy any existing modal elements\n if(modalElement) destroyModalElement();\n\n // create a new scope, so we can destroy it and all child scopes\n // when destroying the modal element\n modalScope = $modal.$scope.$new();\n // Fetch a cloned element linked from template (noop callback is required)\n modalElement = $modal.$element = compileData.link(modalScope, function(clonedElement, scope) {});\n\n if(scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) {\n return;\n }\n\n // Set the initial positioning.\n modalElement.css({display: 'block'}).addClass(options.placement);\n\n // Options: animation\n if(options.animation) {\n if(options.backdrop) {\n backdropElement.addClass(options.backdropAnimation);\n }\n modalElement.addClass(options.animation);\n }\n\n if(options.backdrop) {\n $animate.enter(backdropElement, bodyElement, null);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.enter(modalElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(modalElement, parent, after).then(enterAnimateCallback);\n }\n\n $modal.$isShown = scope.$isShown = true;\n safeDigest(scope);\n // Focus once the enter-animation has started\n // Weird PhantomJS bug hack\n var el = modalElement[0];\n requestAnimationFrame(function() {\n el.focus();\n });\n\n bodyElement.addClass(options.prefixClass + '-open');\n if(options.animation) {\n bodyElement.addClass(options.prefixClass + '-with-' + options.animation);\n }\n\n // Bind events\n bindBackdropEvents();\n bindKeyboardEvents();\n };\n\n function enterAnimateCallback() {\n scope.$emit(options.prefixEvent + '.show', $modal);\n }\n\n $modal.hide = function() {\n if(!$modal.$isShown) return;\n\n if(scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) {\n return;\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.leave(modalElement, leaveAnimateCallback);\n } else {\n $animate.leave(modalElement).then(leaveAnimateCallback);\n }\n\n if(options.backdrop) {\n $animate.leave(backdropElement);\n }\n $modal.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n };\n\n function leaveAnimateCallback() {\n scope.$emit(options.prefixEvent + '.hide', $modal);\n bodyElement.removeClass(options.prefixClass + '-open');\n if(options.animation) {\n bodyElement.removeClass(options.prefixClass + '-with-' + options.animation);\n }\n }\n\n $modal.toggle = function() {\n\n $modal.$isShown ? $modal.hide() : $modal.show();\n\n };\n\n $modal.focus = function() {\n modalElement[0].focus();\n };\n\n // Protected methods\n\n $modal.$onKeyUp = function(evt) {\n\n if (evt.which === 27 && $modal.$isShown) {\n $modal.hide();\n evt.stopPropagation();\n }\n\n };\n\n function bindBackdropEvents() {\n if(options.backdrop) {\n modalElement.on('click', hideOnBackdropClick);\n backdropElement.on('click', hideOnBackdropClick);\n backdropElement.on('wheel', preventEventDefault);\n }\n }\n\n function unbindBackdropEvents() {\n if(options.backdrop) {\n modalElement.off('click', hideOnBackdropClick);\n backdropElement.off('click', hideOnBackdropClick);\n backdropElement.off('wheel', preventEventDefault);\n }\n }\n\n function bindKeyboardEvents() {\n if(options.keyboard) {\n modalElement.on('keyup', $modal.$onKeyUp);\n }\n }\n\n function unbindKeyboardEvents() {\n if(options.keyboard) {\n modalElement.off('keyup', $modal.$onKeyUp);\n }\n }\n\n // Private helpers\n\n function hideOnBackdropClick(evt) {\n if(evt.target !== evt.currentTarget) return;\n options.backdrop === 'static' ? $modal.focus() : $modal.hide();\n }\n\n function preventEventDefault(evt) {\n evt.preventDefault();\n }\n\n function destroyModalElement() {\n if($modal.$isShown && modalElement !== null) {\n // un-bind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n }\n\n if(modalScope) {\n modalScope.$destroy();\n modalScope = null;\n }\n\n if(modalElement) {\n modalElement.remove();\n modalElement = $modal.$element = null;\n }\n }\n\n return $modal;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return ModalFactory;\n\n };\n\n })\n\n .directive('bsModal', function($window, $sce, $modal) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsModal && scope.$watch(attr.bsModal, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize modal\n var modal = $modal(options);\n\n // Trigger\n element.on(attr.trigger || 'click', modal.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (modal) modal.destroy();\n options = null;\n modal = null;\n });\n\n }\n };\n\n });\n\n// Source: helpers/raf.js\n(angular.version.minor < 3 && angular.version.dot < 14) && angular.module('ng')\n\n.factory('$$rAF', function($window, $timeout) {\n\n var requestAnimationFrame = $window.requestAnimationFrame ||\n $window.webkitRequestAnimationFrame ||\n $window.mozRequestAnimationFrame;\n\n var cancelAnimationFrame = $window.cancelAnimationFrame ||\n $window.webkitCancelAnimationFrame ||\n $window.mozCancelAnimationFrame ||\n $window.webkitCancelRequestAnimationFrame;\n\n var rafSupported = !!requestAnimationFrame;\n var raf = rafSupported ?\n function(fn) {\n var id = requestAnimationFrame(fn);\n return function() {\n cancelAnimationFrame(id);\n };\n } :\n function(fn) {\n var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666\n return function() {\n $timeout.cancel(timer);\n };\n };\n\n raf.supported = rafSupported;\n\n return raf;\n\n});\n\n// .factory('$$animateReflow', function($$rAF, $document) {\n\n// var bodyEl = $document[0].body;\n\n// return function(fn) {\n// //the returned function acts as the cancellation function\n// return $$rAF(function() {\n// //the line below will force the browser to perform a repaint\n// //so that all the animated elements within the animation frame\n// //will be properly updated and drawn on screen. This is\n// //required to perform multi-class CSS based animations with\n// //Firefox. DO NOT REMOVE THIS LINE.\n// var a = bodyEl.offsetWidth + 1;\n// fn();\n// });\n// };\n\n// });\n\n// Source: helpers/parse-options.js\nangular.module('mgcrea.ngStrap.helpers.parseOptions', [])\n\n .provider('$parseOptions', function() {\n\n var defaults = this.defaults = {\n regexp: /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n };\n\n this.$get = function($parse, $q) {\n\n function ParseOptionsFactory(attr, config) {\n\n var $parseOptions = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n $parseOptions.$values = [];\n\n // Private vars\n var match, displayFn, valueName, keyName, groupByFn, valueFn, valuesFn;\n\n $parseOptions.init = function() {\n $parseOptions.$match = match = attr.match(options.regexp);\n displayFn = $parse(match[2] || match[1]),\n valueName = match[4] || match[6],\n keyName = match[5],\n groupByFn = $parse(match[3] || ''),\n valueFn = $parse(match[2] ? match[1] : valueName),\n valuesFn = $parse(match[7]);\n };\n\n $parseOptions.valuesFn = function(scope, controller) {\n return $q.when(valuesFn(scope, controller))\n .then(function(values) {\n if(!angular.isArray(values)) {\n values = [];\n }\n $parseOptions.$values = values.length ? parseValues(values, scope) : [];\n return $parseOptions.$values;\n });\n };\n\n $parseOptions.displayValue = function(modelValue) {\n var scope = {};\n scope[valueName] = modelValue;\n return displayFn(scope);\n };\n\n // Private functions\n\n function parseValues(values, scope) {\n return values.map(function(match, index) {\n var locals = {}, label, value;\n locals[valueName] = match;\n label = displayFn(scope, locals);\n value = valueFn(scope, locals);\n return {label: label, value: value, index: index};\n });\n }\n\n $parseOptions.init();\n return $parseOptions;\n\n }\n\n return ParseOptionsFactory;\n\n };\n\n });\n\n// Source: helpers/dimensions.js\nangular.module('mgcrea.ngStrap.helpers.dimensions', [])\n\n .factory('dimensions', function($document, $window) {\n\n var jqLite = angular.element;\n var fn = {};\n\n /**\n * Test the element nodeName\n * @param element\n * @param name\n */\n var nodeName = fn.nodeName = function(element, name) {\n return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase();\n };\n\n /**\n * Returns the element computed style\n * @param element\n * @param prop\n * @param extra\n */\n fn.css = function(element, prop, extra) {\n var value;\n if (element.currentStyle) { //IE\n value = element.currentStyle[prop];\n } else if (window.getComputedStyle) {\n value = window.getComputedStyle(element)[prop];\n } else {\n value = element.style[prop];\n }\n return extra === true ? parseFloat(value) || 0 : value;\n };\n\n /**\n * Provides read-only equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.offset = function(element) {\n var boxRect = element.getBoundingClientRect();\n var docElement = element.ownerDocument;\n return {\n width: boxRect.width || element.offsetWidth,\n height: boxRect.height || element.offsetHeight,\n top: boxRect.top + (window.pageYOffset || docElement.documentElement.scrollTop) - (docElement.documentElement.clientTop || 0),\n left: boxRect.left + (window.pageXOffset || docElement.documentElement.scrollLeft) - (docElement.documentElement.clientLeft || 0)\n };\n };\n \n /**\n * Provides set equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip\n * @url http://api.jquery.com/offset/\n * @param element\n * @param options\n * @param i\n */\n fn.setOffset = function (element, options, i) {\n var curPosition,\n curLeft,\n curCSSTop,\n curTop,\n curOffset,\n curCSSLeft,\n calculatePosition,\n position = fn.css(element, 'position'),\n curElem = angular.element(element),\n props = {};\n \n // Set position first, in-case top/left are set even on static elem\n if (position === 'static') {\n element.style.position = 'relative';\n }\n \n curOffset = fn.offset(element);\n curCSSTop = fn.css(element, 'top');\n curCSSLeft = fn.css(element, 'left');\n calculatePosition = (position === 'absolute' || position === 'fixed') && \n (curCSSTop + curCSSLeft).indexOf('auto') > -1;\n \n // Need to be able to calculate position if either\n // top or left is auto and position is either absolute or fixed\n if (calculatePosition) {\n curPosition = fn.position(element);\n curTop = curPosition.top;\n curLeft = curPosition.left;\n } else {\n curTop = parseFloat(curCSSTop) || 0;\n curLeft = parseFloat(curCSSLeft) || 0;\n }\n \n if (angular.isFunction(options)) {\n options = options.call(element, i, curOffset);\n }\n \n if (options.top !== null ) {\n props.top = (options.top - curOffset.top) + curTop;\n }\n if ( options.left !== null ) {\n props.left = (options.left - curOffset.left) + curLeft;\n }\n\n if ('using' in options) {\n options.using.call(curElem, props);\n } else {\n curElem.css({\n top: props.top + 'px',\n left: props.left + 'px'\n });\n }\n };\n\n /**\n * Provides read-only equivalent of jQuery's position function\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.position = function(element) {\n\n var offsetParentRect = {top: 0, left: 0},\n offsetParentElement,\n offset;\n\n // Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is it's only offset parent\n if (fn.css(element, 'position') === 'fixed') {\n\n // We assume that getBoundingClientRect is available when computed position is fixed\n offset = element.getBoundingClientRect();\n\n } else {\n\n // Get *real* offsetParentElement\n offsetParentElement = offsetParent(element);\n\n // Get correct offsets\n offset = fn.offset(element);\n if (!nodeName(offsetParentElement, 'html')) {\n offsetParentRect = fn.offset(offsetParentElement);\n }\n\n // Add offsetParent borders\n offsetParentRect.top += fn.css(offsetParentElement, 'borderTopWidth', true);\n offsetParentRect.left += fn.css(offsetParentElement, 'borderLeftWidth', true);\n }\n\n // Subtract parent offsets and element margins\n return {\n width: element.offsetWidth,\n height: element.offsetHeight,\n top: offset.top - offsetParentRect.top - fn.css(element, 'marginTop', true),\n left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true)\n };\n\n };\n\n /**\n * Returns the closest, non-statically positioned offsetParent of a given element\n * @required-by fn.position\n * @param element\n */\n var offsetParent = function offsetParentElement(element) {\n var docElement = element.ownerDocument;\n var offsetParent = element.offsetParent || docElement;\n if(nodeName(offsetParent, '#document')) return docElement.documentElement;\n while(offsetParent && !nodeName(offsetParent, 'html') && fn.css(offsetParent, 'position') === 'static') {\n offsetParent = offsetParent.offsetParent;\n }\n return offsetParent || docElement.documentElement;\n };\n\n /**\n * Provides equivalent of jQuery's height function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/height/\n * @param element\n * @param outer\n */\n fn.height = function(element, outer) {\n var value = element.offsetHeight;\n if(outer) {\n value += fn.css(element, 'marginTop', true) + fn.css(element, 'marginBottom', true);\n } else {\n value -= fn.css(element, 'paddingTop', true) + fn.css(element, 'paddingBottom', true) + fn.css(element, 'borderTopWidth', true) + fn.css(element, 'borderBottomWidth', true);\n }\n return value;\n };\n\n /**\n * Provides equivalent of jQuery's width function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/width/\n * @param element\n * @param outer\n */\n fn.width = function(element, outer) {\n var value = element.offsetWidth;\n if(outer) {\n value += fn.css(element, 'marginLeft', true) + fn.css(element, 'marginRight', true);\n } else {\n value -= fn.css(element, 'paddingLeft', true) + fn.css(element, 'paddingRight', true) + fn.css(element, 'borderLeftWidth', true) + fn.css(element, 'borderRightWidth', true);\n }\n return value;\n };\n\n return fn;\n\n });\n\n// Source: helpers/debounce.js\nangular.module('mgcrea.ngStrap.helpers.debounce', [])\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L693\n.factory('debounce', function($timeout) {\n return function(func, wait, immediate) {\n var timeout = null;\n return function() {\n var context = this,\n args = arguments,\n callNow = immediate && !timeout;\n if(timeout) {\n $timeout.cancel(timeout);\n }\n timeout = $timeout(function later() {\n timeout = null;\n if(!immediate) {\n func.apply(context, args);\n }\n }, wait, false);\n if(callNow) {\n func.apply(context, args);\n }\n return timeout;\n };\n };\n})\n\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L661\n.factory('throttle', function($timeout) {\n return function(func, wait, options) {\n var timeout = null;\n options || (options = {});\n return function() {\n var context = this,\n args = arguments;\n if(!timeout) {\n if(options.leading !== false) {\n func.apply(context, args);\n }\n timeout = $timeout(function later() {\n timeout = null;\n if(options.trailing !== false) {\n func.apply(context, args);\n }\n }, wait, false);\n }\n };\n };\n});\n\n// Source: helpers/date-parser.js\nangular.module('mgcrea.ngStrap.helpers.dateParser', [])\n\n.provider('$dateParser', function($localeProvider) {\n\n // define a custom ParseDate object to use instead of native Date\n // to avoid date values wrapping when setting date component values\n function ParseDate() {\n this.year = 1970;\n this.month = 0;\n this.day = 1;\n this.hours = 0;\n this.minutes = 0;\n this.seconds = 0;\n this.milliseconds = 0;\n }\n\n ParseDate.prototype.setMilliseconds = function(value) { this.milliseconds = value; };\n ParseDate.prototype.setSeconds = function(value) { this.seconds = value; };\n ParseDate.prototype.setMinutes = function(value) { this.minutes = value; };\n ParseDate.prototype.setHours = function(value) { this.hours = value; };\n ParseDate.prototype.getHours = function() { return this.hours; };\n ParseDate.prototype.setDate = function(value) { this.day = value; };\n ParseDate.prototype.setMonth = function(value) { this.month = value; };\n ParseDate.prototype.setFullYear = function(value) { this.year = value; };\n ParseDate.prototype.fromDate = function(value) {\n this.year = value.getFullYear();\n this.month = value.getMonth();\n this.day = value.getDate();\n this.hours = value.getHours();\n this.minutes = value.getMinutes();\n this.seconds = value.getSeconds();\n this.milliseconds = value.getMilliseconds();\n return this;\n };\n\n ParseDate.prototype.toDate = function() {\n return new Date(this.year, this.month, this.day, this.hours, this.minutes, this.seconds, this.milliseconds);\n };\n\n var proto = ParseDate.prototype;\n\n function noop() {\n }\n\n function isNumeric(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n\n function indexOfCaseInsensitive(array, value) {\n var len = array.length, str=value.toString().toLowerCase();\n for (var i=0; i 12 when midnight changeover, but then cannot generate\n * midnight datetime, so jump to 1AM, otherwise reset.\n * @param date (Date) the date to check\n * @return (Date) the corrected date\n *\n * __ copied from jquery ui datepicker __\n */\n $dateParser.daylightSavingAdjust = function(date) {\n if (!date) {\n return null;\n }\n date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);\n return date;\n };\n\n /* Correct the date for timezone offset.\n * @param date (Date) the date to adjust\n * @param timezone (string) the timezone to adjust for\n * @param undo (boolean) to add or subtract timezone offset\n * @return (Date) the corrected date\n */\n $dateParser.timezoneOffsetAdjust = function(date, timezone, undo) {\n if (!date) {\n return null;\n }\n // Right now, only 'UTC' is supported.\n if (timezone && timezone === 'UTC') {\n date = new Date(date.getTime());\n date.setMinutes(date.getMinutes() + (undo?-1:1)*date.getTimezoneOffset());\n }\n return date;\n };\n\n // Private functions\n\n function setMapForFormat(format) {\n var keys = Object.keys(setFnMap), i;\n var map = [], sortedMap = [];\n // Map to setFn\n var clonedFormat = format;\n for(i = 0; i < keys.length; i++) {\n if(format.split(keys[i]).length > 1) {\n var index = clonedFormat.search(keys[i]);\n format = format.split(keys[i]).join('');\n if(setFnMap[keys[i]]) {\n map[index] = setFnMap[keys[i]];\n }\n }\n }\n // Sort result map\n angular.forEach(map, function(v) {\n // conditional required since angular.forEach broke around v1.2.21\n // related pr: https://github.com/angular/angular.js/pull/8525\n if(v) sortedMap.push(v);\n });\n return sortedMap;\n }\n\n function escapeReservedSymbols(text) {\n return text.replace(/\\//g, '[\\\\/]').replace('/-/g', '[-]').replace(/\\./g, '[.]').replace(/\\\\s/g, '[\\\\s]');\n }\n\n function regExpForFormat(format) {\n var keys = Object.keys(regExpMap), i;\n\n var re = format;\n // Abstract replaces to avoid collisions\n for(i = 0; i < keys.length; i++) {\n re = re.split(keys[i]).join('${' + i + '}');\n }\n // Replace abstracted values\n for(i = 0; i < keys.length; i++) {\n re = re.split('${' + i + '}').join('(' + regExpMap[keys[i]] + ')');\n }\n format = escapeReservedSymbols(format);\n\n return new RegExp('^' + re + '$', ['i']);\n }\n\n $dateParser.init();\n return $dateParser;\n\n };\n\n return DateParserFactory;\n\n };\n\n});\n\n// Source: helpers/date-formatter.js\nangular.module('mgcrea.ngStrap.helpers.dateFormatter', [])\n\n .service('$dateFormatter', function($locale, dateFilter) {\n\n // The unused `lang` arguments are on purpose. The default implementation does not\n // use them and it always uses the locale loaded into the `$locale` service.\n // Custom implementations might use it, thus allowing different directives to\n // have different languages.\n\n this.getDefaultLocale = function() {\n return $locale.id;\n };\n\n // Format is either a data format name, e.g. \"shortTime\" or \"fullDate\", or a date format\n // Return either the corresponding date format or the given date format.\n this.getDatetimeFormat = function(format, lang) {\n return $locale.DATETIME_FORMATS[format] || format;\n };\n\n this.weekdaysShort = function(lang) {\n return $locale.DATETIME_FORMATS.SHORTDAY;\n };\n\n function splitTimeFormat(format) {\n return /(h+)([:\\.])?(m+)([:\\.])?(s*)[ ]?(a?)/i.exec(format).slice(1);\n }\n\n // h:mm a => h\n this.hoursFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[0];\n };\n\n // h:mm a => mm\n this.minutesFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[2];\n };\n\n // h:mm:ss a => ss\n this.secondsFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => :\n this.timeSeparator = function(timeFormat) {\n return splitTimeFormat(timeFormat)[1];\n };\n\n // h:mm:ss a => true, h:mm a => false\n this.showSeconds = function(timeFormat) {\n return !!splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => true, H.mm => false\n this.showAM = function(timeFormat) {\n return !!splitTimeFormat(timeFormat)[5];\n };\n\n this.formatDate = function(date, format, lang, timezone){\n return dateFilter(date, format, timezone);\n };\n\n });\n\n// Source: helpers/compiler.js\n// NOTICE: This file was forked from the angular-material project (github.com/angular/material)\n// MIT Licensed - Copyright (c) 2014-2015 Google, Inc. http://angularjs.org\n\nangular.module('mgcrea.ngStrap.core', [])\n .service('$bsCompiler', bsCompilerService);\n\nfunction bsCompilerService($q, $http, $injector, $compile, $controller, $templateCache) {\n /* jshint validthis: true */\n\n /*\n * @ngdoc service\n * @name $bsCompiler\n * @module material.core\n * @description\n * The $bsCompiler service is an abstraction of angular's compiler, that allows the developer\n * to easily compile an element with a templateUrl, controller, and locals.\n *\n * @usage\n * \n * $bsCompiler.compile({\n * templateUrl: 'modal.html',\n * controller: 'ModalCtrl',\n * locals: {\n * modal: myModalInstance;\n * }\n * }).then(function(compileData) {\n * compileData.element; // modal.html's template in an element\n * compileData.link(myScope); //attach controller & scope to element\n * });\n * \n */\n\n /*\n * @ngdoc method\n * @name $bsCompiler#compile\n * @description A helper to compile an HTML template/templateUrl with a given controller,\n * locals, and scope.\n * @param {object} options An options object, with the following properties:\n *\n * - `controller` - `{(string=|function()=}` Controller fn that should be associated with\n * newly created scope or the name of a registered controller if passed as a string.\n * - `controllerAs` - `{string=}` A controller alias name. If present the controller will be\n * published to scope under the `controllerAs` name.\n * - `template` - `{string=}` An html template as a string.\n * - `templateUrl` - `{string=}` A path to an html template.\n * - `transformTemplate` - `{function(template)=}` A function which transforms the template after\n * it is loaded. It will be given the template string as a parameter, and should\n * return a a new string representing the transformed template.\n * - `resolve` - `{Object.=}` - An optional map of dependencies which should\n * be injected into the controller. If any of these dependencies are promises, the compiler\n * will wait for them all to be resolved, or if one is rejected before the controller is\n * instantiated `compile()` will fail..\n * * `key` - `{string}`: a name of a dependency to be injected into the controller.\n * * `factory` - `{string|function}`: If `string` then it is an alias for a service.\n * Otherwise if function, then it is injected and the return value is treated as the\n * dependency. If the result is a promise, it is resolved before its value is\n * injected into the controller.\n *\n * @returns {object=} promise A promise, which will be resolved with a `compileData` object.\n * `compileData` has the following properties:\n *\n * - `element` - `{element}`: an uncompiled element matching the provided template.\n * - `link` - `{function(scope)}`: A link function, which, when called, will compile\n * the element and instantiate the provided controller (if given).\n * - `locals` - `{object}`: The locals which will be passed into the controller once `link` is\n * called. If `bindToController` is true, they will be coppied to the ctrl instead\n * - `bindToController` - `bool`: bind the locals to the controller, instead of passing them in.\n */\n this.compile = function(options) {\n\n if(options.template && /\\.html$/.test(options.template)) {\n console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.');\n options.templateUrl = options.template;\n options.template = '';\n }\n\n var templateUrl = options.templateUrl;\n var template = options.template || '';\n var controller = options.controller;\n var controllerAs = options.controllerAs;\n var resolve = angular.copy(options.resolve || {});\n var locals = angular.copy(options.locals || {});\n var transformTemplate = options.transformTemplate || angular.identity;\n var bindToController = options.bindToController;\n\n // Take resolve values and invoke them.\n // Resolves can either be a string (value: 'MyRegisteredAngularConst'),\n // or an invokable 'factory' of sorts: (value: function ValueGetter($dependency) {})\n angular.forEach(resolve, function(value, key) {\n if (angular.isString(value)) {\n resolve[key] = $injector.get(value);\n } else {\n resolve[key] = $injector.invoke(value);\n }\n });\n // Add the locals, which are just straight values to inject\n // eg locals: { three: 3 }, will inject three into the controller\n angular.extend(resolve, locals);\n\n if (template) {\n resolve.$template = $q.when(template);\n } else if (templateUrl) {\n resolve.$template = fetchTemplate(templateUrl);\n } else {\n throw new Error('Missing `template` / `templateUrl` option.');\n }\n\n if (options.contentTemplate) {\n // TODO(mgcrea): deprecate?\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.contentTemplate)])\n .then(function(templates) {\n var templateEl = angular.element(templates[0]);\n var contentEl = findElement('[ng-bind=\"content\"]', templateEl[0]).removeAttr('ng-bind').html(templates[1]);\n // Drop the default footer as you probably don't want it if you use a custom contentTemplate\n if(!options.templateUrl) contentEl.next().remove();\n return templateEl[0].outerHTML;\n });\n }\n\n // Wait for all the resolves to finish if they are promises\n return $q.all(resolve).then(function(locals) {\n\n var template = transformTemplate(locals.$template);\n if (options.html) {\n template = template.replace(/ng-bind=\"/ig, 'ng-bind-html=\"');\n }\n // var element = options.element || angular.element('
').html(template.trim()).contents();\n var element = angular.element('
').html(template.trim()).contents();\n var linkFn = $compile(element);\n\n // Return a linking function that can be used later when the element is ready\n return {\n locals: locals,\n element: element,\n link: function link(scope) {\n locals.$scope = scope;\n\n // Instantiate controller if it exists, because we have scope\n if (controller) {\n var invokeCtrl = $controller(controller, locals, true);\n if (bindToController) {\n angular.extend(invokeCtrl.instance, locals);\n }\n // Support angular@~1.2 invokeCtrl\n var ctrl = angular.isObject(invokeCtrl) ? invokeCtrl : invokeCtrl();\n // See angular-route source for this logic\n element.data('$ngControllerController', ctrl);\n element.children().data('$ngControllerController', ctrl);\n\n if (controllerAs) {\n scope[controllerAs] = ctrl;\n }\n }\n\n return linkFn.apply(null, arguments);\n }\n };\n });\n\n };\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate(template) {\n if(fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache})\n .then(function(res) {\n return res.data;\n }));\n }\n\n}\n\n// Source: dropdown/dropdown.js\nangular.module('mgcrea.ngStrap.dropdown', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$dropdown', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'dropdown',\n prefixEvent: 'dropdown',\n placement: 'bottom-left',\n templateUrl: 'dropdown/dropdown.tpl.html',\n trigger: 'click',\n container: false,\n keyboard: true,\n html: false,\n delay: 0\n };\n\n this.$get = function($window, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var matchesSelector = Element.prototype.matchesSelector || Element.prototype.webkitMatchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector;\n\n function DropdownFactory(element, config) {\n\n var $dropdown = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var scope = $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n $dropdown = $tooltip(element, options);\n var parentEl = element.parent();\n\n // Protected methods\n\n $dropdown.$onKeyDown = function(evt) {\n if (!/(38|40)/.test(evt.keyCode)) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Retrieve focused index\n var items = angular.element($dropdown.$element[0].querySelectorAll('li:not(.divider) a'));\n if (!items.length) return;\n var index;\n angular.forEach(items, function(el, i) {\n if (matchesSelector && matchesSelector.call(el, ':focus')) index = i;\n });\n\n // Navigate with keyboard\n if (evt.keyCode === 38 && index > 0) index--;\n else if (evt.keyCode === 40 && index < items.length - 1) index++;\n else if (angular.isUndefined(index)) index = 0;\n items.eq(index)[0].focus();\n\n };\n\n // Overrides\n\n var show = $dropdown.show;\n $dropdown.show = function() {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n options.keyboard && $dropdown.$element && $dropdown.$element.on('keydown', $dropdown.$onKeyDown);\n bodyEl.on('click', onBodyClick);\n }, 0, false);\n parentEl.hasClass('dropdown') && parentEl.addClass('open');\n };\n\n var hide = $dropdown.hide;\n $dropdown.hide = function() {\n if (!$dropdown.$isShown) return;\n options.keyboard && $dropdown.$element && $dropdown.$element.off('keydown', $dropdown.$onKeyDown);\n bodyEl.off('click', onBodyClick);\n parentEl.hasClass('dropdown') && parentEl.removeClass('open');\n hide();\n };\n\n var destroy = $dropdown.destroy;\n $dropdown.destroy = function() {\n bodyEl.off('click', onBodyClick);\n destroy();\n };\n\n // Private functions\n\n function onBodyClick(evt) {\n if (evt.target === element[0]) return;\n return evt.target !== element[0] && $dropdown.hide();\n }\n\n return $dropdown;\n\n }\n\n return DropdownFactory;\n\n };\n\n })\n\n .directive('bsDropdown', function($window, $sce, $dropdown) {\n\n return {\n restrict: 'EAC',\n scope: true,\n compile: function(tElement, tAttrs) {\n\n // Directive options\n var options = {};\n\n // Support for inlined template (next sibling)\n // It must be fetched before compilation\n if (!tAttrs.bsDropdown) {\n var nextSibling = tElement[0].nextSibling;\n while (nextSibling && nextSibling.nodeType !== 1) {\n nextSibling = nextSibling.nextSibling;\n }\n if (nextSibling.classList.contains('dropdown-menu')) {\n options.template = nextSibling.outerHTML;\n options.templateUrl = undefined;\n nextSibling.parentNode.removeChild(nextSibling);\n }\n }\n\n return function postLink(scope, element, attr) {\n\n // Directive options\n options.scope = scope;\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id'], function(key) {\n if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as an object\n attr.bsDropdown && scope.$watch(attr.bsDropdown, function(newValue, oldValue) {\n scope.content = newValue;\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!dropdown || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i);\n newValue === true ? dropdown.show() : dropdown.hide();\n });\n\n // Initialize dropdown\n var dropdown = $dropdown(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (dropdown) dropdown.destroy();\n options = null;\n dropdown = null;\n });\n\n };\n }\n };\n\n });\n\n// Source: datepicker/datepicker.js\nangular.module('mgcrea.ngStrap.datepicker', [\n 'mgcrea.ngStrap.helpers.dateParser',\n 'mgcrea.ngStrap.helpers.dateFormatter',\n 'mgcrea.ngStrap.tooltip'])\n\n .provider('$datepicker', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n // Uncommenting the following line will break backwards compatability\n // prefixEvent: 'datepicker',\n prefixClass: 'datepicker',\n placement: 'bottom-left',\n templateUrl: 'datepicker/datepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: false,\n dateType: 'date',\n dateFormat: 'shortDate',\n timezone: null,\n modelDateFormat: null,\n dayFormat: 'dd',\n monthFormat: 'MMM',\n yearFormat: 'yyyy',\n monthTitleFormat: 'MMMM yyyy',\n yearTitleFormat: 'yyyy',\n strictFormat: false,\n autoclose: false,\n minDate: -Infinity,\n maxDate: +Infinity,\n startView: 0,\n minView: 0,\n startWeek: 0,\n daysOfWeekDisabled: '',\n iconLeft: 'glyphicon glyphicon-chevron-left',\n iconRight: 'glyphicon glyphicon-chevron-right'\n };\n\n this.$get = function($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) {\n\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale();\n\n function DatepickerFactory(element, controller, config) {\n\n var $datepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $datepicker.$options;\n var scope = $datepicker.$scope;\n if (options.startView) options.startView -= options.minView;\n\n // View vars\n\n var pickerViews = datepickerViews($datepicker);\n $datepicker.$views = pickerViews.views;\n var viewDate = pickerViews.viewDate;\n scope.$mode = options.startView;\n scope.$iconLeft = options.iconLeft;\n scope.$iconRight = options.iconRight;\n var $picker = $datepicker.$views[scope.$mode];\n\n // Scope methods\n\n scope.$select = function(date) {\n $datepicker.select(date);\n };\n scope.$selectPane = function(value) {\n $datepicker.$selectPane(value);\n };\n scope.$toggleMode = function() {\n $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length);\n };\n\n // Public methods\n\n $datepicker.update = function(date) {\n // console.warn('$datepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $datepicker.$date = date;\n $picker.update.call($picker, date);\n }\n // Build only if pristine\n $datepicker.$build(true);\n };\n\n $datepicker.updateDisabledDates = function(dateRanges) {\n options.disabledDateRanges = dateRanges;\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], $datepicker.$setDisabledEl);\n }\n };\n\n $datepicker.select = function(date, keep) {\n // console.warn('$datepicker.select', date, scope.$mode);\n if (!angular.isDate(controller.$dateValue)) controller.$dateValue = new Date(date);\n if (!scope.$mode || keep) {\n controller.$setViewValue(angular.copy(date));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function() { $datepicker.hide(true); });\n }\n } else {\n angular.extend(viewDate, {year: date.getFullYear(), month: date.getMonth(), date: date.getDate()});\n $datepicker.setMode(scope.$mode - 1);\n $datepicker.$build();\n }\n };\n\n $datepicker.setMode = function(mode) {\n // console.warn('$datepicker.setMode', mode);\n scope.$mode = mode;\n $picker = $datepicker.$views[scope.$mode];\n $datepicker.$build();\n };\n\n // Protected methods\n\n $datepicker.$build = function(pristine) {\n // console.warn('$datepicker.$build() viewDate=%o', viewDate);\n if (pristine === true && $picker.built) return;\n if (pristine === false && !$picker.built) return;\n $picker.build.call($picker);\n };\n\n $datepicker.$updateSelected = function() {\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], updateSelected);\n }\n };\n\n $datepicker.$isSelected = function(date) {\n return $picker.isSelected(date);\n };\n\n $datepicker.$setDisabledEl = function(el) {\n el.disabled = $picker.isDisabled(el.date);\n };\n\n $datepicker.$selectPane = function(value) {\n var steps = $picker.steps;\n // set targetDate to first day of month to avoid problems with\n // date values rollover. This assumes the viewDate does not\n // depend on the day of the month\n var targetDate = new Date(Date.UTC(viewDate.year + ((steps.year || 0) * value), viewDate.month + ((steps.month || 0) * value), 1));\n angular.extend(viewDate, {year: targetDate.getUTCFullYear(), month: targetDate.getUTCMonth(), date: targetDate.getUTCDate()});\n $datepicker.$build();\n };\n\n $datepicker.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $datepicker.$onKeyDown = function(evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n if (evt.keyCode === 13) {\n if (!scope.$mode) {\n $datepicker.hide(true);\n } else {\n scope.$apply(function() { $datepicker.setMode(scope.$mode - 1); });\n }\n return;\n }\n\n // Navigate with keyboard\n $picker.onKeyDown(evt);\n parentScope.$digest();\n };\n\n // Private\n\n function updateSelected(el) {\n el.selected = $datepicker.$isSelected(el.date);\n }\n\n function focusElement() {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $datepicker.init;\n $datepicker.init = function() {\n if (isNative && options.useNative) {\n element.prop('type', 'date');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $datepicker.destroy;\n $datepicker.destroy = function() {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $datepicker.show;\n $datepicker.show = function() {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n // if $datepicker is no longer showing, don't setup events\n if (!$datepicker.$isShown) return;\n $datepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $datepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $datepicker.hide;\n $datepicker.hide = function(blur) {\n if (!$datepicker.$isShown) return;\n $datepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $datepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $datepicker;\n\n }\n\n DatepickerFactory.defaults = defaults;\n return DatepickerFactory;\n\n };\n\n })\n\n .directive('bsDatepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) {\n\n var defaults = $datepicker.defaults;\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // Initialize datepicker\n var datepicker = $datepicker(element, controller, options);\n options = datepicker.$options;\n // Set expected iOS format\n if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd';\n\n var lang = options.lang;\n\n var formatDate = function(date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!datepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i);\n newValue === true ? datepicker.show() : datepicker.hide();\n });\n\n // Observe attributes for changes\n angular.forEach(['minDate', 'maxDate'], function(key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) {\n // console.warn('attr.$observe(%s)=%o', key, newValue);\n datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue);\n // Build only if dirty\n !isNaN(datepicker.$options[key]) && datepicker.$build(false);\n validateAgainstMinMaxDate(controller.$dateValue);\n });\n });\n\n // Observe date format\n angular.isDefined(attr.dateFormat) && attr.$observe('dateFormat', function(newValue) {\n datepicker.$options.dateFormat = newValue;\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n datepicker.update(controller.$dateValue);\n }, true);\n\n // Normalize undefined/null/empty array,\n // so that we don't treat changing from undefined->null as a change.\n function normalizeDateRanges(ranges) {\n if (!ranges || !ranges.length) return null;\n return ranges;\n }\n\n if (angular.isDefined(attr.disabledDates)) {\n scope.$watch(attr.disabledDates, function(disabledRanges, previousValue) {\n disabledRanges = normalizeDateRanges(disabledRanges);\n previousValue = normalizeDateRanges(previousValue);\n\n if (disabledRanges) {\n datepicker.updateDisabledDates(disabledRanges);\n }\n });\n }\n\n function validateAgainstMinMaxDate(parsedDate) {\n if (!angular.isDate(parsedDate)) return;\n var isMinValid = isNaN(datepicker.$options.minDate) || parsedDate.getTime() >= datepicker.$options.minDate;\n var isMaxValid = isNaN(datepicker.$options.maxDate) || parsedDate.getTime() <= datepicker.$options.maxDate;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (isValid) controller.$dateValue = parsedDate;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function(viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n controller.$setValidity('date', true);\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n return null;\n }\n var parsedDate = dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedDate || isNaN(parsedDate.getTime())) {\n controller.$setValidity('date', false);\n // return undefined, causes ngModelController to\n // invalidate model value\n return;\n } else {\n validateAgainstMinMaxDate(parsedDate);\n }\n\n if (options.dateType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true);\n return formatDate(date, options.modelDateFormat || options.dateFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.dateType === 'number') {\n return date.getTime();\n } else if (options.dateType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.dateType === 'iso') {\n return date.toISOString();\n } else {\n return new Date(date);\n }\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.dateType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelDateFormat);\n } else if (options.dateType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if (isNaN(date.getTime())) {\n // var today = new Date();\n // date = new Date(today.getFullYear(), today.getMonth(), today.getDate(), 0, 0, 0, 0);\n // }\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getDateFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function() {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getDateFormattedString());\n };\n\n function getDateFormattedString() {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.dateFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (datepicker) datepicker.destroy();\n options = null;\n datepicker = null;\n });\n\n }\n };\n\n })\n\n .provider('datepickerViews', function() {\n\n var defaults = this.defaults = {\n dayFormat: 'dd',\n daySplit: 7\n };\n\n // Split array into smaller arrays\n function split(arr, size) {\n var arrays = [];\n while(arr.length > 0) {\n arrays.push(arr.splice(0, size));\n }\n return arrays;\n }\n\n // Modulus operator\n function mod(n, m) {\n return ((n % m) + m) % m;\n }\n\n this.$get = function($dateFormatter, $dateParser, $sce) {\n\n return function(picker) {\n\n var scope = picker.$scope;\n var options = picker.$options;\n\n var lang = options.lang;\n var formatDate = function(date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n var weekDaysMin = $dateFormatter.weekdaysShort(lang);\n var weekDaysLabels = weekDaysMin.slice(options.startWeek).concat(weekDaysMin.slice(0, options.startWeek));\n var weekDaysLabelsHtml = $sce.trustAsHtml('' + weekDaysLabels.join('') + '');\n\n var startDate = picker.$date || (options.startDate ? dateParser.getDateForAttribute('startDate', options.startDate) : new Date());\n var viewDate = {year: startDate.getFullYear(), month: startDate.getMonth(), date: startDate.getDate()};\n\n var views = [{\n format: options.dayFormat,\n split: 7,\n steps: { month: 1 },\n update: function(date, force) {\n if (!this.built || force || date.getFullYear() !== viewDate.year || date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getDate() !== viewDate.date || date.getDate() === 1) {\n // chaging picker current month will cause viewDate.date to be set to first day of the month,\n // in $datepicker.$selectPane, so picker would not update selected day display if\n // user picks first day of the new month.\n // As a workaround, we are always forcing update when picked date is first day of month.\n viewDate.date = picker.$date.getDate();\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1), firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset();\n var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5), firstDateOffset = firstDate.getTimezoneOffset();\n var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString();\n // Handle daylight time switch\n if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 60e3);\n var days = [], day;\n for(var i = 0; i < 42; i++) { // < 7 * 6\n day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i));\n days.push({date: day, isToday: day.toDateString() === today, label: formatDate(day, this.format), selected: picker.$date && this.isSelected(day), muted: day.getMonth() !== viewDate.month, disabled: this.isDisabled(day)});\n }\n scope.title = formatDate(firstDayOfMonth, options.monthTitleFormat);\n scope.showLabels = true;\n scope.labels = weekDaysLabelsHtml;\n scope.rows = split(days, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth() && date.getDate() === picker.$date.getDate();\n },\n isDisabled: function(date) {\n var time = date.getTime();\n\n // Disabled because of min/max date.\n if (time < options.minDate || time > options.maxDate) return true;\n\n // Disabled due to being a disabled day of the week\n if (options.daysOfWeekDisabled.indexOf(date.getDay()) !== -1) return true;\n\n // Disabled because of disabled date range.\n if (options.disabledDateRanges) {\n for (var i = 0; i < options.disabledDateRanges.length; i++) {\n if (time >= options.disabledDateRanges[i].start && time <= options.disabledDateRanges[i].end) {\n return true;\n }\n }\n }\n\n return false;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualTime = picker.$date.getTime();\n var newDate;\n\n if (evt.keyCode === 37) newDate = new Date(actualTime - 1 * 864e5);\n else if (evt.keyCode === 38) newDate = new Date(actualTime - 7 * 864e5);\n else if (evt.keyCode === 39) newDate = new Date(actualTime + 1 * 864e5);\n else if (evt.keyCode === 40) newDate = new Date(actualTime + 7 * 864e5);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'month',\n format: options.monthFormat,\n split: 4,\n steps: { year: 1 },\n update: function(date, force) {\n if (!this.built || date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstMonth = new Date(viewDate.year, 0, 1);\n var months = [], month;\n for (var i = 0; i < 12; i++) {\n month = new Date(viewDate.year, i, 1);\n months.push({date: month, label: formatDate(month, this.format), selected: picker.$isSelected(month), disabled: this.isDisabled(month)});\n }\n scope.title = formatDate(month, options.yearTitleFormat);\n scope.showLabels = false;\n scope.rows = split(months, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth();\n },\n isDisabled: function(date) {\n var lastDate = +new Date(date.getFullYear(), date.getMonth() + 1, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualMonth = picker.$date.getMonth();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setMonth(actualMonth - 1);\n else if (evt.keyCode === 38) newDate.setMonth(actualMonth - 4);\n else if (evt.keyCode === 39) newDate.setMonth(actualMonth + 1);\n else if (evt.keyCode === 40) newDate.setMonth(actualMonth + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'year',\n format: options.yearFormat,\n split: 4,\n steps: { year: 12 },\n update: function(date, force) {\n if (!this.built || force || parseInt(date.getFullYear()/20, 10) !== parseInt(viewDate.year/20, 10)) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstYear = viewDate.year - viewDate.year % (this.split * 3);\n var years = [], year;\n for (var i = 0; i < 12; i++) {\n year = new Date(firstYear + i, 0, 1);\n years.push({date: year, label: formatDate(year, this.format), selected: picker.$isSelected(year), disabled: this.isDisabled(year)});\n }\n scope.title = years[0].label + '-' + years[years.length - 1].label;\n scope.showLabels = false;\n scope.rows = split(years, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear();\n },\n isDisabled: function(date) {\n var lastDate = +new Date(date.getFullYear() + 1, 0, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualYear = picker.$date.getFullYear(),\n newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setYear(actualYear - 1);\n else if (evt.keyCode === 38) newDate.setYear(actualYear - 4);\n else if (evt.keyCode === 39) newDate.setYear(actualYear + 1);\n else if (evt.keyCode === 40) newDate.setYear(actualYear + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }];\n\n return {\n views: options.minView ? Array.prototype.slice.call(views, options.minView) : views,\n viewDate: viewDate\n };\n\n };\n\n };\n\n });\n\n// Source: collapse/collapse.js\nangular.module('mgcrea.ngStrap.collapse', [])\n\n .provider('$collapse', function() {\n\n var defaults = this.defaults = {\n animation: 'am-collapse',\n disallowToggle: false,\n activeClass: 'in',\n startCollapsed: false,\n allowMultiple: false\n };\n\n var controller = this.controller = function($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'disallowToggle', 'activeClass', 'startCollapsed', 'allowMultiple'], function (key) {\n if(angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['disallowToggle', 'startCollapsed', 'allowMultiple'], function(key) {\n if(angular.isDefined($attrs[key]) && falseValueRegExp.test($attrs[key])) {\n self.$options[key] = false;\n }\n });\n\n self.$toggles = [];\n self.$targets = [];\n\n self.$viewChangeListeners = [];\n\n self.$registerToggle = function(element) {\n self.$toggles.push(element);\n };\n self.$registerTarget = function(element) {\n self.$targets.push(element);\n };\n\n self.$unregisterToggle = function(element) {\n var index = self.$toggles.indexOf(element);\n // remove toggle from $toggles array\n self.$toggles.splice(index, 1);\n };\n self.$unregisterTarget = function(element) {\n var index = self.$targets.indexOf(element);\n\n // remove element from $targets array\n self.$targets.splice(index, 1);\n\n if (self.$options.allowMultiple) {\n // remove target index from $active array values\n deactivateItem(element);\n }\n\n // fix active item indexes\n fixActiveItemIndexes(index);\n\n self.$viewChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n // use array to store all the currently open panels\n self.$targets.$active = !self.$options.startCollapsed ? [0] : [];\n self.$setActive = $scope.$setActive = function(value) {\n if(angular.isArray(value)) {\n self.$targets.$active = value;\n }\n else if(!self.$options.disallowToggle) {\n // toogle element active status\n isActive(value) ? deactivateItem(value) : activateItem(value);\n } else {\n activateItem(value);\n }\n\n self.$viewChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n self.$activeIndexes = function() {\n return self.$options.allowMultiple ? self.$targets.$active :\n self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1;\n };\n\n function fixActiveItemIndexes(index) {\n // item with index was removed, so we\n // need to adjust other items index values\n var activeIndexes = self.$targets.$active;\n for(var i = 0; i < activeIndexes.length; i++) {\n if (index < activeIndexes[i]) {\n activeIndexes[i] = activeIndexes[i] - 1;\n }\n\n // the last item is active, so we need to\n // adjust its index\n if (activeIndexes[i] === self.$targets.length) {\n activeIndexes[i] = self.$targets.length - 1;\n }\n }\n }\n\n function isActive(value) {\n var activeItems = self.$targets.$active;\n return activeItems.indexOf(value) === -1 ? false : true;\n }\n\n function deactivateItem(value) {\n var index = self.$targets.$active.indexOf(value);\n if (index !== -1) {\n self.$targets.$active.splice(index, 1);\n }\n }\n\n function activateItem(value) {\n if (!self.$options.allowMultiple) {\n // remove current selected item\n self.$targets.$active.splice(0, 1);\n }\n\n if (self.$targets.$active.indexOf(value) === -1) {\n self.$targets.$active.push(value);\n }\n }\n\n };\n\n this.$get = function() {\n var $collapse = {};\n $collapse.defaults = defaults;\n $collapse.controller = controller;\n return $collapse;\n };\n\n })\n\n .directive('bsCollapse', function($window, $animate, $collapse) {\n\n var defaults = $collapse.defaults;\n\n return {\n require: ['?ngModel', 'bsCollapse'],\n controller: ['$scope', '$element', '$attrs', $collapse.controller],\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n if(ngModelCtrl) {\n\n // Update the modelValue following\n bsCollapseCtrl.$viewChangeListeners.push(function() {\n ngModelCtrl.$setViewValue(bsCollapseCtrl.$activeIndexes());\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n if (angular.isArray(modelValue)) {\n // model value is an array, so just replace\n // the active items directly\n bsCollapseCtrl.$setActive(modelValue);\n }\n else {\n var activeIndexes = bsCollapseCtrl.$activeIndexes();\n\n if (angular.isArray(activeIndexes)) {\n // we have an array of selected indexes\n if (activeIndexes.indexOf(modelValue * 1) === -1) {\n // item with modelValue index is not active\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n else if (activeIndexes !== modelValue * 1) {\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n return modelValue;\n });\n\n }\n\n }\n };\n\n })\n\n .directive('bsCollapseToggle', function() {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base attr\n element.attr('data-toggle', 'collapse');\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerToggle(element);\n\n // remove toggle from collapse controller when toggle is destroyed\n scope.$on('$destroy', function() {\n bsCollapseCtrl.$unregisterToggle(element);\n });\n\n element.on('click', function() {\n var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element);\n bsCollapseCtrl.$setActive(index * 1);\n scope.$apply();\n });\n\n }\n };\n\n })\n\n .directive('bsCollapseTarget', function($animate) {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n // scope: true,\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base class\n element.addClass('collapse');\n\n // Add animation class\n if(bsCollapseCtrl.$options.animation) {\n element.addClass(bsCollapseCtrl.$options.animation);\n }\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerTarget(element);\n\n // remove pane target from collapse controller when target is destroyed\n scope.$on('$destroy', function() {\n bsCollapseCtrl.$unregisterTarget(element);\n });\n\n function render() {\n var index = bsCollapseCtrl.$targets.indexOf(element);\n var active = bsCollapseCtrl.$activeIndexes();\n var action = 'removeClass';\n if (angular.isArray(active)) {\n if (active.indexOf(index) !== -1) {\n action = 'addClass';\n }\n }\n else if (index === active) {\n action = 'addClass';\n }\n\n $animate[action](element, bsCollapseCtrl.$options.activeClass);\n }\n\n bsCollapseCtrl.$viewChangeListeners.push(function() {\n render();\n });\n render();\n\n }\n };\n\n });\n\n// Source: button/button.js\nangular.module('mgcrea.ngStrap.button', [])\n\n .provider('$button', function() {\n\n var defaults = this.defaults = {\n activeClass:'active',\n toggleEvent:'click'\n };\n\n this.$get = function() {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsCheckboxGroup', function() {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink(element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"checkbox\"]');\n angular.forEach(children, function(child) {\n var childEl = angular.element(child);\n childEl.attr('bs-checkbox', '');\n childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value'));\n });\n }\n\n };\n\n })\n\n .directive('bsCheckbox', function($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support label > input[type=\"checkbox\"]\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true;\n if(constantValueRegExp.test(attr.trueValue)) {\n trueValue = scope.$eval(attr.trueValue);\n }\n var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false;\n if(constantValueRegExp.test(attr.falseValue)) {\n falseValue = scope.$eval(attr.falseValue);\n }\n\n // Parse exotic values\n var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean';\n if(hasExoticValues) {\n controller.$parsers.push(function(viewValue) {\n // console.warn('$parser', element.attr('ng-model'), 'viewValue', viewValue);\n return viewValue ? trueValue : falseValue;\n });\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n return angular.equals(modelValue, trueValue);\n });\n // Fix rendering for exotic values\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n controller.$render();\n });\n }\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$modelValue, trueValue);\n $$rAF(function() {\n if(isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function() {\n scope.$apply(function () {\n // console.warn('!click', element.attr('ng-model'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n if(!isInput) {\n controller.$setViewValue(!activeElement.hasClass('active'));\n }\n if(!hasExoticValues) {\n controller.$render();\n }\n });\n });\n\n }\n\n };\n\n })\n\n .directive('bsRadioGroup', function() {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink(element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"radio\"]');\n angular.forEach(children, function(child) {\n angular.element(child).attr('bs-radio', '');\n angular.element(child).attr('ng-model', attr.ngModel);\n });\n }\n\n };\n\n })\n\n .directive('bsRadio', function($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support `label > input[type=\"radio\"]` markup\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var value;\n attr.$observe('value', function(v) {\n value = constantValueRegExp.test(v) ? scope.$eval(v) : v;\n controller.$render();\n });\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('value'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$modelValue, value);\n $$rAF(function() {\n if(isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function() {\n scope.$apply(function () {\n // console.warn('!click', element.attr('value'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n controller.$setViewValue(value);\n controller.$render();\n });\n });\n\n }\n\n };\n\n });\n\n// Source: aside/aside.js\nangular.module('mgcrea.ngStrap.aside', ['mgcrea.ngStrap.modal'])\n\n .provider('$aside', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade-and-slide-right',\n prefixClass: 'aside',\n prefixEvent: 'aside',\n placement: 'right',\n templateUrl: 'aside/aside.tpl.html',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function($modal) {\n\n function AsideFactory(config) {\n\n var $aside = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $aside = $modal(options);\n\n return $aside;\n\n }\n\n return AsideFactory;\n\n };\n\n })\n\n .directive('bsAside', function($window, $sce, $aside) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsAside && scope.$watch(attr.bsAside, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize aside\n var aside = $aside(options);\n\n // Trigger\n element.on(attr.trigger || 'click', aside.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (aside) aside.destroy();\n options = null;\n aside = null;\n });\n\n }\n };\n\n });\n\n// Source: alert/alert.js\n// @BUG: following snippet won't compile correctly\n// @TODO: submit issue to core\n// ' ' +\n\nangular.module('mgcrea.ngStrap.alert', ['mgcrea.ngStrap.modal'])\n\n .provider('$alert', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'alert',\n prefixEvent: 'alert',\n placement: null,\n templateUrl: 'alert/alert.tpl.html',\n container: false,\n element: null,\n backdrop: false,\n keyboard: true,\n show: true,\n // Specific options\n duration: false,\n type: false,\n dismissable: true\n };\n\n this.$get = function($modal, $timeout) {\n\n function AlertFactory(config) {\n\n var $alert = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $alert = $modal(options);\n\n // Support scope as string options [/*title, content, */ type, dismissable]\n $alert.$scope.dismissable = !!options.dismissable;\n if(options.type) {\n $alert.$scope.type = options.type;\n }\n\n // Support auto-close duration\n var show = $alert.show;\n if(options.duration) {\n $alert.show = function() {\n show();\n $timeout(function() {\n $alert.hide();\n }, options.duration * 1000);\n };\n }\n\n return $alert;\n\n }\n\n return AlertFactory;\n\n };\n\n })\n\n .directive('bsAlert', function($window, $sce, $alert) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'keyboard', 'html', 'container', 'animation', 'duration', 'dismissable'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['keyboard', 'html', 'container', 'dismissable'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')){\n scope.title = '';\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content', 'type'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsAlert && scope.$watch(attr.bsAlert, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize alert\n var alert = $alert(options);\n\n // Trigger\n element.on(attr.trigger || 'click', alert.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (alert) alert.destroy();\n options = null;\n alert = null;\n });\n\n }\n };\n\n });\n\n// Source: affix/affix.js\nangular.module('mgcrea.ngStrap.affix', ['mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce'])\n\n .provider('$affix', function() {\n\n var defaults = this.defaults = {\n offsetTop: 'auto',\n inlineStyles: true\n };\n\n this.$get = function($window, debounce, dimensions) {\n\n var bodyEl = angular.element($window.document.body);\n var windowEl = angular.element($window);\n\n function AffixFactory(element, config) {\n\n var $affix = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var targetEl = options.target;\n\n // Initial private vars\n var reset = 'affix affix-top affix-bottom',\n setWidth = false,\n initialAffixTop = 0,\n initialOffsetTop = 0,\n offsetTop = 0,\n offsetBottom = 0,\n affixed = null,\n unpin = null;\n\n var parent = element.parent();\n // Options: custom parent\n if (options.offsetParent) {\n if (options.offsetParent.match(/^\\d+$/)) {\n for (var i = 0; i < (options.offsetParent * 1) - 1; i++) {\n parent = parent.parent();\n }\n }\n else {\n parent = angular.element(options.offsetParent);\n }\n }\n\n $affix.init = function() {\n\n this.$parseOffsets();\n initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop;\n setWidth = !element[0].style.width;\n\n // Bind events\n targetEl.on('scroll', this.checkPosition);\n targetEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', this.$debouncedOnResize);\n\n // Both of these checkPosition() calls are necessary for the case where\n // the user hits refresh after scrolling to the bottom of the page.\n this.checkPosition();\n this.checkPositionWithEventLoop();\n\n };\n\n $affix.destroy = function() {\n\n // Unbind events\n targetEl.off('scroll', this.checkPosition);\n targetEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', this.$debouncedOnResize);\n\n };\n\n $affix.checkPositionWithEventLoop = function() {\n\n // IE 9 throws an error if we use 'this' instead of '$affix'\n // in this setTimeout call\n setTimeout($affix.checkPosition, 1);\n\n };\n\n $affix.checkPosition = function() {\n // if (!this.$element.is(':visible')) return\n\n var scrollTop = getScrollTop();\n var position = dimensions.offset(element[0]);\n var elementHeight = dimensions.height(element[0]);\n\n // Get required affix class according to position\n var affix = getRequiredAffixClass(unpin, position, elementHeight);\n\n // Did affix status changed this last check?\n if(affixed === affix) return;\n affixed = affix;\n\n if(affix === 'top') {\n unpin = null;\n if(setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n } else if(affix === 'bottom') {\n if (options.offsetUnpin) {\n unpin = -(options.offsetUnpin * 1);\n }\n else {\n // Calculate unpin threshold when affixed to bottom.\n // Hopefully the browser scrolls pixel by pixel.\n unpin = position.top - scrollTop;\n }\n if(setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', (options.offsetParent) ? '' : ((bodyEl[0].offsetHeight - offsetBottom - elementHeight - initialOffsetTop) + 'px'));\n }\n } else { // affix === 'middle'\n unpin = null;\n if(setWidth) {\n element.css('width', element[0].offsetWidth + 'px');\n }\n if (options.inlineStyles) {\n element.css('position', 'fixed');\n element.css('top', initialAffixTop + 'px');\n }\n }\n\n // Add proper affix class\n element.removeClass(reset).addClass('affix' + ((affix !== 'middle') ? '-' + affix : ''));\n\n };\n\n $affix.$onResize = function() {\n $affix.$parseOffsets();\n $affix.checkPosition();\n };\n $affix.$debouncedOnResize = debounce($affix.$onResize, 50);\n\n $affix.$parseOffsets = function() {\n var initialPosition = element.css('position');\n // Reset position to calculate correct offsetTop\n if (options.inlineStyles){\n element.css('position', (options.offsetParent) ? '' : 'relative');\n }\n\n if(options.offsetTop) {\n if(options.offsetTop === 'auto') {\n options.offsetTop = '+0';\n }\n if(options.offsetTop.match(/^[-+]\\d+$/)) {\n initialAffixTop = - options.offsetTop * 1;\n if(options.offsetParent) {\n offsetTop = dimensions.offset(parent[0]).top + (options.offsetTop * 1);\n }\n else {\n offsetTop = dimensions.offset(element[0]).top - dimensions.css(element[0], 'marginTop', true) + (options.offsetTop * 1);\n }\n }\n else {\n offsetTop = options.offsetTop * 1;\n }\n }\n\n if(options.offsetBottom) {\n if(options.offsetParent && options.offsetBottom.match(/^[-+]\\d+$/)) {\n // add 1 pixel due to rounding problems...\n offsetBottom = getScrollHeight() - (dimensions.offset(parent[0]).top + dimensions.height(parent[0])) + (options.offsetBottom * 1) + 1;\n }\n else {\n offsetBottom = options.offsetBottom * 1;\n }\n }\n\n // Bring back the element's position after calculations\n if (options.inlineStyles){\n element.css('position', initialPosition);\n }\n };\n\n // Private methods\n\n function getRequiredAffixClass(unpin, position, elementHeight) {\n\n var scrollTop = getScrollTop();\n var scrollHeight = getScrollHeight();\n\n if(scrollTop <= offsetTop) {\n return 'top';\n } else if(unpin !== null && (scrollTop + unpin <= position.top)) {\n return 'middle';\n } else if(offsetBottom !== null && (position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom)) {\n return 'bottom';\n } else {\n return 'middle';\n }\n\n }\n\n function getScrollTop() {\n return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop;\n }\n\n function getScrollHeight() {\n return targetEl[0] === $window ? $window.document.body.scrollHeight : targetEl[0].scrollHeight;\n }\n\n $affix.init();\n return $affix;\n\n }\n\n return AffixFactory;\n\n };\n\n })\n\n .directive('bsAffix', function($affix, $window) {\n\n return {\n restrict: 'EAC',\n require: '^?bsAffixTarget',\n link: function postLink(scope, element, attr, affixTarget) {\n\n var options = {scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window)};\n angular.forEach(['offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles'], function(key) {\n if(angular.isDefined(attr[key])) {\n var option = attr[key];\n if (/true/i.test(option)) option = true;\n if (/false/i.test(option)) option = false;\n options[key] = option;\n }\n });\n\n var affix = $affix(element, options);\n scope.$on('$destroy', function() {\n affix && affix.destroy();\n options = null;\n affix = null;\n });\n\n }\n };\n\n })\n\n .directive('bsAffixTarget', function() {\n return {\n controller: function($element) {\n this.$element = $element;\n }\n };\n });\n\n// Source: module.js\nangular.module('mgcrea.ngStrap', [\n 'mgcrea.ngStrap.modal',\n 'mgcrea.ngStrap.aside',\n 'mgcrea.ngStrap.alert',\n 'mgcrea.ngStrap.button',\n 'mgcrea.ngStrap.select',\n 'mgcrea.ngStrap.datepicker',\n 'mgcrea.ngStrap.timepicker',\n 'mgcrea.ngStrap.navbar',\n 'mgcrea.ngStrap.tooltip',\n 'mgcrea.ngStrap.popover',\n 'mgcrea.ngStrap.dropdown',\n 'mgcrea.ngStrap.typeahead',\n 'mgcrea.ngStrap.scrollspy',\n 'mgcrea.ngStrap.affix',\n 'mgcrea.ngStrap.tab',\n 'mgcrea.ngStrap.collapse'\n]);\n\n})(window, document);\n","'use strict';\n\nangular.module('mgcrea.ngStrap.typeahead', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$typeahead', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'typeahead',\n prefixEvent: '$typeahead',\n placement: 'bottom-left',\n templateUrl: 'typeahead/typeahead.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n minLength: 1,\n filter: 'bsAsyncFilter',\n limit: 6,\n autoSelect: false,\n comparator: '',\n trimValue: true\n };\n\n this.$get = function($window, $rootScope, $tooltip, $$rAF, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n\n function TypeaheadFactory(element, controller, config) {\n\n var $typeahead = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $typeahead = $tooltip(element, options);\n var parentScope = config.scope;\n var scope = $typeahead.$scope;\n\n scope.$resetMatches = function() {\n scope.$matches = [];\n scope.$activeIndex = options.autoSelect ? 0 : -1; // If set to 0, the first match will be highlighted\n };\n scope.$resetMatches();\n\n scope.$activate = function(index) {\n scope.$$postDigest(function() {\n $typeahead.activate(index);\n });\n };\n\n scope.$select = function(index, evt) {\n scope.$$postDigest(function() {\n $typeahead.select(index);\n });\n };\n\n scope.$isVisible = function() {\n return $typeahead.$isVisible();\n };\n\n // Public methods\n\n $typeahead.update = function(matches) {\n scope.$matches = matches;\n if (scope.$activeIndex >= matches.length) {\n scope.$activeIndex = options.autoSelect ? 0 : -1;\n }\n\n // wrap in a $timeout so the results are updated\n // before repositioning\n safeDigest(scope);\n $$rAF($typeahead.$applyPlacement);\n };\n\n $typeahead.activate = function(index) {\n scope.$activeIndex = index;\n };\n\n $typeahead.select = function(index) {\n if (index === -1) return;\n var value = scope.$matches[index].value;\n // console.log('$setViewValue', value);\n controller.$setViewValue(value);\n controller.$render();\n scope.$resetMatches();\n if (parentScope) parentScope.$digest();\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $typeahead);\n };\n\n // Protected methods\n\n $typeahead.$isVisible = function() {\n if (!options.minLength || !controller) {\n return !!scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength;\n };\n\n $typeahead.$getIndex = function(value) {\n var l = scope.$matches.length,\n i = l;\n if (!l) return;\n for (i = l; i--;) {\n if (scope.$matches[i].value === value) break;\n }\n if (i < 0) return;\n return i;\n };\n\n $typeahead.$onMouseDown = function(evt) {\n // Prevent blur on mousedown\n evt.preventDefault();\n evt.stopPropagation();\n };\n\n $typeahead.$onKeyDown = function(evt) {\n if (!/(38|40|13)/.test(evt.keyCode)) return;\n\n // Let ngSubmit pass if the typeahead tip is hidden or no option is selected\n if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // Select with enter\n if (evt.keyCode === 13 && scope.$matches.length) {\n $typeahead.select(scope.$activeIndex);\n }\n\n // Navigate with keyboard\n else if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n };\n\n // Overrides\n\n var show = $typeahead.show;\n $typeahead.show = function() {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed immediately.\n $timeout(function() {\n $typeahead.$element && $typeahead.$element.on('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n element && element.on('keydown', $typeahead.$onKeyDown);\n }\n }, 0, false);\n };\n\n var hide = $typeahead.hide;\n $typeahead.hide = function() {\n $typeahead.$element && $typeahead.$element.off('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n element && element.off('keydown', $typeahead.$onKeyDown);\n }\n if (!options.autoSelect)\n $typeahead.activate(-1);\n hide();\n };\n\n return $typeahead;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n TypeaheadFactory.defaults = defaults;\n return TypeaheadFactory;\n\n };\n\n })\n\n .filter('bsAsyncFilter', function($filter) {\n return function(array, expression, comparator) {\n if (array && angular.isFunction(array.then)) {\n return array.then(function(results) {\n return $filter('filter')(results, expression, comparator);\n });\n } else {\n return $filter('filter')(array, expression, comparator);\n }\n };\n })\n\n .directive('bsTypeahead', function($window, $parse, $q, $typeahead, $parseOptions) {\n\n var defaults = $typeahead.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'filter', 'limit', 'minLength', 'watchOptions', 'selectMode', 'autoSelect', 'comparator', 'id', 'prefixEvent', 'prefixClass'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'trimValue'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // Disable browser autocompletion\n if (!element.attr('autocomplete')) element.attr('autocomplete', 'off');\n\n // Build proper bsOptions\n var filter = options.filter || defaults.filter;\n var limit = options.limit || defaults.limit;\n var comparator = options.comparator || defaults.comparator;\n\n var bsOptions = attr.bsOptions;\n if (filter) bsOptions += ' | ' + filter + ':$viewValue';\n if (comparator) bsOptions += ':' + comparator;\n if (limit) bsOptions += ' | limitTo:' + limit;\n var parsedOptions = $parseOptions(bsOptions);\n\n // Initialize typeahead\n var typeahead = $typeahead(element, controller, options);\n\n // Watch options on demand\n if (options.watchOptions) {\n // Watch bsOptions values before filtering for changes, drop function calls\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').replace(/\\(.*\\)/g, '').trim();\n scope.$watchCollection(watchedOptions, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller).then(function(values) {\n typeahead.update(values);\n controller.$render();\n });\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('$watch', element.attr('ng-model'), newValue);\n scope.$modelValue = newValue; // Publish modelValue on scope for custom templates\n parsedOptions.valuesFn(scope, controller)\n .then(function(values) {\n // Prevent input with no future prospect if selectMode is truthy\n // @TODO test selectMode\n if (options.selectMode && !values.length && newValue.length > 0) {\n controller.$setViewValue(controller.$viewValue.substring(0, controller.$viewValue.length - 1));\n return;\n }\n if (values.length > limit) values = values.slice(0, limit);\n typeahead.update(values);\n // Queue a new rendering that will leverage collection loading\n controller.$render();\n });\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var displayValue = parsedOptions.displayValue(modelValue);\n\n // If we can determine the displayValue, use that\n if (displayValue) {\n return displayValue;\n }\n\n // If there's no display value, attempt to use the modelValue.\n // If the model is an object not much we can do\n if (modelValue && typeof modelValue !== 'object') {\n return modelValue;\n }\n return '';\n });\n\n // Model rendering in view\n controller.$render = function() {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n if (controller.$isEmpty(controller.$viewValue)) {\n return element.val('');\n }\n var index = typeahead.$getIndex(controller.$modelValue);\n var selected = angular.isDefined(index) ? typeahead.$scope.$matches[index].label : controller.$viewValue;\n selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected;\n var value = selected ? selected.toString().replace(/<(?:.|\\n)*?>/gm, '') : '';\n element.val(options.trimValue === false ? value : value.trim());\n };\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (typeahead) typeahead.destroy();\n options = null;\n typeahead = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\n// NOTICE: This file was forked from the angular-material project (github.com/angular/material)\n// MIT Licensed - Copyright (c) 2014-2015 Google, Inc. http://angularjs.org\n\nangular.module('mgcrea.ngStrap.core', [])\n .service('$bsCompiler', bsCompilerService);\n\nfunction bsCompilerService($q, $http, $injector, $compile, $controller, $templateCache) {\n /* jshint validthis: true */\n\n /*\n * @ngdoc service\n * @name $bsCompiler\n * @module material.core\n * @description\n * The $bsCompiler service is an abstraction of angular's compiler, that allows the developer\n * to easily compile an element with a templateUrl, controller, and locals.\n *\n * @usage\n * \n * $bsCompiler.compile({\n * templateUrl: 'modal.html',\n * controller: 'ModalCtrl',\n * locals: {\n * modal: myModalInstance;\n * }\n * }).then(function(compileData) {\n * compileData.element; // modal.html's template in an element\n * compileData.link(myScope); //attach controller & scope to element\n * });\n * \n */\n\n /*\n * @ngdoc method\n * @name $bsCompiler#compile\n * @description A helper to compile an HTML template/templateUrl with a given controller,\n * locals, and scope.\n * @param {object} options An options object, with the following properties:\n *\n * - `controller` - `{(string=|function()=}` Controller fn that should be associated with\n * newly created scope or the name of a registered controller if passed as a string.\n * - `controllerAs` - `{string=}` A controller alias name. If present the controller will be\n * published to scope under the `controllerAs` name.\n * - `template` - `{string=}` An html template as a string.\n * - `templateUrl` - `{string=}` A path to an html template.\n * - `transformTemplate` - `{function(template)=}` A function which transforms the template after\n * it is loaded. It will be given the template string as a parameter, and should\n * return a a new string representing the transformed template.\n * - `resolve` - `{Object.=}` - An optional map of dependencies which should\n * be injected into the controller. If any of these dependencies are promises, the compiler\n * will wait for them all to be resolved, or if one is rejected before the controller is\n * instantiated `compile()` will fail..\n * * `key` - `{string}`: a name of a dependency to be injected into the controller.\n * * `factory` - `{string|function}`: If `string` then it is an alias for a service.\n * Otherwise if function, then it is injected and the return value is treated as the\n * dependency. If the result is a promise, it is resolved before its value is\n * injected into the controller.\n *\n * @returns {object=} promise A promise, which will be resolved with a `compileData` object.\n * `compileData` has the following properties:\n *\n * - `element` - `{element}`: an uncompiled element matching the provided template.\n * - `link` - `{function(scope)}`: A link function, which, when called, will compile\n * the element and instantiate the provided controller (if given).\n * - `locals` - `{object}`: The locals which will be passed into the controller once `link` is\n * called. If `bindToController` is true, they will be coppied to the ctrl instead\n * - `bindToController` - `bool`: bind the locals to the controller, instead of passing them in.\n */\n this.compile = function(options) {\n\n if(options.template && /\\.html$/.test(options.template)) {\n console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.');\n options.templateUrl = options.template;\n options.template = '';\n }\n\n var templateUrl = options.templateUrl;\n var template = options.template || '';\n var controller = options.controller;\n var controllerAs = options.controllerAs;\n var resolve = angular.copy(options.resolve || {});\n var locals = angular.copy(options.locals || {});\n var transformTemplate = options.transformTemplate || angular.identity;\n var bindToController = options.bindToController;\n\n // Take resolve values and invoke them.\n // Resolves can either be a string (value: 'MyRegisteredAngularConst'),\n // or an invokable 'factory' of sorts: (value: function ValueGetter($dependency) {})\n angular.forEach(resolve, function(value, key) {\n if (angular.isString(value)) {\n resolve[key] = $injector.get(value);\n } else {\n resolve[key] = $injector.invoke(value);\n }\n });\n // Add the locals, which are just straight values to inject\n // eg locals: { three: 3 }, will inject three into the controller\n angular.extend(resolve, locals);\n\n if (template) {\n resolve.$template = $q.when(template);\n } else if (templateUrl) {\n resolve.$template = fetchTemplate(templateUrl);\n } else {\n throw new Error('Missing `template` / `templateUrl` option.');\n }\n\n if (options.contentTemplate) {\n // TODO(mgcrea): deprecate?\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.contentTemplate)])\n .then(function(templates) {\n var templateEl = angular.element(templates[0]);\n var contentEl = findElement('[ng-bind=\"content\"]', templateEl[0]).removeAttr('ng-bind').html(templates[1]);\n // Drop the default footer as you probably don't want it if you use a custom contentTemplate\n if(!options.templateUrl) contentEl.next().remove();\n return templateEl[0].outerHTML;\n });\n }\n\n // Wait for all the resolves to finish if they are promises\n return $q.all(resolve).then(function(locals) {\n\n var template = transformTemplate(locals.$template);\n if (options.html) {\n template = template.replace(/ng-bind=\"/ig, 'ng-bind-html=\"');\n }\n // var element = options.element || angular.element('
').html(template.trim()).contents();\n var element = angular.element('
').html(template.trim()).contents();\n var linkFn = $compile(element);\n\n // Return a linking function that can be used later when the element is ready\n return {\n locals: locals,\n element: element,\n link: function link(scope) {\n locals.$scope = scope;\n\n // Instantiate controller if it exists, because we have scope\n if (controller) {\n var invokeCtrl = $controller(controller, locals, true);\n if (bindToController) {\n angular.extend(invokeCtrl.instance, locals);\n }\n // Support angular@~1.2 invokeCtrl\n var ctrl = angular.isObject(invokeCtrl) ? invokeCtrl : invokeCtrl();\n // See angular-route source for this logic\n element.data('$ngControllerController', ctrl);\n element.children().data('$ngControllerController', ctrl);\n\n if (controllerAs) {\n scope[controllerAs] = ctrl;\n }\n }\n\n return linkFn.apply(null, arguments);\n }\n };\n });\n\n };\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate(template) {\n if(fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache})\n .then(function(res) {\n return res.data;\n }));\n }\n\n}\n","'use strict';\n\nangular.module('mgcrea.ngStrap.dropdown', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$dropdown', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'dropdown',\n prefixEvent: 'dropdown',\n placement: 'bottom-left',\n templateUrl: 'dropdown/dropdown.tpl.html',\n trigger: 'click',\n container: false,\n keyboard: true,\n html: false,\n delay: 0\n };\n\n this.$get = function($window, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var matchesSelector = Element.prototype.matchesSelector || Element.prototype.webkitMatchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector;\n\n function DropdownFactory(element, config) {\n\n var $dropdown = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var scope = $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n $dropdown = $tooltip(element, options);\n var parentEl = element.parent();\n\n // Protected methods\n\n $dropdown.$onKeyDown = function(evt) {\n if (!/(38|40)/.test(evt.keyCode)) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Retrieve focused index\n var items = angular.element($dropdown.$element[0].querySelectorAll('li:not(.divider) a'));\n if (!items.length) return;\n var index;\n angular.forEach(items, function(el, i) {\n if (matchesSelector && matchesSelector.call(el, ':focus')) index = i;\n });\n\n // Navigate with keyboard\n if (evt.keyCode === 38 && index > 0) index--;\n else if (evt.keyCode === 40 && index < items.length - 1) index++;\n else if (angular.isUndefined(index)) index = 0;\n items.eq(index)[0].focus();\n\n };\n\n // Overrides\n\n var show = $dropdown.show;\n $dropdown.show = function() {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n options.keyboard && $dropdown.$element && $dropdown.$element.on('keydown', $dropdown.$onKeyDown);\n bodyEl.on('click', onBodyClick);\n }, 0, false);\n parentEl.hasClass('dropdown') && parentEl.addClass('open');\n };\n\n var hide = $dropdown.hide;\n $dropdown.hide = function() {\n if (!$dropdown.$isShown) return;\n options.keyboard && $dropdown.$element && $dropdown.$element.off('keydown', $dropdown.$onKeyDown);\n bodyEl.off('click', onBodyClick);\n parentEl.hasClass('dropdown') && parentEl.removeClass('open');\n hide();\n };\n\n var destroy = $dropdown.destroy;\n $dropdown.destroy = function() {\n bodyEl.off('click', onBodyClick);\n destroy();\n };\n\n // Private functions\n\n function onBodyClick(evt) {\n if (evt.target === element[0]) return;\n return evt.target !== element[0] && $dropdown.hide();\n }\n\n return $dropdown;\n\n }\n\n return DropdownFactory;\n\n };\n\n })\n\n .directive('bsDropdown', function($window, $sce, $dropdown) {\n\n return {\n restrict: 'EAC',\n scope: true,\n compile: function(tElement, tAttrs) {\n\n // Directive options\n var options = {};\n\n // Support for inlined template (next sibling)\n // It must be fetched before compilation\n if (!tAttrs.bsDropdown) {\n var nextSibling = tElement[0].nextSibling;\n while (nextSibling && nextSibling.nodeType !== 1) {\n nextSibling = nextSibling.nextSibling;\n }\n if (nextSibling.classList.contains('dropdown-menu')) {\n options.template = nextSibling.outerHTML;\n options.templateUrl = undefined;\n nextSibling.parentNode.removeChild(nextSibling);\n }\n }\n\n return function postLink(scope, element, attr) {\n\n // Directive options\n options.scope = scope;\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id'], function(key) {\n if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as an object\n attr.bsDropdown && scope.$watch(attr.bsDropdown, function(newValue, oldValue) {\n scope.content = newValue;\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!dropdown || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i);\n newValue === true ? dropdown.show() : dropdown.hide();\n });\n\n // Initialize dropdown\n var dropdown = $dropdown(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (dropdown) dropdown.destroy();\n options = null;\n dropdown = null;\n });\n\n };\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.tooltip', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$tooltip', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n prefixClass: 'tooltip',\n prefixEvent: 'tooltip',\n container: false,\n target: false,\n placement: 'top',\n templateUrl: 'tooltip/tooltip.tpl.html',\n template: '',\n contentTemplate: false,\n trigger: 'hover focus',\n keyboard: false,\n html: false,\n show: false,\n title: '',\n type: '',\n delay: 0,\n autoClose: false,\n bsEnabled: true,\n viewport: {\n selector: 'body',\n padding: 0\n }\n };\n\n this.$get = function($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) {\n\n var trim = String.prototype.trim;\n var isTouch = 'createTouch' in $window.document;\n var htmlReplaceRegExp = /ng-bind=\"/ig;\n var $body = angular.element($window.document);\n\n function TooltipFactory(element, config) {\n\n var $tooltip = {};\n\n // Common vars\n var options = $tooltip.$options = angular.extend({}, defaults, config);\n var promise = $tooltip.$promise = $bsCompiler.compile(options);\n var scope = $tooltip.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n var nodeName = element[0].nodeName.toLowerCase();\n if(options.delay && angular.isString(options.delay)) {\n var split = options.delay.split(',').map(parseFloat);\n options.delay = split.length > 1 ? {show: split[0], hide: split[1]} : split[0];\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $tooltip.$id = options.id || element.attr('id') || '';\n\n // Support scope as string options\n if(options.title) {\n scope.title = $sce.trustAsHtml(options.title);\n }\n\n // Provide scope helpers\n scope.$setEnabled = function(isEnabled) {\n scope.$$postDigest(function() {\n $tooltip.setEnabled(isEnabled);\n });\n };\n scope.$hide = function() {\n scope.$$postDigest(function() {\n $tooltip.hide();\n });\n };\n scope.$show = function() {\n scope.$$postDigest(function() {\n $tooltip.show();\n });\n };\n scope.$toggle = function() {\n scope.$$postDigest(function() {\n $tooltip.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $tooltip.$isShown = scope.$isShown = false;\n\n // Private vars\n var timeout, hoverState;\n\n // Fetch, compile then initialize tooltip\n var compileData, tipElement, tipContainer, tipScope;\n promise.then(function(data) {\n compileData = data;\n $tooltip.init();\n });\n\n $tooltip.init = function() {\n\n // Options: delay\n if (options.delay && angular.isNumber(options.delay)) {\n options.delay = {\n show: options.delay,\n hide: options.delay\n };\n }\n\n // Replace trigger on touch devices ?\n // if(isTouch && options.trigger === defaults.trigger) {\n // options.trigger.replace(/hover/g, 'click');\n // }\n\n // Options : container\n if(options.container === 'self') {\n tipContainer = element;\n } else if(angular.isElement(options.container)) {\n tipContainer = options.container;\n } else if(options.container) {\n tipContainer = findElement(options.container);\n }\n\n // Options: trigger\n bindTriggerEvents();\n\n // Options: target\n if(options.target) {\n options.target = angular.isElement(options.target) ? options.target : findElement(options.target);\n }\n\n // Options: show\n if(options.show) {\n scope.$$postDigest(function() {\n options.trigger === 'focus' ? element[0].focus() : $tooltip.show();\n });\n }\n\n };\n\n $tooltip.destroy = function() {\n\n // Unbind events\n unbindTriggerEvents();\n\n // Remove element\n destroyTipElement();\n\n // Destroy scope\n scope.$destroy();\n\n };\n\n $tooltip.enter = function() {\n\n clearTimeout(timeout);\n hoverState = 'in';\n if (!options.delay || !options.delay.show) {\n return $tooltip.show();\n }\n\n timeout = setTimeout(function() {\n if (hoverState ==='in') $tooltip.show();\n }, options.delay.show);\n\n };\n\n $tooltip.show = function() {\n if (!options.bsEnabled || $tooltip.$isShown) return;\n\n scope.$emit(options.prefixEvent + '.show.before', $tooltip);\n var parent, after;\n if (options.container) {\n parent = tipContainer;\n if (tipContainer[0].lastChild) {\n after = angular.element(tipContainer[0].lastChild);\n } else {\n after = null;\n }\n } else {\n parent = null;\n after = element;\n }\n\n\n // Hide any existing tipElement\n if(tipElement) destroyTipElement();\n // Fetch a cloned element linked from template\n tipScope = $tooltip.$scope.$new();\n tipElement = $tooltip.$element = compileData.link(tipScope, function(clonedElement, scope) {});\n\n // Set the initial positioning. Make the tooltip invisible\n // so IE doesn't try to focus on it off screen.\n tipElement.css({top: '-9999px', left: '-9999px', right: 'auto', display: 'block', visibility: 'hidden'});\n\n // Options: animation\n if(options.animation) tipElement.addClass(options.animation);\n // Options: type\n if(options.type) tipElement.addClass(options.prefixClass + '-' + options.type);\n // Options: custom classes\n if(options.customClass) tipElement.addClass(options.customClass);\n\n // Append the element, without any animations. If we append\n // using $animate.enter, some of the animations cause the placement\n // to be off due to the transforms.\n after ? after.after(tipElement) : parent.prepend(tipElement);\n\n $tooltip.$isShown = scope.$isShown = true;\n safeDigest(scope);\n\n // Now, apply placement\n $tooltip.$applyPlacement();\n\n // Once placed, animate it.\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.enter(tipElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(tipElement, parent, after).then(enterAnimateCallback);\n }\n safeDigest(scope);\n\n $$rAF(function () {\n // Once the tooltip is placed and the animation starts, make the tooltip visible\n if(tipElement) tipElement.css({visibility: 'visible'});\n\n // Bind events\n if(options.keyboard) {\n if(options.trigger !== 'focus') {\n $tooltip.focus();\n }\n bindKeyboardEvents();\n }\n });\n\n if(options.autoClose) {\n bindAutoCloseEvents();\n }\n\n };\n\n function enterAnimateCallback() {\n scope.$emit(options.prefixEvent + '.show', $tooltip);\n }\n\n $tooltip.leave = function() {\n\n clearTimeout(timeout);\n hoverState = 'out';\n if (!options.delay || !options.delay.hide) {\n return $tooltip.hide();\n }\n timeout = setTimeout(function () {\n if (hoverState === 'out') {\n $tooltip.hide();\n }\n }, options.delay.hide);\n\n };\n\n var _blur;\n var _tipToHide;\n $tooltip.hide = function(blur) {\n\n if(!$tooltip.$isShown) return;\n scope.$emit(options.prefixEvent + '.hide.before', $tooltip);\n\n // store blur value for leaveAnimateCallback to use\n _blur = blur;\n\n // store current tipElement reference to use\n // in leaveAnimateCallback\n _tipToHide = tipElement;\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.leave(tipElement, leaveAnimateCallback);\n } else {\n $animate.leave(tipElement).then(leaveAnimateCallback);\n }\n\n $tooltip.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n if(options.keyboard && tipElement !== null) {\n unbindKeyboardEvents();\n }\n\n if(options.autoClose && tipElement !== null) {\n unbindAutoCloseEvents();\n }\n };\n\n function leaveAnimateCallback() {\n scope.$emit(options.prefixEvent + '.hide', $tooltip);\n\n // check if current tipElement still references\n // the same element when hide was called\n if (tipElement === _tipToHide) {\n // Allow to blur the input when hidden, like when pressing enter key\n if(_blur && options.trigger === 'focus') {\n return element[0].blur();\n }\n\n // clean up child scopes\n destroyTipElement();\n }\n }\n\n $tooltip.toggle = function() {\n $tooltip.$isShown ? $tooltip.leave() : $tooltip.enter();\n };\n\n $tooltip.focus = function() {\n tipElement[0].focus();\n };\n\n $tooltip.setEnabled = function(isEnabled) {\n options.bsEnabled = isEnabled;\n };\n\n $tooltip.setViewport = function(viewport) {\n options.viewport = viewport;\n };\n\n // Protected methods\n\n $tooltip.$applyPlacement = function() {\n if(!tipElement) return;\n\n // Determine if we're doing an auto or normal placement\n var placement = options.placement,\n autoToken = /\\s?auto?\\s?/i,\n autoPlace = autoToken.test(placement);\n\n if (autoPlace) {\n placement = placement.replace(autoToken, '') || defaults.placement;\n }\n\n // Need to add the position class before we get\n // the offsets\n tipElement.addClass(options.placement);\n\n // Get the position of the target element\n // and the height and width of the tooltip so we can center it.\n var elementPosition = getPosition(),\n tipWidth = tipElement.prop('offsetWidth'),\n tipHeight = tipElement.prop('offsetHeight');\n\n // Refresh viewport position\n $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport);\n\n // If we're auto placing, we need to check the positioning\n if (autoPlace) {\n var originalPlacement = placement;\n var viewportPosition = getPosition($tooltip.$viewport);\n\n // Determine if the vertical placement\n if (originalPlacement.indexOf('bottom') >= 0 && elementPosition.bottom + tipHeight > viewportPosition.bottom) {\n placement = originalPlacement.replace('bottom', 'top');\n } else if (originalPlacement.indexOf('top') >= 0 && elementPosition.top - tipHeight < viewportPosition.top) {\n placement = originalPlacement.replace('top', 'bottom');\n }\n\n // Determine the horizontal placement\n // The exotic placements of left and right are opposite of the standard placements. Their arrows are put on the left/right\n // and flow in the opposite direction of their placement.\n if ((originalPlacement === 'right' || originalPlacement === 'bottom-left' || originalPlacement === 'top-left') &&\n elementPosition.right + tipWidth > viewportPosition.width) {\n\n placement = originalPlacement === 'right' ? 'left' : placement.replace('left', 'right');\n } else if ((originalPlacement === 'left' || originalPlacement === 'bottom-right' || originalPlacement === 'top-right') &&\n elementPosition.left - tipWidth < viewportPosition.left) {\n\n placement = originalPlacement === 'left' ? 'right' : placement.replace('right', 'left');\n }\n\n tipElement.removeClass(originalPlacement).addClass(placement);\n }\n\n // Get the tooltip's top and left coordinates to center it with this directive.\n var tipPosition = getCalculatedOffset(placement, elementPosition, tipWidth, tipHeight);\n applyPlacement(tipPosition, placement);\n };\n\n $tooltip.$onKeyUp = function(evt) {\n if (evt.which === 27 && $tooltip.$isShown) {\n $tooltip.hide();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusKeyUp = function(evt) {\n if (evt.which === 27) {\n element[0].blur();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusElementMouseDown = function(evt) {\n evt.preventDefault();\n evt.stopPropagation();\n // Some browsers do not auto-focus buttons (eg. Safari)\n $tooltip.$isShown ? element[0].blur() : element[0].focus();\n };\n\n // bind/unbind events\n function bindTriggerEvents() {\n var triggers = options.trigger.split(' ');\n angular.forEach(triggers, function(trigger) {\n if(trigger === 'click') {\n element.on('click', $tooltip.toggle);\n } else if(trigger !== 'manual') {\n element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n nodeName === 'button' && trigger !== 'hover' && element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n });\n }\n\n function unbindTriggerEvents() {\n var triggers = options.trigger.split(' ');\n for (var i = triggers.length; i--;) {\n var trigger = triggers[i];\n if(trigger === 'click') {\n element.off('click', $tooltip.toggle);\n } else if(trigger !== 'manual') {\n element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n nodeName === 'button' && trigger !== 'hover' && element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n }\n\n function bindKeyboardEvents() {\n if(options.trigger !== 'focus') {\n tipElement.on('keyup', $tooltip.$onKeyUp);\n } else {\n element.on('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n function unbindKeyboardEvents() {\n if(options.trigger !== 'focus') {\n tipElement.off('keyup', $tooltip.$onKeyUp);\n } else {\n element.off('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n var _autoCloseEventsBinded = false;\n function bindAutoCloseEvents() {\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n // Stop propagation when clicking inside tooltip\n tipElement.on('click', stopEventPropagation);\n\n // Hide when clicking outside tooltip\n $body.on('click', $tooltip.hide);\n\n _autoCloseEventsBinded = true;\n }, 0, false);\n }\n\n function unbindAutoCloseEvents() {\n if (_autoCloseEventsBinded) {\n tipElement.off('click', stopEventPropagation);\n $body.off('click', $tooltip.hide);\n _autoCloseEventsBinded = false;\n }\n }\n\n function stopEventPropagation(event) {\n event.stopPropagation();\n }\n\n // Private methods\n\n function getPosition($element) {\n $element = $element || (options.target || element);\n\n var el = $element[0],\n isBody = el.tagName === 'BODY';\n\n var elRect = el.getBoundingClientRect();\n var rect = {};\n\n // IE8 has issues with angular.extend and using elRect directly.\n // By coping the values of elRect into a new object, we can continue to use extend\n for (var p in elRect) {\n // DO NOT use hasOwnProperty when inspecting the return of getBoundingClientRect.\n rect[p] = elRect[p];\n }\n\n if (rect.width === null) {\n // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093\n rect = angular.extend({}, rect, { width: elRect.right - elRect.left, height: elRect.bottom - elRect.top });\n }\n var elOffset = isBody ? { top: 0, left: 0 } : dimensions.offset(el),\n scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0 },\n outerDims = isBody ? { width: document.documentElement.clientWidth, height: $window.innerHeight } : null;\n\n return angular.extend({}, rect, scroll, outerDims, elOffset);\n }\n\n function getCalculatedOffset(placement, position, actualWidth, actualHeight) {\n var offset;\n var split = placement.split('-');\n\n switch (split[0]) {\n case 'right':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left + position.width\n };\n break;\n case 'bottom':\n offset = {\n top: position.top + position.height,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n case 'left':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left - actualWidth\n };\n break;\n default:\n offset = {\n top: position.top - actualHeight,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n }\n\n if(!split[1]) {\n return offset;\n }\n\n // Add support for corners @todo css\n if(split[0] === 'top' || split[0] === 'bottom') {\n switch (split[1]) {\n case 'left':\n offset.left = position.left;\n break;\n case 'right':\n offset.left = position.left + position.width - actualWidth;\n }\n } else if(split[0] === 'left' || split[0] === 'right') {\n switch (split[1]) {\n case 'top':\n offset.top = position.top - actualHeight;\n break;\n case 'bottom':\n offset.top = position.top + position.height;\n }\n }\n\n return offset;\n }\n\n function applyPlacement(offset, placement) {\n var tip = tipElement[0],\n width = tip.offsetWidth,\n height = tip.offsetHeight;\n\n // manually read margins because getBoundingClientRect includes difference\n var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10),\n marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10);\n\n // we must check for NaN for ie 8/9\n if (isNaN(marginTop)) marginTop = 0;\n if (isNaN(marginLeft)) marginLeft = 0;\n\n offset.top = offset.top + marginTop;\n offset.left = offset.left + marginLeft;\n\n // dimensions setOffset doesn't round pixel values\n // so we use setOffset directly with our own function\n dimensions.setOffset(tip, angular.extend({\n using: function (props) {\n tipElement.css({\n top: Math.round(props.top) + 'px',\n left: Math.round(props.left) + 'px',\n right: ''\n });\n }\n }, offset), 0);\n\n // check to see if placing tip in new offset caused the tip to resize itself\n var actualWidth = tip.offsetWidth,\n actualHeight = tip.offsetHeight;\n\n if (placement === 'top' && actualHeight !== height) {\n offset.top = offset.top + height - actualHeight;\n }\n\n // If it's an exotic placement, exit now instead of\n // applying a delta and changing the arrow\n if (/top-left|top-right|bottom-left|bottom-right/.test(placement)) return;\n\n var delta = getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight);\n\n if (delta.left) {\n offset.left += delta.left;\n } else {\n offset.top += delta.top;\n }\n\n dimensions.setOffset(tip, offset);\n\n if (/top|right|bottom|left/.test(placement)) {\n var isVertical = /top|bottom/.test(placement),\n arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight,\n arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight';\n\n replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical);\n }\n }\n\n // @source https://github.com/twbs/bootstrap/blob/v3.3.5/js/tooltip.js#L380\n function getViewportAdjustedDelta(placement, position, actualWidth, actualHeight) {\n var delta = {top: 0, left: 0};\n if (!$tooltip.$viewport) return delta;\n\n var viewportPadding = options.viewport && options.viewport.padding || 0;\n var viewportDimensions = getPosition($tooltip.$viewport);\n\n if (/right|left/.test(placement)) {\n var topEdgeOffset = position.top - viewportPadding - viewportDimensions.scroll;\n var bottomEdgeOffset = position.top + viewportPadding - viewportDimensions.scroll + actualHeight;\n if (topEdgeOffset < viewportDimensions.top) { // top overflow\n delta.top = viewportDimensions.top - topEdgeOffset;\n } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow\n delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset;\n }\n } else {\n var leftEdgeOffset = position.left - viewportPadding;\n var rightEdgeOffset = position.left + viewportPadding + actualWidth;\n if (leftEdgeOffset < viewportDimensions.left) { // left overflow\n delta.left = viewportDimensions.left - leftEdgeOffset;\n } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow\n delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset;\n }\n }\n\n return delta;\n }\n\n function replaceArrow(delta, dimension, isHorizontal) {\n var $arrow = findElement('.tooltip-arrow, .arrow', tipElement[0]);\n\n $arrow.css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')\n .css(isHorizontal ? 'top' : 'left', '');\n }\n\n function destroyTipElement() {\n // Cancel pending callbacks\n clearTimeout(timeout);\n\n if($tooltip.$isShown && tipElement !== null) {\n if(options.autoClose) {\n unbindAutoCloseEvents();\n }\n\n if(options.keyboard) {\n unbindKeyboardEvents();\n }\n }\n\n if(tipScope) {\n tipScope.$destroy();\n tipScope = null;\n }\n\n if(tipElement) {\n tipElement.remove();\n tipElement = $tooltip.$element = null;\n }\n }\n\n return $tooltip;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate(template) {\n if(fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache}).then(function(res) {\n return res.data;\n }));\n }\n\n return TooltipFactory;\n\n };\n\n })\n\n .directive('bsTooltip', function($window, $location, $sce, $tooltip, $$rAF) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if(angular.isDefined(dataTarget)) {\n if(falseValueRegExp.test(dataTarget))\n options.target = false;\n else\n options.target = dataTarget;\n }\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')){\n scope.title = '';\n }\n\n // Observe scope attributes for change\n attr.$observe('title', function(newValue) {\n if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) {\n var oldValue = scope.title;\n scope.title = $sce.trustAsHtml(newValue);\n angular.isDefined(oldValue) && $$rAF(function() {\n tooltip && tooltip.$applyPlacement();\n });\n }\n });\n\n // Support scope as an object\n attr.bsTooltip && scope.$watch(attr.bsTooltip, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.title = newValue;\n }\n angular.isDefined(oldValue) && $$rAF(function() {\n tooltip && tooltip.$applyPlacement();\n });\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if(!tooltip || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i);\n newValue === true ? tooltip.show() : tooltip.hide();\n });\n\n // Enabled binding support\n attr.bsEnabled && scope.$watch(attr.bsEnabled, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.bsEnabled, newValue, oldValue);\n if(!tooltip || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i);\n newValue === false ? tooltip.setEnabled(false) : tooltip.setEnabled(true);\n });\n\n // Viewport support\n attr.viewport && scope.$watch(attr.viewport, function (newValue) {\n if(!tooltip || !angular.isDefined(newValue)) return;\n tooltip.setViewport(newValue);\n });\n\n // Initialize popover\n var tooltip = $tooltip(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if(tooltip) tooltip.destroy();\n options = null;\n tooltip = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.timepicker', ['mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip'])\n\n .provider('$timepicker', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n //uncommenting the following line will break backwards compatability\n // prefixEvent: 'timepicker',\n prefixClass: 'timepicker',\n placement: 'bottom-left',\n templateUrl: 'timepicker/timepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: true,\n timeType: 'date',\n timeFormat: 'shortTime',\n timezone: null,\n modelTimeFormat: null,\n autoclose: false,\n minTime: -Infinity,\n maxTime: +Infinity,\n length: 5,\n hourStep: 1,\n minuteStep: 5,\n secondStep: 5,\n roundDisplay: false,\n iconUp: 'glyphicon glyphicon-chevron-up',\n iconDown: 'glyphicon glyphicon-chevron-down',\n arrowBehavior: 'pager'\n };\n\n this.$get = function($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) {\n\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) {\n defaults.lang = $dateFormatter.getDefaultLocale();\n }\n\n function timepickerFactory(element, controller, config) {\n\n var $timepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $timepicker.$options;\n var scope = $timepicker.$scope;\n\n var lang = options.lang;\n var formatDate = function(date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n function floorMinutes(time) {\n // coeff used to floor current time to nearest minuteStep interval\n var coeff = 1000 * 60 * options.minuteStep;\n return new Date(Math.floor(time.getTime() / coeff) * coeff);\n }\n\n // View vars\n\n var selectedIndex = 0;\n var defaultDate = options.roundDisplay ? floorMinutes(new Date()) : new Date();\n var startDate = controller.$dateValue || defaultDate;\n var viewDate = {\n hour: startDate.getHours(),\n meridian: startDate.getHours() < 12,\n minute: startDate.getMinutes(),\n second: startDate.getSeconds(),\n millisecond: startDate.getMilliseconds()\n };\n\n var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang);\n\n var hoursFormat = $dateFormatter.hoursFormat(format),\n timeSeparator = $dateFormatter.timeSeparator(format),\n minutesFormat = $dateFormatter.minutesFormat(format),\n secondsFormat = $dateFormatter.secondsFormat(format),\n showSeconds = $dateFormatter.showSeconds(format),\n showAM = $dateFormatter.showAM(format);\n\n scope.$iconUp = options.iconUp;\n scope.$iconDown = options.iconDown;\n\n // Scope methods\n\n scope.$select = function(date, index) {\n $timepicker.select(date, index);\n };\n scope.$moveIndex = function(value, index) {\n $timepicker.$moveIndex(value, index);\n };\n scope.$switchMeridian = function(date) {\n $timepicker.switchMeridian(date);\n };\n\n // Public methods\n\n $timepicker.update = function(date) {\n // console.warn('$timepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $timepicker.$date = date;\n angular.extend(viewDate, {\n hour: date.getHours(),\n minute: date.getMinutes(),\n second: date.getSeconds(),\n millisecond: date.getMilliseconds()\n });\n $timepicker.$build();\n } else if (!$timepicker.$isBuilt) {\n $timepicker.$build();\n }\n };\n\n $timepicker.select = function(date, index, keep) {\n // console.warn('$timepicker.select', date, scope.$mode);\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) controller.$dateValue = new Date(1970, 0, 1);\n if (!angular.isDate(date)) date = new Date(date);\n if (index === 0) controller.$dateValue.setHours(date.getHours());\n else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes());\n else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds());\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function() {\n $timepicker.hide(true);\n });\n }\n };\n\n $timepicker.switchMeridian = function(date) {\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n return;\n }\n var hours = (date || controller.$dateValue).getHours();\n controller.$dateValue.setHours(hours < 12 ? hours + 12 : hours - 12);\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n };\n\n // Protected methods\n\n $timepicker.$build = function() {\n // console.warn('$timepicker.$build() viewDate=%o', viewDate);\n var i, midIndex = scope.midIndex = parseInt(options.length / 2, 10);\n var hours = [],\n hour;\n for (i = 0; i < options.length; i++) {\n hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep);\n hours.push({\n date: hour,\n label: formatDate(hour, hoursFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(hour, 0),\n disabled: $timepicker.$isDisabled(hour, 0)\n });\n }\n var minutes = [],\n minute;\n for (i = 0; i < options.length; i++) {\n minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep);\n minutes.push({\n date: minute,\n label: formatDate(minute, minutesFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(minute, 1),\n disabled: $timepicker.$isDisabled(minute, 1)\n });\n }\n var seconds = [],\n second;\n for (i = 0; i < options.length; i++) {\n second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep);\n seconds.push({\n date: second,\n label: formatDate(second, secondsFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(second, 2),\n disabled: $timepicker.$isDisabled(second, 2)\n });\n }\n\n var rows = [];\n for (i = 0; i < options.length; i++) {\n if (showSeconds) {\n rows.push([hours[i], minutes[i], seconds[i]]);\n } else {\n rows.push([hours[i], minutes[i]]);\n }\n }\n scope.rows = rows;\n scope.showSeconds = showSeconds;\n scope.showAM = showAM;\n scope.isAM = ($timepicker.$date || hours[midIndex].date).getHours() < 12;\n scope.timeSeparator = timeSeparator;\n $timepicker.$isBuilt = true;\n };\n\n $timepicker.$isSelected = function(date, index) {\n if (!$timepicker.$date) return false;\n else if (index === 0) {\n return date.getHours() === $timepicker.$date.getHours();\n } else if (index === 1) {\n return date.getMinutes() === $timepicker.$date.getMinutes();\n } else if (index === 2) {\n return date.getSeconds() === $timepicker.$date.getSeconds();\n }\n };\n\n $timepicker.$isDisabled = function(date, index) {\n var selectedTime;\n if (index === 0) {\n selectedTime = date.getTime() + viewDate.minute * 6e4 + viewDate.second * 1e3;\n } else if (index === 1) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.second * 1e3;\n } else if (index === 2) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.minute * 6e4;\n }\n return selectedTime < options.minTime * 1 || selectedTime > options.maxTime * 1;\n };\n\n scope.$arrowAction = function(value, index) {\n if (options.arrowBehavior === 'picker') {\n $timepicker.$setTimeByStep(value, index);\n } else {\n $timepicker.$moveIndex(value, index);\n }\n };\n\n $timepicker.$setTimeByStep = function(value, index) {\n var newDate = new Date($timepicker.$date || startDate);\n var hours = newDate.getHours();\n var minutes = newDate.getMinutes();\n var seconds = newDate.getSeconds();\n if (index === 0) {\n newDate.setHours(hours - (parseInt(options.hourStep, 10) * value));\n } else if (index === 1) {\n newDate.setMinutes(minutes - (parseInt(options.minuteStep, 10) * value));\n } else if (index === 2) {\n newDate.setSeconds(seconds - (parseInt(options.secondStep, 10) * value));\n }\n $timepicker.select(newDate, index, true);\n };\n\n $timepicker.$moveIndex = function(value, index) {\n var targetDate;\n if (index === 0) {\n targetDate = new Date(1970, 0, 1, viewDate.hour + (value * options.length), viewDate.minute, viewDate.second);\n angular.extend(viewDate, {\n hour: targetDate.getHours()\n });\n } else if (index === 1) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute + (value * options.length * options.minuteStep), viewDate.second);\n angular.extend(viewDate, {\n minute: targetDate.getMinutes()\n });\n } else if (index === 2) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute, viewDate.second + (value * options.length * options.secondStep));\n angular.extend(viewDate, {\n second: targetDate.getSeconds()\n });\n }\n $timepicker.$build();\n };\n\n $timepicker.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n if (evt.target.nodeName.toLowerCase() !== 'input') evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $timepicker.$onKeyDown = function(evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Close on enter\n if (evt.keyCode === 13) {\n $timepicker.hide(true);\n return;\n }\n\n // Navigate with keyboard\n var newDate = new Date($timepicker.$date);\n var hours = newDate.getHours(),\n hoursLength = formatDate(newDate, hoursFormat).length;\n var minutes = newDate.getMinutes(),\n minutesLength = formatDate(newDate, minutesFormat).length;\n var seconds = newDate.getSeconds(),\n secondsLength = formatDate(newDate, secondsFormat).length;\n var sepLength = 1;\n var lateralMove = /(37|39)/.test(evt.keyCode);\n var count = 2 + showSeconds * 1 + showAM * 1;\n\n // Navigate indexes (left, right)\n if (lateralMove) {\n if (evt.keyCode === 37) selectedIndex = selectedIndex < 1 ? count - 1 : selectedIndex - 1;\n else if (evt.keyCode === 39) selectedIndex = selectedIndex < count - 1 ? selectedIndex + 1 : 0;\n }\n\n // Update values (up, down)\n var selectRange = [0, hoursLength];\n var incr = 0;\n if (evt.keyCode === 38) incr = -1;\n if (evt.keyCode === 40) incr = +1;\n var isSeconds = selectedIndex === 2 && showSeconds;\n var isMeridian = selectedIndex === 2 && !showSeconds || selectedIndex === 3 && showSeconds;\n if (selectedIndex === 0) {\n newDate.setHours(hours + incr * parseInt(options.hourStep, 10));\n // re-calculate hours length because we have changed hours value\n hoursLength = formatDate(newDate, hoursFormat).length;\n selectRange = [0, hoursLength];\n } else if (selectedIndex === 1) {\n newDate.setMinutes(minutes + incr * parseInt(options.minuteStep, 10));\n // re-calculate minutes length because we have changes minutes value\n minutesLength = formatDate(newDate, minutesFormat).length;\n selectRange = [hoursLength + sepLength, minutesLength];\n } else if (isSeconds) {\n newDate.setSeconds(seconds + incr * parseInt(options.secondStep, 10));\n // re-calculate seconds length because we have changes seconds value\n secondsLength = formatDate(newDate, secondsFormat).length;\n selectRange = [hoursLength + sepLength + minutesLength + sepLength, secondsLength];\n } else if (isMeridian) {\n if (!lateralMove) $timepicker.switchMeridian();\n selectRange = [hoursLength + sepLength + minutesLength + sepLength + (secondsLength + sepLength) * showSeconds, 2];\n }\n $timepicker.select(newDate, selectedIndex, true);\n createSelection(selectRange[0], selectRange[1]);\n parentScope.$digest();\n };\n\n // Private\n\n function createSelection(start, length) {\n var end = start + length;\n if (element[0].createTextRange) {\n var selRange = element[0].createTextRange();\n selRange.collapse(true);\n selRange.moveStart('character', start);\n selRange.moveEnd('character', end);\n selRange.select();\n } else if (element[0].setSelectionRange) {\n element[0].setSelectionRange(start, end);\n } else if (angular.isUndefined(element[0].selectionStart)) {\n element[0].selectionStart = start;\n element[0].selectionEnd = end;\n }\n }\n\n function focusElement() {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $timepicker.init;\n $timepicker.init = function() {\n if (isNative && options.useNative) {\n element.prop('type', 'time');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $timepicker.destroy;\n $timepicker.destroy = function() {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $timepicker.show;\n $timepicker.show = function() {\n if((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n $timepicker.$element && $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n element && element.on('keydown', $timepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $timepicker.hide;\n $timepicker.hide = function(blur) {\n if (!$timepicker.$isShown) return;\n $timepicker.$element && $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n element && element.off('keydown', $timepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $timepicker;\n\n }\n\n timepickerFactory.defaults = defaults;\n return timepickerFactory;\n\n };\n\n })\n\n\n .directive('bsTimepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) {\n\n var defaults = $timepicker.defaults;\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'roundDisplay'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!timepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i);\n newValue === true ? timepicker.show() : timepicker.hide();\n });\n\n // Initialize timepicker\n if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm';\n var timepicker = $timepicker(element, controller, options);\n options = timepicker.$options;\n\n var lang = options.lang;\n var formatDate = function(date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n // Initialize parser\n var dateParser = $dateParser({\n format: options.timeFormat,\n lang: lang\n });\n\n // Observe attributes for changes\n angular.forEach(['minTime', 'maxTime'], function(key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) {\n timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue);\n !isNaN(timepicker.$options[key]) && timepicker.$build();\n validateAgainstMinMaxTime(controller.$dateValue);\n });\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue, controller.$dateValue);\n timepicker.update(controller.$dateValue);\n }, true);\n\n function validateAgainstMinMaxTime(parsedTime) {\n if (!angular.isDate(parsedTime)) return;\n var isMinValid = isNaN(options.minTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) >= options.minTime;\n var isMaxValid = isNaN(options.maxTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) <= options.maxTime;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (!isValid) {\n return;\n }\n controller.$dateValue = parsedTime;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function(viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n controller.$setValidity('date', true);\n return null;\n }\n var parsedTime = angular.isDate(viewValue) ? viewValue : dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedTime || isNaN(parsedTime.getTime())) {\n controller.$setValidity('date', false);\n // Return undefined, causes ngModelController to\n // invalidate model value\n return undefined;\n } else {\n validateAgainstMinMaxTime(parsedTime);\n }\n\n if (options.timeType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true);\n return formatDate(date, options.modelTimeFormat || options.timeFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.timeType === 'number') {\n return date.getTime();\n } else if (options.timeType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.timeType === 'iso') {\n return date.toISOString();\n } else {\n return new Date(date);\n }\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.timeType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelTimeFormat);\n } else if (options.timeType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if(isNaN(date.getTime())) date = new Date(new Date().setMinutes(0) + 36e5);\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getTimeFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function() {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getTimeFormattedString());\n };\n\n function getTimeFormattedString() {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.timeFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (timepicker) timepicker.destroy();\n options = null;\n timepicker = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.tab', [])\n\n .provider('$tab', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n template: 'tab/tab.tpl.html',\n navClass: 'nav-tabs',\n activeClass: 'active'\n };\n\n var controller = this.controller = function($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'navClass', 'activeClass'], function(key) {\n if(angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // Publish options on scope\n $scope.$navClass = self.$options.navClass;\n $scope.$activeClass = self.$options.activeClass;\n\n self.$panes = $scope.$panes = [];\n\n // Please use $activePaneChangeListeners if you use `bsActivePane`\n // Because we removed `ngModel` as default, we rename viewChangeListeners to\n // activePaneChangeListeners to make more sense.\n self.$activePaneChangeListeners = self.$viewChangeListeners = [];\n\n self.$push = function(pane) {\n if(angular.isUndefined(self.$panes.$active)) {\n $scope.$setActive(pane.name || 0);\n }\n self.$panes.push(pane);\n };\n\n self.$remove = function(pane) {\n var index = self.$panes.indexOf(pane);\n var active = self.$panes.$active;\n var activeIndex;\n if(angular.isString(active)) {\n activeIndex = self.$panes.map(function(pane) {\n return pane.name;\n }).indexOf(active);\n } else {\n activeIndex = self.$panes.$active;\n }\n\n // remove pane from $panes array\n self.$panes.splice(index, 1);\n\n if (index < activeIndex) {\n // we removed a pane before the active pane, so we need to\n // decrement the active pane index\n activeIndex--;\n }\n else if (index === activeIndex && activeIndex === self.$panes.length) {\n // we remove the active pane and it was the one at the end,\n // so select the previous one\n activeIndex--;\n }\n if(activeIndex >= 0 && activeIndex < self.$panes.length) {\n self.$setActive(self.$panes[activeIndex].name || activeIndex);\n } else {\n self.$setActive();\n }\n };\n\n self.$setActive = $scope.$setActive = function(value) {\n self.$panes.$active = value;\n self.$activePaneChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n self.$isActive = $scope.$isActive = function($pane, $index) {\n return self.$panes.$active === $pane.name || self.$panes.$active === $index;\n };\n\n };\n\n this.$get = function() {\n var $tab = {};\n $tab.defaults = defaults;\n $tab.controller = controller;\n return $tab;\n };\n\n })\n\n .directive('bsTabs', function($window, $animate, $tab, $parse) {\n\n var defaults = $tab.defaults;\n\n return {\n require: ['?ngModel', 'bsTabs'],\n transclude: true,\n scope: true,\n controller: ['$scope', '$element', '$attrs', $tab.controller],\n templateUrl: function(element, attr) {\n return attr.template || defaults.template;\n },\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // 'ngModel' does interfere with form validation\n // and status, use `bsActivePane` instead to avoid it\n if(ngModelCtrl) {\n\n // Update the modelValue following\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n ngModelCtrl.$setViewValue(bsTabsCtrl.$panes.$active);\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n bsTabsCtrl.$setActive(modelValue);\n return modelValue;\n });\n\n }\n\n if (attrs.bsActivePane) {\n // adapted from angularjs ngModelController bindings\n // https://github.com/angular/angular.js/blob/v1.3.1/src%2Fng%2Fdirective%2Finput.js#L1730\n var parsedBsActivePane = $parse(attrs.bsActivePane);\n\n // Update bsActivePane value with change\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n parsedBsActivePane.assign(scope, bsTabsCtrl.$panes.$active);\n });\n\n // watch bsActivePane for value changes\n scope.$watch(attrs.bsActivePane, function(newValue, oldValue) {\n bsTabsCtrl.$setActive(newValue);\n }, true);\n }\n }\n };\n\n })\n\n .directive('bsPane', function($window, $animate, $sce) {\n\n return {\n require: ['^?ngModel', '^bsTabs'],\n scope: true,\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // Add base class\n element.addClass('tab-pane');\n\n // Observe title attribute for change\n attrs.$observe('title', function(newValue, oldValue) {\n scope.title = $sce.trustAsHtml(newValue);\n });\n\n // Save tab name into scope\n scope.name = attrs.name;\n\n // Add animation class\n if(bsTabsCtrl.$options.animation) {\n element.addClass(bsTabsCtrl.$options.animation);\n }\n\n attrs.$observe('disabled', function(newValue, oldValue) {\n scope.disabled = scope.$eval(newValue);\n });\n\n // Push pane to parent bsTabs controller\n bsTabsCtrl.$push(scope);\n\n // remove pane from tab controller when pane is destroyed\n scope.$on('$destroy', function() {\n bsTabsCtrl.$remove(scope);\n });\n\n function render() {\n var index = bsTabsCtrl.$panes.indexOf(scope);\n $animate[bsTabsCtrl.$isActive(scope, index) ? 'addClass' : 'removeClass'](element, bsTabsCtrl.$options.activeClass);\n }\n\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n render();\n });\n render();\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.select', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$select', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'select',\n prefixEvent: '$select',\n placement: 'bottom-left',\n templateUrl: 'select/select.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n multiple: false,\n allNoneButtons: false,\n sort: true,\n caretHtml: ' ',\n placeholder: 'Choose among the following...',\n allText: 'All',\n noneText: 'None',\n maxLength: 3,\n maxLengthHtml: 'selected',\n iconCheckmark: 'glyphicon glyphicon-ok'\n };\n\n this.$get = function($window, $document, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n\n function SelectFactory(element, controller, config) {\n\n var $select = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $select = $tooltip(element, options);\n var scope = $select.$scope;\n\n scope.$matches = [];\n if (options.multiple) {\n scope.$activeIndex = [];\n }\n else {\n scope.$activeIndex = -1;\n }\n scope.$isMultiple = options.multiple;\n scope.$showAllNoneButtons = options.allNoneButtons && options.multiple;\n scope.$iconCheckmark = options.iconCheckmark;\n scope.$allText = options.allText;\n scope.$noneText = options.noneText;\n\n scope.$activate = function(index) {\n scope.$$postDigest(function() {\n $select.activate(index);\n });\n };\n\n scope.$select = function(index, evt) {\n scope.$$postDigest(function() {\n $select.select(index);\n });\n };\n\n scope.$isVisible = function() {\n return $select.$isVisible();\n };\n\n scope.$isActive = function(index) {\n return $select.$isActive(index);\n };\n\n scope.$selectAll = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (!scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n scope.$selectNone = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n // Public methods\n\n $select.update = function(matches) {\n scope.$matches = matches;\n $select.$updateActiveIndex();\n };\n\n $select.activate = function(index) {\n if(options.multiple) {\n $select.$isActive(index) ? scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1) : scope.$activeIndex.push(index);\n if(options.sort) scope.$activeIndex.sort(function(a, b) { return a - b; }); // use numeric sort instead of default sort\n } else {\n scope.$activeIndex = index;\n }\n return scope.$activeIndex;\n };\n\n $select.select = function(index) {\n var value = scope.$matches[index].value;\n scope.$apply(function() {\n $select.activate(index);\n if(options.multiple) {\n controller.$setViewValue(scope.$activeIndex.map(function(index) {\n if (angular.isUndefined(scope.$matches[index])) {\n return null;\n }\n return scope.$matches[index].value;\n }));\n } else {\n controller.$setViewValue(value);\n // Hide if single select\n $select.hide();\n }\n });\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $select);\n };\n\n // Protected methods\n\n $select.$updateActiveIndex = function() {\n if(options.multiple) {\n if(angular.isArray(controller.$modelValue)) {\n scope.$activeIndex = controller.$modelValue.map(function(value) {\n return $select.$getIndex(value);\n });\n } else {\n scope.$activeIndex = [];\n }\n } else {\n if(angular.isDefined(controller.$modelValue) && scope.$matches.length) {\n scope.$activeIndex = $select.$getIndex(controller.$modelValue);\n } else {\n scope.$activeIndex = -1;\n }\n }\n };\n\n $select.$isVisible = function() {\n if(!options.minLength || !controller) {\n return scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && controller.$viewValue.length >= options.minLength;\n };\n\n $select.$isActive = function(index) {\n if(options.multiple) {\n return scope.$activeIndex.indexOf(index) !== -1;\n } else {\n return scope.$activeIndex === index;\n }\n };\n\n $select.$getIndex = function(value) {\n var l = scope.$matches.length, i = l;\n if(!l) return;\n for(i = l; i--;) {\n if(scope.$matches[i].value === value) break;\n }\n if(i < 0) return;\n return i;\n };\n\n $select.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if(isTouch) {\n var targetEl = angular.element(evt.target);\n targetEl.triggerHandler('click');\n }\n };\n\n $select.$onKeyDown = function(evt) {\n if (!/(9|13|38|40)/.test(evt.keyCode)) return;\n // Let tab propagate\n if (evt.keyCode !== 9) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // release focus on tab\n if (options.multiple && evt.keyCode === 9) {\n return $select.hide();\n }\n\n // Select with enter\n if(!options.multiple && (evt.keyCode === 13 || evt.keyCode === 9)) {\n return $select.select(scope.$activeIndex);\n }\n\n if (!options.multiple) {\n // Navigate with keyboard\n if(evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if(evt.keyCode === 38 && scope.$activeIndex < 0) scope.$activeIndex = scope.$matches.length - 1;\n else if(evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if(angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n }\n };\n\n $select.$isIE = function() {\n var ua = $window.navigator.userAgent;\n return ua.indexOf('MSIE ') > 0 || ua.indexOf('Trident/') > 0 || ua.indexOf('Edge/') > 0;\n };\n\n $select.$selectScrollFix = function(e) {\n if ($document[0].activeElement.tagName === 'UL') {\n e.preventDefault();\n e.stopImmediatePropagation();\n e.target.focus();\n }\n };\n\n // Overrides\n\n var _show = $select.show;\n $select.show = function() {\n _show();\n if(options.multiple) {\n $select.$element.addClass('select-multiple');\n }\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n $select.$element.on(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if(options.keyboard) {\n element.on('keydown', $select.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $select.hide;\n $select.hide = function() {\n if(!options.multiple && angular.isUndefined(controller.$modelValue)) {\n scope.$activeIndex = -1;\n }\n $select.$element.off(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if(options.keyboard) {\n element.off('keydown', $select.$onKeyDown);\n }\n _hide(true);\n };\n\n return $select;\n\n }\n\n SelectFactory.defaults = defaults;\n return SelectFactory;\n\n };\n\n })\n\n .directive('bsSelect', function($window, $parse, $q, $select, $parseOptions) {\n\n var defaults = $select.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope, placeholder: defaults.placeholder};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'allNoneButtons', 'sort'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Only parse data-multiple. Angular sets existence attributes to true (multiple/required/etc), they apply this\n // to data-multiple as well for some reason, so we'll parse this ourselves and disregard multiple\n var dataMultiple = element.attr('data-multiple');\n if(angular.isDefined(dataMultiple)) {\n if(falseValueRegExp.test(dataMultiple))\n options.multiple = false;\n else\n options.multiple = dataMultiple;\n }\n\n // Add support for select markup\n if(element[0].nodeName.toLowerCase() === 'select') {\n var inputEl = element;\n inputEl.css('display', 'none');\n element = angular.element('');\n inputEl.after(element);\n }\n\n // Build proper bsOptions\n var parsedOptions = $parseOptions(attr.bsOptions);\n\n // Initialize select\n var select = $select(element, controller, options);\n\n if (select.$isIE()) {\n element[0].addEventListener('blur', select.$selectScrollFix);\n }\n\n // Watch bsOptions values before filtering for changes\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').trim();\n scope.$watchCollection(watchedOptions, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller)\n .then(function(values) {\n select.update(values);\n controller.$render();\n });\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue);\n select.$updateActiveIndex();\n controller.$render();\n }, true);\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var selected, index;\n if(options.multiple && angular.isArray(controller.$modelValue)) {\n selected = controller.$modelValue.map(function(value) {\n index = select.$getIndex(value);\n return angular.isDefined(index) ? select.$scope.$matches[index].label : false;\n }).filter(angular.isDefined);\n if(selected.length > (options.maxLength || defaults.maxLength)) {\n selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml);\n } else {\n selected = selected.join(', ');\n }\n } else {\n index = select.$getIndex(controller.$modelValue);\n selected = angular.isDefined(index) ? select.$scope.$matches[index].label : false;\n }\n element.html((selected ? selected : options.placeholder) + (options.caretHtml ? options.caretHtml : defaults.caretHtml));\n };\n\n if(options.multiple){\n controller.$isEmpty = function(value){\n return !value || value.length === 0;\n };\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (select) select.destroy();\n options = null;\n select = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.scrollspy', ['mgcrea.ngStrap.helpers.debounce', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$scrollspy', function() {\n\n // Pool of registered spies\n var spies = this.$$spies = {};\n\n var defaults = this.defaults = {\n debounce: 150,\n throttle: 100,\n offset: 100\n };\n\n this.$get = function($window, $document, $rootScope, dimensions, debounce, throttle) {\n\n var windowEl = angular.element($window);\n var docEl = angular.element($document.prop('documentElement'));\n var bodyEl = angular.element($window.document.body);\n\n // Helper functions\n\n function nodeName(element, name) {\n return element[0].nodeName && element[0].nodeName.toLowerCase() === name.toLowerCase();\n }\n\n function ScrollSpyFactory(config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n if(!options.element) options.element = bodyEl;\n var isWindowSpy = nodeName(options.element, 'body');\n var scrollEl = isWindowSpy ? windowEl : options.element;\n var scrollId = isWindowSpy ? 'window' : options.id;\n\n // Use existing spy\n if(spies[scrollId]) {\n spies[scrollId].$$count++;\n return spies[scrollId];\n }\n\n var $scrollspy = {};\n\n // Private vars\n var unbindViewContentLoaded, unbindIncludeContentLoaded;\n var trackedElements = $scrollspy.$trackedElements = [];\n var sortedElements = [];\n var activeTarget;\n var debouncedCheckPosition;\n var throttledCheckPosition;\n var debouncedCheckOffsets;\n var viewportHeight;\n var scrollTop;\n\n $scrollspy.init = function() {\n\n // Setup internal ref counter\n this.$$count = 1;\n\n // Bind events\n debouncedCheckPosition = debounce(this.checkPosition, options.debounce);\n throttledCheckPosition = throttle(this.checkPosition, options.throttle);\n scrollEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', debouncedCheckPosition);\n scrollEl.on('scroll', throttledCheckPosition);\n\n debouncedCheckOffsets = debounce(this.checkOffsets, options.debounce);\n unbindViewContentLoaded = $rootScope.$on('$viewContentLoaded', debouncedCheckOffsets);\n unbindIncludeContentLoaded = $rootScope.$on('$includeContentLoaded', debouncedCheckOffsets);\n debouncedCheckOffsets();\n\n // Register spy for reuse\n if(scrollId) {\n spies[scrollId] = $scrollspy;\n }\n\n };\n\n $scrollspy.destroy = function() {\n\n // Check internal ref counter\n this.$$count--;\n if(this.$$count > 0) {\n return;\n }\n\n // Unbind events\n scrollEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', debouncedCheckPosition);\n scrollEl.off('scroll', throttledCheckPosition);\n unbindViewContentLoaded();\n unbindIncludeContentLoaded();\n if (scrollId) {\n delete spies[scrollId];\n }\n };\n\n $scrollspy.checkPosition = function() {\n\n // Not ready yet\n if(!sortedElements.length) return;\n\n // Calculate the scroll position\n scrollTop = (isWindowSpy ? $window.pageYOffset : scrollEl.prop('scrollTop')) || 0;\n\n // Calculate the viewport height for use by the components\n viewportHeight = Math.max($window.innerHeight, docEl.prop('clientHeight'));\n\n // Activate first element if scroll is smaller\n if(scrollTop < sortedElements[0].offsetTop && activeTarget !== sortedElements[0].target) {\n return $scrollspy.$activateElement(sortedElements[0]);\n }\n\n // Activate proper element\n for (var i = sortedElements.length; i--;) {\n if(angular.isUndefined(sortedElements[i].offsetTop) || sortedElements[i].offsetTop === null) continue;\n if(activeTarget === sortedElements[i].target) continue;\n if(scrollTop < sortedElements[i].offsetTop) continue;\n if(sortedElements[i + 1] && scrollTop > sortedElements[i + 1].offsetTop) continue;\n return $scrollspy.$activateElement(sortedElements[i]);\n }\n\n };\n\n $scrollspy.checkPositionWithEventLoop = function() {\n // IE 9 throws an error if we use 'this' instead of '$scrollspy'\n // in this setTimeout call\n setTimeout($scrollspy.checkPosition, 1);\n };\n\n // Protected methods\n\n $scrollspy.$activateElement = function(element) {\n if(activeTarget) {\n var activeElement = $scrollspy.$getTrackedElement(activeTarget);\n if(activeElement) {\n activeElement.source.removeClass('active');\n if(nodeName(activeElement.source, 'li') && nodeName(activeElement.source.parent().parent(), 'li')) {\n activeElement.source.parent().parent().removeClass('active');\n }\n }\n }\n activeTarget = element.target;\n element.source.addClass('active');\n if(nodeName(element.source, 'li') && nodeName(element.source.parent().parent(), 'li')) {\n element.source.parent().parent().addClass('active');\n }\n };\n\n $scrollspy.$getTrackedElement = function(target) {\n return trackedElements.filter(function(obj) {\n return obj.target === target;\n })[0];\n };\n\n // Track offsets behavior\n\n $scrollspy.checkOffsets = function() {\n\n angular.forEach(trackedElements, function(trackedElement) {\n var targetElement = document.querySelector(trackedElement.target);\n trackedElement.offsetTop = targetElement ? dimensions.offset(targetElement).top : null;\n if(options.offset && trackedElement.offsetTop !== null) trackedElement.offsetTop -= options.offset * 1;\n });\n\n sortedElements = trackedElements\n .filter(function(el) {\n return el.offsetTop !== null;\n })\n .sort(function(a, b) {\n return a.offsetTop - b.offsetTop;\n });\n\n debouncedCheckPosition();\n\n };\n\n $scrollspy.trackElement = function(target, source) {\n trackedElements.push({target: target, source: source});\n };\n\n $scrollspy.untrackElement = function(target, source) {\n var toDelete;\n for (var i = trackedElements.length; i--;) {\n if(trackedElements[i].target === target && trackedElements[i].source === source) {\n toDelete = i;\n break;\n }\n }\n trackedElements = trackedElements.splice(toDelete, 1);\n };\n\n $scrollspy.activate = function(i) {\n trackedElements[i].addClass('active');\n };\n\n // Initialize plugin\n\n $scrollspy.init();\n return $scrollspy;\n\n }\n\n return ScrollSpyFactory;\n\n };\n\n })\n\n .directive('bsScrollspy', function($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'EAC',\n link: function postLink(scope, element, attr) {\n\n var options = {scope: scope};\n angular.forEach(['offset', 'target'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n var scrollspy = $scrollspy(options);\n scrollspy.trackElement(options.target, element);\n\n scope.$on('$destroy', function() {\n if (scrollspy) {\n scrollspy.untrackElement(options.target, element);\n scrollspy.destroy();\n }\n options = null;\n scrollspy = null;\n });\n\n }\n };\n\n })\n\n\n .directive('bsScrollspyList', function($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'A',\n compile: function postLink(element, attr) {\n var children = element[0].querySelectorAll('li > a[href]');\n angular.forEach(children, function(child) {\n var childEl = angular.element(child);\n childEl.parent().attr('bs-scrollspy', '').attr('data-target', childEl.attr('href'));\n });\n }\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.popover', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$popover', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n // uncommenting the next two lines will break backwards compatability\n // prefixClass: 'popover',\n // prefixEvent: 'popover',\n container: false,\n target: false,\n placement: 'right',\n templateUrl: 'popover/popover.tpl.html',\n contentTemplate: false,\n trigger: 'click',\n keyboard: true,\n html: false,\n title: '',\n content: '',\n delay: 0,\n autoClose: false\n };\n\n this.$get = function($tooltip) {\n\n function PopoverFactory(element, config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n var $popover = $tooltip(element, options);\n\n // Support scope as string options [/*title, */content]\n if(options.content) {\n $popover.$scope.content = options.content;\n }\n\n return $popover;\n\n }\n\n return PopoverFactory;\n\n };\n\n })\n\n .directive('bsPopover', function($window, $sce, $popover) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoClose'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if(angular.isDefined(dataTarget)) {\n if(falseValueRegExp.test(dataTarget))\n options.target = false;\n else\n options.target = dataTarget;\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n angular.isDefined(oldValue) && requestAnimationFrame(function() {\n popover && popover.$applyPlacement();\n });\n });\n });\n\n // Support scope as an object\n attr.bsPopover && scope.$watch(attr.bsPopover, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n angular.isDefined(oldValue) && requestAnimationFrame(function() {\n popover && popover.$applyPlacement();\n });\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if(!popover || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i);\n newValue === true ? popover.show() : popover.hide();\n });\n\n // Viewport support\n attr.viewport && scope.$watch(attr.viewport, function (newValue) {\n if(!popover || !angular.isDefined(newValue)) return;\n popover.setViewport(newValue);\n });\n\n // Initialize popover\n var popover = $popover(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (popover) popover.destroy();\n options = null;\n popover = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.navbar', [])\n\n .provider('$navbar', function() {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n routeAttr: 'data-match-route',\n strict: false\n };\n\n this.$get = function() {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsNavbar', function($window, $location, $navbar) {\n\n var defaults = $navbar.defaults;\n\n return {\n restrict: 'A',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = angular.copy(defaults);\n angular.forEach(Object.keys(defaults), function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Watch for the $location\n scope.$watch(function() {\n\n return $location.path();\n\n }, function(newValue, oldValue) {\n\n var liElements = element[0].querySelectorAll('li[' + options.routeAttr + ']');\n\n angular.forEach(liElements, function(li) {\n\n var liElement = angular.element(li);\n var pattern = liElement.attr(options.routeAttr).replace('/', '\\\\/');\n if(options.strict) {\n pattern = '^' + pattern + '$';\n }\n var regexp = new RegExp(pattern, 'i');\n\n if(regexp.test(newValue)) {\n liElement.addClass(options.activeClass);\n } else {\n liElement.removeClass(options.activeClass);\n }\n\n });\n\n });\n\n }\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.modal', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$modal', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n backdropAnimation: 'am-fade',\n prefixClass: 'modal',\n prefixEvent: 'modal',\n placement: 'top',\n templateUrl: 'modal/modal.tpl.html',\n template: '',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) {\n\n var forEach = angular.forEach;\n var trim = String.prototype.trim;\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n var bodyElement = angular.element($window.document.body);\n\n function ModalFactory(config) {\n\n var $modal = {};\n\n // Common vars\n var options = $modal.$options = angular.extend({}, defaults, config);\n var promise = $modal.$promise = $bsCompiler.compile(options);\n var scope = $modal.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n if(!options.element && !options.container) {\n options.container = 'body';\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $modal.$id = options.id || options.element && options.element.attr('id') || '';\n\n // Support scope as string options\n forEach(['title', 'content'], function(key) {\n if(options[key]) scope[key] = $sce.trustAsHtml(options[key]);\n });\n\n // Provide scope helpers\n scope.$hide = function() {\n scope.$$postDigest(function() {\n $modal.hide();\n });\n };\n scope.$show = function() {\n scope.$$postDigest(function() {\n $modal.show();\n });\n };\n scope.$toggle = function() {\n scope.$$postDigest(function() {\n $modal.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $modal.$isShown = scope.$isShown = false;\n\n // Fetch, compile then initialize modal\n var compileData, modalElement, modalScope;\n var backdropElement = angular.element('
');\n backdropElement.css({position:'fixed', top:'0px', left:'0px', bottom:'0px', right:'0px', 'z-index': 1038});\n promise.then(function(data) {\n compileData = data;\n $modal.init();\n });\n\n $modal.init = function() {\n\n // Options: show\n if(options.show) {\n scope.$$postDigest(function() {\n $modal.show();\n });\n }\n\n };\n\n $modal.destroy = function() {\n\n // Remove element\n destroyModalElement();\n\n // remove backdrop element\n if(backdropElement) {\n backdropElement.remove();\n backdropElement = null;\n }\n\n // Destroy scope\n scope.$destroy();\n };\n\n $modal.show = function() {\n if($modal.$isShown) return;\n\n var parent, after;\n if(angular.isElement(options.container)) {\n parent = options.container;\n after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null;\n } else {\n if (options.container) {\n parent = findElement(options.container);\n after = parent[0] && parent[0].lastChild ? angular.element(parent[0].lastChild) : null;\n } else {\n parent = null;\n after = options.element;\n }\n }\n\n // destroy any existing modal elements\n if(modalElement) destroyModalElement();\n\n // create a new scope, so we can destroy it and all child scopes\n // when destroying the modal element\n modalScope = $modal.$scope.$new();\n // Fetch a cloned element linked from template (noop callback is required)\n modalElement = $modal.$element = compileData.link(modalScope, function(clonedElement, scope) {});\n\n if(scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) {\n return;\n }\n\n // Set the initial positioning.\n modalElement.css({display: 'block'}).addClass(options.placement);\n\n // Options: animation\n if(options.animation) {\n if(options.backdrop) {\n backdropElement.addClass(options.backdropAnimation);\n }\n modalElement.addClass(options.animation);\n }\n\n if(options.backdrop) {\n $animate.enter(backdropElement, bodyElement, null);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.enter(modalElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(modalElement, parent, after).then(enterAnimateCallback);\n }\n\n $modal.$isShown = scope.$isShown = true;\n safeDigest(scope);\n // Focus once the enter-animation has started\n // Weird PhantomJS bug hack\n var el = modalElement[0];\n requestAnimationFrame(function() {\n el.focus();\n });\n\n bodyElement.addClass(options.prefixClass + '-open');\n if(options.animation) {\n bodyElement.addClass(options.prefixClass + '-with-' + options.animation);\n }\n\n // Bind events\n bindBackdropEvents();\n bindKeyboardEvents();\n };\n\n function enterAnimateCallback() {\n scope.$emit(options.prefixEvent + '.show', $modal);\n }\n\n $modal.hide = function() {\n if(!$modal.$isShown) return;\n\n if(scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) {\n return;\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.leave(modalElement, leaveAnimateCallback);\n } else {\n $animate.leave(modalElement).then(leaveAnimateCallback);\n }\n\n if(options.backdrop) {\n $animate.leave(backdropElement);\n }\n $modal.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n };\n\n function leaveAnimateCallback() {\n scope.$emit(options.prefixEvent + '.hide', $modal);\n bodyElement.removeClass(options.prefixClass + '-open');\n if(options.animation) {\n bodyElement.removeClass(options.prefixClass + '-with-' + options.animation);\n }\n }\n\n $modal.toggle = function() {\n\n $modal.$isShown ? $modal.hide() : $modal.show();\n\n };\n\n $modal.focus = function() {\n modalElement[0].focus();\n };\n\n // Protected methods\n\n $modal.$onKeyUp = function(evt) {\n\n if (evt.which === 27 && $modal.$isShown) {\n $modal.hide();\n evt.stopPropagation();\n }\n\n };\n\n function bindBackdropEvents() {\n if(options.backdrop) {\n modalElement.on('click', hideOnBackdropClick);\n backdropElement.on('click', hideOnBackdropClick);\n backdropElement.on('wheel', preventEventDefault);\n }\n }\n\n function unbindBackdropEvents() {\n if(options.backdrop) {\n modalElement.off('click', hideOnBackdropClick);\n backdropElement.off('click', hideOnBackdropClick);\n backdropElement.off('wheel', preventEventDefault);\n }\n }\n\n function bindKeyboardEvents() {\n if(options.keyboard) {\n modalElement.on('keyup', $modal.$onKeyUp);\n }\n }\n\n function unbindKeyboardEvents() {\n if(options.keyboard) {\n modalElement.off('keyup', $modal.$onKeyUp);\n }\n }\n\n // Private helpers\n\n function hideOnBackdropClick(evt) {\n if(evt.target !== evt.currentTarget) return;\n options.backdrop === 'static' ? $modal.focus() : $modal.hide();\n }\n\n function preventEventDefault(evt) {\n evt.preventDefault();\n }\n\n function destroyModalElement() {\n if($modal.$isShown && modalElement !== null) {\n // un-bind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n }\n\n if(modalScope) {\n modalScope.$destroy();\n modalScope = null;\n }\n\n if(modalElement) {\n modalElement.remove();\n modalElement = $modal.$element = null;\n }\n }\n\n return $modal;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return ModalFactory;\n\n };\n\n })\n\n .directive('bsModal', function($window, $sce, $modal) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsModal && scope.$watch(attr.bsModal, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize modal\n var modal = $modal(options);\n\n // Trigger\n element.on(attr.trigger || 'click', modal.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (modal) modal.destroy();\n options = null;\n modal = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\n(angular.version.minor < 3 && angular.version.dot < 14) && angular.module('ng')\n\n.factory('$$rAF', function($window, $timeout) {\n\n var requestAnimationFrame = $window.requestAnimationFrame ||\n $window.webkitRequestAnimationFrame ||\n $window.mozRequestAnimationFrame;\n\n var cancelAnimationFrame = $window.cancelAnimationFrame ||\n $window.webkitCancelAnimationFrame ||\n $window.mozCancelAnimationFrame ||\n $window.webkitCancelRequestAnimationFrame;\n\n var rafSupported = !!requestAnimationFrame;\n var raf = rafSupported ?\n function(fn) {\n var id = requestAnimationFrame(fn);\n return function() {\n cancelAnimationFrame(id);\n };\n } :\n function(fn) {\n var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666\n return function() {\n $timeout.cancel(timer);\n };\n };\n\n raf.supported = rafSupported;\n\n return raf;\n\n});\n\n// .factory('$$animateReflow', function($$rAF, $document) {\n\n// var bodyEl = $document[0].body;\n\n// return function(fn) {\n// //the returned function acts as the cancellation function\n// return $$rAF(function() {\n// //the line below will force the browser to perform a repaint\n// //so that all the animated elements within the animation frame\n// //will be properly updated and drawn on screen. This is\n// //required to perform multi-class CSS based animations with\n// //Firefox. DO NOT REMOVE THIS LINE.\n// var a = bodyEl.offsetWidth + 1;\n// fn();\n// });\n// };\n\n// });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.parseOptions', [])\n\n .provider('$parseOptions', function() {\n\n var defaults = this.defaults = {\n regexp: /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n };\n\n this.$get = function($parse, $q) {\n\n function ParseOptionsFactory(attr, config) {\n\n var $parseOptions = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n $parseOptions.$values = [];\n\n // Private vars\n var match, displayFn, valueName, keyName, groupByFn, valueFn, valuesFn;\n\n $parseOptions.init = function() {\n $parseOptions.$match = match = attr.match(options.regexp);\n displayFn = $parse(match[2] || match[1]),\n valueName = match[4] || match[6],\n keyName = match[5],\n groupByFn = $parse(match[3] || ''),\n valueFn = $parse(match[2] ? match[1] : valueName),\n valuesFn = $parse(match[7]);\n };\n\n $parseOptions.valuesFn = function(scope, controller) {\n return $q.when(valuesFn(scope, controller))\n .then(function(values) {\n if(!angular.isArray(values)) {\n values = [];\n }\n $parseOptions.$values = values.length ? parseValues(values, scope) : [];\n return $parseOptions.$values;\n });\n };\n\n $parseOptions.displayValue = function(modelValue) {\n var scope = {};\n scope[valueName] = modelValue;\n return displayFn(scope);\n };\n\n // Private functions\n\n function parseValues(values, scope) {\n return values.map(function(match, index) {\n var locals = {}, label, value;\n locals[valueName] = match;\n label = displayFn(scope, locals);\n value = valueFn(scope, locals);\n return {label: label, value: value, index: index};\n });\n }\n\n $parseOptions.init();\n return $parseOptions;\n\n }\n\n return ParseOptionsFactory;\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dimensions', [])\n\n .factory('dimensions', function($document, $window) {\n\n var jqLite = angular.element;\n var fn = {};\n\n /**\n * Test the element nodeName\n * @param element\n * @param name\n */\n var nodeName = fn.nodeName = function(element, name) {\n return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase();\n };\n\n /**\n * Returns the element computed style\n * @param element\n * @param prop\n * @param extra\n */\n fn.css = function(element, prop, extra) {\n var value;\n if (element.currentStyle) { //IE\n value = element.currentStyle[prop];\n } else if (window.getComputedStyle) {\n value = window.getComputedStyle(element)[prop];\n } else {\n value = element.style[prop];\n }\n return extra === true ? parseFloat(value) || 0 : value;\n };\n\n /**\n * Provides read-only equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.offset = function(element) {\n var boxRect = element.getBoundingClientRect();\n var docElement = element.ownerDocument;\n return {\n width: boxRect.width || element.offsetWidth,\n height: boxRect.height || element.offsetHeight,\n top: boxRect.top + (window.pageYOffset || docElement.documentElement.scrollTop) - (docElement.documentElement.clientTop || 0),\n left: boxRect.left + (window.pageXOffset || docElement.documentElement.scrollLeft) - (docElement.documentElement.clientLeft || 0)\n };\n };\n \n /**\n * Provides set equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip\n * @url http://api.jquery.com/offset/\n * @param element\n * @param options\n * @param i\n */\n fn.setOffset = function (element, options, i) {\n var curPosition,\n curLeft,\n curCSSTop,\n curTop,\n curOffset,\n curCSSLeft,\n calculatePosition,\n position = fn.css(element, 'position'),\n curElem = angular.element(element),\n props = {};\n \n // Set position first, in-case top/left are set even on static elem\n if (position === 'static') {\n element.style.position = 'relative';\n }\n \n curOffset = fn.offset(element);\n curCSSTop = fn.css(element, 'top');\n curCSSLeft = fn.css(element, 'left');\n calculatePosition = (position === 'absolute' || position === 'fixed') && \n (curCSSTop + curCSSLeft).indexOf('auto') > -1;\n \n // Need to be able to calculate position if either\n // top or left is auto and position is either absolute or fixed\n if (calculatePosition) {\n curPosition = fn.position(element);\n curTop = curPosition.top;\n curLeft = curPosition.left;\n } else {\n curTop = parseFloat(curCSSTop) || 0;\n curLeft = parseFloat(curCSSLeft) || 0;\n }\n \n if (angular.isFunction(options)) {\n options = options.call(element, i, curOffset);\n }\n \n if (options.top !== null ) {\n props.top = (options.top - curOffset.top) + curTop;\n }\n if ( options.left !== null ) {\n props.left = (options.left - curOffset.left) + curLeft;\n }\n\n if ('using' in options) {\n options.using.call(curElem, props);\n } else {\n curElem.css({\n top: props.top + 'px',\n left: props.left + 'px'\n });\n }\n };\n\n /**\n * Provides read-only equivalent of jQuery's position function\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.position = function(element) {\n\n var offsetParentRect = {top: 0, left: 0},\n offsetParentElement,\n offset;\n\n // Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is it's only offset parent\n if (fn.css(element, 'position') === 'fixed') {\n\n // We assume that getBoundingClientRect is available when computed position is fixed\n offset = element.getBoundingClientRect();\n\n } else {\n\n // Get *real* offsetParentElement\n offsetParentElement = offsetParent(element);\n\n // Get correct offsets\n offset = fn.offset(element);\n if (!nodeName(offsetParentElement, 'html')) {\n offsetParentRect = fn.offset(offsetParentElement);\n }\n\n // Add offsetParent borders\n offsetParentRect.top += fn.css(offsetParentElement, 'borderTopWidth', true);\n offsetParentRect.left += fn.css(offsetParentElement, 'borderLeftWidth', true);\n }\n\n // Subtract parent offsets and element margins\n return {\n width: element.offsetWidth,\n height: element.offsetHeight,\n top: offset.top - offsetParentRect.top - fn.css(element, 'marginTop', true),\n left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true)\n };\n\n };\n\n /**\n * Returns the closest, non-statically positioned offsetParent of a given element\n * @required-by fn.position\n * @param element\n */\n var offsetParent = function offsetParentElement(element) {\n var docElement = element.ownerDocument;\n var offsetParent = element.offsetParent || docElement;\n if(nodeName(offsetParent, '#document')) return docElement.documentElement;\n while(offsetParent && !nodeName(offsetParent, 'html') && fn.css(offsetParent, 'position') === 'static') {\n offsetParent = offsetParent.offsetParent;\n }\n return offsetParent || docElement.documentElement;\n };\n\n /**\n * Provides equivalent of jQuery's height function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/height/\n * @param element\n * @param outer\n */\n fn.height = function(element, outer) {\n var value = element.offsetHeight;\n if(outer) {\n value += fn.css(element, 'marginTop', true) + fn.css(element, 'marginBottom', true);\n } else {\n value -= fn.css(element, 'paddingTop', true) + fn.css(element, 'paddingBottom', true) + fn.css(element, 'borderTopWidth', true) + fn.css(element, 'borderBottomWidth', true);\n }\n return value;\n };\n\n /**\n * Provides equivalent of jQuery's width function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/width/\n * @param element\n * @param outer\n */\n fn.width = function(element, outer) {\n var value = element.offsetWidth;\n if(outer) {\n value += fn.css(element, 'marginLeft', true) + fn.css(element, 'marginRight', true);\n } else {\n value -= fn.css(element, 'paddingLeft', true) + fn.css(element, 'paddingRight', true) + fn.css(element, 'borderLeftWidth', true) + fn.css(element, 'borderRightWidth', true);\n }\n return value;\n };\n\n return fn;\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.debounce', [])\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L693\n.factory('debounce', function($timeout) {\n return function(func, wait, immediate) {\n var timeout = null;\n return function() {\n var context = this,\n args = arguments,\n callNow = immediate && !timeout;\n if(timeout) {\n $timeout.cancel(timeout);\n }\n timeout = $timeout(function later() {\n timeout = null;\n if(!immediate) {\n func.apply(context, args);\n }\n }, wait, false);\n if(callNow) {\n func.apply(context, args);\n }\n return timeout;\n };\n };\n})\n\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L661\n.factory('throttle', function($timeout) {\n return function(func, wait, options) {\n var timeout = null;\n options || (options = {});\n return function() {\n var context = this,\n args = arguments;\n if(!timeout) {\n if(options.leading !== false) {\n func.apply(context, args);\n }\n timeout = $timeout(function later() {\n timeout = null;\n if(options.trailing !== false) {\n func.apply(context, args);\n }\n }, wait, false);\n }\n };\n };\n});\n\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateParser', [])\n\n.provider('$dateParser', function($localeProvider) {\n\n // define a custom ParseDate object to use instead of native Date\n // to avoid date values wrapping when setting date component values\n function ParseDate() {\n this.year = 1970;\n this.month = 0;\n this.day = 1;\n this.hours = 0;\n this.minutes = 0;\n this.seconds = 0;\n this.milliseconds = 0;\n }\n\n ParseDate.prototype.setMilliseconds = function(value) { this.milliseconds = value; };\n ParseDate.prototype.setSeconds = function(value) { this.seconds = value; };\n ParseDate.prototype.setMinutes = function(value) { this.minutes = value; };\n ParseDate.prototype.setHours = function(value) { this.hours = value; };\n ParseDate.prototype.getHours = function() { return this.hours; };\n ParseDate.prototype.setDate = function(value) { this.day = value; };\n ParseDate.prototype.setMonth = function(value) { this.month = value; };\n ParseDate.prototype.setFullYear = function(value) { this.year = value; };\n ParseDate.prototype.fromDate = function(value) {\n this.year = value.getFullYear();\n this.month = value.getMonth();\n this.day = value.getDate();\n this.hours = value.getHours();\n this.minutes = value.getMinutes();\n this.seconds = value.getSeconds();\n this.milliseconds = value.getMilliseconds();\n return this;\n };\n\n ParseDate.prototype.toDate = function() {\n return new Date(this.year, this.month, this.day, this.hours, this.minutes, this.seconds, this.milliseconds);\n };\n\n var proto = ParseDate.prototype;\n\n function noop() {\n }\n\n function isNumeric(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n\n function indexOfCaseInsensitive(array, value) {\n var len = array.length, str=value.toString().toLowerCase();\n for (var i=0; i 12 when midnight changeover, but then cannot generate\n * midnight datetime, so jump to 1AM, otherwise reset.\n * @param date (Date) the date to check\n * @return (Date) the corrected date\n *\n * __ copied from jquery ui datepicker __\n */\n $dateParser.daylightSavingAdjust = function(date) {\n if (!date) {\n return null;\n }\n date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);\n return date;\n };\n\n /* Correct the date for timezone offset.\n * @param date (Date) the date to adjust\n * @param timezone (string) the timezone to adjust for\n * @param undo (boolean) to add or subtract timezone offset\n * @return (Date) the corrected date\n */\n $dateParser.timezoneOffsetAdjust = function(date, timezone, undo) {\n if (!date) {\n return null;\n }\n // Right now, only 'UTC' is supported.\n if (timezone && timezone === 'UTC') {\n date = new Date(date.getTime());\n date.setMinutes(date.getMinutes() + (undo?-1:1)*date.getTimezoneOffset());\n }\n return date;\n };\n\n // Private functions\n\n function setMapForFormat(format) {\n var keys = Object.keys(setFnMap), i;\n var map = [], sortedMap = [];\n // Map to setFn\n var clonedFormat = format;\n for(i = 0; i < keys.length; i++) {\n if(format.split(keys[i]).length > 1) {\n var index = clonedFormat.search(keys[i]);\n format = format.split(keys[i]).join('');\n if(setFnMap[keys[i]]) {\n map[index] = setFnMap[keys[i]];\n }\n }\n }\n // Sort result map\n angular.forEach(map, function(v) {\n // conditional required since angular.forEach broke around v1.2.21\n // related pr: https://github.com/angular/angular.js/pull/8525\n if(v) sortedMap.push(v);\n });\n return sortedMap;\n }\n\n function escapeReservedSymbols(text) {\n return text.replace(/\\//g, '[\\\\/]').replace('/-/g', '[-]').replace(/\\./g, '[.]').replace(/\\\\s/g, '[\\\\s]');\n }\n\n function regExpForFormat(format) {\n var keys = Object.keys(regExpMap), i;\n\n var re = format;\n // Abstract replaces to avoid collisions\n for(i = 0; i < keys.length; i++) {\n re = re.split(keys[i]).join('${' + i + '}');\n }\n // Replace abstracted values\n for(i = 0; i < keys.length; i++) {\n re = re.split('${' + i + '}').join('(' + regExpMap[keys[i]] + ')');\n }\n format = escapeReservedSymbols(format);\n\n return new RegExp('^' + re + '$', ['i']);\n }\n\n $dateParser.init();\n return $dateParser;\n\n };\n\n return DateParserFactory;\n\n };\n\n});\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateFormatter', [])\n\n .service('$dateFormatter', function($locale, dateFilter) {\n\n // The unused `lang` arguments are on purpose. The default implementation does not\n // use them and it always uses the locale loaded into the `$locale` service.\n // Custom implementations might use it, thus allowing different directives to\n // have different languages.\n\n this.getDefaultLocale = function() {\n return $locale.id;\n };\n\n // Format is either a data format name, e.g. \"shortTime\" or \"fullDate\", or a date format\n // Return either the corresponding date format or the given date format.\n this.getDatetimeFormat = function(format, lang) {\n return $locale.DATETIME_FORMATS[format] || format;\n };\n\n this.weekdaysShort = function(lang) {\n return $locale.DATETIME_FORMATS.SHORTDAY;\n };\n\n function splitTimeFormat(format) {\n return /(h+)([:\\.])?(m+)([:\\.])?(s*)[ ]?(a?)/i.exec(format).slice(1);\n }\n\n // h:mm a => h\n this.hoursFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[0];\n };\n\n // h:mm a => mm\n this.minutesFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[2];\n };\n\n // h:mm:ss a => ss\n this.secondsFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => :\n this.timeSeparator = function(timeFormat) {\n return splitTimeFormat(timeFormat)[1];\n };\n\n // h:mm:ss a => true, h:mm a => false\n this.showSeconds = function(timeFormat) {\n return !!splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => true, H.mm => false\n this.showAM = function(timeFormat) {\n return !!splitTimeFormat(timeFormat)[5];\n };\n\n this.formatDate = function(date, format, lang, timezone){\n return dateFilter(date, format, timezone);\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.datepicker', [\n 'mgcrea.ngStrap.helpers.dateParser',\n 'mgcrea.ngStrap.helpers.dateFormatter',\n 'mgcrea.ngStrap.tooltip'])\n\n .provider('$datepicker', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n // Uncommenting the following line will break backwards compatability\n // prefixEvent: 'datepicker',\n prefixClass: 'datepicker',\n placement: 'bottom-left',\n templateUrl: 'datepicker/datepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: false,\n dateType: 'date',\n dateFormat: 'shortDate',\n timezone: null,\n modelDateFormat: null,\n dayFormat: 'dd',\n monthFormat: 'MMM',\n yearFormat: 'yyyy',\n monthTitleFormat: 'MMMM yyyy',\n yearTitleFormat: 'yyyy',\n strictFormat: false,\n autoclose: false,\n minDate: -Infinity,\n maxDate: +Infinity,\n startView: 0,\n minView: 0,\n startWeek: 0,\n daysOfWeekDisabled: '',\n iconLeft: 'glyphicon glyphicon-chevron-left',\n iconRight: 'glyphicon glyphicon-chevron-right'\n };\n\n this.$get = function($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) {\n\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale();\n\n function DatepickerFactory(element, controller, config) {\n\n var $datepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $datepicker.$options;\n var scope = $datepicker.$scope;\n if (options.startView) options.startView -= options.minView;\n\n // View vars\n\n var pickerViews = datepickerViews($datepicker);\n $datepicker.$views = pickerViews.views;\n var viewDate = pickerViews.viewDate;\n scope.$mode = options.startView;\n scope.$iconLeft = options.iconLeft;\n scope.$iconRight = options.iconRight;\n var $picker = $datepicker.$views[scope.$mode];\n\n // Scope methods\n\n scope.$select = function(date) {\n $datepicker.select(date);\n };\n scope.$selectPane = function(value) {\n $datepicker.$selectPane(value);\n };\n scope.$toggleMode = function() {\n $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length);\n };\n\n // Public methods\n\n $datepicker.update = function(date) {\n // console.warn('$datepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $datepicker.$date = date;\n $picker.update.call($picker, date);\n }\n // Build only if pristine\n $datepicker.$build(true);\n };\n\n $datepicker.updateDisabledDates = function(dateRanges) {\n options.disabledDateRanges = dateRanges;\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], $datepicker.$setDisabledEl);\n }\n };\n\n $datepicker.select = function(date, keep) {\n // console.warn('$datepicker.select', date, scope.$mode);\n if (!angular.isDate(controller.$dateValue)) controller.$dateValue = new Date(date);\n if (!scope.$mode || keep) {\n controller.$setViewValue(angular.copy(date));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function() { $datepicker.hide(true); });\n }\n } else {\n angular.extend(viewDate, {year: date.getFullYear(), month: date.getMonth(), date: date.getDate()});\n $datepicker.setMode(scope.$mode - 1);\n $datepicker.$build();\n }\n };\n\n $datepicker.setMode = function(mode) {\n // console.warn('$datepicker.setMode', mode);\n scope.$mode = mode;\n $picker = $datepicker.$views[scope.$mode];\n $datepicker.$build();\n };\n\n // Protected methods\n\n $datepicker.$build = function(pristine) {\n // console.warn('$datepicker.$build() viewDate=%o', viewDate);\n if (pristine === true && $picker.built) return;\n if (pristine === false && !$picker.built) return;\n $picker.build.call($picker);\n };\n\n $datepicker.$updateSelected = function() {\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], updateSelected);\n }\n };\n\n $datepicker.$isSelected = function(date) {\n return $picker.isSelected(date);\n };\n\n $datepicker.$setDisabledEl = function(el) {\n el.disabled = $picker.isDisabled(el.date);\n };\n\n $datepicker.$selectPane = function(value) {\n var steps = $picker.steps;\n // set targetDate to first day of month to avoid problems with\n // date values rollover. This assumes the viewDate does not\n // depend on the day of the month\n var targetDate = new Date(Date.UTC(viewDate.year + ((steps.year || 0) * value), viewDate.month + ((steps.month || 0) * value), 1));\n angular.extend(viewDate, {year: targetDate.getUTCFullYear(), month: targetDate.getUTCMonth(), date: targetDate.getUTCDate()});\n $datepicker.$build();\n };\n\n $datepicker.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $datepicker.$onKeyDown = function(evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n if (evt.keyCode === 13) {\n if (!scope.$mode) {\n $datepicker.hide(true);\n } else {\n scope.$apply(function() { $datepicker.setMode(scope.$mode - 1); });\n }\n return;\n }\n\n // Navigate with keyboard\n $picker.onKeyDown(evt);\n parentScope.$digest();\n };\n\n // Private\n\n function updateSelected(el) {\n el.selected = $datepicker.$isSelected(el.date);\n }\n\n function focusElement() {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $datepicker.init;\n $datepicker.init = function() {\n if (isNative && options.useNative) {\n element.prop('type', 'date');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $datepicker.destroy;\n $datepicker.destroy = function() {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $datepicker.show;\n $datepicker.show = function() {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n // if $datepicker is no longer showing, don't setup events\n if (!$datepicker.$isShown) return;\n $datepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $datepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $datepicker.hide;\n $datepicker.hide = function(blur) {\n if (!$datepicker.$isShown) return;\n $datepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $datepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $datepicker;\n\n }\n\n DatepickerFactory.defaults = defaults;\n return DatepickerFactory;\n\n };\n\n })\n\n .directive('bsDatepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) {\n\n var defaults = $datepicker.defaults;\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // Initialize datepicker\n var datepicker = $datepicker(element, controller, options);\n options = datepicker.$options;\n // Set expected iOS format\n if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd';\n\n var lang = options.lang;\n\n var formatDate = function(date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!datepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i);\n newValue === true ? datepicker.show() : datepicker.hide();\n });\n\n // Observe attributes for changes\n angular.forEach(['minDate', 'maxDate'], function(key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) {\n // console.warn('attr.$observe(%s)=%o', key, newValue);\n datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue);\n // Build only if dirty\n !isNaN(datepicker.$options[key]) && datepicker.$build(false);\n validateAgainstMinMaxDate(controller.$dateValue);\n });\n });\n\n // Observe date format\n angular.isDefined(attr.dateFormat) && attr.$observe('dateFormat', function(newValue) {\n datepicker.$options.dateFormat = newValue;\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n datepicker.update(controller.$dateValue);\n }, true);\n\n // Normalize undefined/null/empty array,\n // so that we don't treat changing from undefined->null as a change.\n function normalizeDateRanges(ranges) {\n if (!ranges || !ranges.length) return null;\n return ranges;\n }\n\n if (angular.isDefined(attr.disabledDates)) {\n scope.$watch(attr.disabledDates, function(disabledRanges, previousValue) {\n disabledRanges = normalizeDateRanges(disabledRanges);\n previousValue = normalizeDateRanges(previousValue);\n\n if (disabledRanges) {\n datepicker.updateDisabledDates(disabledRanges);\n }\n });\n }\n\n function validateAgainstMinMaxDate(parsedDate) {\n if (!angular.isDate(parsedDate)) return;\n var isMinValid = isNaN(datepicker.$options.minDate) || parsedDate.getTime() >= datepicker.$options.minDate;\n var isMaxValid = isNaN(datepicker.$options.maxDate) || parsedDate.getTime() <= datepicker.$options.maxDate;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (isValid) controller.$dateValue = parsedDate;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function(viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n controller.$setValidity('date', true);\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n return null;\n }\n var parsedDate = dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedDate || isNaN(parsedDate.getTime())) {\n controller.$setValidity('date', false);\n // return undefined, causes ngModelController to\n // invalidate model value\n return;\n } else {\n validateAgainstMinMaxDate(parsedDate);\n }\n\n if (options.dateType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true);\n return formatDate(date, options.modelDateFormat || options.dateFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.dateType === 'number') {\n return date.getTime();\n } else if (options.dateType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.dateType === 'iso') {\n return date.toISOString();\n } else {\n return new Date(date);\n }\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.dateType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelDateFormat);\n } else if (options.dateType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if (isNaN(date.getTime())) {\n // var today = new Date();\n // date = new Date(today.getFullYear(), today.getMonth(), today.getDate(), 0, 0, 0, 0);\n // }\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getDateFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function() {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getDateFormattedString());\n };\n\n function getDateFormattedString() {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.dateFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (datepicker) datepicker.destroy();\n options = null;\n datepicker = null;\n });\n\n }\n };\n\n })\n\n .provider('datepickerViews', function() {\n\n var defaults = this.defaults = {\n dayFormat: 'dd',\n daySplit: 7\n };\n\n // Split array into smaller arrays\n function split(arr, size) {\n var arrays = [];\n while(arr.length > 0) {\n arrays.push(arr.splice(0, size));\n }\n return arrays;\n }\n\n // Modulus operator\n function mod(n, m) {\n return ((n % m) + m) % m;\n }\n\n this.$get = function($dateFormatter, $dateParser, $sce) {\n\n return function(picker) {\n\n var scope = picker.$scope;\n var options = picker.$options;\n\n var lang = options.lang;\n var formatDate = function(date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n var weekDaysMin = $dateFormatter.weekdaysShort(lang);\n var weekDaysLabels = weekDaysMin.slice(options.startWeek).concat(weekDaysMin.slice(0, options.startWeek));\n var weekDaysLabelsHtml = $sce.trustAsHtml('' + weekDaysLabels.join('') + '');\n\n var startDate = picker.$date || (options.startDate ? dateParser.getDateForAttribute('startDate', options.startDate) : new Date());\n var viewDate = {year: startDate.getFullYear(), month: startDate.getMonth(), date: startDate.getDate()};\n\n var views = [{\n format: options.dayFormat,\n split: 7,\n steps: { month: 1 },\n update: function(date, force) {\n if (!this.built || force || date.getFullYear() !== viewDate.year || date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getDate() !== viewDate.date || date.getDate() === 1) {\n // chaging picker current month will cause viewDate.date to be set to first day of the month,\n // in $datepicker.$selectPane, so picker would not update selected day display if\n // user picks first day of the new month.\n // As a workaround, we are always forcing update when picked date is first day of month.\n viewDate.date = picker.$date.getDate();\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1), firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset();\n var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5), firstDateOffset = firstDate.getTimezoneOffset();\n var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString();\n // Handle daylight time switch\n if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 60e3);\n var days = [], day;\n for(var i = 0; i < 42; i++) { // < 7 * 6\n day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i));\n days.push({date: day, isToday: day.toDateString() === today, label: formatDate(day, this.format), selected: picker.$date && this.isSelected(day), muted: day.getMonth() !== viewDate.month, disabled: this.isDisabled(day)});\n }\n scope.title = formatDate(firstDayOfMonth, options.monthTitleFormat);\n scope.showLabels = true;\n scope.labels = weekDaysLabelsHtml;\n scope.rows = split(days, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth() && date.getDate() === picker.$date.getDate();\n },\n isDisabled: function(date) {\n var time = date.getTime();\n\n // Disabled because of min/max date.\n if (time < options.minDate || time > options.maxDate) return true;\n\n // Disabled due to being a disabled day of the week\n if (options.daysOfWeekDisabled.indexOf(date.getDay()) !== -1) return true;\n\n // Disabled because of disabled date range.\n if (options.disabledDateRanges) {\n for (var i = 0; i < options.disabledDateRanges.length; i++) {\n if (time >= options.disabledDateRanges[i].start && time <= options.disabledDateRanges[i].end) {\n return true;\n }\n }\n }\n\n return false;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualTime = picker.$date.getTime();\n var newDate;\n\n if (evt.keyCode === 37) newDate = new Date(actualTime - 1 * 864e5);\n else if (evt.keyCode === 38) newDate = new Date(actualTime - 7 * 864e5);\n else if (evt.keyCode === 39) newDate = new Date(actualTime + 1 * 864e5);\n else if (evt.keyCode === 40) newDate = new Date(actualTime + 7 * 864e5);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'month',\n format: options.monthFormat,\n split: 4,\n steps: { year: 1 },\n update: function(date, force) {\n if (!this.built || date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstMonth = new Date(viewDate.year, 0, 1);\n var months = [], month;\n for (var i = 0; i < 12; i++) {\n month = new Date(viewDate.year, i, 1);\n months.push({date: month, label: formatDate(month, this.format), selected: picker.$isSelected(month), disabled: this.isDisabled(month)});\n }\n scope.title = formatDate(month, options.yearTitleFormat);\n scope.showLabels = false;\n scope.rows = split(months, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth();\n },\n isDisabled: function(date) {\n var lastDate = +new Date(date.getFullYear(), date.getMonth() + 1, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualMonth = picker.$date.getMonth();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setMonth(actualMonth - 1);\n else if (evt.keyCode === 38) newDate.setMonth(actualMonth - 4);\n else if (evt.keyCode === 39) newDate.setMonth(actualMonth + 1);\n else if (evt.keyCode === 40) newDate.setMonth(actualMonth + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'year',\n format: options.yearFormat,\n split: 4,\n steps: { year: 12 },\n update: function(date, force) {\n if (!this.built || force || parseInt(date.getFullYear()/20, 10) !== parseInt(viewDate.year/20, 10)) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstYear = viewDate.year - viewDate.year % (this.split * 3);\n var years = [], year;\n for (var i = 0; i < 12; i++) {\n year = new Date(firstYear + i, 0, 1);\n years.push({date: year, label: formatDate(year, this.format), selected: picker.$isSelected(year), disabled: this.isDisabled(year)});\n }\n scope.title = years[0].label + '-' + years[years.length - 1].label;\n scope.showLabels = false;\n scope.rows = split(years, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear();\n },\n isDisabled: function(date) {\n var lastDate = +new Date(date.getFullYear() + 1, 0, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualYear = picker.$date.getFullYear(),\n newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setYear(actualYear - 1);\n else if (evt.keyCode === 38) newDate.setYear(actualYear - 4);\n else if (evt.keyCode === 39) newDate.setYear(actualYear + 1);\n else if (evt.keyCode === 40) newDate.setYear(actualYear + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }];\n\n return {\n views: options.minView ? Array.prototype.slice.call(views, options.minView) : views,\n viewDate: viewDate\n };\n\n };\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.collapse', [])\n\n .provider('$collapse', function() {\n\n var defaults = this.defaults = {\n animation: 'am-collapse',\n disallowToggle: false,\n activeClass: 'in',\n startCollapsed: false,\n allowMultiple: false\n };\n\n var controller = this.controller = function($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'disallowToggle', 'activeClass', 'startCollapsed', 'allowMultiple'], function (key) {\n if(angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['disallowToggle', 'startCollapsed', 'allowMultiple'], function(key) {\n if(angular.isDefined($attrs[key]) && falseValueRegExp.test($attrs[key])) {\n self.$options[key] = false;\n }\n });\n\n self.$toggles = [];\n self.$targets = [];\n\n self.$viewChangeListeners = [];\n\n self.$registerToggle = function(element) {\n self.$toggles.push(element);\n };\n self.$registerTarget = function(element) {\n self.$targets.push(element);\n };\n\n self.$unregisterToggle = function(element) {\n var index = self.$toggles.indexOf(element);\n // remove toggle from $toggles array\n self.$toggles.splice(index, 1);\n };\n self.$unregisterTarget = function(element) {\n var index = self.$targets.indexOf(element);\n\n // remove element from $targets array\n self.$targets.splice(index, 1);\n\n if (self.$options.allowMultiple) {\n // remove target index from $active array values\n deactivateItem(element);\n }\n\n // fix active item indexes\n fixActiveItemIndexes(index);\n\n self.$viewChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n // use array to store all the currently open panels\n self.$targets.$active = !self.$options.startCollapsed ? [0] : [];\n self.$setActive = $scope.$setActive = function(value) {\n if(angular.isArray(value)) {\n self.$targets.$active = value;\n }\n else if(!self.$options.disallowToggle) {\n // toogle element active status\n isActive(value) ? deactivateItem(value) : activateItem(value);\n } else {\n activateItem(value);\n }\n\n self.$viewChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n self.$activeIndexes = function() {\n return self.$options.allowMultiple ? self.$targets.$active :\n self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1;\n };\n\n function fixActiveItemIndexes(index) {\n // item with index was removed, so we\n // need to adjust other items index values\n var activeIndexes = self.$targets.$active;\n for(var i = 0; i < activeIndexes.length; i++) {\n if (index < activeIndexes[i]) {\n activeIndexes[i] = activeIndexes[i] - 1;\n }\n\n // the last item is active, so we need to\n // adjust its index\n if (activeIndexes[i] === self.$targets.length) {\n activeIndexes[i] = self.$targets.length - 1;\n }\n }\n }\n\n function isActive(value) {\n var activeItems = self.$targets.$active;\n return activeItems.indexOf(value) === -1 ? false : true;\n }\n\n function deactivateItem(value) {\n var index = self.$targets.$active.indexOf(value);\n if (index !== -1) {\n self.$targets.$active.splice(index, 1);\n }\n }\n\n function activateItem(value) {\n if (!self.$options.allowMultiple) {\n // remove current selected item\n self.$targets.$active.splice(0, 1);\n }\n\n if (self.$targets.$active.indexOf(value) === -1) {\n self.$targets.$active.push(value);\n }\n }\n\n };\n\n this.$get = function() {\n var $collapse = {};\n $collapse.defaults = defaults;\n $collapse.controller = controller;\n return $collapse;\n };\n\n })\n\n .directive('bsCollapse', function($window, $animate, $collapse) {\n\n var defaults = $collapse.defaults;\n\n return {\n require: ['?ngModel', 'bsCollapse'],\n controller: ['$scope', '$element', '$attrs', $collapse.controller],\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n if(ngModelCtrl) {\n\n // Update the modelValue following\n bsCollapseCtrl.$viewChangeListeners.push(function() {\n ngModelCtrl.$setViewValue(bsCollapseCtrl.$activeIndexes());\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n if (angular.isArray(modelValue)) {\n // model value is an array, so just replace\n // the active items directly\n bsCollapseCtrl.$setActive(modelValue);\n }\n else {\n var activeIndexes = bsCollapseCtrl.$activeIndexes();\n\n if (angular.isArray(activeIndexes)) {\n // we have an array of selected indexes\n if (activeIndexes.indexOf(modelValue * 1) === -1) {\n // item with modelValue index is not active\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n else if (activeIndexes !== modelValue * 1) {\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n return modelValue;\n });\n\n }\n\n }\n };\n\n })\n\n .directive('bsCollapseToggle', function() {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base attr\n element.attr('data-toggle', 'collapse');\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerToggle(element);\n\n // remove toggle from collapse controller when toggle is destroyed\n scope.$on('$destroy', function() {\n bsCollapseCtrl.$unregisterToggle(element);\n });\n\n element.on('click', function() {\n var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element);\n bsCollapseCtrl.$setActive(index * 1);\n scope.$apply();\n });\n\n }\n };\n\n })\n\n .directive('bsCollapseTarget', function($animate) {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n // scope: true,\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base class\n element.addClass('collapse');\n\n // Add animation class\n if(bsCollapseCtrl.$options.animation) {\n element.addClass(bsCollapseCtrl.$options.animation);\n }\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerTarget(element);\n\n // remove pane target from collapse controller when target is destroyed\n scope.$on('$destroy', function() {\n bsCollapseCtrl.$unregisterTarget(element);\n });\n\n function render() {\n var index = bsCollapseCtrl.$targets.indexOf(element);\n var active = bsCollapseCtrl.$activeIndexes();\n var action = 'removeClass';\n if (angular.isArray(active)) {\n if (active.indexOf(index) !== -1) {\n action = 'addClass';\n }\n }\n else if (index === active) {\n action = 'addClass';\n }\n\n $animate[action](element, bsCollapseCtrl.$options.activeClass);\n }\n\n bsCollapseCtrl.$viewChangeListeners.push(function() {\n render();\n });\n render();\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.button', [])\n\n .provider('$button', function() {\n\n var defaults = this.defaults = {\n activeClass:'active',\n toggleEvent:'click'\n };\n\n this.$get = function() {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsCheckboxGroup', function() {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink(element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"checkbox\"]');\n angular.forEach(children, function(child) {\n var childEl = angular.element(child);\n childEl.attr('bs-checkbox', '');\n childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value'));\n });\n }\n\n };\n\n })\n\n .directive('bsCheckbox', function($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support label > input[type=\"checkbox\"]\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true;\n if(constantValueRegExp.test(attr.trueValue)) {\n trueValue = scope.$eval(attr.trueValue);\n }\n var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false;\n if(constantValueRegExp.test(attr.falseValue)) {\n falseValue = scope.$eval(attr.falseValue);\n }\n\n // Parse exotic values\n var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean';\n if(hasExoticValues) {\n controller.$parsers.push(function(viewValue) {\n // console.warn('$parser', element.attr('ng-model'), 'viewValue', viewValue);\n return viewValue ? trueValue : falseValue;\n });\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n return angular.equals(modelValue, trueValue);\n });\n // Fix rendering for exotic values\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n controller.$render();\n });\n }\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$modelValue, trueValue);\n $$rAF(function() {\n if(isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function() {\n scope.$apply(function () {\n // console.warn('!click', element.attr('ng-model'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n if(!isInput) {\n controller.$setViewValue(!activeElement.hasClass('active'));\n }\n if(!hasExoticValues) {\n controller.$render();\n }\n });\n });\n\n }\n\n };\n\n })\n\n .directive('bsRadioGroup', function() {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink(element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"radio\"]');\n angular.forEach(children, function(child) {\n angular.element(child).attr('bs-radio', '');\n angular.element(child).attr('ng-model', attr.ngModel);\n });\n }\n\n };\n\n })\n\n .directive('bsRadio', function($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support `label > input[type=\"radio\"]` markup\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var value;\n attr.$observe('value', function(v) {\n value = constantValueRegExp.test(v) ? scope.$eval(v) : v;\n controller.$render();\n });\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('value'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$modelValue, value);\n $$rAF(function() {\n if(isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function() {\n scope.$apply(function () {\n // console.warn('!click', element.attr('value'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n controller.$setViewValue(value);\n controller.$render();\n });\n });\n\n }\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.aside', ['mgcrea.ngStrap.modal'])\n\n .provider('$aside', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade-and-slide-right',\n prefixClass: 'aside',\n prefixEvent: 'aside',\n placement: 'right',\n templateUrl: 'aside/aside.tpl.html',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function($modal) {\n\n function AsideFactory(config) {\n\n var $aside = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $aside = $modal(options);\n\n return $aside;\n\n }\n\n return AsideFactory;\n\n };\n\n })\n\n .directive('bsAside', function($window, $sce, $aside) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsAside && scope.$watch(attr.bsAside, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize aside\n var aside = $aside(options);\n\n // Trigger\n element.on(attr.trigger || 'click', aside.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (aside) aside.destroy();\n options = null;\n aside = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\n// @BUG: following snippet won't compile correctly\n// @TODO: submit issue to core\n// ' ' +\n\nangular.module('mgcrea.ngStrap.alert', ['mgcrea.ngStrap.modal'])\n\n .provider('$alert', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'alert',\n prefixEvent: 'alert',\n placement: null,\n templateUrl: 'alert/alert.tpl.html',\n container: false,\n element: null,\n backdrop: false,\n keyboard: true,\n show: true,\n // Specific options\n duration: false,\n type: false,\n dismissable: true\n };\n\n this.$get = function($modal, $timeout) {\n\n function AlertFactory(config) {\n\n var $alert = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $alert = $modal(options);\n\n // Support scope as string options [/*title, content, */ type, dismissable]\n $alert.$scope.dismissable = !!options.dismissable;\n if(options.type) {\n $alert.$scope.type = options.type;\n }\n\n // Support auto-close duration\n var show = $alert.show;\n if(options.duration) {\n $alert.show = function() {\n show();\n $timeout(function() {\n $alert.hide();\n }, options.duration * 1000);\n };\n }\n\n return $alert;\n\n }\n\n return AlertFactory;\n\n };\n\n })\n\n .directive('bsAlert', function($window, $sce, $alert) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'keyboard', 'html', 'container', 'animation', 'duration', 'dismissable'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['keyboard', 'html', 'container', 'dismissable'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')){\n scope.title = '';\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content', 'type'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsAlert && scope.$watch(attr.bsAlert, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize alert\n var alert = $alert(options);\n\n // Trigger\n element.on(attr.trigger || 'click', alert.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (alert) alert.destroy();\n options = null;\n alert = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.affix', ['mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce'])\n\n .provider('$affix', function() {\n\n var defaults = this.defaults = {\n offsetTop: 'auto',\n inlineStyles: true\n };\n\n this.$get = function($window, debounce, dimensions) {\n\n var bodyEl = angular.element($window.document.body);\n var windowEl = angular.element($window);\n\n function AffixFactory(element, config) {\n\n var $affix = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var targetEl = options.target;\n\n // Initial private vars\n var reset = 'affix affix-top affix-bottom',\n setWidth = false,\n initialAffixTop = 0,\n initialOffsetTop = 0,\n offsetTop = 0,\n offsetBottom = 0,\n affixed = null,\n unpin = null;\n\n var parent = element.parent();\n // Options: custom parent\n if (options.offsetParent) {\n if (options.offsetParent.match(/^\\d+$/)) {\n for (var i = 0; i < (options.offsetParent * 1) - 1; i++) {\n parent = parent.parent();\n }\n }\n else {\n parent = angular.element(options.offsetParent);\n }\n }\n\n $affix.init = function() {\n\n this.$parseOffsets();\n initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop;\n setWidth = !element[0].style.width;\n\n // Bind events\n targetEl.on('scroll', this.checkPosition);\n targetEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', this.$debouncedOnResize);\n\n // Both of these checkPosition() calls are necessary for the case where\n // the user hits refresh after scrolling to the bottom of the page.\n this.checkPosition();\n this.checkPositionWithEventLoop();\n\n };\n\n $affix.destroy = function() {\n\n // Unbind events\n targetEl.off('scroll', this.checkPosition);\n targetEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', this.$debouncedOnResize);\n\n };\n\n $affix.checkPositionWithEventLoop = function() {\n\n // IE 9 throws an error if we use 'this' instead of '$affix'\n // in this setTimeout call\n setTimeout($affix.checkPosition, 1);\n\n };\n\n $affix.checkPosition = function() {\n // if (!this.$element.is(':visible')) return\n\n var scrollTop = getScrollTop();\n var position = dimensions.offset(element[0]);\n var elementHeight = dimensions.height(element[0]);\n\n // Get required affix class according to position\n var affix = getRequiredAffixClass(unpin, position, elementHeight);\n\n // Did affix status changed this last check?\n if(affixed === affix) return;\n affixed = affix;\n\n if(affix === 'top') {\n unpin = null;\n if(setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n } else if(affix === 'bottom') {\n if (options.offsetUnpin) {\n unpin = -(options.offsetUnpin * 1);\n }\n else {\n // Calculate unpin threshold when affixed to bottom.\n // Hopefully the browser scrolls pixel by pixel.\n unpin = position.top - scrollTop;\n }\n if(setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', (options.offsetParent) ? '' : ((bodyEl[0].offsetHeight - offsetBottom - elementHeight - initialOffsetTop) + 'px'));\n }\n } else { // affix === 'middle'\n unpin = null;\n if(setWidth) {\n element.css('width', element[0].offsetWidth + 'px');\n }\n if (options.inlineStyles) {\n element.css('position', 'fixed');\n element.css('top', initialAffixTop + 'px');\n }\n }\n\n // Add proper affix class\n element.removeClass(reset).addClass('affix' + ((affix !== 'middle') ? '-' + affix : ''));\n\n };\n\n $affix.$onResize = function() {\n $affix.$parseOffsets();\n $affix.checkPosition();\n };\n $affix.$debouncedOnResize = debounce($affix.$onResize, 50);\n\n $affix.$parseOffsets = function() {\n var initialPosition = element.css('position');\n // Reset position to calculate correct offsetTop\n if (options.inlineStyles){\n element.css('position', (options.offsetParent) ? '' : 'relative');\n }\n\n if(options.offsetTop) {\n if(options.offsetTop === 'auto') {\n options.offsetTop = '+0';\n }\n if(options.offsetTop.match(/^[-+]\\d+$/)) {\n initialAffixTop = - options.offsetTop * 1;\n if(options.offsetParent) {\n offsetTop = dimensions.offset(parent[0]).top + (options.offsetTop * 1);\n }\n else {\n offsetTop = dimensions.offset(element[0]).top - dimensions.css(element[0], 'marginTop', true) + (options.offsetTop * 1);\n }\n }\n else {\n offsetTop = options.offsetTop * 1;\n }\n }\n\n if(options.offsetBottom) {\n if(options.offsetParent && options.offsetBottom.match(/^[-+]\\d+$/)) {\n // add 1 pixel due to rounding problems...\n offsetBottom = getScrollHeight() - (dimensions.offset(parent[0]).top + dimensions.height(parent[0])) + (options.offsetBottom * 1) + 1;\n }\n else {\n offsetBottom = options.offsetBottom * 1;\n }\n }\n\n // Bring back the element's position after calculations\n if (options.inlineStyles){\n element.css('position', initialPosition);\n }\n };\n\n // Private methods\n\n function getRequiredAffixClass(unpin, position, elementHeight) {\n\n var scrollTop = getScrollTop();\n var scrollHeight = getScrollHeight();\n\n if(scrollTop <= offsetTop) {\n return 'top';\n } else if(unpin !== null && (scrollTop + unpin <= position.top)) {\n return 'middle';\n } else if(offsetBottom !== null && (position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom)) {\n return 'bottom';\n } else {\n return 'middle';\n }\n\n }\n\n function getScrollTop() {\n return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop;\n }\n\n function getScrollHeight() {\n return targetEl[0] === $window ? $window.document.body.scrollHeight : targetEl[0].scrollHeight;\n }\n\n $affix.init();\n return $affix;\n\n }\n\n return AffixFactory;\n\n };\n\n })\n\n .directive('bsAffix', function($affix, $window) {\n\n return {\n restrict: 'EAC',\n require: '^?bsAffixTarget',\n link: function postLink(scope, element, attr, affixTarget) {\n\n var options = {scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window)};\n angular.forEach(['offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles'], function(key) {\n if(angular.isDefined(attr[key])) {\n var option = attr[key];\n if (/true/i.test(option)) option = true;\n if (/false/i.test(option)) option = false;\n options[key] = option;\n }\n });\n\n var affix = $affix(element, options);\n scope.$on('$destroy', function() {\n affix && affix.destroy();\n options = null;\n affix = null;\n });\n\n }\n };\n\n })\n\n .directive('bsAffixTarget', function() {\n return {\n controller: function($element) {\n this.$element = $element;\n }\n };\n });\n","\nangular.module('mgcrea.ngStrap', [\n 'mgcrea.ngStrap.modal',\n 'mgcrea.ngStrap.aside',\n 'mgcrea.ngStrap.alert',\n 'mgcrea.ngStrap.button',\n 'mgcrea.ngStrap.select',\n 'mgcrea.ngStrap.datepicker',\n 'mgcrea.ngStrap.timepicker',\n 'mgcrea.ngStrap.navbar',\n 'mgcrea.ngStrap.tooltip',\n 'mgcrea.ngStrap.popover',\n 'mgcrea.ngStrap.dropdown',\n 'mgcrea.ngStrap.typeahead',\n 'mgcrea.ngStrap.scrollspy',\n 'mgcrea.ngStrap.affix',\n 'mgcrea.ngStrap.tab',\n 'mgcrea.ngStrap.collapse'\n]);\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["angular-strap.js","typeahead/typeahead.js","helpers/compiler.js","dropdown/dropdown.js","tooltip/tooltip.js","timepicker/timepicker.js","tab/tab.js","select/select.js","scrollspy/scrollspy.js","popover/popover.js","navbar/navbar.js","modal/modal.js","helpers/raf.js","helpers/parse-options.js","helpers/dimensions.js","helpers/debounce.js","helpers/date-parser.js","helpers/date-formatter.js","button/button.js","datepicker/datepicker.js","collapse/collapse.js","aside/aside.js","alert/alert.js","affix/affix.js","module.js"],"names":["prefixEvent","undefined","templateUrl","options","cache","$templateCache","then","element","res","fetchTemplate","template","fetchPromises","module","$http","get","defaults","animation","prefixClass","data","this","compile","controller","console","controllerAs","resolve","locals","bindToController","angular","forEach","value","isString","$injector","key","transformTemplate","identity","extend","$template","when","$q","all","templateEl","Error","findElement","contentTemplate","templates","contentEl","removeAttr","html","replace","next","remove","outerHTML","link","scope","trim","contents","linkFn","invokeCtrl","children","instance","ctrl","isObject","arguments","apply","bsCompilerService","$inject","trigger","provider","container","keyboard","delay","minLength","filter","limit","autoSelect","comparator","trimValue","$get","$typeahead","parentScope","$resetMatches","$window","$rootScope","$tooltip","$$rAF","$timeout","$matches","config","$$postDigest","$scope","activate","$activeIndex","select","index","$select","evt","matches","$isVisible","safeDigest","update","$render","$emit","onSelect","$setViewValue","length","isDefined","isFunction","equals","$onMouseDown","preventDefault","stopPropagation","clientTop","active","offsetTop","clientBottom","highWatermark","scrollTop","Math","lowWatermark","clientHeight","keyCode","$$updateScrollTop","$digest","show","isUndefined","$element","hide","$onKeyDown","$$phase","TypeaheadFactory","array","$filter","expression","directive","results","postLink","attr","restrict","require","falseValueRegExp","bsKey","charAt","toUpperCase","slice","test","bsOptions","parsedOptions","$parseOptions","typeahead","watchedOptions","$watchCollection","values","watchOptions","$match","$watch","ngModel","newValue","oldValue","$modelValue","valuesFn","selectMode","$viewValue","substring","displayValue","$formatters","push","modelValue","selected","ss","selectionStart","val","selectionEnd","setSelectionRange","sd","label","destroy","$on","target","placement","titleTemplate","title","type","autoClose","bsEnabled","mouseDownPreventDefault","mouseDownStopPropagation","viewport","selector","padding","TooltipFactory","promise","$bsCompiler","split","clearTimeout","hoverState","onShow","leaveAnimateCallback","destroyTipElement","onHide","_tipToHide","triggers","nodeName","on","isTouch","toggle","enter","unbindTriggerEvents","$onFocusElementMouseDown","off","i","bindKeyboardEvents","tipElement","$onKeyUp","unbindKeyboardEvents","_autoCloseEventsBinded","bindAutoCloseEvents","$body","unbindAutoCloseEvents","stopEventPropagation","event","getPosition","elRect","getBoundingClientRect","rect","p","width","height","top","left","dimensions","offset","el","scroll","outerDims","isBody","document","documentElement","body","prop","clientWidth","innerHeight","position","actualWidth","actualHeight","marginLeft","isNaN","setOffset","using","parseInt","props","css","right","marginTop","tip","offsetHeight","getViewportAdjustedDelta","delta","offsetWidth","isVertical","arrowDelta","replaceArrow","arrowOffsetPosition","viewportDimensions","$viewport","topEdgeOffset","bottomEdgeOffset","viewportPadding","leftEdgeOffset","rightEdgeOffset","dimension","isHorizontal","$arrow","timeout","$isShown","tipScope","$destroy","$options","$promise","$new","toLowerCase","$id","map","parseFloat","$sce","trustAsHtml","$setEnabled","id","$hide","setEnabled","isEnabled","compileData","tipContainer","init","bindTriggerEvents","isElement","focus","onBeforeShow","after","parent","lastChild","display","visibility","clonedElement","addClass","version","prepend","$applyPlacement","$animate","enterAnimateCallback","_blur","onBeforeHide","blur","leave","autoPlace","autoToken","elementPosition","tipWidth","tipHeight","viewportPosition","originalPlacement","removeClass","tipPosition","getCalculatedOffset","applyPlacement","$root","query","querySelectorAll","isNative","tooltip","transclusion","$eval","dataTarget","hasOwnProperty","$observe","bsTooltip","bsShow","match","setViewport","useNative","timeType","timeFormat","timezone","modelTimeFormat","autoclose","minTime","maxTime","Infinity","hourStep","minuteStep","secondStep","roundDisplay","iconUp","iconDown","arrowBehavior","$timepicker","timepickerFactory","formatDate","format","viewDate","hour","startDate","getHours","meridian","coeff","selRange","end","start","collapse","moveStart","moveEnd","focusElement","_init","floorMinutes","time","floor","lang","selectedIndex","date","defaultDate","second","getSeconds","millisecond","getMilliseconds","$dateValue","hoursFormat","$dateFormatter","timeSeparator","minute","minutesFormat","secondsFormat","showSeconds","$iconUp","$moveIndex","$switchMeridian","switchMeridian","isDate","getMinutes","getTime","$build","$isBuilt","keep","Date","setHours","setMinutes","setSeconds","copy","midIndex","hours","disabled","minutes","$date","seconds","rows","$isDisabled","showAM","$isSelected","isAM","selectedTime","$arrowAction","$setTimeByStep","newDate","targetDate","targetEl","triggerHandler","hoursLength","minutesLength","sepLength","lateralMove","count","selectRange","incr","isSeconds","isMeridian","secondsLength","createSelection","createTextRange","_destroy","_show","_hide","navigator","userAgent","isMaxValid","isValid","parsedTime","isMinValid","setFullYear","$setValidity","$parsers","unshift","viewValue","getTimeFormattedString","timepicker","dateParser","$dateParser","validateAgainstMinMaxTime","timezoneOffsetAdjust","NaN","parse","self","navClass","activeClass","$activeClass","$panes","$activePaneChangeListeners","$push","pane","$active","$attrs","$navClass","$remove","activeIndex","indexOf","splice","$setActive","name","fn","$pane","$tab","transclude","ngModelCtrl","bsTabsCtrl","attrs","bsActivePane","parsedBsActivePane","assign","$parse","controllers","render","$isActive","multiple","allNoneButtons","sort","caretHtml","placeholder","allText","noneText","maxLength","maxLengthHtml","iconCheckmark","SelectFactory","$isMultiple","$showAllNoneButtons","$iconCheckmark","$allText","$activate","$selectNone","$updateActiveIndex","a","b","$apply","isArray","$getIndex","anchor","anchorCandidate","$selectScrollFix","$isIE","stopImmediatePropagation","ua","activeElement","tagName","e","dataMultiple","inputEl","addEventListener","join","$isEmpty","spies","$document","windowEl","debounce","bodyEl","throttle","ScrollSpyFactory","scrollEl","isWindowSpy","scrollId","$$count","$scrollspy","unbindViewContentLoaded","unbindIncludeContentLoaded","trackedElements","sortedElements","activeTarget","debouncedCheckOffsets","viewportHeight","debouncedCheckPosition","throttledCheckPosition","checkPositionWithEventLoop","checkOffsets","checkPosition","docEl","$activateElement","setTimeout","source","$getTrackedElement","targetElement","querySelector","trackedElement","trackElement","toDelete","untrackElement","scrollspy","childEl","child","content","$popover","PopoverFactory","requestAnimationFrame","bsPopover","popover","routeAttr","$navbar","liElements","li","liElement","pattern","path","RegExp","regexp","backdrop","size","zIndex","bodyElement","backdropCount","dialogBaseZindex","validSizes","ModalFactory","$modal","unbindBackdropEvents","modalElement","hideOnBackdropClick","backdropElement","preventEventDefault","modalScope","backdropBaseZindex","$show","bottom","destroyModalElement","z-index","defaultPrevented","customClass","minor","backdropAnimation","bindBackdropEvents","which","lg","sm","modalClass","bsModal","modal","dot","cancelAnimationFrame","webkitRequestAnimationFrame","mozRequestAnimationFrame","rafSupported","raf","timer","cancel","supported","$values","valueName","keyName","displayFn","valueFn","ParseOptionsFactory","groupByFn","factory","offsetParent","docElement","ownerDocument","outer","window","getComputedStyle","extra","currentStyle","boxRect","style","curPosition","curLeft","curCSSTop","curTop","calculatePosition","curElem","curCSSLeft","curOffset","call","offsetParentRect","offsetParentEl","offsetParentElement","func","immediate","args","context","callNow","leading","wait","trailing","year","$localeProvider","milliseconds","ParseDate","prototype","noop","indexOfCaseInsensitive","len","str","toString","strict","day","month","getFullYear","getMonth","proto","isNumeric","toDate","regExpMap","sss","$locale","dateFilter","mm","literalRegex","formatParts","escapedLiteralFormat","dateElements","keys","escapedFormat","isFormatStringLiteral","part","trimLiteralEscapeChars","dateRegexParts","escapeReservedSymbols","text","re","abstractRegex","buildDateParseRegex","setMapForFormat","buildDateAbstractRegex","buildDateParseValuesMap","Object","valuesMatch","keyIndex","valueKey","valuesFunctionMap","valueFunction","valuesRegex","exec","setFnMap","DateParserFactory","setMap","m","HH","H","hh","h","EEEE","EEE","DATETIME_FORMATS","SHORTDAY","dd","d","MMMM","MMM","SHORTMONTH","MM","M","DAY","yyyy","yy","y","MONTH","s","setDate","setMonth","regex","$format","regExpForFormat","baseDate","formatRegex","formatSetMap","fromDate","getDate","getDateForAttribute","today","substr","getTimeForAttribute","daylightSavingAdjust","undo","getTimezoneOffset","getDefaultLocale","getDatetimeFormat","weekdaysShort","splitTimeFormat","service","matchesSelector","DropdownFactory","$dropdown","onBodyClick","items","parentEl","hasClass","tAttrs","nextSibling","nodeType","parentNode","removeChild","dropdown","$button","constantValueRegExp","isInput","trueValue","falseValue","hasExoticValues","isActive","checked","toggleClass","bind","toggleEvent","v","dateType","dateFormat","modelDateFormat","dayFormat","monthFormat","yearFormat","monthTitleFormat","yearTitleFormat","strictFormat","minDate","maxDate","startView","minView","startWeek","daysOfWeekDisabled","hasToday","hasClear","iconLeft","iconRight","DatepickerFactory","$datepicker","pickerViews","views","$iconLeft","$iconRight","$hasToday","$picker","$selectPane","$views","$mode","$toggleMode","$setToday","$clear","setMode","updateDisabledDates","disabledDateRanges","dateRanges","$setDisabledEl","mode","pristine","$updateSelected","built","isDisabled","steps","getUTCFullYear","getUTCMonth","UTC","getUTCDate","shiftKey","altKey","onKeyDown","datepickerViews","previousValue","normalizeDateRanges","ranges","disabledRanges","datepicker","parsedDate","getDateFormattedString","validateAgainstMinMaxDate","disabledDates","arrays","mod","n","arr","weekDaysMin","weekDaysLabelsHtml","picker","weekDaysLabels","concat","firstDayOfMonth","firstDayOfMonthOffset","firstDate","firstDateOffset","build","days","isToday","toDateString","muted","showLabels","labels","isTodayDisabled","isSelected","getDay","months","lastDate","actualMonth","firstYear","years","actualYear","Array","setYear","startCollapsed","allowMultiple","activeIndexes","$targets","activeItems","activateItem","$collapse","$viewChangeListeners","$registerToggle","$toggles","$unregisterToggle","$unregisterTarget","deactivateItem","fixActiveItemIndexes","disallowToggle","$activeIndexes","bsCollapseCtrl","bsCollapseToggle","$registerTarget","action","AsideFactory","$aside","bsAside","aside","duration","dismissable","AlertFactory","$alert","bsAlert","alert","AffixFactory","inlineStyles","$affix","setWidth","reset","initialOffsetTop","getRequiredAffixClass","_unpin","getScrollTop","scrollHeight","getScrollHeight","pageYOffset","offsetBottom","elementHeight","initialAffixTop","affixed","unpin","$parseOffsets","affix","offsetUnpin","$onResize","initialTop","$debouncedOnResize","initialPosition","affixTarget","option"],"mappings":"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","file":"angular-strap.min.js","sourcesContent":["(function(window, document, undefined) {\n'use strict';\n\n// Source: typeahead/typeahead.js\nangular.module('mgcrea.ngStrap.typeahead', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$typeahead', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'typeahead',\n prefixEvent: '$typeahead',\n placement: 'bottom-left',\n templateUrl: 'typeahead/typeahead.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n minLength: 1,\n filter: 'bsAsyncFilter',\n limit: 6,\n autoSelect: false,\n comparator: '',\n trimValue: true\n };\n\n this.$get = function ($window, $rootScope, $tooltip, $$rAF, $timeout) {\n\n function TypeaheadFactory (element, controller, config) {\n\n var $typeahead = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $typeahead = $tooltip(element, options);\n var parentScope = config.scope;\n var scope = $typeahead.$scope;\n\n scope.$resetMatches = function () {\n scope.$matches = [];\n scope.$activeIndex = options.autoSelect ? 0 : -1; // If set to 0, the first match will be highlighted\n };\n scope.$resetMatches();\n\n scope.$activate = function (index) {\n scope.$$postDigest(function () {\n $typeahead.activate(index);\n });\n };\n\n scope.$select = function (index, evt) {\n scope.$$postDigest(function () {\n $typeahead.select(index);\n });\n };\n\n scope.$isVisible = function () {\n return $typeahead.$isVisible();\n };\n\n // Public methods\n\n $typeahead.update = function (matches) {\n scope.$matches = matches;\n if (scope.$activeIndex >= matches.length) {\n scope.$activeIndex = options.autoSelect ? 0 : -1;\n }\n\n // wrap in a $timeout so the results are updated\n // before repositioning\n safeDigest(scope);\n $$rAF($typeahead.$applyPlacement);\n };\n\n $typeahead.activate = function (index) {\n scope.$activeIndex = index;\n };\n\n $typeahead.select = function (index) {\n if (index === -1) return;\n var value = scope.$matches[index].value;\n // console.log('$setViewValue', value);\n controller.$setViewValue(value);\n controller.$render();\n scope.$resetMatches();\n if (parentScope) parentScope.$digest();\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $typeahead);\n if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) {\n options.onSelect(value, index, $typeahead);\n }\n };\n\n // Protected methods\n\n $typeahead.$isVisible = function () {\n if (!options.minLength || !controller) {\n return !!scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength;\n };\n\n $typeahead.$getIndex = function (value) {\n var index;\n for (index = scope.$matches.length; index--;) {\n if (angular.equals(scope.$matches[index].value, value)) break;\n }\n return index;\n };\n\n $typeahead.$onMouseDown = function (evt) {\n // Prevent blur on mousedown\n evt.preventDefault();\n evt.stopPropagation();\n };\n\n $typeahead.$$updateScrollTop = function (container, index) {\n if (index > -1 && index < container.children.length) {\n var active = container.children[index];\n var clientTop = active.offsetTop;\n var clientBottom = active.offsetTop + active.clientHeight;\n var highWatermark = container.scrollTop;\n var lowWatermark = container.scrollTop + container.clientHeight;\n\n // active entry overlaps top border\n if (clientBottom >= highWatermark && clientTop < highWatermark) {\n container.scrollTop = Math.max(0, container.scrollTop - container.clientHeight);\n } else if (clientBottom > lowWatermark) {\n // top of active element is invisible because it's below the bottom of the visible container window\n container.scrollTop = clientTop;\n }\n }\n };\n\n $typeahead.$onKeyDown = function (evt) {\n if (!/(38|40|13)/.test(evt.keyCode)) return;\n\n // Let ngSubmit pass if the typeahead tip is hidden or no option is selected\n if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // Select with enter\n if (evt.keyCode === 13 && scope.$matches.length) {\n $typeahead.select(scope.$activeIndex);\n // Navigate with keyboard\n } else if (evt.keyCode === 38 && scope.$activeIndex > 0) {\n scope.$activeIndex--;\n } else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) {\n scope.$activeIndex++;\n } else if (angular.isUndefined(scope.$activeIndex)) {\n scope.$activeIndex = 0;\n }\n\n // update scrollTop property on $typeahead when scope.$activeIndex is not in visible area\n $typeahead.$$updateScrollTop($typeahead.$element[0], scope.$activeIndex);\n scope.$digest();\n };\n\n // Overrides\n\n var show = $typeahead.show;\n $typeahead.show = function () {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed immediately.\n $timeout(function () {\n if ($typeahead.$element) {\n $typeahead.$element.on('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n if (element) element.on('keydown', $typeahead.$onKeyDown);\n }\n }\n }, 0, false);\n };\n\n var hide = $typeahead.hide;\n $typeahead.hide = function () {\n if ($typeahead.$element) $typeahead.$element.off('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n if (element) element.off('keydown', $typeahead.$onKeyDown);\n }\n if (!options.autoSelect) {\n $typeahead.activate(-1);\n }\n hide();\n };\n\n return $typeahead;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n TypeaheadFactory.defaults = defaults;\n return TypeaheadFactory;\n\n };\n\n })\n\n .filter('bsAsyncFilter', function ($filter) {\n return function (array, expression, comparator) {\n if (array && angular.isFunction(array.then)) {\n return array.then(function (results) {\n return $filter('filter')(results, expression, comparator);\n });\n }\n return $filter('filter')(array, expression, comparator);\n };\n })\n\n .directive('bsTypeahead', function ($window, $parse, $q, $typeahead, $parseOptions) {\n\n var defaults = $typeahead.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Fixes firefox bug when using objects in model with typeahead\n // Yes this breaks any other directive using a 'change' event on this input,\n // but if it is using the 'change' event why is it used with typeahead?\n element.off('change');\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'filter', 'limit', 'minLength', 'watchOptions', 'selectMode', 'autoSelect', 'comparator', 'id', 'prefixEvent', 'prefixClass'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'trimValue', 'filter'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show, hide and select events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Disable browser autocompletion\n if (!element.attr('autocomplete')) element.attr('autocomplete', 'off');\n\n // Build proper bsOptions\n var filter = angular.isDefined(options.filter) ? options.filter : defaults.filter;\n var limit = options.limit || defaults.limit;\n var comparator = options.comparator || defaults.comparator;\n\n var bsOptions = attr.bsOptions;\n if (filter) {\n bsOptions += ' | ' + filter + ':$viewValue';\n if (comparator) bsOptions += ':' + comparator;\n }\n if (limit) bsOptions += ' | limitTo:' + limit;\n var parsedOptions = $parseOptions(bsOptions);\n\n // Initialize typeahead\n var typeahead = $typeahead(element, controller, options);\n\n // Watch options on demand\n if (options.watchOptions) {\n // Watch bsOptions values before filtering for changes, drop function calls\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').replace(/\\(.*\\)/g, '').trim();\n scope.$watchCollection(watchedOptions, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller).then(function (values) {\n typeahead.update(values);\n controller.$render();\n });\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('$watch', element.attr('ng-model'), newValue);\n scope.$modelValue = newValue; // Publish modelValue on scope for custom templates\n parsedOptions.valuesFn(scope, controller)\n .then(function (values) {\n // Prevent input with no future prospect if selectMode is truthy\n // @TODO test selectMode\n if (options.selectMode && !values.length && newValue.length > 0) {\n controller.$setViewValue(controller.$viewValue.substring(0, controller.$viewValue.length - 1));\n return;\n }\n if (values.length > limit) values = values.slice(0, limit);\n typeahead.update(values);\n // Queue a new rendering that will leverage collection loading\n controller.$render();\n });\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var displayValue = parsedOptions.displayValue(modelValue);\n\n // If we can determine the displayValue, use that\n if (displayValue) {\n return displayValue;\n }\n\n // If there's no display value, attempt to use the modelValue.\n // If the model is an object not much we can do\n if (angular.isDefined(modelValue) && typeof modelValue !== 'object') {\n return modelValue;\n }\n return '';\n });\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n if (controller.$isEmpty(controller.$viewValue)) {\n return element.val('');\n }\n var index = typeahead.$getIndex(controller.$modelValue);\n var selected = index !== -1 ? typeahead.$scope.$matches[index].label : controller.$viewValue;\n selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected;\n var value = selected ? selected.toString().replace(/<(?:.|\\n)*?>/gm, '') : '';\n var ss = element[0].selectionStart;\n var sd = element[0].selectionEnd;\n element.val(options.trimValue === false ? value : value.trim());\n element[0].setSelectionRange(ss, sd);\n };\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (typeahead) typeahead.destroy();\n options = null;\n typeahead = null;\n });\n\n }\n };\n\n });\n\n// Source: tooltip/tooltip.js\nangular.module('mgcrea.ngStrap.tooltip', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$tooltip', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n prefixClass: 'tooltip',\n prefixEvent: 'tooltip',\n container: false,\n target: false,\n placement: 'top',\n templateUrl: 'tooltip/tooltip.tpl.html',\n template: '',\n titleTemplate: false,\n trigger: 'hover focus',\n keyboard: false,\n html: false,\n show: false,\n title: '',\n type: '',\n delay: 0,\n autoClose: false,\n bsEnabled: true,\n mouseDownPreventDefault: true,\n mouseDownStopPropagation: true,\n viewport: {\n selector: 'body',\n padding: 0\n }\n };\n\n this.$get = function ($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n var $body = angular.element($window.document);\n\n function TooltipFactory (element, config) {\n\n var $tooltip = {};\n\n // Common vars\n var options = $tooltip.$options = angular.extend({}, defaults, config);\n var promise = $tooltip.$promise = $bsCompiler.compile(options);\n var scope = $tooltip.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n var nodeName = element[0].nodeName.toLowerCase();\n if (options.delay && angular.isString(options.delay)) {\n var split = options.delay.split(',').map(parseFloat);\n options.delay = split.length > 1 ? {show: split[0], hide: split[1]} : split[0];\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $tooltip.$id = options.id || element.attr('id') || '';\n\n // Support scope as string options\n if (options.title) {\n scope.title = $sce.trustAsHtml(options.title);\n }\n\n // Provide scope helpers\n scope.$setEnabled = function (isEnabled) {\n scope.$$postDigest(function () {\n $tooltip.setEnabled(isEnabled);\n });\n };\n scope.$hide = function () {\n scope.$$postDigest(function () {\n $tooltip.hide();\n });\n };\n scope.$show = function () {\n scope.$$postDigest(function () {\n $tooltip.show();\n });\n };\n scope.$toggle = function () {\n scope.$$postDigest(function () {\n $tooltip.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $tooltip.$isShown = scope.$isShown = false;\n\n // Private vars\n var timeout;\n var hoverState;\n\n // Fetch, compile then initialize tooltip\n var compileData;\n var tipElement;\n var tipContainer;\n var tipScope;\n promise.then(function (data) {\n compileData = data;\n $tooltip.init();\n });\n\n $tooltip.init = function () {\n\n // Options: delay\n if (options.delay && angular.isNumber(options.delay)) {\n options.delay = {\n show: options.delay,\n hide: options.delay\n };\n }\n\n // Replace trigger on touch devices ?\n // if(isTouch && options.trigger === defaults.trigger) {\n // options.trigger.replace(/hover/g, 'click');\n // }\n\n // Options : container\n if (options.container === 'self') {\n tipContainer = element;\n } else if (angular.isElement(options.container)) {\n tipContainer = options.container;\n } else if (options.container) {\n tipContainer = findElement(options.container);\n }\n\n // Options: trigger\n bindTriggerEvents();\n\n // Options: target\n if (options.target) {\n options.target = angular.isElement(options.target) ? options.target : findElement(options.target);\n }\n\n // Options: show\n if (options.show) {\n scope.$$postDigest(function () {\n if (options.trigger === 'focus') {\n element[0].focus();\n } else {\n $tooltip.show();\n }\n });\n }\n\n };\n\n $tooltip.destroy = function () {\n\n // Unbind events\n unbindTriggerEvents();\n\n // Remove element\n destroyTipElement();\n\n // Destroy scope\n scope.$destroy();\n\n };\n\n $tooltip.enter = function () {\n\n clearTimeout(timeout);\n hoverState = 'in';\n if (!options.delay || !options.delay.show) {\n return $tooltip.show();\n }\n\n timeout = setTimeout(function () {\n if (hoverState === 'in') $tooltip.show();\n }, options.delay.show);\n\n };\n\n $tooltip.show = function () {\n if (!options.bsEnabled || $tooltip.$isShown) return;\n\n scope.$emit(options.prefixEvent + '.show.before', $tooltip);\n if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) {\n options.onBeforeShow($tooltip);\n }\n var parent;\n var after;\n if (options.container) {\n parent = tipContainer;\n if (tipContainer[0].lastChild) {\n after = angular.element(tipContainer[0].lastChild);\n } else {\n after = null;\n }\n } else {\n parent = null;\n after = element;\n }\n\n\n // Hide any existing tipElement\n if (tipElement) destroyTipElement();\n // Fetch a cloned element linked from template\n tipScope = $tooltip.$scope.$new();\n tipElement = $tooltip.$element = compileData.link(tipScope, function (clonedElement, scope) {});\n\n // Set the initial positioning. Make the tooltip invisible\n // so IE doesn't try to focus on it off screen.\n tipElement.css({top: '-9999px', left: '-9999px', right: 'auto', display: 'block', visibility: 'hidden'});\n\n // Options: animation\n if (options.animation) tipElement.addClass(options.animation);\n // Options: type\n if (options.type) tipElement.addClass(options.prefixClass + '-' + options.type);\n // Options: custom classes\n if (options.customClass) tipElement.addClass(options.customClass);\n\n // Append the element, without any animations. If we append\n // using $animate.enter, some of the animations cause the placement\n // to be off due to the transforms.\n if (after) {\n after.after(tipElement);\n } else {\n parent.prepend(tipElement);\n }\n\n $tooltip.$isShown = scope.$isShown = true;\n safeDigest(scope);\n\n // Now, apply placement\n $tooltip.$applyPlacement();\n\n // Once placed, animate it.\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.enter(tipElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(tipElement, parent, after).then(enterAnimateCallback);\n }\n safeDigest(scope);\n\n $$rAF(function () {\n // Once the tooltip is placed and the animation starts, make the tooltip visible\n if (tipElement) tipElement.css({visibility: 'visible'});\n\n // Bind events\n if (options.keyboard) {\n if (options.trigger !== 'focus') {\n $tooltip.focus();\n }\n bindKeyboardEvents();\n }\n });\n\n if (options.autoClose) {\n bindAutoCloseEvents();\n }\n\n };\n\n function enterAnimateCallback () {\n scope.$emit(options.prefixEvent + '.show', $tooltip);\n if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) {\n options.onShow($tooltip);\n }\n }\n\n $tooltip.leave = function () {\n\n clearTimeout(timeout);\n hoverState = 'out';\n if (!options.delay || !options.delay.hide) {\n return $tooltip.hide();\n }\n timeout = setTimeout(function () {\n if (hoverState === 'out') {\n $tooltip.hide();\n }\n }, options.delay.hide);\n\n };\n\n var _blur;\n var _tipToHide;\n $tooltip.hide = function (blur) {\n\n if (!$tooltip.$isShown) return;\n scope.$emit(options.prefixEvent + '.hide.before', $tooltip);\n if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) {\n options.onBeforeHide($tooltip);\n }\n\n // store blur value for leaveAnimateCallback to use\n _blur = blur;\n\n // store current tipElement reference to use\n // in leaveAnimateCallback\n _tipToHide = tipElement;\n\n if (tipElement !== null) {\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.leave(tipElement, leaveAnimateCallback);\n } else {\n $animate.leave(tipElement).then(leaveAnimateCallback);\n }\n }\n\n $tooltip.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n if (options.keyboard && tipElement !== null) {\n unbindKeyboardEvents();\n }\n\n if (options.autoClose && tipElement !== null) {\n unbindAutoCloseEvents();\n }\n };\n\n function leaveAnimateCallback () {\n scope.$emit(options.prefixEvent + '.hide', $tooltip);\n if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) {\n options.onHide($tooltip);\n }\n\n // check if current tipElement still references\n // the same element when hide was called\n if (tipElement === _tipToHide) {\n // Allow to blur the input when hidden, like when pressing enter key\n if (_blur && options.trigger === 'focus') {\n return element[0].blur();\n }\n\n // clean up child scopes\n destroyTipElement();\n }\n }\n\n $tooltip.toggle = function (evt) {\n if (evt) { evt.preventDefault(); }\n if ($tooltip.$isShown) {\n $tooltip.leave();\n } else {\n $tooltip.enter();\n }\n };\n\n $tooltip.focus = function () {\n tipElement[0].focus();\n };\n\n $tooltip.setEnabled = function (isEnabled) {\n options.bsEnabled = isEnabled;\n };\n\n $tooltip.setViewport = function (viewport) {\n options.viewport = viewport;\n };\n\n // Protected methods\n\n $tooltip.$applyPlacement = function () {\n if (!tipElement) return;\n\n // Determine if we're doing an auto or normal placement\n var placement = options.placement;\n var autoToken = /\\s?auto?\\s?/i;\n var autoPlace = autoToken.test(placement);\n\n if (autoPlace) {\n placement = placement.replace(autoToken, '') || defaults.placement;\n }\n\n // Need to add the position class before we get\n // the offsets\n tipElement.addClass(options.placement);\n\n // Get the position of the target element\n // and the height and width of the tooltip so we can center it.\n var elementPosition = getPosition();\n var tipWidth = tipElement.prop('offsetWidth');\n var tipHeight = tipElement.prop('offsetHeight');\n\n // Refresh viewport position\n $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport);\n\n // If we're auto placing, we need to check the positioning\n if (autoPlace) {\n var originalPlacement = placement;\n var viewportPosition = getPosition($tooltip.$viewport);\n\n if (/bottom/.test(originalPlacement) && elementPosition.bottom + tipHeight > viewportPosition.bottom) {\n placement = originalPlacement.replace('bottom', 'top');\n } else if (/top/.test(originalPlacement) && elementPosition.top - tipHeight < viewportPosition.top) {\n placement = originalPlacement.replace('top', 'bottom');\n }\n\n if (/left/.test(originalPlacement) && elementPosition.left - tipWidth < viewportPosition.left) {\n placement = placement.replace('left', 'right');\n } else if (/right/.test(originalPlacement) && elementPosition.right + tipWidth > viewportPosition.width) {\n placement = placement.replace('right', 'left');\n }\n\n tipElement.removeClass(originalPlacement).addClass(placement);\n }\n\n // Get the tooltip's top and left coordinates to center it with this directive.\n var tipPosition = getCalculatedOffset(placement, elementPosition, tipWidth, tipHeight);\n applyPlacement(tipPosition, placement);\n };\n\n $tooltip.$onKeyUp = function (evt) {\n if (evt.which === 27 && $tooltip.$isShown) {\n $tooltip.hide();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusKeyUp = function (evt) {\n if (evt.which === 27) {\n element[0].blur();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusElementMouseDown = function (evt) {\n if (options.mouseDownPreventDefault) { evt.preventDefault(); }\n if (options.mouseDownStopPropagation) { evt.stopPropagation(); }\n // Some browsers do not auto-focus buttons (eg. Safari)\n if ($tooltip.$isShown) {\n element[0].blur();\n } else {\n element[0].focus();\n }\n };\n\n // bind/unbind events\n function bindTriggerEvents () {\n var triggers = options.trigger.split(' ');\n angular.forEach(triggers, function (trigger) {\n if (trigger === 'click' || trigger === 'contextmenu') {\n element.on(trigger, $tooltip.toggle);\n } else if (trigger !== 'manual') {\n element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n if (nodeName === 'button' && trigger !== 'hover') {\n element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n });\n }\n\n function unbindTriggerEvents () {\n var triggers = options.trigger.split(' ');\n for (var i = triggers.length; i--;) {\n var trigger = triggers[i];\n if (trigger === 'click' || trigger === 'contextmenu') {\n element.off(trigger, $tooltip.toggle);\n } else if (trigger !== 'manual') {\n element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n if (nodeName === 'button' && trigger !== 'hover') {\n element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n }\n }\n\n function bindKeyboardEvents () {\n if (options.trigger !== 'focus') {\n tipElement.on('keyup', $tooltip.$onKeyUp);\n } else {\n element.on('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n function unbindKeyboardEvents () {\n if (options.trigger !== 'focus') {\n tipElement.off('keyup', $tooltip.$onKeyUp);\n } else {\n element.off('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n var _autoCloseEventsBinded = false;\n function bindAutoCloseEvents () {\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n // Stop propagation when clicking inside tooltip\n tipElement.on('click', stopEventPropagation);\n\n // Hide when clicking outside tooltip\n $body.on('click', $tooltip.hide);\n\n _autoCloseEventsBinded = true;\n }, 0, false);\n }\n\n function unbindAutoCloseEvents () {\n if (_autoCloseEventsBinded) {\n tipElement.off('click', stopEventPropagation);\n $body.off('click', $tooltip.hide);\n _autoCloseEventsBinded = false;\n }\n }\n\n function stopEventPropagation (event) {\n event.stopPropagation();\n }\n\n // Private methods\n\n function getPosition ($element) {\n $element = $element || (options.target || element);\n\n var el = $element[0];\n var isBody = el.tagName === 'BODY';\n\n var elRect = el.getBoundingClientRect();\n var rect = {};\n\n // IE8 has issues with angular.extend and using elRect directly.\n // By coping the values of elRect into a new object, we can continue to use extend\n /* eslint-disable guard-for-in */\n for (var p in elRect) { // eslint-disable-line\n // DO NOT use hasOwnProperty when inspecting the return of getBoundingClientRect.\n rect[p] = elRect[p];\n }\n /* eslint-enable guard-for-in */\n\n if (rect.width === null) {\n // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093\n rect = angular.extend({}, rect, {width: elRect.right - elRect.left, height: elRect.bottom - elRect.top});\n }\n var elOffset = isBody ? {top: 0, left: 0} : dimensions.offset(el);\n var scroll = {scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0};\n var outerDims = isBody ? {width: document.documentElement.clientWidth, height: $window.innerHeight} : null;\n\n return angular.extend({}, rect, scroll, outerDims, elOffset);\n }\n\n function getCalculatedOffset (placement, position, actualWidth, actualHeight) {\n var offset;\n var split = placement.split('-');\n\n switch (split[0]) {\n case 'right':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left + position.width\n };\n break;\n case 'bottom':\n offset = {\n top: position.top + position.height,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n case 'left':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left - actualWidth\n };\n break;\n default:\n offset = {\n top: position.top - actualHeight,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n }\n\n if (!split[1]) {\n return offset;\n }\n\n // Add support for corners @todo css\n if (split[0] === 'top' || split[0] === 'bottom') {\n switch (split[1]) {\n case 'left':\n offset.left = position.left;\n break;\n case 'right':\n offset.left = position.left + position.width - actualWidth;\n break;\n default:\n break;\n }\n } else if (split[0] === 'left' || split[0] === 'right') {\n switch (split[1]) {\n case 'top':\n offset.top = position.top - actualHeight + position.height;\n break;\n case 'bottom':\n offset.top = position.top;\n break;\n default:\n break;\n }\n }\n\n return offset;\n }\n\n function applyPlacement (offset, placement) {\n var tip = tipElement[0];\n var width = tip.offsetWidth;\n var height = tip.offsetHeight;\n\n // manually read margins because getBoundingClientRect includes difference\n var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10);\n var marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10);\n\n // we must check for NaN for ie 8/9\n if (isNaN(marginTop)) marginTop = 0;\n if (isNaN(marginLeft)) marginLeft = 0;\n\n offset.top = offset.top + marginTop;\n offset.left = offset.left + marginLeft;\n\n // dimensions setOffset doesn't round pixel values\n // so we use setOffset directly with our own function\n dimensions.setOffset(tip, angular.extend({\n using: function (props) {\n tipElement.css({\n top: Math.round(props.top) + 'px',\n left: Math.round(props.left) + 'px',\n right: ''\n });\n }\n }, offset), 0);\n\n // check to see if placing tip in new offset caused the tip to resize itself\n var actualWidth = tip.offsetWidth;\n var actualHeight = tip.offsetHeight;\n\n if (placement === 'top' && actualHeight !== height) {\n offset.top = offset.top + height - actualHeight;\n }\n\n // If it's an exotic placement, exit now instead of\n // applying a delta and changing the arrow\n if (/top-left|top-right|bottom-left|bottom-right/.test(placement)) return;\n\n var delta = getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight);\n\n if (delta.left) {\n offset.left += delta.left;\n } else {\n offset.top += delta.top;\n }\n\n dimensions.setOffset(tip, offset);\n\n if (/top|right|bottom|left/.test(placement)) {\n var isVertical = /top|bottom/.test(placement);\n var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight;\n var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight';\n\n replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical);\n }\n }\n\n // @source https://github.com/twbs/bootstrap/blob/v3.3.5/js/tooltip.js#L380\n function getViewportAdjustedDelta (placement, position, actualWidth, actualHeight) {\n var delta = {top: 0, left: 0};\n if (!$tooltip.$viewport) return delta;\n\n var viewportPadding = options.viewport && options.viewport.padding || 0;\n var viewportDimensions = getPosition($tooltip.$viewport);\n\n if (/right|left/.test(placement)) {\n var topEdgeOffset = position.top - viewportPadding - viewportDimensions.scroll;\n var bottomEdgeOffset = position.top + viewportPadding - viewportDimensions.scroll + actualHeight;\n if (topEdgeOffset < viewportDimensions.top) { // top overflow\n delta.top = viewportDimensions.top - topEdgeOffset;\n } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow\n delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset;\n }\n } else {\n var leftEdgeOffset = position.left - viewportPadding;\n var rightEdgeOffset = position.left + viewportPadding + actualWidth;\n if (leftEdgeOffset < viewportDimensions.left) { // left overflow\n delta.left = viewportDimensions.left - leftEdgeOffset;\n } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow\n delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset;\n }\n }\n\n return delta;\n }\n\n function replaceArrow (delta, dimension, isHorizontal) {\n var $arrow = findElement('.tooltip-arrow, .arrow', tipElement[0]);\n\n $arrow.css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')\n .css(isHorizontal ? 'top' : 'left', '');\n }\n\n function destroyTipElement () {\n // Cancel pending callbacks\n clearTimeout(timeout);\n\n if ($tooltip.$isShown && tipElement !== null) {\n if (options.autoClose) {\n unbindAutoCloseEvents();\n }\n\n if (options.keyboard) {\n unbindKeyboardEvents();\n }\n }\n\n if (tipScope) {\n tipScope.$destroy();\n tipScope = null;\n }\n\n if (tipElement) {\n tipElement.remove();\n tipElement = $tooltip.$element = null;\n }\n }\n\n return $tooltip;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return TooltipFactory;\n\n };\n\n })\n\n .directive('bsTooltip', function ($window, $location, $sce, $parse, $tooltip, $$rAF) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n var tooltip;\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'titleTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if (angular.isDefined(dataTarget)) {\n if (falseValueRegExp.test(dataTarget)) {\n options.target = false;\n } else {\n options.target = dataTarget;\n }\n }\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')) {\n scope.title = '';\n }\n\n // Observe scope attributes for change\n attr.$observe('title', function (newValue) {\n if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) {\n var oldValue = scope.title;\n scope.title = $sce.trustAsHtml(newValue);\n if (angular.isDefined(oldValue)) {\n $$rAF(function () {\n if (tooltip) tooltip.$applyPlacement();\n });\n }\n }\n });\n\n attr.$observe('disabled', function (newValue) {\n if (newValue && tooltip.$isShown) {\n tooltip.hide();\n }\n });\n\n // Support scope as an object\n if (attr.bsTooltip) {\n scope.$watch(attr.bsTooltip, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.title = newValue;\n }\n if (angular.isDefined(oldValue)) {\n $$rAF(function () {\n if (tooltip) tooltip.$applyPlacement();\n });\n }\n }, true);\n }\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!tooltip || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i);\n if (newValue === true) {\n tooltip.show();\n } else {\n tooltip.hide();\n }\n });\n }\n\n // Enabled binding support\n if (attr.bsEnabled) {\n scope.$watch(attr.bsEnabled, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.bsEnabled, newValue, oldValue);\n if (!tooltip || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i);\n if (newValue === false) {\n tooltip.setEnabled(false);\n } else {\n tooltip.setEnabled(true);\n }\n });\n }\n\n // Viewport support\n if (attr.viewport) {\n scope.$watch(attr.viewport, function (newValue) {\n if (!tooltip || !angular.isDefined(newValue)) return;\n tooltip.setViewport(newValue);\n });\n }\n\n // Initialize popover\n tooltip = $tooltip(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (tooltip) tooltip.destroy();\n options = null;\n tooltip = null;\n });\n\n }\n };\n\n });\n\n// Source: timepicker/timepicker.js\nangular.module('mgcrea.ngStrap.timepicker', ['mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip'])\n\n .provider('$timepicker', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n defaultDate: 'auto',\n // uncommenting the following line will break backwards compatability\n // prefixEvent: 'timepicker',\n prefixClass: 'timepicker',\n placement: 'bottom-left',\n templateUrl: 'timepicker/timepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: true,\n timeType: 'date',\n timeFormat: 'shortTime',\n timezone: null,\n modelTimeFormat: null,\n autoclose: false,\n minTime: -Infinity,\n maxTime: +Infinity,\n length: 5,\n hourStep: 1,\n minuteStep: 5,\n secondStep: 5,\n roundDisplay: false,\n iconUp: 'glyphicon glyphicon-chevron-up',\n iconDown: 'glyphicon glyphicon-chevron-down',\n arrowBehavior: 'pager'\n };\n\n this.$get = function ($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) {\n defaults.lang = $dateFormatter.getDefaultLocale();\n }\n\n function timepickerFactory (element, controller, config) {\n\n var $timepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $timepicker.$options;\n var scope = $timepicker.$scope;\n\n var lang = options.lang;\n var formatDate = function (date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n function floorMinutes (time) {\n // coeff used to floor current time to nearest minuteStep interval\n var coeff = 1000 * 60 * options.minuteStep;\n return new Date(Math.floor(time.getTime() / coeff) * coeff);\n }\n\n // View vars\n\n var selectedIndex = 0;\n var defaultDate = options.roundDisplay ? floorMinutes(new Date()) : new Date();\n var startDate = controller.$dateValue || defaultDate;\n var viewDate = {\n hour: startDate.getHours(),\n meridian: startDate.getHours() < 12,\n minute: startDate.getMinutes(),\n second: startDate.getSeconds(),\n millisecond: startDate.getMilliseconds()\n };\n\n var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang);\n\n var hoursFormat = $dateFormatter.hoursFormat(format);\n var timeSeparator = $dateFormatter.timeSeparator(format);\n var minutesFormat = $dateFormatter.minutesFormat(format);\n var secondsFormat = $dateFormatter.secondsFormat(format);\n var showSeconds = $dateFormatter.showSeconds(format);\n var showAM = $dateFormatter.showAM(format);\n\n scope.$iconUp = options.iconUp;\n scope.$iconDown = options.iconDown;\n\n // Scope methods\n\n scope.$select = function (date, index) {\n $timepicker.select(date, index);\n };\n scope.$moveIndex = function (value, index) {\n $timepicker.$moveIndex(value, index);\n };\n scope.$switchMeridian = function (date) {\n $timepicker.switchMeridian(date);\n };\n\n // Public methods\n\n $timepicker.update = function (date) {\n // console.warn('$timepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $timepicker.$date = date;\n angular.extend(viewDate, {\n hour: date.getHours(),\n minute: date.getMinutes(),\n second: date.getSeconds(),\n millisecond: date.getMilliseconds()\n });\n $timepicker.$build();\n } else if (!$timepicker.$isBuilt) {\n $timepicker.$build();\n }\n };\n\n $timepicker.select = function (date, index, keep) {\n // console.warn('$timepicker.select', date, scope.$mode);\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n controller.$dateValue = options.defaultDate === 'today' ? new Date() : new Date(1970, 0, 1);\n }\n\n if (!angular.isDate(date)) date = new Date(date);\n if (index === 0) controller.$dateValue.setHours(date.getHours());\n else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes());\n else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds());\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function () {\n $timepicker.hide(true);\n });\n }\n };\n\n $timepicker.switchMeridian = function (date) {\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n return;\n }\n var hours = (date || controller.$dateValue).getHours();\n controller.$dateValue.setHours(hours < 12 ? hours + 12 : hours - 12);\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n };\n\n // Protected methods\n\n $timepicker.$build = function () {\n // console.warn('$timepicker.$build() viewDate=%o', viewDate);\n var i;\n var midIndex = scope.midIndex = parseInt(options.length / 2, 10);\n var hours = [];\n var hour;\n for (i = 0; i < options.length; i++) {\n hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep);\n hours.push({\n date: hour,\n label: formatDate(hour, hoursFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(hour, 0),\n disabled: $timepicker.$isDisabled(hour, 0)\n });\n }\n var minutes = [];\n var minute;\n for (i = 0; i < options.length; i++) {\n minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep);\n minutes.push({\n date: minute,\n label: formatDate(minute, minutesFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(minute, 1),\n disabled: $timepicker.$isDisabled(minute, 1)\n });\n }\n var seconds = [];\n var second;\n for (i = 0; i < options.length; i++) {\n second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep);\n seconds.push({\n date: second,\n label: formatDate(second, secondsFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(second, 2),\n disabled: $timepicker.$isDisabled(second, 2)\n });\n }\n\n var rows = [];\n for (i = 0; i < options.length; i++) {\n if (showSeconds) {\n rows.push([hours[i], minutes[i], seconds[i]]);\n } else {\n rows.push([hours[i], minutes[i]]);\n }\n }\n scope.rows = rows;\n scope.showSeconds = showSeconds;\n scope.showAM = showAM;\n scope.isAM = ($timepicker.$date || hours[midIndex].date).getHours() < 12;\n scope.timeSeparator = timeSeparator;\n $timepicker.$isBuilt = true;\n };\n\n $timepicker.$isSelected = function (date, index) {\n if (!$timepicker.$date) return false;\n else if (index === 0) {\n return date.getHours() === $timepicker.$date.getHours();\n } else if (index === 1) {\n return date.getMinutes() === $timepicker.$date.getMinutes();\n } else if (index === 2) {\n return date.getSeconds() === $timepicker.$date.getSeconds();\n }\n };\n\n $timepicker.$isDisabled = function (date, index) {\n var selectedTime;\n if (index === 0) {\n selectedTime = date.getTime() + viewDate.minute * 6e4 + viewDate.second * 1e3;\n } else if (index === 1) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.second * 1e3;\n } else if (index === 2) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.minute * 6e4;\n }\n return selectedTime < options.minTime * 1 || selectedTime > options.maxTime * 1;\n };\n\n scope.$arrowAction = function (value, index) {\n if (options.arrowBehavior === 'picker') {\n $timepicker.$setTimeByStep(value, index);\n } else {\n $timepicker.$moveIndex(value, index);\n }\n };\n\n $timepicker.$setTimeByStep = function (value, index) {\n var newDate = new Date($timepicker.$date || startDate);\n var hours = newDate.getHours();\n var minutes = newDate.getMinutes();\n var seconds = newDate.getSeconds();\n if (index === 0) {\n newDate.setHours(hours - (parseInt(options.hourStep, 10) * value));\n } else if (index === 1) {\n newDate.setMinutes(minutes - (parseInt(options.minuteStep, 10) * value));\n } else if (index === 2) {\n newDate.setSeconds(seconds - (parseInt(options.secondStep, 10) * value));\n }\n $timepicker.select(newDate, index, true);\n };\n\n $timepicker.$moveIndex = function (value, index) {\n var targetDate;\n if (index === 0) {\n targetDate = new Date(1970, 0, 1, viewDate.hour + (value * options.length), viewDate.minute, viewDate.second);\n angular.extend(viewDate, {\n hour: targetDate.getHours()\n });\n } else if (index === 1) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute + (value * options.length * options.minuteStep), viewDate.second);\n angular.extend(viewDate, {\n minute: targetDate.getMinutes()\n });\n } else if (index === 2) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute, viewDate.second + (value * options.length * options.secondStep));\n angular.extend(viewDate, {\n second: targetDate.getSeconds()\n });\n }\n $timepicker.$build();\n };\n\n $timepicker.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n if (evt.target.nodeName.toLowerCase() !== 'input') evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $timepicker.$onKeyDown = function (evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Close on enter\n if (evt.keyCode === 13) {\n $timepicker.hide(true);\n return;\n }\n\n // Navigate with keyboard\n var newDate = new Date($timepicker.$date);\n var hours = newDate.getHours();\n var hoursLength = formatDate(newDate, hoursFormat).length;\n var minutes = newDate.getMinutes();\n var minutesLength = formatDate(newDate, minutesFormat).length;\n var seconds = newDate.getSeconds();\n var secondsLength = formatDate(newDate, secondsFormat).length;\n var sepLength = 1;\n var lateralMove = /(37|39)/.test(evt.keyCode);\n var count = 2 + showSeconds * 1 + showAM * 1;\n\n // Navigate indexes (left, right)\n if (lateralMove) {\n if (evt.keyCode === 37) selectedIndex = selectedIndex < 1 ? count - 1 : selectedIndex - 1;\n else if (evt.keyCode === 39) selectedIndex = selectedIndex < count - 1 ? selectedIndex + 1 : 0;\n }\n\n // Update values (up, down)\n var selectRange = [0, hoursLength];\n var incr = 0;\n if (evt.keyCode === 38) incr = -1;\n if (evt.keyCode === 40) incr = +1;\n var isSeconds = selectedIndex === 2 && showSeconds;\n var isMeridian = selectedIndex === 2 && !showSeconds || selectedIndex === 3 && showSeconds;\n if (selectedIndex === 0) {\n newDate.setHours(hours + incr * parseInt(options.hourStep, 10));\n // re-calculate hours length because we have changed hours value\n hoursLength = formatDate(newDate, hoursFormat).length;\n selectRange = [0, hoursLength];\n } else if (selectedIndex === 1) {\n newDate.setMinutes(minutes + incr * parseInt(options.minuteStep, 10));\n // re-calculate minutes length because we have changes minutes value\n minutesLength = formatDate(newDate, minutesFormat).length;\n selectRange = [hoursLength + sepLength, minutesLength];\n } else if (isSeconds) {\n newDate.setSeconds(seconds + incr * parseInt(options.secondStep, 10));\n // re-calculate seconds length because we have changes seconds value\n secondsLength = formatDate(newDate, secondsFormat).length;\n selectRange = [hoursLength + sepLength + minutesLength + sepLength, secondsLength];\n } else if (isMeridian) {\n if (!lateralMove) $timepicker.switchMeridian();\n selectRange = [hoursLength + sepLength + minutesLength + sepLength + (secondsLength + sepLength) * showSeconds, 2];\n }\n $timepicker.select(newDate, selectedIndex, true);\n createSelection(selectRange[0], selectRange[1]);\n parentScope.$digest();\n };\n\n // Private\n\n function createSelection (start, length) {\n var end = start + length;\n if (element[0].createTextRange) {\n var selRange = element[0].createTextRange();\n selRange.collapse(true);\n selRange.moveStart('character', start);\n selRange.moveEnd('character', end);\n selRange.select();\n } else if (element[0].setSelectionRange) {\n element[0].setSelectionRange(start, end);\n } else if (angular.isUndefined(element[0].selectionStart)) {\n element[0].selectionStart = start;\n element[0].selectionEnd = end;\n }\n }\n\n function focusElement () {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $timepicker.init;\n $timepicker.init = function () {\n if (isNative && options.useNative) {\n element.prop('type', 'time');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $timepicker.destroy;\n $timepicker.destroy = function () {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $timepicker.show;\n $timepicker.show = function () {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n if ($timepicker.$element) $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n if (element) element.on('keydown', $timepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $timepicker.hide;\n $timepicker.hide = function (blur) {\n if (!$timepicker.$isShown) return;\n if ($timepicker.$element) $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n if (element) element.off('keydown', $timepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $timepicker;\n\n }\n\n timepickerFactory.defaults = defaults;\n return timepickerFactory;\n\n };\n\n })\n\n\n .directive('bsTimepicker', function ($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) {\n\n var defaults = $timepicker.defaults;\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent', 'defaultDate'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'roundDisplay'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Initialize timepicker\n if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm';\n var timepicker = $timepicker(element, controller, options);\n options = timepicker.$options;\n\n var lang = options.lang;\n var formatDate = function (date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!timepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i);\n if (newValue === true) {\n timepicker.show();\n } else {\n timepicker.hide();\n }\n });\n }\n\n // Initialize parser\n var dateParser = $dateParser({\n format: options.timeFormat,\n lang: lang\n });\n\n // Observe attributes for changes\n angular.forEach(['minTime', 'maxTime'], function (key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n if (angular.isDefined(attr[key])) {\n attr.$observe(key, function (newValue) {\n timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue);\n if (!isNaN(timepicker.$options[key])) timepicker.$build();\n validateAgainstMinMaxTime(controller.$dateValue);\n });\n }\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue, controller.$dateValue);\n timepicker.update(controller.$dateValue);\n }, true);\n\n function validateAgainstMinMaxTime (parsedTime) {\n if (!angular.isDate(parsedTime)) return;\n var isMinValid = isNaN(options.minTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) >= options.minTime;\n var isMaxValid = isNaN(options.maxTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) <= options.maxTime;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (!isValid) {\n return;\n }\n controller.$dateValue = parsedTime;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function (viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n controller.$setValidity('date', true);\n return null;\n }\n var parsedTime = angular.isDate(viewValue) ? viewValue : dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedTime || isNaN(parsedTime.getTime())) {\n controller.$setValidity('date', false);\n // Return undefined, causes ngModelController to\n // invalidate model value\n return undefined;\n }\n validateAgainstMinMaxTime(parsedTime);\n\n if (options.timeType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true);\n return formatDate(date, options.modelTimeFormat || options.timeFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.timeType === 'number') {\n return date.getTime();\n } else if (options.timeType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.timeType === 'iso') {\n return date.toISOString();\n }\n return new Date(date);\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.timeType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelTimeFormat);\n } else if (options.timeType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if(isNaN(date.getTime())) date = new Date(new Date().setMinutes(0) + 36e5);\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getTimeFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function () {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getTimeFormattedString());\n };\n\n function getTimeFormattedString () {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.timeFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (timepicker) timepicker.destroy();\n options = null;\n timepicker = null;\n });\n\n }\n };\n\n });\n\n// Source: tab/tab.js\nangular.module('mgcrea.ngStrap.tab', [])\n\n .provider('$tab', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n template: 'tab/tab.tpl.html',\n navClass: 'nav-tabs',\n activeClass: 'active'\n };\n\n var controller = this.controller = function ($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'navClass', 'activeClass'], function (key) {\n if (angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // Publish options on scope\n $scope.$navClass = self.$options.navClass;\n $scope.$activeClass = self.$options.activeClass;\n\n self.$panes = $scope.$panes = [];\n\n // Please use $activePaneChangeListeners if you use `bsActivePane`\n // Because we removed `ngModel` as default, we rename viewChangeListeners to\n // activePaneChangeListeners to make more sense.\n self.$activePaneChangeListeners = self.$viewChangeListeners = [];\n\n self.$push = function (pane) {\n if (angular.isUndefined(self.$panes.$active)) {\n $scope.$setActive(pane.name || 0);\n }\n self.$panes.push(pane);\n };\n\n self.$remove = function (pane) {\n var index = self.$panes.indexOf(pane);\n var active = self.$panes.$active;\n var activeIndex;\n if (angular.isString(active)) {\n activeIndex = self.$panes.map(function (pane) {\n return pane.name;\n }).indexOf(active);\n } else {\n activeIndex = self.$panes.$active;\n }\n\n // remove pane from $panes array\n self.$panes.splice(index, 1);\n\n if (index < activeIndex) {\n // we removed a pane before the active pane, so we need to\n // decrement the active pane index\n activeIndex--;\n } else if (index === activeIndex && activeIndex === self.$panes.length) {\n // we remove the active pane and it was the one at the end,\n // so select the previous one\n activeIndex--;\n }\n if (activeIndex >= 0 && activeIndex < self.$panes.length) {\n self.$setActive(self.$panes[activeIndex].name || activeIndex);\n } else {\n self.$setActive();\n }\n };\n\n self.$setActive = $scope.$setActive = function (value) {\n self.$panes.$active = value;\n self.$activePaneChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n self.$isActive = $scope.$isActive = function ($pane, $index) {\n return self.$panes.$active === $pane.name || self.$panes.$active === $index;\n };\n\n };\n\n this.$get = function () {\n var $tab = {};\n $tab.defaults = defaults;\n $tab.controller = controller;\n return $tab;\n };\n\n })\n\n .directive('bsTabs', function ($window, $animate, $tab, $parse) {\n\n var defaults = $tab.defaults;\n\n return {\n require: ['?ngModel', 'bsTabs'],\n transclude: true,\n scope: true,\n controller: ['$scope', '$element', '$attrs', $tab.controller],\n templateUrl: function (element, attr) {\n return attr.template || defaults.template;\n },\n link: function postLink (scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // 'ngModel' does interfere with form validation\n // and status, use `bsActivePane` instead to avoid it\n if (ngModelCtrl) {\n\n // Update the modelValue following\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n ngModelCtrl.$setViewValue(bsTabsCtrl.$panes.$active);\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n bsTabsCtrl.$setActive(modelValue);\n return modelValue;\n });\n\n }\n\n if (attrs.bsActivePane) {\n // adapted from angularjs ngModelController bindings\n // https://github.com/angular/angular.js/blob/v1.3.1/src%2Fng%2Fdirective%2Finput.js#L1730\n var parsedBsActivePane = $parse(attrs.bsActivePane);\n\n // Update bsActivePane value with change\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n parsedBsActivePane.assign(scope, bsTabsCtrl.$panes.$active);\n });\n\n // watch bsActivePane for value changes\n scope.$watch(attrs.bsActivePane, function (newValue, oldValue) {\n bsTabsCtrl.$setActive(newValue);\n }, true);\n }\n }\n };\n\n })\n\n .directive('bsPane', function ($window, $animate, $sce) {\n\n return {\n require: ['^?ngModel', '^bsTabs'],\n scope: true,\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // Add base class\n element.addClass('tab-pane');\n\n // Observe title attribute for change\n attrs.$observe('title', function (newValue, oldValue) {\n scope.title = $sce.trustAsHtml(newValue);\n });\n\n // Save tab name into scope\n scope.name = attrs.name;\n\n // Add animation class\n if (bsTabsCtrl.$options.animation) {\n element.addClass(bsTabsCtrl.$options.animation);\n }\n\n attrs.$observe('disabled', function (newValue, oldValue) {\n scope.disabled = scope.$eval(newValue);\n });\n\n // Push pane to parent bsTabs controller\n bsTabsCtrl.$push(scope);\n\n // remove pane from tab controller when pane is destroyed\n scope.$on('$destroy', function () {\n bsTabsCtrl.$remove(scope);\n });\n\n function render () {\n var index = bsTabsCtrl.$panes.indexOf(scope);\n $animate[bsTabsCtrl.$isActive(scope, index) ? 'addClass' : 'removeClass'](element, bsTabsCtrl.$options.activeClass);\n }\n\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n render();\n });\n render();\n\n }\n };\n\n });\n\n// Source: select/select.js\nangular.module('mgcrea.ngStrap.select', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$select', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'select',\n prefixEvent: '$select',\n placement: 'bottom-left',\n templateUrl: 'select/select.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n multiple: false,\n allNoneButtons: false,\n sort: true,\n caretHtml: ' ',\n placeholder: 'Choose among the following...',\n allText: 'All',\n noneText: 'None',\n maxLength: 3,\n maxLengthHtml: 'selected',\n iconCheckmark: 'glyphicon glyphicon-ok',\n toggle: false\n };\n\n this.$get = function ($window, $document, $rootScope, $tooltip, $timeout) {\n\n // var bodyEl = angular.element($window.document.body);\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n\n function SelectFactory (element, controller, config) {\n\n var $select = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $select = $tooltip(element, options);\n var scope = $select.$scope;\n\n scope.$matches = [];\n if (options.multiple) {\n scope.$activeIndex = [];\n } else {\n scope.$activeIndex = -1;\n }\n scope.$isMultiple = options.multiple;\n scope.$showAllNoneButtons = options.allNoneButtons && options.multiple;\n scope.$iconCheckmark = options.iconCheckmark;\n scope.$allText = options.allText;\n scope.$noneText = options.noneText;\n\n scope.$activate = function (index) {\n scope.$$postDigest(function () {\n $select.activate(index);\n });\n };\n\n scope.$select = function (index, evt) {\n scope.$$postDigest(function () {\n $select.select(index);\n });\n };\n\n scope.$isVisible = function () {\n return $select.$isVisible();\n };\n\n scope.$isActive = function (index) {\n return $select.$isActive(index);\n };\n\n scope.$selectAll = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (!scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n scope.$selectNone = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n // Public methods\n\n $select.update = function (matches) {\n scope.$matches = matches;\n $select.$updateActiveIndex();\n };\n\n $select.activate = function (index) {\n if (options.multiple) {\n if ($select.$isActive(index)) {\n scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1);\n } else {\n scope.$activeIndex.push(index);\n }\n if (options.sort) scope.$activeIndex.sort(function (a, b) { return a - b; }); // use numeric sort instead of default sort\n } else {\n scope.$activeIndex = index;\n }\n return scope.$activeIndex;\n };\n\n $select.select = function (index) {\n if (angular.isUndefined(index) || index < 0 || index >= scope.$matches.length) { return; }\n var value = scope.$matches[index].value;\n scope.$apply(function () {\n $select.activate(index);\n if (options.multiple) {\n controller.$setViewValue(scope.$activeIndex.map(function (index) {\n if (angular.isUndefined(scope.$matches[index])) {\n return null;\n }\n return scope.$matches[index].value;\n }));\n } else {\n if (options.toggle) {\n controller.$setViewValue((value === controller.$modelValue) ? undefined : value);\n } else {\n controller.$setViewValue(value);\n }\n // Hide if single select\n $select.hide();\n }\n });\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $select);\n if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) {\n options.onSelect(value, index, $select);\n }\n };\n\n // Protected methods\n\n $select.$updateActiveIndex = function () {\n if (options.multiple) {\n if (angular.isArray(controller.$modelValue)) {\n scope.$activeIndex = controller.$modelValue.map(function (value) {\n return $select.$getIndex(value);\n });\n } else {\n scope.$activeIndex = [];\n }\n } else {\n if (angular.isDefined(controller.$modelValue) && scope.$matches.length) {\n scope.$activeIndex = $select.$getIndex(controller.$modelValue);\n } else {\n scope.$activeIndex = -1;\n }\n }\n };\n\n $select.$isVisible = function () {\n if (!options.minLength || !controller) {\n return scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && controller.$viewValue.length >= options.minLength;\n };\n\n $select.$isActive = function (index) {\n if (options.multiple) {\n return scope.$activeIndex.indexOf(index) !== -1;\n }\n return scope.$activeIndex === index;\n };\n\n $select.$getIndex = function (value) {\n var index;\n for (index = scope.$matches.length; index--;) {\n if (angular.equals(scope.$matches[index].value, value)) break;\n }\n return index;\n };\n\n $select.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n var anchor;\n\n if (evt.target.nodeName !== 'A') {\n var anchorCandidate = targetEl.parent();\n while (!anchor && anchorCandidate.length > 0) {\n if (anchorCandidate[0].nodeName === 'A') {\n anchor = anchorCandidate;\n }\n anchorCandidate = anchorCandidate.parent();\n }\n }\n\n if (anchor) {\n angular.element(anchor).triggerHandler('click');\n } else {\n targetEl.triggerHandler('click');\n }\n }\n };\n\n $select.$onKeyDown = function (evt) {\n if (!/(9|13|38|40)/.test(evt.keyCode)) return;\n // Let tab propagate\n if (evt.keyCode !== 9) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // release focus on tab\n if (options.multiple && evt.keyCode === 9) {\n return $select.hide();\n }\n\n // Select with enter\n if (!options.multiple && (evt.keyCode === 13 || evt.keyCode === 9)) {\n return $select.select(scope.$activeIndex);\n }\n\n if (!options.multiple) {\n // Navigate with keyboard\n if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if (evt.keyCode === 38 && scope.$activeIndex < 0) scope.$activeIndex = scope.$matches.length - 1;\n else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n }\n };\n\n $select.$isIE = function () {\n var ua = $window.navigator.userAgent;\n return ua.indexOf('MSIE ') > 0 || ua.indexOf('Trident/') > 0 || ua.indexOf('Edge/') > 0;\n };\n\n $select.$selectScrollFix = function (e) {\n if ($document[0].activeElement.tagName === 'UL') {\n e.preventDefault();\n e.stopImmediatePropagation();\n e.target.focus();\n }\n };\n\n // Overrides\n\n var _show = $select.show;\n $select.show = function () {\n _show();\n if (options.multiple) {\n $select.$element.addClass('select-multiple');\n }\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n $select.$element.on(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $select.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $select.hide;\n $select.hide = function () {\n if (!options.multiple && angular.isUndefined(controller.$modelValue)) {\n scope.$activeIndex = -1;\n }\n $select.$element.off(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $select.$onKeyDown);\n }\n _hide(true);\n };\n\n return $select;\n\n }\n\n SelectFactory.defaults = defaults;\n return SelectFactory;\n\n };\n\n })\n\n .directive('bsSelect', function ($window, $parse, $q, $select, $parseOptions) {\n\n var defaults = $select.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope, placeholder: defaults.placeholder};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent', 'toggle'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'allNoneButtons', 'sort'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show, hide and select events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Only parse data-multiple. Angular sets existence attributes to true (multiple/required/etc), they apply this\n // to data-multiple as well for some reason, so we'll parse this ourselves and disregard multiple\n var dataMultiple = element.attr('data-multiple');\n if (angular.isDefined(dataMultiple)) {\n if (falseValueRegExp.test(dataMultiple)) {\n options.multiple = false;\n } else {\n options.multiple = dataMultiple;\n }\n }\n\n // Add support for select markup\n if (element[0].nodeName.toLowerCase() === 'select') {\n var inputEl = element;\n inputEl.css('display', 'none');\n element = angular.element('');\n inputEl.after(element);\n }\n\n // Build proper bsOptions\n var parsedOptions = $parseOptions(attr.bsOptions);\n\n // Initialize select\n var select = $select(element, controller, options);\n\n if (select.$isIE()) {\n element[0].addEventListener('blur', select.$selectScrollFix);\n }\n\n // Watch bsOptions values before filtering for changes\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').trim();\n scope.$watch(watchedOptions, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller)\n .then(function (values) {\n select.update(values);\n controller.$render();\n });\n }, true);\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue);\n select.$updateActiveIndex();\n controller.$render();\n }, true);\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var selected;\n var index;\n if (options.multiple && angular.isArray(controller.$modelValue)) {\n selected = controller.$modelValue.map(function (value) {\n index = select.$getIndex(value);\n return index !== -1 ? select.$scope.$matches[index].label : false;\n }).filter(angular.isDefined);\n if (selected.length > (options.maxLength || defaults.maxLength)) {\n selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml);\n } else {\n selected = selected.join(', ');\n }\n } else {\n index = select.$getIndex(controller.$modelValue);\n selected = index !== -1 ? select.$scope.$matches[index].label : false;\n }\n element.html((selected || options.placeholder) + (options.caretHtml || defaults.caretHtml));\n };\n\n if (options.multiple) {\n controller.$isEmpty = function (value) {\n return !value || value.length === 0;\n };\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (select) select.destroy();\n options = null;\n select = null;\n });\n\n }\n };\n\n });\n\n// Source: scrollspy/scrollspy.js\nangular.module('mgcrea.ngStrap.scrollspy', ['mgcrea.ngStrap.helpers.debounce', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$scrollspy', function () {\n\n // Pool of registered spies\n var spies = this.$$spies = {};\n\n var defaults = this.defaults = {\n debounce: 150,\n throttle: 100,\n offset: 100\n };\n\n this.$get = function ($window, $document, $rootScope, dimensions, debounce, throttle) {\n\n var windowEl = angular.element($window);\n var docEl = angular.element($document.prop('documentElement'));\n var bodyEl = angular.element($window.document.body);\n\n // Helper functions\n\n function nodeName (element, name) {\n return element[0].nodeName && element[0].nodeName.toLowerCase() === name.toLowerCase();\n }\n\n function ScrollSpyFactory (config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n if (!options.element) options.element = bodyEl;\n var isWindowSpy = nodeName(options.element, 'body');\n var scrollEl = isWindowSpy ? windowEl : options.element;\n var scrollId = isWindowSpy ? 'window' : options.id;\n\n // Use existing spy\n if (spies[scrollId]) {\n spies[scrollId].$$count++;\n return spies[scrollId];\n }\n\n var $scrollspy = {};\n\n // Private vars\n var unbindViewContentLoaded;\n var unbindIncludeContentLoaded;\n var trackedElements = $scrollspy.$trackedElements = [];\n var sortedElements = [];\n var activeTarget;\n var debouncedCheckPosition;\n var throttledCheckPosition;\n var debouncedCheckOffsets;\n /* eslint-disable no-unused-vars */\n var viewportHeight;\n /* eslint-enable no-unused-vars */\n var scrollTop;\n\n $scrollspy.init = function () {\n\n // Setup internal ref counter\n this.$$count = 1;\n\n // Bind events\n debouncedCheckPosition = debounce(this.checkPosition, options.debounce);\n throttledCheckPosition = throttle(this.checkPosition, options.throttle);\n scrollEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', debouncedCheckPosition);\n scrollEl.on('scroll', throttledCheckPosition);\n\n debouncedCheckOffsets = debounce(this.checkOffsets, options.debounce);\n unbindViewContentLoaded = $rootScope.$on('$viewContentLoaded', debouncedCheckOffsets);\n unbindIncludeContentLoaded = $rootScope.$on('$includeContentLoaded', debouncedCheckOffsets);\n debouncedCheckOffsets();\n\n // Register spy for reuse\n if (scrollId) {\n spies[scrollId] = $scrollspy;\n }\n\n };\n\n $scrollspy.destroy = function () {\n\n // Check internal ref counter\n this.$$count--;\n if (this.$$count > 0) {\n return;\n }\n\n // Unbind events\n scrollEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', debouncedCheckPosition);\n scrollEl.off('scroll', throttledCheckPosition);\n unbindViewContentLoaded();\n unbindIncludeContentLoaded();\n if (scrollId) {\n delete spies[scrollId];\n }\n };\n\n $scrollspy.checkPosition = function () {\n\n // Not ready yet\n if (!sortedElements.length) return;\n\n // Calculate the scroll position\n scrollTop = (isWindowSpy ? $window.pageYOffset : scrollEl.prop('scrollTop')) || 0;\n\n // Calculate the viewport height for use by the components\n viewportHeight = Math.max($window.innerHeight, docEl.prop('clientHeight'));\n\n // Activate first element if scroll is smaller\n if (scrollTop < sortedElements[0].offsetTop && activeTarget !== sortedElements[0].target) {\n return $scrollspy.$activateElement(sortedElements[0]);\n }\n\n // Activate proper element\n for (var i = sortedElements.length; i--;) {\n if (angular.isUndefined(sortedElements[i].offsetTop) || sortedElements[i].offsetTop === null) continue;\n if (activeTarget === sortedElements[i].target) continue;\n if (scrollTop < sortedElements[i].offsetTop) continue;\n if (sortedElements[i + 1] && scrollTop > sortedElements[i + 1].offsetTop) continue;\n return $scrollspy.$activateElement(sortedElements[i]);\n }\n\n };\n\n $scrollspy.checkPositionWithEventLoop = function () {\n // IE 9 throws an error if we use 'this' instead of '$scrollspy'\n // in this setTimeout call\n setTimeout($scrollspy.checkPosition, 1);\n };\n\n // Protected methods\n\n $scrollspy.$activateElement = function (element) {\n if (activeTarget) {\n var activeElement = $scrollspy.$getTrackedElement(activeTarget);\n if (activeElement) {\n activeElement.source.removeClass('active');\n if (nodeName(activeElement.source, 'li') && nodeName(activeElement.source.parent().parent(), 'li')) {\n activeElement.source.parent().parent().removeClass('active');\n }\n }\n }\n activeTarget = element.target;\n element.source.addClass('active');\n if (nodeName(element.source, 'li') && nodeName(element.source.parent().parent(), 'li')) {\n element.source.parent().parent().addClass('active');\n }\n };\n\n $scrollspy.$getTrackedElement = function (target) {\n return trackedElements.filter(function (obj) {\n return obj.target === target;\n })[0];\n };\n\n // Track offsets behavior\n\n $scrollspy.checkOffsets = function () {\n\n angular.forEach(trackedElements, function (trackedElement) {\n var targetElement = document.querySelector(trackedElement.target);\n trackedElement.offsetTop = targetElement ? dimensions.offset(targetElement).top : null;\n if (options.offset && trackedElement.offsetTop !== null) trackedElement.offsetTop -= options.offset * 1;\n });\n\n sortedElements = trackedElements\n .filter(function (el) {\n return el.offsetTop !== null;\n })\n .sort(function (a, b) {\n return a.offsetTop - b.offsetTop;\n });\n\n debouncedCheckPosition();\n\n };\n\n $scrollspy.trackElement = function (target, source) {\n trackedElements.push({target: target, source: source});\n };\n\n $scrollspy.untrackElement = function (target, source) {\n var toDelete;\n for (var i = trackedElements.length; i--;) {\n if (trackedElements[i].target === target && trackedElements[i].source === source) {\n toDelete = i;\n break;\n }\n }\n trackedElements.splice(toDelete, 1);\n };\n\n $scrollspy.activate = function (i) {\n trackedElements[i].addClass('active');\n };\n\n // Initialize plugin\n\n $scrollspy.init();\n return $scrollspy;\n\n }\n\n return ScrollSpyFactory;\n\n };\n\n })\n\n .directive('bsScrollspy', function ($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'EAC',\n link: function postLink (scope, element, attr) {\n\n var options = {scope: scope};\n angular.forEach(['offset', 'target'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n var scrollspy = $scrollspy(options);\n scrollspy.trackElement(options.target, element);\n\n scope.$on('$destroy', function () {\n if (scrollspy) {\n scrollspy.untrackElement(options.target, element);\n scrollspy.destroy();\n }\n options = null;\n scrollspy = null;\n });\n\n }\n };\n\n })\n\n\n .directive('bsScrollspyList', function ($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'A',\n compile: function postLink (element, attr) {\n var children = element[0].querySelectorAll('li > a[href]');\n angular.forEach(children, function (child) {\n var childEl = angular.element(child);\n childEl.parent().attr('bs-scrollspy', '').attr('data-target', childEl.attr('href'));\n });\n }\n\n };\n\n });\n\n// Source: popover/popover.js\nangular.module('mgcrea.ngStrap.popover', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$popover', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n // uncommenting the next two lines will break backwards compatability\n // prefixClass: 'popover',\n // prefixEvent: 'popover',\n container: false,\n target: false,\n placement: 'right',\n templateUrl: 'popover/popover.tpl.html',\n contentTemplate: false,\n trigger: 'click',\n keyboard: true,\n html: false,\n title: '',\n content: '',\n delay: 0,\n autoClose: false\n };\n\n this.$get = function ($tooltip) {\n\n function PopoverFactory (element, config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n var $popover = $tooltip(element, options);\n\n // Support scope as string options [/*title, */content]\n if (options.content) {\n $popover.$scope.content = options.content;\n }\n\n return $popover;\n\n }\n\n return PopoverFactory;\n\n };\n\n })\n\n .directive('bsPopover', function ($window, $sce, $popover) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr) {\n\n var popover;\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent', 'bsEnabled'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoClose'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if (angular.isDefined(dataTarget)) {\n if (falseValueRegExp.test(dataTarget)) {\n options.target = false;\n } else {\n options.target = dataTarget;\n }\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n if (angular.isDefined(oldValue)) {\n requestAnimationFrame(function () {\n if (popover) popover.$applyPlacement();\n });\n }\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsPopover) {\n scope.$watch(attr.bsPopover, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n if (angular.isDefined(oldValue)) {\n requestAnimationFrame(function () {\n if (popover) popover.$applyPlacement();\n });\n }\n }, true);\n }\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i);\n if (newValue === true) {\n popover.show();\n } else {\n popover.hide();\n }\n });\n }\n\n // Enabled binding support\n if (attr.bsEnabled) {\n scope.$watch(attr.bsEnabled, function (newValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(popover),?/i);\n if (newValue === false) {\n popover.setEnabled(false);\n } else {\n popover.setEnabled(true);\n }\n });\n }\n\n // Viewport support\n if (attr.viewport) {\n scope.$watch(attr.viewport, function (newValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n popover.setViewport(newValue);\n });\n }\n\n // Initialize popover\n popover = $popover(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (popover) popover.destroy();\n options = null;\n popover = null;\n });\n\n }\n };\n\n });\n\n// Source: navbar/navbar.js\nangular.module('mgcrea.ngStrap.navbar', [])\n\n .provider('$navbar', function () {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n routeAttr: 'data-match-route',\n strict: false\n };\n\n this.$get = function () {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsNavbar', function ($window, $location, $navbar) {\n\n var defaults = $navbar.defaults;\n\n return {\n restrict: 'A',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = angular.copy(defaults);\n angular.forEach(Object.keys(defaults), function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Watch for the $location\n scope.$watch(function () {\n\n return $location.path();\n\n }, function (newValue, oldValue) {\n\n var liElements = element[0].querySelectorAll('li[' + options.routeAttr + ']');\n\n angular.forEach(liElements, function (li) {\n\n var liElement = angular.element(li);\n var pattern = liElement.attr(options.routeAttr).replace('/', '\\\\/');\n if (options.strict) {\n pattern = '^' + pattern + '$';\n }\n var regexp = new RegExp(pattern, 'i');\n\n if (regexp.test(newValue)) {\n liElement.addClass(options.activeClass);\n } else {\n liElement.removeClass(options.activeClass);\n }\n\n });\n\n });\n\n }\n\n };\n\n });\n\n// Source: modal/modal.js\nangular.module('mgcrea.ngStrap.modal', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$modal', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n backdropAnimation: 'am-fade',\n customClass: '',\n prefixClass: 'modal',\n prefixEvent: 'modal',\n placement: 'top',\n templateUrl: 'modal/modal.tpl.html',\n template: '',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true,\n size: null,\n zIndex: null\n };\n\n this.$get = function ($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) {\n\n var forEach = angular.forEach;\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n var bodyElement = angular.element($window.document.body);\n\n var backdropCount = 0;\n var dialogBaseZindex = 1050;\n var backdropBaseZindex = 1040;\n\n var validSizes = {\n lg: 'modal-lg',\n sm: 'modal-sm'\n };\n\n function ModalFactory (config) {\n\n var $modal = {};\n\n // Common vars\n var options = $modal.$options = angular.extend({}, defaults, config);\n var promise = $modal.$promise = $bsCompiler.compile(options);\n var scope = $modal.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n if (!options.element && !options.container) {\n options.container = 'body';\n }\n\n if (options.zIndex) {\n dialogBaseZindex = parseInt(options.zIndex, 10);\n backdropBaseZindex = dialogBaseZindex - 10;\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $modal.$id = options.id || options.element && options.element.attr('id') || '';\n\n // Support scope as string options\n forEach(['title', 'content'], function (key) {\n if (options[key]) scope[key] = $sce.trustAsHtml(options[key]);\n });\n\n // Provide scope helpers\n scope.$hide = function () {\n scope.$$postDigest(function () {\n $modal.hide();\n });\n };\n scope.$show = function () {\n scope.$$postDigest(function () {\n $modal.show();\n });\n };\n scope.$toggle = function () {\n scope.$$postDigest(function () {\n $modal.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $modal.$isShown = scope.$isShown = false;\n\n // Fetch, compile then initialize modal\n var compileData;\n var modalElement;\n var modalScope;\n var backdropElement = angular.element('
');\n backdropElement.css({position: 'fixed', top: '0px', left: '0px', bottom: '0px', right: '0px'});\n promise.then(function (data) {\n compileData = data;\n $modal.init();\n });\n\n $modal.init = function () {\n\n // Options: show\n if (options.show) {\n scope.$$postDigest(function () {\n $modal.show();\n });\n }\n\n };\n\n $modal.destroy = function () {\n\n // Remove element\n destroyModalElement();\n\n // remove backdrop element\n if (backdropElement) {\n backdropElement.remove();\n backdropElement = null;\n }\n\n // Destroy scope\n scope.$destroy();\n };\n\n $modal.show = function () {\n if ($modal.$isShown) return;\n\n var parent;\n var after;\n if (angular.isElement(options.container)) {\n parent = options.container;\n after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null;\n } else {\n if (options.container) {\n parent = findElement(options.container);\n after = parent[0] && parent[0].lastChild ? angular.element(parent[0].lastChild) : null;\n } else {\n parent = null;\n after = options.element;\n }\n }\n\n // destroy any existing modal elements\n if (modalElement) destroyModalElement();\n\n // create a new scope, so we can destroy it and all child scopes\n // when destroying the modal element\n modalScope = $modal.$scope.$new();\n // Fetch a cloned element linked from template (noop callback is required)\n modalElement = $modal.$element = compileData.link(modalScope, function (clonedElement, scope) {});\n\n if (options.backdrop) {\n // set z-index\n modalElement.css({'z-index': dialogBaseZindex + (backdropCount * 20)});\n backdropElement.css({'z-index': backdropBaseZindex + (backdropCount * 20)});\n\n // increment number of backdrops\n backdropCount++;\n }\n\n if (scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) {\n return;\n }\n if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) {\n options.onBeforeShow($modal);\n }\n\n // Set the initial positioning.\n modalElement.css({display: 'block'}).addClass(options.placement);\n\n // Options: customClass\n if (options.customClass) {\n modalElement.addClass(options.customClass);\n }\n\n // Options: size\n if (options.size && validSizes[options.size]) {\n angular.element(findElement('.modal-dialog', modalElement[0])).addClass(validSizes[options.size]);\n }\n\n // Options: animation\n if (options.animation) {\n if (options.backdrop) {\n backdropElement.addClass(options.backdropAnimation);\n }\n modalElement.addClass(options.animation);\n }\n\n if (options.backdrop) {\n $animate.enter(backdropElement, bodyElement, null);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.enter(modalElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(modalElement, parent, after).then(enterAnimateCallback);\n }\n\n $modal.$isShown = scope.$isShown = true;\n safeDigest(scope);\n // Focus once the enter-animation has started\n // Weird PhantomJS bug hack\n var el = modalElement[0];\n requestAnimationFrame(function () {\n el.focus();\n });\n\n bodyElement.addClass(options.prefixClass + '-open');\n if (options.animation) {\n bodyElement.addClass(options.prefixClass + '-with-' + options.animation);\n }\n\n // Bind events\n bindBackdropEvents();\n bindKeyboardEvents();\n };\n\n function enterAnimateCallback () {\n scope.$emit(options.prefixEvent + '.show', $modal);\n if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) {\n options.onShow($modal);\n }\n }\n\n $modal.hide = function () {\n if (!$modal.$isShown) return;\n\n if (scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) {\n return;\n }\n if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) {\n options.onBeforeHide($modal);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.leave(modalElement, leaveAnimateCallback);\n } else {\n $animate.leave(modalElement).then(leaveAnimateCallback);\n }\n\n if (options.backdrop) {\n // decrement number of backdrops\n backdropCount--;\n $animate.leave(backdropElement);\n }\n $modal.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n };\n\n function leaveAnimateCallback () {\n scope.$emit(options.prefixEvent + '.hide', $modal);\n if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) {\n options.onHide($modal);\n }\n if (findElement('.modal').length <= 0) {\n bodyElement.removeClass(options.prefixClass + '-open');\n }\n if (options.animation) {\n bodyElement.removeClass(options.prefixClass + '-with-' + options.animation);\n }\n }\n\n $modal.toggle = function () {\n if ($modal.$isShown) {\n $modal.hide();\n } else {\n $modal.show();\n }\n };\n\n $modal.focus = function () {\n modalElement[0].focus();\n };\n\n // Protected methods\n\n $modal.$onKeyUp = function (evt) {\n\n if (evt.which === 27 && $modal.$isShown) {\n $modal.hide();\n evt.stopPropagation();\n }\n\n };\n\n function bindBackdropEvents () {\n if (options.backdrop) {\n modalElement.on('click', hideOnBackdropClick);\n backdropElement.on('click', hideOnBackdropClick);\n backdropElement.on('wheel', preventEventDefault);\n }\n }\n\n function unbindBackdropEvents () {\n if (options.backdrop) {\n modalElement.off('click', hideOnBackdropClick);\n backdropElement.off('click', hideOnBackdropClick);\n backdropElement.off('wheel', preventEventDefault);\n }\n }\n\n function bindKeyboardEvents () {\n if (options.keyboard) {\n modalElement.on('keyup', $modal.$onKeyUp);\n }\n }\n\n function unbindKeyboardEvents () {\n if (options.keyboard) {\n modalElement.off('keyup', $modal.$onKeyUp);\n }\n }\n\n // Private helpers\n\n function hideOnBackdropClick (evt) {\n if (evt.target !== evt.currentTarget) return;\n if (options.backdrop === 'static') {\n $modal.focus();\n } else {\n $modal.hide();\n }\n }\n\n function preventEventDefault (evt) {\n evt.preventDefault();\n }\n\n function destroyModalElement () {\n if ($modal.$isShown && modalElement !== null) {\n // un-bind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n }\n\n if (modalScope) {\n modalScope.$destroy();\n modalScope = null;\n }\n\n if (modalElement) {\n modalElement.remove();\n modalElement = $modal.$element = null;\n }\n }\n\n return $modal;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return ModalFactory;\n\n };\n\n })\n\n .directive('bsModal', function ($window, $sce, $parse, $modal) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass', 'customClass', 'modalClass', 'size', 'zIndex'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Options: alias modalClass to customClass\n if (options.modalClass) {\n options.customClass = options.modalClass;\n }\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsModal) {\n scope.$watch(attr.bsModal, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize modal\n var modal = $modal(options);\n\n // Trigger\n element.on(attr.trigger || 'click', modal.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (modal) modal.destroy();\n options = null;\n modal = null;\n });\n\n }\n };\n\n });\n\n// Source: helpers/raf.js\nif (angular.version.minor < 3 && angular.version.dot < 14) {\n angular.module('ng')\n\n .factory('$$rAF', function ($window, $timeout) {\n\n var requestAnimationFrame = $window.requestAnimationFrame ||\n $window.webkitRequestAnimationFrame ||\n $window.mozRequestAnimationFrame;\n\n var cancelAnimationFrame = $window.cancelAnimationFrame ||\n $window.webkitCancelAnimationFrame ||\n $window.mozCancelAnimationFrame ||\n $window.webkitCancelRequestAnimationFrame;\n\n var rafSupported = !!requestAnimationFrame;\n var raf = rafSupported ?\n function (fn) {\n var id = requestAnimationFrame(fn);\n return function () {\n cancelAnimationFrame(id);\n };\n } :\n function (fn) {\n var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666\n return function () {\n $timeout.cancel(timer);\n };\n };\n\n raf.supported = rafSupported;\n\n return raf;\n\n });\n}\n\n// Source: helpers/parse-options.js\nangular.module('mgcrea.ngStrap.helpers.parseOptions', [])\n\n .provider('$parseOptions', function () {\n\n var defaults = this.defaults = {\n regexp: /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n };\n\n this.$get = function ($parse, $q) {\n\n function ParseOptionsFactory (attr, config) {\n\n var $parseOptions = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n $parseOptions.$values = [];\n\n // Private vars\n var match;\n var displayFn;\n var valueName;\n /* eslint-disable no-unused-vars */\n var keyName;\n var groupByFn;\n /* eslint-enable no-unused-vars */\n var valueFn;\n var valuesFn;\n\n $parseOptions.init = function () {\n $parseOptions.$match = match = attr.match(options.regexp);\n displayFn = $parse(match[2] || match[1]);\n valueName = match[4] || match[6];\n keyName = match[5];\n groupByFn = $parse(match[3] || '');\n valueFn = $parse(match[2] ? match[1] : valueName);\n valuesFn = $parse(match[7]);\n };\n\n $parseOptions.valuesFn = function (scope, controller) {\n return $q.when(valuesFn(scope, controller))\n .then(function (values) {\n if (!angular.isArray(values)) {\n values = [];\n }\n $parseOptions.$values = values.length ? parseValues(values, scope) : [];\n return $parseOptions.$values;\n });\n };\n\n $parseOptions.displayValue = function (modelValue) {\n var scope = {};\n scope[valueName] = modelValue;\n return displayFn(scope);\n };\n\n // Private functions\n\n function parseValues (values, scope) {\n return values.map(function (match, index) {\n var locals = {};\n var label;\n var value;\n locals[valueName] = match;\n label = displayFn(scope, locals);\n value = valueFn(scope, locals);\n return {label: label, value: value, index: index};\n });\n }\n\n $parseOptions.init();\n return $parseOptions;\n\n }\n\n return ParseOptionsFactory;\n\n };\n\n });\n\n// Source: helpers/dimensions.js\nangular.module('mgcrea.ngStrap.helpers.dimensions', [])\n\n .factory('dimensions', function () {\n\n var fn = {};\n\n /**\n * Test the element nodeName\n * @param element\n * @param name\n */\n var nodeName = fn.nodeName = function (element, name) {\n return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase();\n };\n\n /**\n * Returns the element computed style\n * @param element\n * @param prop\n * @param extra\n */\n fn.css = function (element, prop, extra) {\n var value;\n if (element.currentStyle) { // IE\n value = element.currentStyle[prop];\n } else if (window.getComputedStyle) {\n value = window.getComputedStyle(element)[prop];\n } else {\n value = element.style[prop];\n }\n return extra === true ? parseFloat(value) || 0 : value;\n };\n\n /**\n * Provides read-only equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.offset = function (element) {\n var boxRect = element.getBoundingClientRect();\n var docElement = element.ownerDocument;\n return {\n width: boxRect.width || element.offsetWidth,\n height: boxRect.height || element.offsetHeight,\n top: boxRect.top + (window.pageYOffset || docElement.documentElement.scrollTop) - (docElement.documentElement.clientTop || 0),\n left: boxRect.left + (window.pageXOffset || docElement.documentElement.scrollLeft) - (docElement.documentElement.clientLeft || 0)\n };\n };\n\n /**\n * Provides set equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip\n * @url http://api.jquery.com/offset/\n * @param element\n * @param options\n * @param i\n */\n fn.setOffset = function (element, options, i) {\n var curPosition;\n var curLeft;\n var curCSSTop;\n var curTop;\n var curOffset;\n var curCSSLeft;\n var calculatePosition;\n var position = fn.css(element, 'position');\n var curElem = angular.element(element);\n var props = {};\n\n // Set position first, in-case top/left are set even on static elem\n if (position === 'static') {\n element.style.position = 'relative';\n }\n\n curOffset = fn.offset(element);\n curCSSTop = fn.css(element, 'top');\n curCSSLeft = fn.css(element, 'left');\n calculatePosition = (position === 'absolute' || position === 'fixed') &&\n (curCSSTop + curCSSLeft).indexOf('auto') > -1;\n\n // Need to be able to calculate position if either\n // top or left is auto and position is either absolute or fixed\n if (calculatePosition) {\n curPosition = fn.position(element);\n curTop = curPosition.top;\n curLeft = curPosition.left;\n } else {\n curTop = parseFloat(curCSSTop) || 0;\n curLeft = parseFloat(curCSSLeft) || 0;\n }\n\n if (angular.isFunction(options)) {\n options = options.call(element, i, curOffset);\n }\n\n if (options.top !== null) {\n props.top = (options.top - curOffset.top) + curTop;\n }\n if (options.left !== null) {\n props.left = (options.left - curOffset.left) + curLeft;\n }\n\n if ('using' in options) {\n options.using.call(curElem, props);\n } else {\n curElem.css({\n top: props.top + 'px',\n left: props.left + 'px'\n });\n }\n };\n\n /**\n * Provides read-only equivalent of jQuery's position function\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.position = function (element) {\n\n var offsetParentRect = {top: 0, left: 0};\n var offsetParentEl;\n var offset;\n\n // Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is it's only offset parent\n if (fn.css(element, 'position') === 'fixed') {\n\n // We assume that getBoundingClientRect is available when computed position is fixed\n offset = element.getBoundingClientRect();\n\n } else {\n\n // Get *real* offsetParentEl\n offsetParentEl = offsetParentElement(element);\n\n // Get correct offsets\n offset = fn.offset(element);\n if (!nodeName(offsetParentEl, 'html')) {\n offsetParentRect = fn.offset(offsetParentEl);\n }\n\n // Add offsetParent borders\n offsetParentRect.top += fn.css(offsetParentEl, 'borderTopWidth', true);\n offsetParentRect.left += fn.css(offsetParentEl, 'borderLeftWidth', true);\n }\n\n // Subtract parent offsets and element margins\n return {\n width: element.offsetWidth,\n height: element.offsetHeight,\n top: offset.top - offsetParentRect.top - fn.css(element, 'marginTop', true),\n left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true)\n };\n\n };\n\n /**\n * Returns the closest, non-statically positioned offsetParent of a given element\n * @required-by fn.position\n * @param element\n */\n function offsetParentElement (element) {\n var docElement = element.ownerDocument;\n var offsetParent = element.offsetParent || docElement;\n if (nodeName(offsetParent, '#document')) return docElement.documentElement;\n while (offsetParent && !nodeName(offsetParent, 'html') && fn.css(offsetParent, 'position') === 'static') {\n offsetParent = offsetParent.offsetParent;\n }\n return offsetParent || docElement.documentElement;\n }\n\n /**\n * Provides equivalent of jQuery's height function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/height/\n * @param element\n * @param outer\n */\n fn.height = function (element, outer) {\n var value = element.offsetHeight;\n if (outer) {\n value += fn.css(element, 'marginTop', true) + fn.css(element, 'marginBottom', true);\n } else {\n value -= fn.css(element, 'paddingTop', true) + fn.css(element, 'paddingBottom', true) + fn.css(element, 'borderTopWidth', true) + fn.css(element, 'borderBottomWidth', true);\n }\n return value;\n };\n\n /**\n * Provides equivalent of jQuery's width function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/width/\n * @param element\n * @param outer\n */\n fn.width = function (element, outer) {\n var value = element.offsetWidth;\n if (outer) {\n value += fn.css(element, 'marginLeft', true) + fn.css(element, 'marginRight', true);\n } else {\n value -= fn.css(element, 'paddingLeft', true) + fn.css(element, 'paddingRight', true) + fn.css(element, 'borderLeftWidth', true) + fn.css(element, 'borderRightWidth', true);\n }\n return value;\n };\n\n return fn;\n\n });\n\n// Source: helpers/debounce.js\nangular.module('mgcrea.ngStrap.helpers.debounce', [])\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L693\n.factory('debounce', function ($timeout) {\n return function (func, wait, immediate) {\n var timeout = null;\n return function () {\n var context = this;\n var args = arguments;\n var callNow = immediate && !timeout;\n if (timeout) {\n $timeout.cancel(timeout);\n }\n timeout = $timeout(function later () {\n timeout = null;\n if (!immediate) {\n func.apply(context, args);\n }\n }, wait, false);\n if (callNow) {\n func.apply(context, args);\n }\n return timeout;\n };\n };\n})\n\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L661\n.factory('throttle', function ($timeout) {\n return function (func, wait, options) {\n var timeout = null;\n if (!options) options = {};\n return function () {\n var context = this;\n var args = arguments;\n if (!timeout) {\n if (options.leading !== false) {\n func.apply(context, args);\n }\n timeout = $timeout(function later () {\n timeout = null;\n if (options.trailing !== false) {\n func.apply(context, args);\n }\n }, wait, false);\n }\n };\n };\n});\n\n// Source: helpers/date-parser.js\nangular.module('mgcrea.ngStrap.helpers.dateParser', [])\n\n.provider('$dateParser', function ($localeProvider) {\n\n // define a custom ParseDate object to use instead of native Date\n // to avoid date values wrapping when setting date component values\n function ParseDate () {\n this.year = 1970;\n this.month = 0;\n this.day = 1;\n this.hours = 0;\n this.minutes = 0;\n this.seconds = 0;\n this.milliseconds = 0;\n }\n\n ParseDate.prototype.setMilliseconds = function (value) { this.milliseconds = value; };\n ParseDate.prototype.setSeconds = function (value) { this.seconds = value; };\n ParseDate.prototype.setMinutes = function (value) { this.minutes = value; };\n ParseDate.prototype.setHours = function (value) { this.hours = value; };\n ParseDate.prototype.getHours = function () { return this.hours; };\n ParseDate.prototype.setDate = function (value) { this.day = value; };\n ParseDate.prototype.setMonth = function (value) { this.month = value; };\n ParseDate.prototype.setFullYear = function (value) { this.year = value; };\n ParseDate.prototype.fromDate = function (value) {\n this.year = value.getFullYear();\n this.month = value.getMonth();\n this.day = value.getDate();\n this.hours = value.getHours();\n this.minutes = value.getMinutes();\n this.seconds = value.getSeconds();\n this.milliseconds = value.getMilliseconds();\n return this;\n };\n\n ParseDate.prototype.toDate = function () {\n return new Date(this.year, this.month, this.day, this.hours, this.minutes, this.seconds, this.milliseconds);\n };\n\n var proto = ParseDate.prototype;\n\n function noop () {\n }\n\n function isNumeric (n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n\n function indexOfCaseInsensitive (array, value) {\n var len = array.length;\n var str = value.toString().toLowerCase();\n for (var i = 0; i < len; i++) {\n if (array[i].toLowerCase() === str) { return i; }\n }\n return -1; // Return -1 per the \"Array.indexOf()\" method.\n }\n\n var defaults = this.defaults = {\n format: 'shortDate',\n strict: false\n };\n\n this.$get = function ($locale, dateFilter) {\n\n var DateParserFactory = function (config) {\n\n var options = angular.extend({}, defaults, config);\n\n var $dateParser = {};\n\n /* eslint-disable key-spacing, quote-props */\n var regExpMap = {\n 'sss' : '[0-9]{3}',\n 'ss' : '[0-5][0-9]',\n 's' : options.strict ? '[1-5]?[0-9]' : '[0-9]|[0-5][0-9]',\n 'mm' : '[0-5][0-9]',\n 'm' : options.strict ? '[1-5]?[0-9]' : '[0-9]|[0-5][0-9]',\n 'HH' : '[01][0-9]|2[0-3]',\n 'H' : options.strict ? '1?[0-9]|2[0-3]' : '[01]?[0-9]|2[0-3]',\n 'hh' : '[0][1-9]|[1][012]',\n 'h' : options.strict ? '[1-9]|1[012]' : '0?[1-9]|1[012]',\n 'a' : 'AM|PM',\n 'EEEE' : $locale.DATETIME_FORMATS.DAY.join('|'),\n 'EEE' : $locale.DATETIME_FORMATS.SHORTDAY.join('|'),\n 'dd' : '0[1-9]|[12][0-9]|3[01]',\n 'd' : options.strict ? '[1-9]|[1-2][0-9]|3[01]' : '0?[1-9]|[1-2][0-9]|3[01]',\n 'MMMM' : $locale.DATETIME_FORMATS.MONTH.join('|'),\n 'MMM' : $locale.DATETIME_FORMATS.SHORTMONTH.join('|'),\n 'MM' : '0[1-9]|1[012]',\n 'M' : options.strict ? '[1-9]|1[012]' : '0?[1-9]|1[012]',\n 'yyyy' : '[1]{1}[0-9]{3}|[2]{1}[0-9]{3}',\n 'yy' : '[0-9]{2}',\n 'y' : options.strict ? '-?(0|[1-9][0-9]{0,3})' : '-?0*[0-9]{1,4}'\n };\n\n var setFnMap = {\n 'sss' : proto.setMilliseconds,\n 'ss' : proto.setSeconds,\n 's' : proto.setSeconds,\n 'mm' : proto.setMinutes,\n 'm' : proto.setMinutes,\n 'HH' : proto.setHours,\n 'H' : proto.setHours,\n 'hh' : proto.setHours,\n 'h' : proto.setHours,\n 'EEEE' : noop,\n 'EEE' : noop,\n 'dd' : proto.setDate,\n 'd' : proto.setDate,\n 'a' : function (value) { var hours = this.getHours() % 12; return this.setHours(value.match(/pm/i) ? hours + 12 : hours); },\n 'MMMM' : function (value) { return this.setMonth(indexOfCaseInsensitive($locale.DATETIME_FORMATS.MONTH, value)); },\n 'MMM' : function (value) { return this.setMonth(indexOfCaseInsensitive($locale.DATETIME_FORMATS.SHORTMONTH, value)); },\n 'MM' : function (value) { return this.setMonth(1 * value - 1); },\n 'M' : function (value) { return this.setMonth(1 * value - 1); },\n 'yyyy' : proto.setFullYear,\n 'yy' : function (value) { return this.setFullYear(2000 + 1 * value); },\n 'y' : function (value) { return (1 * value <= 50 && value.length === 2) ? this.setFullYear(2000 + 1 * value) : this.setFullYear(1 * value); }\n };\n /* eslint-enable key-spacing, quote-props */\n\n var regex;\n var setMap;\n\n $dateParser.init = function () {\n $dateParser.$format = $locale.DATETIME_FORMATS[options.format] || options.format;\n regex = regExpForFormat($dateParser.$format);\n setMap = setMapForFormat($dateParser.$format);\n };\n\n $dateParser.isValid = function (date) {\n if (angular.isDate(date)) return !isNaN(date.getTime());\n return regex.test(date);\n };\n\n $dateParser.parse = function (value, baseDate, format, timezone) {\n // check for date format special names\n if (format) format = $locale.DATETIME_FORMATS[format] || format;\n if (angular.isDate(value)) value = dateFilter(value, format || $dateParser.$format, timezone);\n var formatRegex = format ? regExpForFormat(format) : regex;\n var formatSetMap = format ? setMapForFormat(format) : setMap;\n var matches = formatRegex.exec(value);\n if (!matches) return false;\n // use custom ParseDate object to set parsed values\n var date = baseDate && !isNaN(baseDate.getTime()) ? new ParseDate().fromDate(baseDate) : new ParseDate().fromDate(new Date(1970, 0, 1, 0));\n for (var i = 0; i < matches.length - 1; i++) {\n if (formatSetMap[i]) formatSetMap[i].call(date, matches[i + 1]);\n }\n // convert back to native Date object\n var newDate = date.toDate();\n\n // check new native Date object for day values overflow\n if (parseInt(date.day, 10) !== newDate.getDate()) {\n return false;\n }\n\n return newDate;\n };\n\n $dateParser.getDateForAttribute = function (key, value) {\n var date;\n\n if (value === 'today') {\n var today = new Date();\n date = new Date(today.getFullYear(), today.getMonth(), today.getDate() + (key === 'maxDate' ? 1 : 0), 0, 0, 0, (key === 'minDate' ? 0 : -1));\n } else if (angular.isString(value) && value.match(/^\".+\"$/)) { // Support {{ dateObj }}\n date = new Date(value.substr(1, value.length - 2));\n } else if (isNumeric(value)) {\n date = new Date(parseInt(value, 10));\n } else if (angular.isString(value) && value.length === 0) { // Reset date\n date = key === 'minDate' ? -Infinity : +Infinity;\n } else {\n date = new Date(value);\n }\n\n return date;\n };\n\n $dateParser.getTimeForAttribute = function (key, value) {\n var time;\n\n if (value === 'now') {\n time = new Date().setFullYear(1970, 0, 1);\n } else if (angular.isString(value) && value.match(/^\".+\"$/)) {\n time = new Date(value.substr(1, value.length - 2)).setFullYear(1970, 0, 1);\n } else if (isNumeric(value)) {\n time = new Date(parseInt(value, 10)).setFullYear(1970, 0, 1);\n } else if (angular.isString(value) && value.length === 0) { // Reset time\n time = key === 'minTime' ? -Infinity : +Infinity;\n } else {\n time = $dateParser.parse(value, new Date(1970, 0, 1, 0));\n }\n\n return time;\n };\n\n /* Handle switch to/from daylight saving.\n * Hours may be non-zero on daylight saving cut-over:\n * > 12 when midnight changeover, but then cannot generate\n * midnight datetime, so jump to 1AM, otherwise reset.\n * @param date (Date) the date to check\n * @return (Date) the corrected date\n *\n * __ copied from jquery ui datepicker __\n */\n $dateParser.daylightSavingAdjust = function (date) {\n if (!date) {\n return null;\n }\n date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);\n return date;\n };\n\n /* Correct the date for timezone offset.\n * @param date (Date) the date to adjust\n * @param timezone (string) the timezone to adjust for\n * @param undo (boolean) to add or subtract timezone offset\n * @return (Date) the corrected date\n */\n $dateParser.timezoneOffsetAdjust = function (date, timezone, undo) {\n if (!date) {\n return null;\n }\n // Right now, only 'UTC' is supported.\n if (timezone && timezone === 'UTC') {\n date = new Date(date.getTime());\n date.setMinutes(date.getMinutes() + (undo ? -1 : 1) * date.getTimezoneOffset());\n }\n return date;\n };\n\n // Private functions\n\n function regExpForFormat (format) {\n // `format` string can contain literal values.\n // These need to be escaped by surrounding with\n // single quotes (e.g. `\"h 'in the morning'\"`).\n // In order to output a single quote, escape it - i.e.,\n // two single quotes in a sequence (e.g. `\"h 'o''clock'\"`).\n\n var re = buildDateAbstractRegex(format);\n return buildDateParseRegex(re);\n }\n\n function buildDateAbstractRegex (format) {\n var escapedFormat = escapeReservedSymbols(format);\n var escapedLiteralFormat = escapedFormat.replace(/''/g, '\\\\\\'');\n var literalRegex = /('(?:\\\\'|.)*?')/;\n var formatParts = escapedLiteralFormat.split(literalRegex);\n var dateElements = Object.keys(regExpMap);\n var dateRegexParts = [];\n\n angular.forEach(formatParts, function (part) {\n if (isFormatStringLiteral(part)) {\n part = trimLiteralEscapeChars(part);\n } else {\n // Abstract replaces to avoid collisions\n for (var i = 0; i < dateElements.length; i++) {\n part = part.split(dateElements[i]).join('${' + i + '}');\n }\n }\n dateRegexParts.push(part);\n });\n\n return dateRegexParts.join('');\n }\n\n function escapeReservedSymbols (text) {\n return text.replace(/\\\\/g, '[\\\\\\\\]')\n .replace(/-/g, '[-]')\n .replace(/\\./g, '[.]')\n .replace(/\\*/g, '[*]')\n .replace(/\\+/g, '[+]')\n .replace(/\\?/g, '[?]')\n .replace(/\\$/g, '[$]')\n .replace(/\\^/g, '[^]')\n .replace(/\\//g, '[/]')\n .replace(/\\\\s/g, '[\\\\s]');\n }\n\n function isFormatStringLiteral (text) {\n return /^'.*'$/.test(text);\n }\n\n function trimLiteralEscapeChars (text) {\n return text.replace(/^'(.*)'$/, '$1');\n }\n\n function buildDateParseRegex (abstractRegex) {\n var dateElements = Object.keys(regExpMap);\n var re = abstractRegex;\n\n // Replace abstracted values\n for (var i = 0; i < dateElements.length; i++) {\n re = re.split('${' + i + '}').join('(' + regExpMap[dateElements[i]] + ')');\n }\n\n return new RegExp('^' + re + '$', ['i']);\n }\n\n function setMapForFormat (format) {\n var re = buildDateAbstractRegex(format);\n return buildDateParseValuesMap(re);\n }\n\n function buildDateParseValuesMap (abstractRegex) {\n var dateElements = Object.keys(regExpMap);\n var valuesRegex = new RegExp('\\\\${(\\\\d+)}', 'g');\n var valuesMatch;\n var keyIndex;\n var valueKey;\n var valueFunction;\n var valuesFunctionMap = [];\n\n /* eslint-disable no-cond-assign */\n while ((valuesMatch = valuesRegex.exec(abstractRegex)) !== null) {\n keyIndex = valuesMatch[1];\n valueKey = dateElements[keyIndex];\n valueFunction = setFnMap[valueKey];\n\n valuesFunctionMap.push(valueFunction);\n }\n\n return valuesFunctionMap;\n }\n\n $dateParser.init();\n return $dateParser;\n\n };\n\n return DateParserFactory;\n\n };\n\n});\n\n// Source: helpers/date-formatter.js\nangular.module('mgcrea.ngStrap.helpers.dateFormatter', [])\n\n .service('$dateFormatter', function ($locale, dateFilter) {\n\n // The unused `lang` arguments are on purpose. The default implementation does not\n // use them and it always uses the locale loaded into the `$locale` service.\n // Custom implementations might use it, thus allowing different directives to\n // have different languages.\n\n this.getDefaultLocale = function () {\n return $locale.id;\n };\n\n // Format is either a data format name, e.g. \"shortTime\" or \"fullDate\", or a date format\n // Return either the corresponding date format or the given date format.\n this.getDatetimeFormat = function (format, lang) {\n return $locale.DATETIME_FORMATS[format] || format;\n };\n\n this.weekdaysShort = function (lang) {\n return $locale.DATETIME_FORMATS.SHORTDAY;\n };\n\n function splitTimeFormat (format) {\n return /(h+)([:\\.])?(m+)([:\\.])?(s*)[ ]?(a?)/i.exec(format).slice(1);\n }\n\n // h:mm a => h\n this.hoursFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[0];\n };\n\n // h:mm a => mm\n this.minutesFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[2];\n };\n\n // h:mm:ss a => ss\n this.secondsFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => :\n this.timeSeparator = function (timeFormat) {\n return splitTimeFormat(timeFormat)[1];\n };\n\n // h:mm:ss a => true, h:mm a => false\n this.showSeconds = function (timeFormat) {\n return !!splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => true, H.mm => false\n this.showAM = function (timeFormat) {\n return !!splitTimeFormat(timeFormat)[5];\n };\n\n this.formatDate = function (date, format, lang, timezone) {\n return dateFilter(date, format, timezone);\n };\n\n });\n\n// Source: helpers/compiler.js\n// NOTICE: This file was forked from the angular-material project (github.com/angular/material)\n// MIT Licensed - Copyright (c) 2014-2015 Google, Inc. http://angularjs.org\n\nangular.module('mgcrea.ngStrap.core', [])\n .service('$bsCompiler', bsCompilerService);\n\nfunction bsCompilerService ($q, $http, $injector, $compile, $controller, $templateCache) {\n\n /*\n * @ngdoc service\n * @name $bsCompiler\n * @module material.core\n * @description\n * The $bsCompiler service is an abstraction of angular's compiler, that allows the developer\n * to easily compile an element with a templateUrl, controller, and locals.\n *\n * @usage\n * \n * $bsCompiler.compile({\n * templateUrl: 'modal.html',\n * controller: 'ModalCtrl',\n * locals: {\n * modal: myModalInstance;\n * }\n * }).then(function(compileData) {\n * compileData.element; // modal.html's template in an element\n * compileData.link(myScope); //attach controller & scope to element\n * });\n * \n */\n\n /*\n * @ngdoc method\n * @name $bsCompiler#compile\n * @description A helper to compile an HTML template/templateUrl with a given controller,\n * locals, and scope.\n * @param {object} options An options object, with the following properties:\n *\n * - `controller` - `{(string=|function()=}` Controller fn that should be associated with\n * newly created scope or the name of a registered controller if passed as a string.\n * - `controllerAs` - `{string=}` A controller alias name. If present the controller will be\n * published to scope under the `controllerAs` name.\n * - `template` - `{string=}` An html template as a string.\n * - `templateUrl` - `{string=}` A path to an html template.\n * - `transformTemplate` - `{function(template)=}` A function which transforms the template after\n * it is loaded. It will be given the template string as a parameter, and should\n * return a a new string representing the transformed template.\n * - `resolve` - `{Object.=}` - An optional map of dependencies which should\n * be injected into the controller. If any of these dependencies are promises, the compiler\n * will wait for them all to be resolved, or if one is rejected before the controller is\n * instantiated `compile()` will fail..\n * * `key` - `{string}`: a name of a dependency to be injected into the controller.\n * * `factory` - `{string|function}`: If `string` then it is an alias for a service.\n * Otherwise if function, then it is injected and the return value is treated as the\n * dependency. If the result is a promise, it is resolved before its value is\n * injected into the controller.\n *\n * @returns {object=} promise A promise, which will be resolved with a `compileData` object.\n * `compileData` has the following properties:\n *\n * - `element` - `{element}`: an uncompiled element matching the provided template.\n * - `link` - `{function(scope)}`: A link function, which, when called, will compile\n * the element and instantiate the provided controller (if given).\n * - `locals` - `{object}`: The locals which will be passed into the controller once `link` is\n * called. If `bindToController` is true, they will be coppied to the ctrl instead\n * - `bindToController` - `bool`: bind the locals to the controller, instead of passing them in.\n */\n this.compile = function (options) {\n\n if (options.template && /\\.html$/.test(options.template)) {\n console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.');\n options.templateUrl = options.template;\n options.template = '';\n }\n\n var templateUrl = options.templateUrl;\n var template = options.template || '';\n var controller = options.controller;\n var controllerAs = options.controllerAs;\n var resolve = options.resolve || {};\n var locals = options.locals || {};\n var transformTemplate = options.transformTemplate || angular.identity;\n var bindToController = options.bindToController;\n\n // Take resolve values and invoke them.\n // Resolves can either be a string (value: 'MyRegisteredAngularConst'),\n // or an invokable 'factory' of sorts: (value: function ValueGetter($dependency) {})\n angular.forEach(resolve, function (value, key) {\n if (angular.isString(value)) {\n resolve[key] = $injector.get(value);\n } else {\n resolve[key] = $injector.invoke(value);\n }\n });\n // Add the locals, which are just straight values to inject\n // eg locals: { three: 3 }, will inject three into the controller\n angular.extend(resolve, locals);\n\n if (template) {\n resolve.$template = $q.when(template);\n } else if (templateUrl) {\n resolve.$template = fetchTemplate(templateUrl);\n } else {\n throw new Error('Missing `template` / `templateUrl` option.');\n }\n\n if (options.titleTemplate) {\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.titleTemplate)])\n .then(function (templates) {\n var templateEl = angular.element(templates[0]);\n findElement('[ng-bind=\"title\"]', templateEl[0])\n .removeAttr('ng-bind')\n .html(templates[1]);\n return templateEl[0].outerHTML;\n });\n }\n\n if (options.contentTemplate) {\n // TODO(mgcrea): deprecate?\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.contentTemplate)])\n .then(function (templates) {\n var templateEl = angular.element(templates[0]);\n var contentEl = findElement('[ng-bind=\"content\"]', templateEl[0])\n .removeAttr('ng-bind')\n .html(templates[1]);\n // Drop the default footer as you probably don't want it if you use a custom contentTemplate\n if (!options.templateUrl) contentEl.next().remove();\n return templateEl[0].outerHTML;\n });\n }\n\n // Wait for all the resolves to finish if they are promises\n return $q.all(resolve).then(function (locals) {\n\n var template = transformTemplate(locals.$template);\n if (options.html) {\n template = template.replace(/ng-bind=\"/ig, 'ng-bind-html=\"');\n }\n // var element = options.element || angular.element('
').html(template.trim()).contents();\n var element = angular.element('
').html(template.trim()).contents();\n var linkFn = $compile(element);\n\n // Return a linking function that can be used later when the element is ready\n return {\n locals: locals,\n element: element,\n link: function link (scope) {\n locals.$scope = scope;\n\n // Instantiate controller if it exists, because we have scope\n if (controller) {\n var invokeCtrl = $controller(controller, locals, true);\n if (bindToController) {\n angular.extend(invokeCtrl.instance, locals);\n }\n // Support angular@~1.2 invokeCtrl\n var ctrl = angular.isObject(invokeCtrl) ? invokeCtrl : invokeCtrl();\n // See angular-route source for this logic\n element.data('$ngControllerController', ctrl);\n element.children().data('$ngControllerController', ctrl);\n\n if (controllerAs) {\n scope[controllerAs] = ctrl;\n }\n }\n\n return linkFn.apply(null, arguments);\n }\n };\n });\n\n };\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate (template) {\n if (fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache})\n .then(function (res) {\n return res.data;\n }));\n }\n\n}\n\n// Source: dropdown/dropdown.js\nangular.module('mgcrea.ngStrap.dropdown', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$dropdown', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'dropdown',\n prefixEvent: 'dropdown',\n placement: 'bottom-left',\n templateUrl: 'dropdown/dropdown.tpl.html',\n trigger: 'click',\n container: false,\n keyboard: true,\n html: false,\n delay: 0\n };\n\n this.$get = function ($window, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var matchesSelector = Element.prototype.matchesSelector || Element.prototype.webkitMatchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector;\n\n function DropdownFactory (element, config) {\n\n var $dropdown = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n /* var scope = */$dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n $dropdown = $tooltip(element, options);\n var parentEl = element.parent();\n\n // Protected methods\n\n $dropdown.$onKeyDown = function (evt) {\n if (!/(38|40)/.test(evt.keyCode)) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Retrieve focused index\n var items = angular.element($dropdown.$element[0].querySelectorAll('li:not(.divider) a'));\n if (!items.length) return;\n var index;\n angular.forEach(items, function (el, i) {\n if (matchesSelector && matchesSelector.call(el, ':focus')) index = i;\n });\n\n // Navigate with keyboard\n if (evt.keyCode === 38 && index > 0) index--;\n else if (evt.keyCode === 40 && index < items.length - 1) index++;\n else if (angular.isUndefined(index)) index = 0;\n items.eq(index)[0].focus();\n\n };\n\n // Overrides\n\n var show = $dropdown.show;\n $dropdown.show = function () {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n if (options.keyboard && $dropdown.$element) $dropdown.$element.on('keydown', $dropdown.$onKeyDown);\n bodyEl.on('click', onBodyClick);\n }, 0, false);\n if (parentEl.hasClass('dropdown')) parentEl.addClass('open');\n };\n\n var hide = $dropdown.hide;\n $dropdown.hide = function () {\n if (!$dropdown.$isShown) return;\n if (options.keyboard && $dropdown.$element) $dropdown.$element.off('keydown', $dropdown.$onKeyDown);\n bodyEl.off('click', onBodyClick);\n if (parentEl.hasClass('dropdown')) parentEl.removeClass('open');\n hide();\n };\n\n var destroy = $dropdown.destroy;\n $dropdown.destroy = function () {\n bodyEl.off('click', onBodyClick);\n destroy();\n };\n\n // Private functions\n\n function onBodyClick (evt) {\n if (evt.target === element[0]) return;\n return evt.target !== element[0] && $dropdown.hide();\n }\n\n return $dropdown;\n\n }\n\n return DropdownFactory;\n\n };\n\n })\n\n .directive('bsDropdown', function ($window, $sce, $dropdown) {\n\n return {\n restrict: 'EAC',\n scope: true,\n compile: function (tElement, tAttrs) {\n\n // Support for inlined template (next sibling)\n // It must be fetched before compilation\n if (!tAttrs.bsDropdown) {\n var nextSibling = tElement[0].nextSibling;\n while (nextSibling && nextSibling.nodeType !== 1) {\n nextSibling = nextSibling.nextSibling;\n }\n if (nextSibling && nextSibling.className.split(' ').indexOf('dropdown-menu') >= 0) {\n tAttrs.template = nextSibling.outerHTML;\n tAttrs.templateUrl = undefined;\n nextSibling.parentNode.removeChild(nextSibling);\n }\n }\n\n return function postLink (scope, element, attr) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id', 'autoClose'], function (key) {\n if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as an object\n if (attr.bsDropdown) {\n scope.$watch(attr.bsDropdown, function (newValue, oldValue) {\n scope.content = newValue;\n }, true);\n }\n\n // Initialize dropdown\n var dropdown = $dropdown(element, options);\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!dropdown || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i);\n if (newValue === true) {\n dropdown.show();\n } else {\n dropdown.hide();\n }\n });\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (dropdown) dropdown.destroy();\n options = null;\n dropdown = null;\n });\n\n };\n }\n };\n\n });\n\n// Source: button/button.js\nangular.module('mgcrea.ngStrap.button', [])\n\n .provider('$button', function () {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n toggleEvent: 'click'\n };\n\n this.$get = function () {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsCheckboxGroup', function () {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink (element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"checkbox\"]');\n angular.forEach(children, function (child) {\n var childEl = angular.element(child);\n childEl.attr('bs-checkbox', '');\n childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value'));\n });\n }\n\n };\n\n })\n\n .directive('bsCheckbox', function ($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support label > input[type=\"checkbox\"]\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true;\n if (constantValueRegExp.test(attr.trueValue)) {\n trueValue = scope.$eval(attr.trueValue);\n }\n var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false;\n if (constantValueRegExp.test(attr.falseValue)) {\n falseValue = scope.$eval(attr.falseValue);\n }\n\n // Parse exotic values\n var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean';\n if (hasExoticValues) {\n controller.$parsers.push(function (viewValue) {\n // console.warn('$parser', element.attr('ng-model'), 'viewValue', viewValue);\n return viewValue ? trueValue : falseValue;\n });\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n return angular.equals(modelValue, trueValue);\n });\n }\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = !!controller.$viewValue;\n $$rAF(function () {\n if (isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function () {\n scope.$apply(function () {\n // console.warn('!click', element.attr('ng-model'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n if (!isInput) {\n controller.$setViewValue(!activeElement.hasClass('active'));\n }\n controller.$render();\n });\n });\n\n }\n\n };\n\n })\n\n .directive('bsRadioGroup', function () {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink (element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"radio\"]');\n angular.forEach(children, function (child) {\n angular.element(child).attr('bs-radio', '');\n angular.element(child).attr('ng-model', attr.ngModel);\n });\n }\n\n };\n\n })\n\n .directive('bsRadio', function ($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support `label > input[type=\"radio\"]` markup\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var value;\n attr.$observe('value', function (v) {\n if (typeof v !== 'boolean' && constantValueRegExp.test(v)) {\n value = scope.$eval(v);\n } else {\n value = v;\n }\n controller.$render();\n });\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('value'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$viewValue, value);\n $$rAF(function () {\n if (isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function () {\n scope.$apply(function () {\n // console.warn('!click', element.attr('value'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n controller.$setViewValue(value);\n controller.$render();\n });\n });\n\n }\n\n };\n\n });\n\n// Source: datepicker/datepicker.js\nangular.module('mgcrea.ngStrap.datepicker', [\n 'mgcrea.ngStrap.helpers.dateParser',\n 'mgcrea.ngStrap.helpers.dateFormatter',\n 'mgcrea.ngStrap.tooltip'])\n\n .provider('$datepicker', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n // Uncommenting the following line will break backwards compatability\n // prefixEvent: 'datepicker',\n prefixClass: 'datepicker',\n placement: 'bottom-left',\n templateUrl: 'datepicker/datepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: false,\n dateType: 'date',\n dateFormat: 'shortDate',\n timezone: null,\n modelDateFormat: null,\n dayFormat: 'dd',\n monthFormat: 'MMM',\n yearFormat: 'yyyy',\n monthTitleFormat: 'MMMM yyyy',\n yearTitleFormat: 'yyyy',\n strictFormat: false,\n autoclose: false,\n minDate: -Infinity,\n maxDate: +Infinity,\n startView: 0,\n minView: 0,\n startWeek: 0,\n daysOfWeekDisabled: '',\n hasToday: false,\n hasClear: false,\n iconLeft: 'glyphicon glyphicon-chevron-left',\n iconRight: 'glyphicon glyphicon-chevron-right'\n };\n\n this.$get = function ($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale();\n\n function DatepickerFactory (element, controller, config) {\n\n var $datepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $datepicker.$options;\n var scope = $datepicker.$scope;\n if (options.startView) options.startView -= options.minView;\n\n // View vars\n\n var pickerViews = datepickerViews($datepicker);\n $datepicker.$views = pickerViews.views;\n var viewDate = pickerViews.viewDate;\n scope.$mode = options.startView;\n scope.$iconLeft = options.iconLeft;\n scope.$iconRight = options.iconRight;\n scope.$hasToday = options.hasToday;\n scope.$hasClear = options.hasClear;\n var $picker = $datepicker.$views[scope.$mode];\n\n // Scope methods\n\n scope.$select = function (date, disabled) {\n if (disabled) return;\n $datepicker.select(date);\n };\n scope.$selectPane = function (value) {\n $datepicker.$selectPane(value);\n };\n scope.$toggleMode = function () {\n $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length);\n };\n scope.$setToday = function () {\n if (options.autoclose) {\n $datepicker.setMode(0);\n $datepicker.select(new Date());\n } else {\n $datepicker.select(new Date(), true);\n }\n };\n scope.$clear = function () {\n if (options.autoclose) {\n $datepicker.setMode(0);\n $datepicker.select(null);\n } else {\n $datepicker.select(null, true);\n }\n };\n\n // Public methods\n\n $datepicker.update = function (date) {\n // console.warn('$datepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $datepicker.$date = date;\n $picker.update.call($picker, date);\n }\n // Build only if pristine\n $datepicker.$build(true);\n };\n\n $datepicker.updateDisabledDates = function (dateRanges) {\n options.disabledDateRanges = dateRanges;\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], $datepicker.$setDisabledEl);\n }\n };\n\n $datepicker.select = function (date, keep) {\n // console.warn('$datepicker.select', date, scope.$mode);\n if (angular.isDate(date)) {\n if (!angular.isDate(controller.$dateValue) || isNaN(controller.$dateValue.getTime())) {\n controller.$dateValue = new Date(date);\n }\n } else {\n controller.$dateValue = null;\n }\n if (!scope.$mode || keep) {\n controller.$setViewValue(angular.copy(date));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function () { $datepicker.hide(true); });\n }\n } else {\n angular.extend(viewDate, {year: date.getFullYear(), month: date.getMonth(), date: date.getDate()});\n $datepicker.setMode(scope.$mode - 1);\n $datepicker.$build();\n }\n };\n\n $datepicker.setMode = function (mode) {\n // console.warn('$datepicker.setMode', mode);\n scope.$mode = mode;\n $picker = $datepicker.$views[scope.$mode];\n $datepicker.$build();\n };\n\n // Protected methods\n\n $datepicker.$build = function (pristine) {\n // console.warn('$datepicker.$build() viewDate=%o', viewDate);\n if (pristine === true && $picker.built) return;\n if (pristine === false && !$picker.built) return;\n $picker.build.call($picker);\n };\n\n $datepicker.$updateSelected = function () {\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], updateSelected);\n }\n };\n\n $datepicker.$isSelected = function (date) {\n return $picker.isSelected(date);\n };\n\n $datepicker.$setDisabledEl = function (el) {\n el.disabled = $picker.isDisabled(el.date);\n };\n\n $datepicker.$selectPane = function (value) {\n var steps = $picker.steps;\n // set targetDate to first day of month to avoid problems with\n // date values rollover. This assumes the viewDate does not\n // depend on the day of the month\n var targetDate = new Date(Date.UTC(viewDate.year + ((steps.year || 0) * value), viewDate.month + ((steps.month || 0) * value), 1));\n angular.extend(viewDate, {year: targetDate.getUTCFullYear(), month: targetDate.getUTCMonth(), date: targetDate.getUTCDate()});\n $datepicker.$build();\n };\n\n $datepicker.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $datepicker.$onKeyDown = function (evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n if (evt.keyCode === 13) {\n if (!scope.$mode) {\n $datepicker.hide(true);\n } else {\n scope.$apply(function () { $datepicker.setMode(scope.$mode - 1); });\n }\n return;\n }\n\n // Navigate with keyboard\n $picker.onKeyDown(evt);\n parentScope.$digest();\n };\n\n // Private\n\n function updateSelected (el) {\n el.selected = $datepicker.$isSelected(el.date);\n }\n\n function focusElement () {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $datepicker.init;\n $datepicker.init = function () {\n if (isNative && options.useNative) {\n element.prop('type', 'date');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $datepicker.destroy;\n $datepicker.destroy = function () {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $datepicker.show;\n $datepicker.show = function () {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n // if $datepicker is no longer showing, don't setup events\n if (!$datepicker.$isShown) return;\n $datepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $datepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $datepicker.hide;\n $datepicker.hide = function (blur) {\n if (!$datepicker.$isShown) return;\n $datepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $datepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $datepicker;\n\n }\n\n DatepickerFactory.defaults = defaults;\n return DatepickerFactory;\n\n };\n\n })\n\n .directive('bsDatepicker', function ($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) {\n\n // var defaults = $datepicker.defaults;\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent', 'hasToday', 'hasClear'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'hasToday', 'hasClear'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Initialize datepicker\n var datepicker = $datepicker(element, controller, options);\n options = datepicker.$options;\n // Set expected iOS format\n if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd';\n\n var lang = options.lang;\n\n var formatDate = function (date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!datepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i);\n if (newValue === true) {\n datepicker.show();\n } else {\n datepicker.hide();\n }\n });\n }\n\n // Observe attributes for changes\n angular.forEach(['minDate', 'maxDate'], function (key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n if (angular.isDefined(attr[key])) {\n attr.$observe(key, function (newValue) {\n // console.warn('attr.$observe(%s)=%o', key, newValue);\n datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue);\n // Build only if dirty\n if (!isNaN(datepicker.$options[key])) datepicker.$build(false);\n validateAgainstMinMaxDate(controller.$dateValue);\n });\n }\n });\n\n // Observe date format\n if (angular.isDefined(attr.dateFormat)) {\n attr.$observe('dateFormat', function (newValue) {\n datepicker.$options.dateFormat = newValue;\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n datepicker.update(controller.$dateValue);\n }, true);\n\n // Normalize undefined/null/empty array,\n // so that we don't treat changing from undefined->null as a change.\n function normalizeDateRanges (ranges) {\n if (!ranges || !ranges.length) return null;\n return ranges;\n }\n\n if (angular.isDefined(attr.disabledDates)) {\n scope.$watch(attr.disabledDates, function (disabledRanges, previousValue) {\n disabledRanges = normalizeDateRanges(disabledRanges);\n previousValue = normalizeDateRanges(previousValue);\n\n if (disabledRanges) {\n datepicker.updateDisabledDates(disabledRanges);\n }\n });\n }\n\n function validateAgainstMinMaxDate (parsedDate) {\n if (!angular.isDate(parsedDate)) return;\n var isMinValid = isNaN(datepicker.$options.minDate) || parsedDate.getTime() >= datepicker.$options.minDate;\n var isMaxValid = isNaN(datepicker.$options.maxDate) || parsedDate.getTime() <= datepicker.$options.maxDate;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (isValid) controller.$dateValue = parsedDate;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function (viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n controller.$setValidity('date', true);\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n return null;\n }\n var parsedDate = dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedDate || isNaN(parsedDate.getTime())) {\n controller.$setValidity('date', false);\n // return undefined, causes ngModelController to\n // invalidate model value\n return;\n }\n validateAgainstMinMaxDate(parsedDate);\n\n if (options.dateType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true);\n return formatDate(date, options.modelDateFormat || options.dateFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.dateType === 'number') {\n return date.getTime();\n } else if (options.dateType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.dateType === 'iso') {\n return date.toISOString();\n }\n return new Date(date);\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.dateType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelDateFormat);\n } else if (options.dateType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if (isNaN(date.getTime())) {\n // var today = new Date();\n // date = new Date(today.getFullYear(), today.getMonth(), today.getDate(), 0, 0, 0, 0);\n // }\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getDateFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function () {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getDateFormattedString());\n };\n\n function getDateFormattedString () {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.dateFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (datepicker) datepicker.destroy();\n options = null;\n datepicker = null;\n });\n\n }\n };\n\n })\n\n .provider('datepickerViews', function () {\n\n // var defaults = this.defaults = {\n // dayFormat: 'dd',\n // daySplit: 7\n // };\n\n // Split array into smaller arrays\n function split (arr, size) {\n var arrays = [];\n while (arr.length > 0) {\n arrays.push(arr.splice(0, size));\n }\n return arrays;\n }\n\n // Modulus operator\n function mod (n, m) {\n return ((n % m) + m) % m;\n }\n\n this.$get = function ($dateFormatter, $dateParser, $sce) {\n\n return function (picker) {\n\n var scope = picker.$scope;\n var options = picker.$options;\n\n var lang = options.lang;\n var formatDate = function (date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n var weekDaysMin = $dateFormatter.weekdaysShort(lang);\n var weekDaysLabels = weekDaysMin.slice(options.startWeek).concat(weekDaysMin.slice(0, options.startWeek));\n var weekDaysLabelsHtml = $sce.trustAsHtml('' + weekDaysLabels.join('') + '');\n\n var startDate = picker.$date || (options.startDate ? dateParser.getDateForAttribute('startDate', options.startDate) : new Date());\n var viewDate = {year: startDate.getFullYear(), month: startDate.getMonth(), date: startDate.getDate()};\n\n var views = [{\n format: options.dayFormat,\n split: 7,\n steps: {month: 1},\n update: function (date, force) {\n if (!this.built || force || date.getFullYear() !== viewDate.year || date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getDate() !== viewDate.date || date.getDate() === 1) {\n // chaging picker current month will cause viewDate.date to be set to first day of the month,\n // in $datepicker.$selectPane, so picker would not update selected day display if\n // user picks first day of the new month.\n // As a workaround, we are always forcing update when picked date is first day of month.\n viewDate.date = picker.$date.getDate();\n picker.$updateSelected();\n }\n },\n build: function () {\n var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1);\n var firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset();\n var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5);\n var firstDateOffset = firstDate.getTimezoneOffset();\n var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString();\n // Handle daylight time switch\n if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 60e3);\n var days = [];\n var day;\n for (var i = 0; i < 42; i++) { // < 7 * 6\n day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i));\n days.push({date: day, isToday: day.toDateString() === today, label: formatDate(day, this.format), selected: picker.$date && this.isSelected(day), muted: day.getMonth() !== viewDate.month, disabled: this.isDisabled(day)});\n }\n scope.title = formatDate(firstDayOfMonth, options.monthTitleFormat);\n scope.showLabels = true;\n scope.labels = weekDaysLabelsHtml;\n scope.rows = split(days, this.split);\n scope.isTodayDisabled = this.isDisabled(new Date());\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth() && date.getDate() === picker.$date.getDate();\n },\n isDisabled: function (date) {\n var time = date.getTime();\n\n // Disabled because of min/max date.\n if (time < options.minDate || time > options.maxDate) return true;\n\n // Disabled due to being a disabled day of the week\n if (options.daysOfWeekDisabled.indexOf(date.getDay()) !== -1) return true;\n\n // Disabled because of disabled date range.\n if (options.disabledDateRanges) {\n for (var i = 0; i < options.disabledDateRanges.length; i++) {\n if (time >= options.disabledDateRanges[i].start && time <= options.disabledDateRanges[i].end) {\n return true;\n }\n }\n }\n\n return false;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualTime = picker.$date.getTime();\n var newDate;\n\n if (evt.keyCode === 37) newDate = new Date(actualTime - 1 * 864e5);\n else if (evt.keyCode === 38) newDate = new Date(actualTime - 7 * 864e5);\n else if (evt.keyCode === 39) newDate = new Date(actualTime + 1 * 864e5);\n else if (evt.keyCode === 40) newDate = new Date(actualTime + 7 * 864e5);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'month',\n format: options.monthFormat,\n split: 4,\n steps: {year: 1},\n update: function (date, force) {\n if (!this.built || date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function () {\n // var firstMonth = new Date(viewDate.year, 0, 1);\n var months = [];\n var month;\n for (var i = 0; i < 12; i++) {\n month = new Date(viewDate.year, i, 1);\n months.push({date: month, label: formatDate(month, this.format), selected: picker.$isSelected(month), disabled: this.isDisabled(month)});\n }\n scope.title = formatDate(month, options.yearTitleFormat);\n scope.showLabels = false;\n scope.rows = split(months, this.split);\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth();\n },\n isDisabled: function (date) {\n var lastDate = +new Date(date.getFullYear(), date.getMonth() + 1, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualMonth = picker.$date.getMonth();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setMonth(actualMonth - 1);\n else if (evt.keyCode === 38) newDate.setMonth(actualMonth - 4);\n else if (evt.keyCode === 39) newDate.setMonth(actualMonth + 1);\n else if (evt.keyCode === 40) newDate.setMonth(actualMonth + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'year',\n format: options.yearFormat,\n split: 4,\n steps: {year: 12},\n update: function (date, force) {\n if (!this.built || force || parseInt(date.getFullYear() / 20, 10) !== parseInt(viewDate.year / 20, 10)) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function () {\n var firstYear = viewDate.year - viewDate.year % (this.split * 3);\n var years = [];\n var year;\n for (var i = 0; i < 12; i++) {\n year = new Date(firstYear + i, 0, 1);\n years.push({date: year, label: formatDate(year, this.format), selected: picker.$isSelected(year), disabled: this.isDisabled(year)});\n }\n scope.title = years[0].label + '-' + years[years.length - 1].label;\n scope.showLabels = false;\n scope.rows = split(years, this.split);\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear();\n },\n isDisabled: function (date) {\n var lastDate = +new Date(date.getFullYear() + 1, 0, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualYear = picker.$date.getFullYear();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setYear(actualYear - 1);\n else if (evt.keyCode === 38) newDate.setYear(actualYear - 4);\n else if (evt.keyCode === 39) newDate.setYear(actualYear + 1);\n else if (evt.keyCode === 40) newDate.setYear(actualYear + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }];\n\n return {\n views: options.minView ? Array.prototype.slice.call(views, options.minView) : views,\n viewDate: viewDate\n };\n\n };\n\n };\n\n });\n\n// Source: collapse/collapse.js\nangular.module('mgcrea.ngStrap.collapse', [])\n\n .provider('$collapse', function () {\n\n var defaults = this.defaults = {\n animation: 'am-collapse',\n disallowToggle: false,\n activeClass: 'in',\n startCollapsed: false,\n allowMultiple: false\n };\n\n var controller = this.controller = function ($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'disallowToggle', 'activeClass', 'startCollapsed', 'allowMultiple'], function (key) {\n if (angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['disallowToggle', 'startCollapsed', 'allowMultiple'], function (key) {\n if (angular.isDefined($attrs[key]) && falseValueRegExp.test($attrs[key])) {\n self.$options[key] = false;\n }\n });\n\n self.$toggles = [];\n self.$targets = [];\n\n self.$viewChangeListeners = [];\n\n self.$registerToggle = function (element) {\n self.$toggles.push(element);\n };\n self.$registerTarget = function (element) {\n self.$targets.push(element);\n };\n\n self.$unregisterToggle = function (element) {\n var index = self.$toggles.indexOf(element);\n // remove toggle from $toggles array\n self.$toggles.splice(index, 1);\n };\n self.$unregisterTarget = function (element) {\n var index = self.$targets.indexOf(element);\n\n // remove element from $targets array\n self.$targets.splice(index, 1);\n\n if (self.$options.allowMultiple) {\n // remove target index from $active array values\n deactivateItem(element);\n }\n\n // fix active item indexes\n fixActiveItemIndexes(index);\n\n self.$viewChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n // use array to store all the currently open panels\n self.$targets.$active = !self.$options.startCollapsed ? [0] : [];\n self.$setActive = $scope.$setActive = function (value) {\n if (angular.isArray(value)) {\n self.$targets.$active = value;\n } else if (!self.$options.disallowToggle && isActive(value)) {\n deactivateItem(value);\n } else {\n activateItem(value);\n }\n\n self.$viewChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n self.$activeIndexes = function () {\n if (self.$options.allowMultiple) {\n return self.$targets.$active;\n }\n return self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1;\n };\n\n function fixActiveItemIndexes (index) {\n // item with index was removed, so we\n // need to adjust other items index values\n var activeIndexes = self.$targets.$active;\n for (var i = 0; i < activeIndexes.length; i++) {\n if (index < activeIndexes[i]) {\n activeIndexes[i] = activeIndexes[i] - 1;\n }\n\n // the last item is active, so we need to\n // adjust its index\n if (activeIndexes[i] === self.$targets.length) {\n activeIndexes[i] = self.$targets.length - 1;\n }\n }\n }\n\n function isActive (value) {\n var activeItems = self.$targets.$active;\n return activeItems.indexOf(value) !== -1;\n }\n\n function deactivateItem (value) {\n var index = self.$targets.$active.indexOf(value);\n if (index !== -1) {\n self.$targets.$active.splice(index, 1);\n }\n }\n\n function activateItem (value) {\n if (!self.$options.allowMultiple) {\n // remove current selected item\n self.$targets.$active.splice(0, 1);\n }\n\n if (self.$targets.$active.indexOf(value) === -1) {\n self.$targets.$active.push(value);\n }\n }\n\n };\n\n this.$get = function () {\n var $collapse = {};\n $collapse.defaults = defaults;\n $collapse.controller = controller;\n return $collapse;\n };\n\n })\n\n .directive('bsCollapse', function ($window, $animate, $collapse) {\n\n return {\n require: ['?ngModel', 'bsCollapse'],\n controller: ['$scope', '$element', '$attrs', $collapse.controller],\n link: function postLink (scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n if (ngModelCtrl) {\n\n // Update the modelValue following\n bsCollapseCtrl.$viewChangeListeners.push(function () {\n ngModelCtrl.$setViewValue(bsCollapseCtrl.$activeIndexes());\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n if (angular.isArray(modelValue)) {\n // model value is an array, so just replace\n // the active items directly\n bsCollapseCtrl.$setActive(modelValue);\n } else {\n var activeIndexes = bsCollapseCtrl.$activeIndexes();\n\n if (angular.isArray(activeIndexes)) {\n // we have an array of selected indexes\n if (activeIndexes.indexOf(modelValue * 1) === -1) {\n // item with modelValue index is not active\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n } else if (activeIndexes !== modelValue * 1) {\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n return modelValue;\n });\n\n }\n\n }\n };\n\n })\n\n .directive('bsCollapseToggle', function () {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base attr\n element.attr('data-toggle', 'collapse');\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerToggle(element);\n\n // remove toggle from collapse controller when toggle is destroyed\n scope.$on('$destroy', function () {\n bsCollapseCtrl.$unregisterToggle(element);\n });\n\n element.on('click', function () {\n if (!attrs.disabled) {\n var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element);\n bsCollapseCtrl.$setActive(index * 1);\n scope.$apply();\n }\n });\n\n }\n };\n\n })\n\n .directive('bsCollapseTarget', function ($animate) {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n // scope: true,\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base class\n element.addClass('collapse');\n\n // Add animation class\n if (bsCollapseCtrl.$options.animation) {\n element.addClass(bsCollapseCtrl.$options.animation);\n }\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerTarget(element);\n\n // remove pane target from collapse controller when target is destroyed\n scope.$on('$destroy', function () {\n bsCollapseCtrl.$unregisterTarget(element);\n });\n\n function render () {\n var index = bsCollapseCtrl.$targets.indexOf(element);\n var active = bsCollapseCtrl.$activeIndexes();\n var action = 'removeClass';\n if (angular.isArray(active)) {\n if (active.indexOf(index) !== -1) {\n action = 'addClass';\n }\n } else if (index === active) {\n action = 'addClass';\n }\n\n $animate[action](element, bsCollapseCtrl.$options.activeClass);\n }\n\n bsCollapseCtrl.$viewChangeListeners.push(function () {\n render();\n });\n render();\n\n }\n };\n\n });\n\n// Source: aside/aside.js\nangular.module('mgcrea.ngStrap.aside', ['mgcrea.ngStrap.modal'])\n\n .provider('$aside', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade-and-slide-right',\n prefixClass: 'aside',\n prefixEvent: 'aside',\n placement: 'right',\n templateUrl: 'aside/aside.tpl.html',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function ($modal) {\n\n function AsideFactory (config) {\n\n var $aside = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $aside = $modal(options);\n\n return $aside;\n\n }\n\n return AsideFactory;\n\n };\n\n })\n\n .directive('bsAside', function ($window, $sce, $aside) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsAside) {\n scope.$watch(attr.bsAside, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize aside\n var aside = $aside(options);\n\n // Trigger\n element.on(attr.trigger || 'click', aside.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (aside) aside.destroy();\n options = null;\n aside = null;\n });\n\n }\n };\n\n });\n\n// Source: alert/alert.js\n// @BUG: following snippet won't compile correctly\n\nangular.module('mgcrea.ngStrap.alert', ['mgcrea.ngStrap.modal'])\n\n .provider('$alert', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'alert',\n prefixEvent: 'alert',\n placement: null,\n templateUrl: 'alert/alert.tpl.html',\n container: false,\n element: null,\n backdrop: false,\n keyboard: true,\n show: true,\n // Specific options\n duration: false,\n type: false,\n dismissable: true\n };\n\n this.$get = function ($modal, $timeout) {\n\n function AlertFactory (config) {\n\n var $alert = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $alert = $modal(options);\n\n // Support scope as string options [/*title, content, */ type, dismissable]\n $alert.$scope.dismissable = !!options.dismissable;\n if (options.type) {\n $alert.$scope.type = options.type;\n }\n\n // Support auto-close duration\n var show = $alert.show;\n if (options.duration) {\n $alert.show = function () {\n show();\n $timeout(function () {\n $alert.hide();\n }, options.duration * 1000);\n };\n }\n\n return $alert;\n\n }\n\n return AlertFactory;\n\n };\n\n })\n\n .directive('bsAlert', function ($window, $sce, $alert) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'keyboard', 'html', 'container', 'animation', 'duration', 'dismissable'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['keyboard', 'html', 'container', 'dismissable'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')) {\n scope.title = '';\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content', 'type'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsAlert) {\n scope.$watch(attr.bsAlert, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize alert\n var alert = $alert(options);\n\n // Trigger\n element.on(attr.trigger || 'click', alert.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (alert) alert.destroy();\n options = null;\n alert = null;\n });\n\n }\n };\n\n });\n\n// Source: affix/affix.js\nangular.module('mgcrea.ngStrap.affix', ['mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce'])\n\n .provider('$affix', function () {\n\n var defaults = this.defaults = {\n offsetTop: 'auto',\n inlineStyles: true,\n setWidth: true\n };\n\n this.$get = function ($window, debounce, dimensions) {\n\n var bodyEl = angular.element($window.document.body);\n var windowEl = angular.element($window);\n\n function AffixFactory (element, config) {\n\n var $affix = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var targetEl = options.target;\n\n // Initial private vars\n var reset = 'affix affix-top affix-bottom';\n var setWidth = false;\n var initialAffixTop = 0;\n var initialOffsetTop = 0;\n var offsetTop = 0;\n var offsetBottom = 0;\n var affixed = null;\n var unpin = null;\n\n var parent = element.parent();\n // Options: custom parent\n if (options.offsetParent) {\n if (options.offsetParent.match(/^\\d+$/)) {\n for (var i = 0; i < (options.offsetParent * 1) - 1; i++) {\n parent = parent.parent();\n }\n } else {\n parent = angular.element(options.offsetParent);\n }\n }\n\n $affix.init = function () {\n\n this.$parseOffsets();\n initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop;\n setWidth = options.setWidth && !element[0].style.width;\n\n // Bind events\n targetEl.on('scroll', this.checkPosition);\n targetEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', this.$debouncedOnResize);\n\n // Both of these checkPosition() calls are necessary for the case where\n // the user hits refresh after scrolling to the bottom of the page.\n this.checkPosition();\n this.checkPositionWithEventLoop();\n\n };\n\n $affix.destroy = function () {\n\n // Unbind events\n targetEl.off('scroll', this.checkPosition);\n targetEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', this.$debouncedOnResize);\n\n };\n\n $affix.checkPositionWithEventLoop = function () {\n\n // IE 9 throws an error if we use 'this' instead of '$affix'\n // in this setTimeout call\n setTimeout($affix.checkPosition, 1);\n\n };\n\n $affix.checkPosition = function () {\n // if (!this.$element.is(':visible')) return\n\n var scrollTop = getScrollTop();\n var position = dimensions.offset(element[0]);\n var elementHeight = dimensions.height(element[0]);\n\n // Get required affix class according to position\n var affix = getRequiredAffixClass(unpin, position, elementHeight);\n\n // Did affix status changed this last check?\n if (affixed === affix) return;\n affixed = affix;\n\n if (affix === 'top') {\n unpin = null;\n if (setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n } else if (affix === 'bottom') {\n if (options.offsetUnpin) {\n unpin = -(options.offsetUnpin * 1);\n } else {\n // Calculate unpin threshold when affixed to bottom.\n // Hopefully the browser scrolls pixel by pixel.\n unpin = position.top - scrollTop;\n }\n if (setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', (options.offsetParent) ? '' : ((bodyEl[0].offsetHeight - offsetBottom - elementHeight - initialOffsetTop) + 'px'));\n }\n } else { // affix === 'middle'\n unpin = null;\n if (setWidth) {\n element.css('width', element[0].offsetWidth + 'px');\n }\n if (options.inlineStyles) {\n element.css('position', 'fixed');\n element.css('top', initialAffixTop + 'px');\n }\n }\n\n // Add proper affix class\n element.removeClass(reset).addClass('affix' + ((affix !== 'middle') ? '-' + affix : ''));\n\n };\n\n $affix.$onResize = function () {\n $affix.$parseOffsets();\n $affix.checkPosition();\n };\n $affix.$debouncedOnResize = debounce($affix.$onResize, 50);\n\n $affix.$parseOffsets = function () {\n var initialPosition = element[0].style.position;\n var initialTop = element[0].style.top;\n // Reset position to calculate correct offsetTop\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n\n if (options.offsetTop) {\n if (options.offsetTop === 'auto') {\n options.offsetTop = '+0';\n }\n if (options.offsetTop.match(/^[-+]\\d+$/)) {\n initialAffixTop = - options.offsetTop * 1;\n if (options.offsetParent) {\n offsetTop = dimensions.offset(parent[0]).top + (options.offsetTop * 1);\n } else {\n offsetTop = dimensions.offset(element[0]).top - dimensions.css(element[0], 'marginTop', true) + (options.offsetTop * 1);\n }\n } else {\n offsetTop = options.offsetTop * 1;\n }\n }\n\n if (options.offsetBottom) {\n if (options.offsetParent && options.offsetBottom.match(/^[-+]\\d+$/)) {\n // add 1 pixel due to rounding problems...\n offsetBottom = getScrollHeight() - (dimensions.offset(parent[0]).top + dimensions.height(parent[0])) + (options.offsetBottom * 1) + 1;\n } else {\n offsetBottom = options.offsetBottom * 1;\n }\n }\n\n // Bring back the element's position after calculations\n if (options.inlineStyles) {\n element.css('position', initialPosition);\n element.css('top', initialTop);\n }\n };\n\n // Private methods\n\n function getRequiredAffixClass (_unpin, position, elementHeight) {\n var scrollTop = getScrollTop();\n var scrollHeight = getScrollHeight();\n\n if (scrollTop <= offsetTop) {\n return 'top';\n } else if (_unpin !== null) {\n return scrollTop + _unpin <= position.top ? 'middle' : 'bottom';\n } else if (offsetBottom !== null && (position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom)) {\n return 'bottom';\n }\n return 'middle';\n }\n\n function getScrollTop () {\n return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop;\n }\n\n function getScrollHeight () {\n return targetEl[0] === $window ? $window.document.body.scrollHeight : targetEl[0].scrollHeight;\n }\n\n $affix.init();\n return $affix;\n\n }\n\n return AffixFactory;\n\n };\n\n })\n\n .directive('bsAffix', function ($affix, $window, $timeout) {\n\n return {\n restrict: 'EAC',\n require: '^?bsAffixTarget',\n link: function postLink (scope, element, attr, affixTarget) {\n\n var options = {scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window)};\n angular.forEach(['offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles', 'setWidth'], function (key) {\n if (angular.isDefined(attr[key])) {\n var option = attr[key];\n if (/true/i.test(option)) option = true;\n if (/false/i.test(option)) option = false;\n options[key] = option;\n }\n });\n\n var affix;\n $timeout(function () { affix = $affix(element, options); });\n scope.$on('$destroy', function () {\n if (affix) affix.destroy();\n options = null;\n affix = null;\n });\n\n }\n };\n\n })\n\n .directive('bsAffixTarget', function () {\n return {\n controller: function ($element) {\n this.$element = $element;\n }\n };\n });\n\n// Source: module.js\nangular.module('mgcrea.ngStrap', [\n 'mgcrea.ngStrap.modal',\n 'mgcrea.ngStrap.aside',\n 'mgcrea.ngStrap.alert',\n 'mgcrea.ngStrap.button',\n 'mgcrea.ngStrap.select',\n 'mgcrea.ngStrap.datepicker',\n 'mgcrea.ngStrap.timepicker',\n 'mgcrea.ngStrap.navbar',\n 'mgcrea.ngStrap.tooltip',\n 'mgcrea.ngStrap.popover',\n 'mgcrea.ngStrap.dropdown',\n 'mgcrea.ngStrap.typeahead',\n 'mgcrea.ngStrap.scrollspy',\n 'mgcrea.ngStrap.affix',\n 'mgcrea.ngStrap.tab',\n 'mgcrea.ngStrap.collapse'\n]);\n\n})(window, document);\n","'use strict';\n\nangular.module('mgcrea.ngStrap.typeahead', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$typeahead', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'typeahead',\n prefixEvent: '$typeahead',\n placement: 'bottom-left',\n templateUrl: 'typeahead/typeahead.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n minLength: 1,\n filter: 'bsAsyncFilter',\n limit: 6,\n autoSelect: false,\n comparator: '',\n trimValue: true\n };\n\n this.$get = function ($window, $rootScope, $tooltip, $$rAF, $timeout) {\n\n function TypeaheadFactory (element, controller, config) {\n\n var $typeahead = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $typeahead = $tooltip(element, options);\n var parentScope = config.scope;\n var scope = $typeahead.$scope;\n\n scope.$resetMatches = function () {\n scope.$matches = [];\n scope.$activeIndex = options.autoSelect ? 0 : -1; // If set to 0, the first match will be highlighted\n };\n scope.$resetMatches();\n\n scope.$activate = function (index) {\n scope.$$postDigest(function () {\n $typeahead.activate(index);\n });\n };\n\n scope.$select = function (index, evt) {\n scope.$$postDigest(function () {\n $typeahead.select(index);\n });\n };\n\n scope.$isVisible = function () {\n return $typeahead.$isVisible();\n };\n\n // Public methods\n\n $typeahead.update = function (matches) {\n scope.$matches = matches;\n if (scope.$activeIndex >= matches.length) {\n scope.$activeIndex = options.autoSelect ? 0 : -1;\n }\n\n // wrap in a $timeout so the results are updated\n // before repositioning\n safeDigest(scope);\n $$rAF($typeahead.$applyPlacement);\n };\n\n $typeahead.activate = function (index) {\n scope.$activeIndex = index;\n };\n\n $typeahead.select = function (index) {\n if (index === -1) return;\n var value = scope.$matches[index].value;\n // console.log('$setViewValue', value);\n controller.$setViewValue(value);\n controller.$render();\n scope.$resetMatches();\n if (parentScope) parentScope.$digest();\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $typeahead);\n if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) {\n options.onSelect(value, index, $typeahead);\n }\n };\n\n // Protected methods\n\n $typeahead.$isVisible = function () {\n if (!options.minLength || !controller) {\n return !!scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength;\n };\n\n $typeahead.$getIndex = function (value) {\n var index;\n for (index = scope.$matches.length; index--;) {\n if (angular.equals(scope.$matches[index].value, value)) break;\n }\n return index;\n };\n\n $typeahead.$onMouseDown = function (evt) {\n // Prevent blur on mousedown\n evt.preventDefault();\n evt.stopPropagation();\n };\n\n $typeahead.$$updateScrollTop = function (container, index) {\n if (index > -1 && index < container.children.length) {\n var active = container.children[index];\n var clientTop = active.offsetTop;\n var clientBottom = active.offsetTop + active.clientHeight;\n var highWatermark = container.scrollTop;\n var lowWatermark = container.scrollTop + container.clientHeight;\n\n // active entry overlaps top border\n if (clientBottom >= highWatermark && clientTop < highWatermark) {\n container.scrollTop = Math.max(0, container.scrollTop - container.clientHeight);\n } else if (clientBottom > lowWatermark) {\n // top of active element is invisible because it's below the bottom of the visible container window\n container.scrollTop = clientTop;\n }\n }\n };\n\n $typeahead.$onKeyDown = function (evt) {\n if (!/(38|40|13)/.test(evt.keyCode)) return;\n\n // Let ngSubmit pass if the typeahead tip is hidden or no option is selected\n if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // Select with enter\n if (evt.keyCode === 13 && scope.$matches.length) {\n $typeahead.select(scope.$activeIndex);\n // Navigate with keyboard\n } else if (evt.keyCode === 38 && scope.$activeIndex > 0) {\n scope.$activeIndex--;\n } else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) {\n scope.$activeIndex++;\n } else if (angular.isUndefined(scope.$activeIndex)) {\n scope.$activeIndex = 0;\n }\n\n // update scrollTop property on $typeahead when scope.$activeIndex is not in visible area\n $typeahead.$$updateScrollTop($typeahead.$element[0], scope.$activeIndex);\n scope.$digest();\n };\n\n // Overrides\n\n var show = $typeahead.show;\n $typeahead.show = function () {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed immediately.\n $timeout(function () {\n if ($typeahead.$element) {\n $typeahead.$element.on('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n if (element) element.on('keydown', $typeahead.$onKeyDown);\n }\n }\n }, 0, false);\n };\n\n var hide = $typeahead.hide;\n $typeahead.hide = function () {\n if ($typeahead.$element) $typeahead.$element.off('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n if (element) element.off('keydown', $typeahead.$onKeyDown);\n }\n if (!options.autoSelect) {\n $typeahead.activate(-1);\n }\n hide();\n };\n\n return $typeahead;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n TypeaheadFactory.defaults = defaults;\n return TypeaheadFactory;\n\n };\n\n })\n\n .filter('bsAsyncFilter', function ($filter) {\n return function (array, expression, comparator) {\n if (array && angular.isFunction(array.then)) {\n return array.then(function (results) {\n return $filter('filter')(results, expression, comparator);\n });\n }\n return $filter('filter')(array, expression, comparator);\n };\n })\n\n .directive('bsTypeahead', function ($window, $parse, $q, $typeahead, $parseOptions) {\n\n var defaults = $typeahead.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Fixes firefox bug when using objects in model with typeahead\n // Yes this breaks any other directive using a 'change' event on this input,\n // but if it is using the 'change' event why is it used with typeahead?\n element.off('change');\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'filter', 'limit', 'minLength', 'watchOptions', 'selectMode', 'autoSelect', 'comparator', 'id', 'prefixEvent', 'prefixClass'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'trimValue', 'filter'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show, hide and select events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Disable browser autocompletion\n if (!element.attr('autocomplete')) element.attr('autocomplete', 'off');\n\n // Build proper bsOptions\n var filter = angular.isDefined(options.filter) ? options.filter : defaults.filter;\n var limit = options.limit || defaults.limit;\n var comparator = options.comparator || defaults.comparator;\n\n var bsOptions = attr.bsOptions;\n if (filter) {\n bsOptions += ' | ' + filter + ':$viewValue';\n if (comparator) bsOptions += ':' + comparator;\n }\n if (limit) bsOptions += ' | limitTo:' + limit;\n var parsedOptions = $parseOptions(bsOptions);\n\n // Initialize typeahead\n var typeahead = $typeahead(element, controller, options);\n\n // Watch options on demand\n if (options.watchOptions) {\n // Watch bsOptions values before filtering for changes, drop function calls\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').replace(/\\(.*\\)/g, '').trim();\n scope.$watchCollection(watchedOptions, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller).then(function (values) {\n typeahead.update(values);\n controller.$render();\n });\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('$watch', element.attr('ng-model'), newValue);\n scope.$modelValue = newValue; // Publish modelValue on scope for custom templates\n parsedOptions.valuesFn(scope, controller)\n .then(function (values) {\n // Prevent input with no future prospect if selectMode is truthy\n // @TODO test selectMode\n if (options.selectMode && !values.length && newValue.length > 0) {\n controller.$setViewValue(controller.$viewValue.substring(0, controller.$viewValue.length - 1));\n return;\n }\n if (values.length > limit) values = values.slice(0, limit);\n typeahead.update(values);\n // Queue a new rendering that will leverage collection loading\n controller.$render();\n });\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var displayValue = parsedOptions.displayValue(modelValue);\n\n // If we can determine the displayValue, use that\n if (displayValue) {\n return displayValue;\n }\n\n // If there's no display value, attempt to use the modelValue.\n // If the model is an object not much we can do\n if (angular.isDefined(modelValue) && typeof modelValue !== 'object') {\n return modelValue;\n }\n return '';\n });\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n if (controller.$isEmpty(controller.$viewValue)) {\n return element.val('');\n }\n var index = typeahead.$getIndex(controller.$modelValue);\n var selected = index !== -1 ? typeahead.$scope.$matches[index].label : controller.$viewValue;\n selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected;\n var value = selected ? selected.toString().replace(/<(?:.|\\n)*?>/gm, '') : '';\n var ss = element[0].selectionStart;\n var sd = element[0].selectionEnd;\n element.val(options.trimValue === false ? value : value.trim());\n element[0].setSelectionRange(ss, sd);\n };\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (typeahead) typeahead.destroy();\n options = null;\n typeahead = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\n// NOTICE: This file was forked from the angular-material project (github.com/angular/material)\n// MIT Licensed - Copyright (c) 2014-2015 Google, Inc. http://angularjs.org\n\nangular.module('mgcrea.ngStrap.core', [])\n .service('$bsCompiler', bsCompilerService);\n\nfunction bsCompilerService ($q, $http, $injector, $compile, $controller, $templateCache) {\n\n /*\n * @ngdoc service\n * @name $bsCompiler\n * @module material.core\n * @description\n * The $bsCompiler service is an abstraction of angular's compiler, that allows the developer\n * to easily compile an element with a templateUrl, controller, and locals.\n *\n * @usage\n * \n * $bsCompiler.compile({\n * templateUrl: 'modal.html',\n * controller: 'ModalCtrl',\n * locals: {\n * modal: myModalInstance;\n * }\n * }).then(function(compileData) {\n * compileData.element; // modal.html's template in an element\n * compileData.link(myScope); //attach controller & scope to element\n * });\n * \n */\n\n /*\n * @ngdoc method\n * @name $bsCompiler#compile\n * @description A helper to compile an HTML template/templateUrl with a given controller,\n * locals, and scope.\n * @param {object} options An options object, with the following properties:\n *\n * - `controller` - `{(string=|function()=}` Controller fn that should be associated with\n * newly created scope or the name of a registered controller if passed as a string.\n * - `controllerAs` - `{string=}` A controller alias name. If present the controller will be\n * published to scope under the `controllerAs` name.\n * - `template` - `{string=}` An html template as a string.\n * - `templateUrl` - `{string=}` A path to an html template.\n * - `transformTemplate` - `{function(template)=}` A function which transforms the template after\n * it is loaded. It will be given the template string as a parameter, and should\n * return a a new string representing the transformed template.\n * - `resolve` - `{Object.=}` - An optional map of dependencies which should\n * be injected into the controller. If any of these dependencies are promises, the compiler\n * will wait for them all to be resolved, or if one is rejected before the controller is\n * instantiated `compile()` will fail..\n * * `key` - `{string}`: a name of a dependency to be injected into the controller.\n * * `factory` - `{string|function}`: If `string` then it is an alias for a service.\n * Otherwise if function, then it is injected and the return value is treated as the\n * dependency. If the result is a promise, it is resolved before its value is\n * injected into the controller.\n *\n * @returns {object=} promise A promise, which will be resolved with a `compileData` object.\n * `compileData` has the following properties:\n *\n * - `element` - `{element}`: an uncompiled element matching the provided template.\n * - `link` - `{function(scope)}`: A link function, which, when called, will compile\n * the element and instantiate the provided controller (if given).\n * - `locals` - `{object}`: The locals which will be passed into the controller once `link` is\n * called. If `bindToController` is true, they will be coppied to the ctrl instead\n * - `bindToController` - `bool`: bind the locals to the controller, instead of passing them in.\n */\n this.compile = function (options) {\n\n if (options.template && /\\.html$/.test(options.template)) {\n console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.');\n options.templateUrl = options.template;\n options.template = '';\n }\n\n var templateUrl = options.templateUrl;\n var template = options.template || '';\n var controller = options.controller;\n var controllerAs = options.controllerAs;\n var resolve = options.resolve || {};\n var locals = options.locals || {};\n var transformTemplate = options.transformTemplate || angular.identity;\n var bindToController = options.bindToController;\n\n // Take resolve values and invoke them.\n // Resolves can either be a string (value: 'MyRegisteredAngularConst'),\n // or an invokable 'factory' of sorts: (value: function ValueGetter($dependency) {})\n angular.forEach(resolve, function (value, key) {\n if (angular.isString(value)) {\n resolve[key] = $injector.get(value);\n } else {\n resolve[key] = $injector.invoke(value);\n }\n });\n // Add the locals, which are just straight values to inject\n // eg locals: { three: 3 }, will inject three into the controller\n angular.extend(resolve, locals);\n\n if (template) {\n resolve.$template = $q.when(template);\n } else if (templateUrl) {\n resolve.$template = fetchTemplate(templateUrl);\n } else {\n throw new Error('Missing `template` / `templateUrl` option.');\n }\n\n if (options.titleTemplate) {\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.titleTemplate)])\n .then(function (templates) {\n var templateEl = angular.element(templates[0]);\n findElement('[ng-bind=\"title\"]', templateEl[0])\n .removeAttr('ng-bind')\n .html(templates[1]);\n return templateEl[0].outerHTML;\n });\n }\n\n if (options.contentTemplate) {\n // TODO(mgcrea): deprecate?\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.contentTemplate)])\n .then(function (templates) {\n var templateEl = angular.element(templates[0]);\n var contentEl = findElement('[ng-bind=\"content\"]', templateEl[0])\n .removeAttr('ng-bind')\n .html(templates[1]);\n // Drop the default footer as you probably don't want it if you use a custom contentTemplate\n if (!options.templateUrl) contentEl.next().remove();\n return templateEl[0].outerHTML;\n });\n }\n\n // Wait for all the resolves to finish if they are promises\n return $q.all(resolve).then(function (locals) {\n\n var template = transformTemplate(locals.$template);\n if (options.html) {\n template = template.replace(/ng-bind=\"/ig, 'ng-bind-html=\"');\n }\n // var element = options.element || angular.element('
').html(template.trim()).contents();\n var element = angular.element('
').html(template.trim()).contents();\n var linkFn = $compile(element);\n\n // Return a linking function that can be used later when the element is ready\n return {\n locals: locals,\n element: element,\n link: function link (scope) {\n locals.$scope = scope;\n\n // Instantiate controller if it exists, because we have scope\n if (controller) {\n var invokeCtrl = $controller(controller, locals, true);\n if (bindToController) {\n angular.extend(invokeCtrl.instance, locals);\n }\n // Support angular@~1.2 invokeCtrl\n var ctrl = angular.isObject(invokeCtrl) ? invokeCtrl : invokeCtrl();\n // See angular-route source for this logic\n element.data('$ngControllerController', ctrl);\n element.children().data('$ngControllerController', ctrl);\n\n if (controllerAs) {\n scope[controllerAs] = ctrl;\n }\n }\n\n return linkFn.apply(null, arguments);\n }\n };\n });\n\n };\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate (template) {\n if (fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache})\n .then(function (res) {\n return res.data;\n }));\n }\n\n}\n","'use strict';\n\nangular.module('mgcrea.ngStrap.dropdown', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$dropdown', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'dropdown',\n prefixEvent: 'dropdown',\n placement: 'bottom-left',\n templateUrl: 'dropdown/dropdown.tpl.html',\n trigger: 'click',\n container: false,\n keyboard: true,\n html: false,\n delay: 0\n };\n\n this.$get = function ($window, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var matchesSelector = Element.prototype.matchesSelector || Element.prototype.webkitMatchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector;\n\n function DropdownFactory (element, config) {\n\n var $dropdown = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n /* var scope = */$dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n $dropdown = $tooltip(element, options);\n var parentEl = element.parent();\n\n // Protected methods\n\n $dropdown.$onKeyDown = function (evt) {\n if (!/(38|40)/.test(evt.keyCode)) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Retrieve focused index\n var items = angular.element($dropdown.$element[0].querySelectorAll('li:not(.divider) a'));\n if (!items.length) return;\n var index;\n angular.forEach(items, function (el, i) {\n if (matchesSelector && matchesSelector.call(el, ':focus')) index = i;\n });\n\n // Navigate with keyboard\n if (evt.keyCode === 38 && index > 0) index--;\n else if (evt.keyCode === 40 && index < items.length - 1) index++;\n else if (angular.isUndefined(index)) index = 0;\n items.eq(index)[0].focus();\n\n };\n\n // Overrides\n\n var show = $dropdown.show;\n $dropdown.show = function () {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n if (options.keyboard && $dropdown.$element) $dropdown.$element.on('keydown', $dropdown.$onKeyDown);\n bodyEl.on('click', onBodyClick);\n }, 0, false);\n if (parentEl.hasClass('dropdown')) parentEl.addClass('open');\n };\n\n var hide = $dropdown.hide;\n $dropdown.hide = function () {\n if (!$dropdown.$isShown) return;\n if (options.keyboard && $dropdown.$element) $dropdown.$element.off('keydown', $dropdown.$onKeyDown);\n bodyEl.off('click', onBodyClick);\n if (parentEl.hasClass('dropdown')) parentEl.removeClass('open');\n hide();\n };\n\n var destroy = $dropdown.destroy;\n $dropdown.destroy = function () {\n bodyEl.off('click', onBodyClick);\n destroy();\n };\n\n // Private functions\n\n function onBodyClick (evt) {\n if (evt.target === element[0]) return;\n return evt.target !== element[0] && $dropdown.hide();\n }\n\n return $dropdown;\n\n }\n\n return DropdownFactory;\n\n };\n\n })\n\n .directive('bsDropdown', function ($window, $sce, $dropdown) {\n\n return {\n restrict: 'EAC',\n scope: true,\n compile: function (tElement, tAttrs) {\n\n // Support for inlined template (next sibling)\n // It must be fetched before compilation\n if (!tAttrs.bsDropdown) {\n var nextSibling = tElement[0].nextSibling;\n while (nextSibling && nextSibling.nodeType !== 1) {\n nextSibling = nextSibling.nextSibling;\n }\n if (nextSibling && nextSibling.className.split(' ').indexOf('dropdown-menu') >= 0) {\n tAttrs.template = nextSibling.outerHTML;\n tAttrs.templateUrl = undefined;\n nextSibling.parentNode.removeChild(nextSibling);\n }\n }\n\n return function postLink (scope, element, attr) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id', 'autoClose'], function (key) {\n if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as an object\n if (attr.bsDropdown) {\n scope.$watch(attr.bsDropdown, function (newValue, oldValue) {\n scope.content = newValue;\n }, true);\n }\n\n // Initialize dropdown\n var dropdown = $dropdown(element, options);\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!dropdown || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i);\n if (newValue === true) {\n dropdown.show();\n } else {\n dropdown.hide();\n }\n });\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (dropdown) dropdown.destroy();\n options = null;\n dropdown = null;\n });\n\n };\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.tooltip', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$tooltip', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n prefixClass: 'tooltip',\n prefixEvent: 'tooltip',\n container: false,\n target: false,\n placement: 'top',\n templateUrl: 'tooltip/tooltip.tpl.html',\n template: '',\n titleTemplate: false,\n trigger: 'hover focus',\n keyboard: false,\n html: false,\n show: false,\n title: '',\n type: '',\n delay: 0,\n autoClose: false,\n bsEnabled: true,\n mouseDownPreventDefault: true,\n mouseDownStopPropagation: true,\n viewport: {\n selector: 'body',\n padding: 0\n }\n };\n\n this.$get = function ($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n var $body = angular.element($window.document);\n\n function TooltipFactory (element, config) {\n\n var $tooltip = {};\n\n // Common vars\n var options = $tooltip.$options = angular.extend({}, defaults, config);\n var promise = $tooltip.$promise = $bsCompiler.compile(options);\n var scope = $tooltip.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n var nodeName = element[0].nodeName.toLowerCase();\n if (options.delay && angular.isString(options.delay)) {\n var split = options.delay.split(',').map(parseFloat);\n options.delay = split.length > 1 ? {show: split[0], hide: split[1]} : split[0];\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $tooltip.$id = options.id || element.attr('id') || '';\n\n // Support scope as string options\n if (options.title) {\n scope.title = $sce.trustAsHtml(options.title);\n }\n\n // Provide scope helpers\n scope.$setEnabled = function (isEnabled) {\n scope.$$postDigest(function () {\n $tooltip.setEnabled(isEnabled);\n });\n };\n scope.$hide = function () {\n scope.$$postDigest(function () {\n $tooltip.hide();\n });\n };\n scope.$show = function () {\n scope.$$postDigest(function () {\n $tooltip.show();\n });\n };\n scope.$toggle = function () {\n scope.$$postDigest(function () {\n $tooltip.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $tooltip.$isShown = scope.$isShown = false;\n\n // Private vars\n var timeout;\n var hoverState;\n\n // Fetch, compile then initialize tooltip\n var compileData;\n var tipElement;\n var tipContainer;\n var tipScope;\n promise.then(function (data) {\n compileData = data;\n $tooltip.init();\n });\n\n $tooltip.init = function () {\n\n // Options: delay\n if (options.delay && angular.isNumber(options.delay)) {\n options.delay = {\n show: options.delay,\n hide: options.delay\n };\n }\n\n // Replace trigger on touch devices ?\n // if(isTouch && options.trigger === defaults.trigger) {\n // options.trigger.replace(/hover/g, 'click');\n // }\n\n // Options : container\n if (options.container === 'self') {\n tipContainer = element;\n } else if (angular.isElement(options.container)) {\n tipContainer = options.container;\n } else if (options.container) {\n tipContainer = findElement(options.container);\n }\n\n // Options: trigger\n bindTriggerEvents();\n\n // Options: target\n if (options.target) {\n options.target = angular.isElement(options.target) ? options.target : findElement(options.target);\n }\n\n // Options: show\n if (options.show) {\n scope.$$postDigest(function () {\n if (options.trigger === 'focus') {\n element[0].focus();\n } else {\n $tooltip.show();\n }\n });\n }\n\n };\n\n $tooltip.destroy = function () {\n\n // Unbind events\n unbindTriggerEvents();\n\n // Remove element\n destroyTipElement();\n\n // Destroy scope\n scope.$destroy();\n\n };\n\n $tooltip.enter = function () {\n\n clearTimeout(timeout);\n hoverState = 'in';\n if (!options.delay || !options.delay.show) {\n return $tooltip.show();\n }\n\n timeout = setTimeout(function () {\n if (hoverState === 'in') $tooltip.show();\n }, options.delay.show);\n\n };\n\n $tooltip.show = function () {\n if (!options.bsEnabled || $tooltip.$isShown) return;\n\n scope.$emit(options.prefixEvent + '.show.before', $tooltip);\n if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) {\n options.onBeforeShow($tooltip);\n }\n var parent;\n var after;\n if (options.container) {\n parent = tipContainer;\n if (tipContainer[0].lastChild) {\n after = angular.element(tipContainer[0].lastChild);\n } else {\n after = null;\n }\n } else {\n parent = null;\n after = element;\n }\n\n\n // Hide any existing tipElement\n if (tipElement) destroyTipElement();\n // Fetch a cloned element linked from template\n tipScope = $tooltip.$scope.$new();\n tipElement = $tooltip.$element = compileData.link(tipScope, function (clonedElement, scope) {});\n\n // Set the initial positioning. Make the tooltip invisible\n // so IE doesn't try to focus on it off screen.\n tipElement.css({top: '-9999px', left: '-9999px', right: 'auto', display: 'block', visibility: 'hidden'});\n\n // Options: animation\n if (options.animation) tipElement.addClass(options.animation);\n // Options: type\n if (options.type) tipElement.addClass(options.prefixClass + '-' + options.type);\n // Options: custom classes\n if (options.customClass) tipElement.addClass(options.customClass);\n\n // Append the element, without any animations. If we append\n // using $animate.enter, some of the animations cause the placement\n // to be off due to the transforms.\n if (after) {\n after.after(tipElement);\n } else {\n parent.prepend(tipElement);\n }\n\n $tooltip.$isShown = scope.$isShown = true;\n safeDigest(scope);\n\n // Now, apply placement\n $tooltip.$applyPlacement();\n\n // Once placed, animate it.\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.enter(tipElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(tipElement, parent, after).then(enterAnimateCallback);\n }\n safeDigest(scope);\n\n $$rAF(function () {\n // Once the tooltip is placed and the animation starts, make the tooltip visible\n if (tipElement) tipElement.css({visibility: 'visible'});\n\n // Bind events\n if (options.keyboard) {\n if (options.trigger !== 'focus') {\n $tooltip.focus();\n }\n bindKeyboardEvents();\n }\n });\n\n if (options.autoClose) {\n bindAutoCloseEvents();\n }\n\n };\n\n function enterAnimateCallback () {\n scope.$emit(options.prefixEvent + '.show', $tooltip);\n if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) {\n options.onShow($tooltip);\n }\n }\n\n $tooltip.leave = function () {\n\n clearTimeout(timeout);\n hoverState = 'out';\n if (!options.delay || !options.delay.hide) {\n return $tooltip.hide();\n }\n timeout = setTimeout(function () {\n if (hoverState === 'out') {\n $tooltip.hide();\n }\n }, options.delay.hide);\n\n };\n\n var _blur;\n var _tipToHide;\n $tooltip.hide = function (blur) {\n\n if (!$tooltip.$isShown) return;\n scope.$emit(options.prefixEvent + '.hide.before', $tooltip);\n if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) {\n options.onBeforeHide($tooltip);\n }\n\n // store blur value for leaveAnimateCallback to use\n _blur = blur;\n\n // store current tipElement reference to use\n // in leaveAnimateCallback\n _tipToHide = tipElement;\n\n if (tipElement !== null) {\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.leave(tipElement, leaveAnimateCallback);\n } else {\n $animate.leave(tipElement).then(leaveAnimateCallback);\n }\n }\n\n $tooltip.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n if (options.keyboard && tipElement !== null) {\n unbindKeyboardEvents();\n }\n\n if (options.autoClose && tipElement !== null) {\n unbindAutoCloseEvents();\n }\n };\n\n function leaveAnimateCallback () {\n scope.$emit(options.prefixEvent + '.hide', $tooltip);\n if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) {\n options.onHide($tooltip);\n }\n\n // check if current tipElement still references\n // the same element when hide was called\n if (tipElement === _tipToHide) {\n // Allow to blur the input when hidden, like when pressing enter key\n if (_blur && options.trigger === 'focus') {\n return element[0].blur();\n }\n\n // clean up child scopes\n destroyTipElement();\n }\n }\n\n $tooltip.toggle = function (evt) {\n if (evt) { evt.preventDefault(); }\n if ($tooltip.$isShown) {\n $tooltip.leave();\n } else {\n $tooltip.enter();\n }\n };\n\n $tooltip.focus = function () {\n tipElement[0].focus();\n };\n\n $tooltip.setEnabled = function (isEnabled) {\n options.bsEnabled = isEnabled;\n };\n\n $tooltip.setViewport = function (viewport) {\n options.viewport = viewport;\n };\n\n // Protected methods\n\n $tooltip.$applyPlacement = function () {\n if (!tipElement) return;\n\n // Determine if we're doing an auto or normal placement\n var placement = options.placement;\n var autoToken = /\\s?auto?\\s?/i;\n var autoPlace = autoToken.test(placement);\n\n if (autoPlace) {\n placement = placement.replace(autoToken, '') || defaults.placement;\n }\n\n // Need to add the position class before we get\n // the offsets\n tipElement.addClass(options.placement);\n\n // Get the position of the target element\n // and the height and width of the tooltip so we can center it.\n var elementPosition = getPosition();\n var tipWidth = tipElement.prop('offsetWidth');\n var tipHeight = tipElement.prop('offsetHeight');\n\n // Refresh viewport position\n $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport);\n\n // If we're auto placing, we need to check the positioning\n if (autoPlace) {\n var originalPlacement = placement;\n var viewportPosition = getPosition($tooltip.$viewport);\n\n if (/bottom/.test(originalPlacement) && elementPosition.bottom + tipHeight > viewportPosition.bottom) {\n placement = originalPlacement.replace('bottom', 'top');\n } else if (/top/.test(originalPlacement) && elementPosition.top - tipHeight < viewportPosition.top) {\n placement = originalPlacement.replace('top', 'bottom');\n }\n\n if (/left/.test(originalPlacement) && elementPosition.left - tipWidth < viewportPosition.left) {\n placement = placement.replace('left', 'right');\n } else if (/right/.test(originalPlacement) && elementPosition.right + tipWidth > viewportPosition.width) {\n placement = placement.replace('right', 'left');\n }\n\n tipElement.removeClass(originalPlacement).addClass(placement);\n }\n\n // Get the tooltip's top and left coordinates to center it with this directive.\n var tipPosition = getCalculatedOffset(placement, elementPosition, tipWidth, tipHeight);\n applyPlacement(tipPosition, placement);\n };\n\n $tooltip.$onKeyUp = function (evt) {\n if (evt.which === 27 && $tooltip.$isShown) {\n $tooltip.hide();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusKeyUp = function (evt) {\n if (evt.which === 27) {\n element[0].blur();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusElementMouseDown = function (evt) {\n if (options.mouseDownPreventDefault) { evt.preventDefault(); }\n if (options.mouseDownStopPropagation) { evt.stopPropagation(); }\n // Some browsers do not auto-focus buttons (eg. Safari)\n if ($tooltip.$isShown) {\n element[0].blur();\n } else {\n element[0].focus();\n }\n };\n\n // bind/unbind events\n function bindTriggerEvents () {\n var triggers = options.trigger.split(' ');\n angular.forEach(triggers, function (trigger) {\n if (trigger === 'click' || trigger === 'contextmenu') {\n element.on(trigger, $tooltip.toggle);\n } else if (trigger !== 'manual') {\n element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n if (nodeName === 'button' && trigger !== 'hover') {\n element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n });\n }\n\n function unbindTriggerEvents () {\n var triggers = options.trigger.split(' ');\n for (var i = triggers.length; i--;) {\n var trigger = triggers[i];\n if (trigger === 'click' || trigger === 'contextmenu') {\n element.off(trigger, $tooltip.toggle);\n } else if (trigger !== 'manual') {\n element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n if (nodeName === 'button' && trigger !== 'hover') {\n element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n }\n }\n\n function bindKeyboardEvents () {\n if (options.trigger !== 'focus') {\n tipElement.on('keyup', $tooltip.$onKeyUp);\n } else {\n element.on('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n function unbindKeyboardEvents () {\n if (options.trigger !== 'focus') {\n tipElement.off('keyup', $tooltip.$onKeyUp);\n } else {\n element.off('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n var _autoCloseEventsBinded = false;\n function bindAutoCloseEvents () {\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n // Stop propagation when clicking inside tooltip\n tipElement.on('click', stopEventPropagation);\n\n // Hide when clicking outside tooltip\n $body.on('click', $tooltip.hide);\n\n _autoCloseEventsBinded = true;\n }, 0, false);\n }\n\n function unbindAutoCloseEvents () {\n if (_autoCloseEventsBinded) {\n tipElement.off('click', stopEventPropagation);\n $body.off('click', $tooltip.hide);\n _autoCloseEventsBinded = false;\n }\n }\n\n function stopEventPropagation (event) {\n event.stopPropagation();\n }\n\n // Private methods\n\n function getPosition ($element) {\n $element = $element || (options.target || element);\n\n var el = $element[0];\n var isBody = el.tagName === 'BODY';\n\n var elRect = el.getBoundingClientRect();\n var rect = {};\n\n // IE8 has issues with angular.extend and using elRect directly.\n // By coping the values of elRect into a new object, we can continue to use extend\n /* eslint-disable guard-for-in */\n for (var p in elRect) { // eslint-disable-line\n // DO NOT use hasOwnProperty when inspecting the return of getBoundingClientRect.\n rect[p] = elRect[p];\n }\n /* eslint-enable guard-for-in */\n\n if (rect.width === null) {\n // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093\n rect = angular.extend({}, rect, {width: elRect.right - elRect.left, height: elRect.bottom - elRect.top});\n }\n var elOffset = isBody ? {top: 0, left: 0} : dimensions.offset(el);\n var scroll = {scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0};\n var outerDims = isBody ? {width: document.documentElement.clientWidth, height: $window.innerHeight} : null;\n\n return angular.extend({}, rect, scroll, outerDims, elOffset);\n }\n\n function getCalculatedOffset (placement, position, actualWidth, actualHeight) {\n var offset;\n var split = placement.split('-');\n\n switch (split[0]) {\n case 'right':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left + position.width\n };\n break;\n case 'bottom':\n offset = {\n top: position.top + position.height,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n case 'left':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left - actualWidth\n };\n break;\n default:\n offset = {\n top: position.top - actualHeight,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n }\n\n if (!split[1]) {\n return offset;\n }\n\n // Add support for corners @todo css\n if (split[0] === 'top' || split[0] === 'bottom') {\n switch (split[1]) {\n case 'left':\n offset.left = position.left;\n break;\n case 'right':\n offset.left = position.left + position.width - actualWidth;\n break;\n default:\n break;\n }\n } else if (split[0] === 'left' || split[0] === 'right') {\n switch (split[1]) {\n case 'top':\n offset.top = position.top - actualHeight + position.height;\n break;\n case 'bottom':\n offset.top = position.top;\n break;\n default:\n break;\n }\n }\n\n return offset;\n }\n\n function applyPlacement (offset, placement) {\n var tip = tipElement[0];\n var width = tip.offsetWidth;\n var height = tip.offsetHeight;\n\n // manually read margins because getBoundingClientRect includes difference\n var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10);\n var marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10);\n\n // we must check for NaN for ie 8/9\n if (isNaN(marginTop)) marginTop = 0;\n if (isNaN(marginLeft)) marginLeft = 0;\n\n offset.top = offset.top + marginTop;\n offset.left = offset.left + marginLeft;\n\n // dimensions setOffset doesn't round pixel values\n // so we use setOffset directly with our own function\n dimensions.setOffset(tip, angular.extend({\n using: function (props) {\n tipElement.css({\n top: Math.round(props.top) + 'px',\n left: Math.round(props.left) + 'px',\n right: ''\n });\n }\n }, offset), 0);\n\n // check to see if placing tip in new offset caused the tip to resize itself\n var actualWidth = tip.offsetWidth;\n var actualHeight = tip.offsetHeight;\n\n if (placement === 'top' && actualHeight !== height) {\n offset.top = offset.top + height - actualHeight;\n }\n\n // If it's an exotic placement, exit now instead of\n // applying a delta and changing the arrow\n if (/top-left|top-right|bottom-left|bottom-right/.test(placement)) return;\n\n var delta = getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight);\n\n if (delta.left) {\n offset.left += delta.left;\n } else {\n offset.top += delta.top;\n }\n\n dimensions.setOffset(tip, offset);\n\n if (/top|right|bottom|left/.test(placement)) {\n var isVertical = /top|bottom/.test(placement);\n var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight;\n var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight';\n\n replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical);\n }\n }\n\n // @source https://github.com/twbs/bootstrap/blob/v3.3.5/js/tooltip.js#L380\n function getViewportAdjustedDelta (placement, position, actualWidth, actualHeight) {\n var delta = {top: 0, left: 0};\n if (!$tooltip.$viewport) return delta;\n\n var viewportPadding = options.viewport && options.viewport.padding || 0;\n var viewportDimensions = getPosition($tooltip.$viewport);\n\n if (/right|left/.test(placement)) {\n var topEdgeOffset = position.top - viewportPadding - viewportDimensions.scroll;\n var bottomEdgeOffset = position.top + viewportPadding - viewportDimensions.scroll + actualHeight;\n if (topEdgeOffset < viewportDimensions.top) { // top overflow\n delta.top = viewportDimensions.top - topEdgeOffset;\n } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow\n delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset;\n }\n } else {\n var leftEdgeOffset = position.left - viewportPadding;\n var rightEdgeOffset = position.left + viewportPadding + actualWidth;\n if (leftEdgeOffset < viewportDimensions.left) { // left overflow\n delta.left = viewportDimensions.left - leftEdgeOffset;\n } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow\n delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset;\n }\n }\n\n return delta;\n }\n\n function replaceArrow (delta, dimension, isHorizontal) {\n var $arrow = findElement('.tooltip-arrow, .arrow', tipElement[0]);\n\n $arrow.css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')\n .css(isHorizontal ? 'top' : 'left', '');\n }\n\n function destroyTipElement () {\n // Cancel pending callbacks\n clearTimeout(timeout);\n\n if ($tooltip.$isShown && tipElement !== null) {\n if (options.autoClose) {\n unbindAutoCloseEvents();\n }\n\n if (options.keyboard) {\n unbindKeyboardEvents();\n }\n }\n\n if (tipScope) {\n tipScope.$destroy();\n tipScope = null;\n }\n\n if (tipElement) {\n tipElement.remove();\n tipElement = $tooltip.$element = null;\n }\n }\n\n return $tooltip;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return TooltipFactory;\n\n };\n\n })\n\n .directive('bsTooltip', function ($window, $location, $sce, $parse, $tooltip, $$rAF) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n var tooltip;\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'titleTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if (angular.isDefined(dataTarget)) {\n if (falseValueRegExp.test(dataTarget)) {\n options.target = false;\n } else {\n options.target = dataTarget;\n }\n }\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')) {\n scope.title = '';\n }\n\n // Observe scope attributes for change\n attr.$observe('title', function (newValue) {\n if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) {\n var oldValue = scope.title;\n scope.title = $sce.trustAsHtml(newValue);\n if (angular.isDefined(oldValue)) {\n $$rAF(function () {\n if (tooltip) tooltip.$applyPlacement();\n });\n }\n }\n });\n\n attr.$observe('disabled', function (newValue) {\n if (newValue && tooltip.$isShown) {\n tooltip.hide();\n }\n });\n\n // Support scope as an object\n if (attr.bsTooltip) {\n scope.$watch(attr.bsTooltip, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.title = newValue;\n }\n if (angular.isDefined(oldValue)) {\n $$rAF(function () {\n if (tooltip) tooltip.$applyPlacement();\n });\n }\n }, true);\n }\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!tooltip || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i);\n if (newValue === true) {\n tooltip.show();\n } else {\n tooltip.hide();\n }\n });\n }\n\n // Enabled binding support\n if (attr.bsEnabled) {\n scope.$watch(attr.bsEnabled, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.bsEnabled, newValue, oldValue);\n if (!tooltip || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i);\n if (newValue === false) {\n tooltip.setEnabled(false);\n } else {\n tooltip.setEnabled(true);\n }\n });\n }\n\n // Viewport support\n if (attr.viewport) {\n scope.$watch(attr.viewport, function (newValue) {\n if (!tooltip || !angular.isDefined(newValue)) return;\n tooltip.setViewport(newValue);\n });\n }\n\n // Initialize popover\n tooltip = $tooltip(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (tooltip) tooltip.destroy();\n options = null;\n tooltip = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.timepicker', ['mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip'])\n\n .provider('$timepicker', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n defaultDate: 'auto',\n // uncommenting the following line will break backwards compatability\n // prefixEvent: 'timepicker',\n prefixClass: 'timepicker',\n placement: 'bottom-left',\n templateUrl: 'timepicker/timepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: true,\n timeType: 'date',\n timeFormat: 'shortTime',\n timezone: null,\n modelTimeFormat: null,\n autoclose: false,\n minTime: -Infinity,\n maxTime: +Infinity,\n length: 5,\n hourStep: 1,\n minuteStep: 5,\n secondStep: 5,\n roundDisplay: false,\n iconUp: 'glyphicon glyphicon-chevron-up',\n iconDown: 'glyphicon glyphicon-chevron-down',\n arrowBehavior: 'pager'\n };\n\n this.$get = function ($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) {\n defaults.lang = $dateFormatter.getDefaultLocale();\n }\n\n function timepickerFactory (element, controller, config) {\n\n var $timepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $timepicker.$options;\n var scope = $timepicker.$scope;\n\n var lang = options.lang;\n var formatDate = function (date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n function floorMinutes (time) {\n // coeff used to floor current time to nearest minuteStep interval\n var coeff = 1000 * 60 * options.minuteStep;\n return new Date(Math.floor(time.getTime() / coeff) * coeff);\n }\n\n // View vars\n\n var selectedIndex = 0;\n var defaultDate = options.roundDisplay ? floorMinutes(new Date()) : new Date();\n var startDate = controller.$dateValue || defaultDate;\n var viewDate = {\n hour: startDate.getHours(),\n meridian: startDate.getHours() < 12,\n minute: startDate.getMinutes(),\n second: startDate.getSeconds(),\n millisecond: startDate.getMilliseconds()\n };\n\n var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang);\n\n var hoursFormat = $dateFormatter.hoursFormat(format);\n var timeSeparator = $dateFormatter.timeSeparator(format);\n var minutesFormat = $dateFormatter.minutesFormat(format);\n var secondsFormat = $dateFormatter.secondsFormat(format);\n var showSeconds = $dateFormatter.showSeconds(format);\n var showAM = $dateFormatter.showAM(format);\n\n scope.$iconUp = options.iconUp;\n scope.$iconDown = options.iconDown;\n\n // Scope methods\n\n scope.$select = function (date, index) {\n $timepicker.select(date, index);\n };\n scope.$moveIndex = function (value, index) {\n $timepicker.$moveIndex(value, index);\n };\n scope.$switchMeridian = function (date) {\n $timepicker.switchMeridian(date);\n };\n\n // Public methods\n\n $timepicker.update = function (date) {\n // console.warn('$timepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $timepicker.$date = date;\n angular.extend(viewDate, {\n hour: date.getHours(),\n minute: date.getMinutes(),\n second: date.getSeconds(),\n millisecond: date.getMilliseconds()\n });\n $timepicker.$build();\n } else if (!$timepicker.$isBuilt) {\n $timepicker.$build();\n }\n };\n\n $timepicker.select = function (date, index, keep) {\n // console.warn('$timepicker.select', date, scope.$mode);\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n controller.$dateValue = options.defaultDate === 'today' ? new Date() : new Date(1970, 0, 1);\n }\n\n if (!angular.isDate(date)) date = new Date(date);\n if (index === 0) controller.$dateValue.setHours(date.getHours());\n else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes());\n else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds());\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function () {\n $timepicker.hide(true);\n });\n }\n };\n\n $timepicker.switchMeridian = function (date) {\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n return;\n }\n var hours = (date || controller.$dateValue).getHours();\n controller.$dateValue.setHours(hours < 12 ? hours + 12 : hours - 12);\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n };\n\n // Protected methods\n\n $timepicker.$build = function () {\n // console.warn('$timepicker.$build() viewDate=%o', viewDate);\n var i;\n var midIndex = scope.midIndex = parseInt(options.length / 2, 10);\n var hours = [];\n var hour;\n for (i = 0; i < options.length; i++) {\n hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep);\n hours.push({\n date: hour,\n label: formatDate(hour, hoursFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(hour, 0),\n disabled: $timepicker.$isDisabled(hour, 0)\n });\n }\n var minutes = [];\n var minute;\n for (i = 0; i < options.length; i++) {\n minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep);\n minutes.push({\n date: minute,\n label: formatDate(minute, minutesFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(minute, 1),\n disabled: $timepicker.$isDisabled(minute, 1)\n });\n }\n var seconds = [];\n var second;\n for (i = 0; i < options.length; i++) {\n second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep);\n seconds.push({\n date: second,\n label: formatDate(second, secondsFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(second, 2),\n disabled: $timepicker.$isDisabled(second, 2)\n });\n }\n\n var rows = [];\n for (i = 0; i < options.length; i++) {\n if (showSeconds) {\n rows.push([hours[i], minutes[i], seconds[i]]);\n } else {\n rows.push([hours[i], minutes[i]]);\n }\n }\n scope.rows = rows;\n scope.showSeconds = showSeconds;\n scope.showAM = showAM;\n scope.isAM = ($timepicker.$date || hours[midIndex].date).getHours() < 12;\n scope.timeSeparator = timeSeparator;\n $timepicker.$isBuilt = true;\n };\n\n $timepicker.$isSelected = function (date, index) {\n if (!$timepicker.$date) return false;\n else if (index === 0) {\n return date.getHours() === $timepicker.$date.getHours();\n } else if (index === 1) {\n return date.getMinutes() === $timepicker.$date.getMinutes();\n } else if (index === 2) {\n return date.getSeconds() === $timepicker.$date.getSeconds();\n }\n };\n\n $timepicker.$isDisabled = function (date, index) {\n var selectedTime;\n if (index === 0) {\n selectedTime = date.getTime() + viewDate.minute * 6e4 + viewDate.second * 1e3;\n } else if (index === 1) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.second * 1e3;\n } else if (index === 2) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.minute * 6e4;\n }\n return selectedTime < options.minTime * 1 || selectedTime > options.maxTime * 1;\n };\n\n scope.$arrowAction = function (value, index) {\n if (options.arrowBehavior === 'picker') {\n $timepicker.$setTimeByStep(value, index);\n } else {\n $timepicker.$moveIndex(value, index);\n }\n };\n\n $timepicker.$setTimeByStep = function (value, index) {\n var newDate = new Date($timepicker.$date || startDate);\n var hours = newDate.getHours();\n var minutes = newDate.getMinutes();\n var seconds = newDate.getSeconds();\n if (index === 0) {\n newDate.setHours(hours - (parseInt(options.hourStep, 10) * value));\n } else if (index === 1) {\n newDate.setMinutes(minutes - (parseInt(options.minuteStep, 10) * value));\n } else if (index === 2) {\n newDate.setSeconds(seconds - (parseInt(options.secondStep, 10) * value));\n }\n $timepicker.select(newDate, index, true);\n };\n\n $timepicker.$moveIndex = function (value, index) {\n var targetDate;\n if (index === 0) {\n targetDate = new Date(1970, 0, 1, viewDate.hour + (value * options.length), viewDate.minute, viewDate.second);\n angular.extend(viewDate, {\n hour: targetDate.getHours()\n });\n } else if (index === 1) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute + (value * options.length * options.minuteStep), viewDate.second);\n angular.extend(viewDate, {\n minute: targetDate.getMinutes()\n });\n } else if (index === 2) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute, viewDate.second + (value * options.length * options.secondStep));\n angular.extend(viewDate, {\n second: targetDate.getSeconds()\n });\n }\n $timepicker.$build();\n };\n\n $timepicker.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n if (evt.target.nodeName.toLowerCase() !== 'input') evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $timepicker.$onKeyDown = function (evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Close on enter\n if (evt.keyCode === 13) {\n $timepicker.hide(true);\n return;\n }\n\n // Navigate with keyboard\n var newDate = new Date($timepicker.$date);\n var hours = newDate.getHours();\n var hoursLength = formatDate(newDate, hoursFormat).length;\n var minutes = newDate.getMinutes();\n var minutesLength = formatDate(newDate, minutesFormat).length;\n var seconds = newDate.getSeconds();\n var secondsLength = formatDate(newDate, secondsFormat).length;\n var sepLength = 1;\n var lateralMove = /(37|39)/.test(evt.keyCode);\n var count = 2 + showSeconds * 1 + showAM * 1;\n\n // Navigate indexes (left, right)\n if (lateralMove) {\n if (evt.keyCode === 37) selectedIndex = selectedIndex < 1 ? count - 1 : selectedIndex - 1;\n else if (evt.keyCode === 39) selectedIndex = selectedIndex < count - 1 ? selectedIndex + 1 : 0;\n }\n\n // Update values (up, down)\n var selectRange = [0, hoursLength];\n var incr = 0;\n if (evt.keyCode === 38) incr = -1;\n if (evt.keyCode === 40) incr = +1;\n var isSeconds = selectedIndex === 2 && showSeconds;\n var isMeridian = selectedIndex === 2 && !showSeconds || selectedIndex === 3 && showSeconds;\n if (selectedIndex === 0) {\n newDate.setHours(hours + incr * parseInt(options.hourStep, 10));\n // re-calculate hours length because we have changed hours value\n hoursLength = formatDate(newDate, hoursFormat).length;\n selectRange = [0, hoursLength];\n } else if (selectedIndex === 1) {\n newDate.setMinutes(minutes + incr * parseInt(options.minuteStep, 10));\n // re-calculate minutes length because we have changes minutes value\n minutesLength = formatDate(newDate, minutesFormat).length;\n selectRange = [hoursLength + sepLength, minutesLength];\n } else if (isSeconds) {\n newDate.setSeconds(seconds + incr * parseInt(options.secondStep, 10));\n // re-calculate seconds length because we have changes seconds value\n secondsLength = formatDate(newDate, secondsFormat).length;\n selectRange = [hoursLength + sepLength + minutesLength + sepLength, secondsLength];\n } else if (isMeridian) {\n if (!lateralMove) $timepicker.switchMeridian();\n selectRange = [hoursLength + sepLength + minutesLength + sepLength + (secondsLength + sepLength) * showSeconds, 2];\n }\n $timepicker.select(newDate, selectedIndex, true);\n createSelection(selectRange[0], selectRange[1]);\n parentScope.$digest();\n };\n\n // Private\n\n function createSelection (start, length) {\n var end = start + length;\n if (element[0].createTextRange) {\n var selRange = element[0].createTextRange();\n selRange.collapse(true);\n selRange.moveStart('character', start);\n selRange.moveEnd('character', end);\n selRange.select();\n } else if (element[0].setSelectionRange) {\n element[0].setSelectionRange(start, end);\n } else if (angular.isUndefined(element[0].selectionStart)) {\n element[0].selectionStart = start;\n element[0].selectionEnd = end;\n }\n }\n\n function focusElement () {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $timepicker.init;\n $timepicker.init = function () {\n if (isNative && options.useNative) {\n element.prop('type', 'time');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $timepicker.destroy;\n $timepicker.destroy = function () {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $timepicker.show;\n $timepicker.show = function () {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n if ($timepicker.$element) $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n if (element) element.on('keydown', $timepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $timepicker.hide;\n $timepicker.hide = function (blur) {\n if (!$timepicker.$isShown) return;\n if ($timepicker.$element) $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n if (element) element.off('keydown', $timepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $timepicker;\n\n }\n\n timepickerFactory.defaults = defaults;\n return timepickerFactory;\n\n };\n\n })\n\n\n .directive('bsTimepicker', function ($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) {\n\n var defaults = $timepicker.defaults;\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent', 'defaultDate'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'roundDisplay'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Initialize timepicker\n if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm';\n var timepicker = $timepicker(element, controller, options);\n options = timepicker.$options;\n\n var lang = options.lang;\n var formatDate = function (date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!timepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i);\n if (newValue === true) {\n timepicker.show();\n } else {\n timepicker.hide();\n }\n });\n }\n\n // Initialize parser\n var dateParser = $dateParser({\n format: options.timeFormat,\n lang: lang\n });\n\n // Observe attributes for changes\n angular.forEach(['minTime', 'maxTime'], function (key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n if (angular.isDefined(attr[key])) {\n attr.$observe(key, function (newValue) {\n timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue);\n if (!isNaN(timepicker.$options[key])) timepicker.$build();\n validateAgainstMinMaxTime(controller.$dateValue);\n });\n }\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue, controller.$dateValue);\n timepicker.update(controller.$dateValue);\n }, true);\n\n function validateAgainstMinMaxTime (parsedTime) {\n if (!angular.isDate(parsedTime)) return;\n var isMinValid = isNaN(options.minTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) >= options.minTime;\n var isMaxValid = isNaN(options.maxTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) <= options.maxTime;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (!isValid) {\n return;\n }\n controller.$dateValue = parsedTime;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function (viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n controller.$setValidity('date', true);\n return null;\n }\n var parsedTime = angular.isDate(viewValue) ? viewValue : dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedTime || isNaN(parsedTime.getTime())) {\n controller.$setValidity('date', false);\n // Return undefined, causes ngModelController to\n // invalidate model value\n return undefined;\n }\n validateAgainstMinMaxTime(parsedTime);\n\n if (options.timeType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true);\n return formatDate(date, options.modelTimeFormat || options.timeFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.timeType === 'number') {\n return date.getTime();\n } else if (options.timeType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.timeType === 'iso') {\n return date.toISOString();\n }\n return new Date(date);\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.timeType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelTimeFormat);\n } else if (options.timeType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if(isNaN(date.getTime())) date = new Date(new Date().setMinutes(0) + 36e5);\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getTimeFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function () {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getTimeFormattedString());\n };\n\n function getTimeFormattedString () {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.timeFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (timepicker) timepicker.destroy();\n options = null;\n timepicker = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.tab', [])\n\n .provider('$tab', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n template: 'tab/tab.tpl.html',\n navClass: 'nav-tabs',\n activeClass: 'active'\n };\n\n var controller = this.controller = function ($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'navClass', 'activeClass'], function (key) {\n if (angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // Publish options on scope\n $scope.$navClass = self.$options.navClass;\n $scope.$activeClass = self.$options.activeClass;\n\n self.$panes = $scope.$panes = [];\n\n // Please use $activePaneChangeListeners if you use `bsActivePane`\n // Because we removed `ngModel` as default, we rename viewChangeListeners to\n // activePaneChangeListeners to make more sense.\n self.$activePaneChangeListeners = self.$viewChangeListeners = [];\n\n self.$push = function (pane) {\n if (angular.isUndefined(self.$panes.$active)) {\n $scope.$setActive(pane.name || 0);\n }\n self.$panes.push(pane);\n };\n\n self.$remove = function (pane) {\n var index = self.$panes.indexOf(pane);\n var active = self.$panes.$active;\n var activeIndex;\n if (angular.isString(active)) {\n activeIndex = self.$panes.map(function (pane) {\n return pane.name;\n }).indexOf(active);\n } else {\n activeIndex = self.$panes.$active;\n }\n\n // remove pane from $panes array\n self.$panes.splice(index, 1);\n\n if (index < activeIndex) {\n // we removed a pane before the active pane, so we need to\n // decrement the active pane index\n activeIndex--;\n } else if (index === activeIndex && activeIndex === self.$panes.length) {\n // we remove the active pane and it was the one at the end,\n // so select the previous one\n activeIndex--;\n }\n if (activeIndex >= 0 && activeIndex < self.$panes.length) {\n self.$setActive(self.$panes[activeIndex].name || activeIndex);\n } else {\n self.$setActive();\n }\n };\n\n self.$setActive = $scope.$setActive = function (value) {\n self.$panes.$active = value;\n self.$activePaneChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n self.$isActive = $scope.$isActive = function ($pane, $index) {\n return self.$panes.$active === $pane.name || self.$panes.$active === $index;\n };\n\n };\n\n this.$get = function () {\n var $tab = {};\n $tab.defaults = defaults;\n $tab.controller = controller;\n return $tab;\n };\n\n })\n\n .directive('bsTabs', function ($window, $animate, $tab, $parse) {\n\n var defaults = $tab.defaults;\n\n return {\n require: ['?ngModel', 'bsTabs'],\n transclude: true,\n scope: true,\n controller: ['$scope', '$element', '$attrs', $tab.controller],\n templateUrl: function (element, attr) {\n return attr.template || defaults.template;\n },\n link: function postLink (scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // 'ngModel' does interfere with form validation\n // and status, use `bsActivePane` instead to avoid it\n if (ngModelCtrl) {\n\n // Update the modelValue following\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n ngModelCtrl.$setViewValue(bsTabsCtrl.$panes.$active);\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n bsTabsCtrl.$setActive(modelValue);\n return modelValue;\n });\n\n }\n\n if (attrs.bsActivePane) {\n // adapted from angularjs ngModelController bindings\n // https://github.com/angular/angular.js/blob/v1.3.1/src%2Fng%2Fdirective%2Finput.js#L1730\n var parsedBsActivePane = $parse(attrs.bsActivePane);\n\n // Update bsActivePane value with change\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n parsedBsActivePane.assign(scope, bsTabsCtrl.$panes.$active);\n });\n\n // watch bsActivePane for value changes\n scope.$watch(attrs.bsActivePane, function (newValue, oldValue) {\n bsTabsCtrl.$setActive(newValue);\n }, true);\n }\n }\n };\n\n })\n\n .directive('bsPane', function ($window, $animate, $sce) {\n\n return {\n require: ['^?ngModel', '^bsTabs'],\n scope: true,\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // Add base class\n element.addClass('tab-pane');\n\n // Observe title attribute for change\n attrs.$observe('title', function (newValue, oldValue) {\n scope.title = $sce.trustAsHtml(newValue);\n });\n\n // Save tab name into scope\n scope.name = attrs.name;\n\n // Add animation class\n if (bsTabsCtrl.$options.animation) {\n element.addClass(bsTabsCtrl.$options.animation);\n }\n\n attrs.$observe('disabled', function (newValue, oldValue) {\n scope.disabled = scope.$eval(newValue);\n });\n\n // Push pane to parent bsTabs controller\n bsTabsCtrl.$push(scope);\n\n // remove pane from tab controller when pane is destroyed\n scope.$on('$destroy', function () {\n bsTabsCtrl.$remove(scope);\n });\n\n function render () {\n var index = bsTabsCtrl.$panes.indexOf(scope);\n $animate[bsTabsCtrl.$isActive(scope, index) ? 'addClass' : 'removeClass'](element, bsTabsCtrl.$options.activeClass);\n }\n\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n render();\n });\n render();\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.select', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$select', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'select',\n prefixEvent: '$select',\n placement: 'bottom-left',\n templateUrl: 'select/select.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n multiple: false,\n allNoneButtons: false,\n sort: true,\n caretHtml: ' ',\n placeholder: 'Choose among the following...',\n allText: 'All',\n noneText: 'None',\n maxLength: 3,\n maxLengthHtml: 'selected',\n iconCheckmark: 'glyphicon glyphicon-ok',\n toggle: false\n };\n\n this.$get = function ($window, $document, $rootScope, $tooltip, $timeout) {\n\n // var bodyEl = angular.element($window.document.body);\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n\n function SelectFactory (element, controller, config) {\n\n var $select = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $select = $tooltip(element, options);\n var scope = $select.$scope;\n\n scope.$matches = [];\n if (options.multiple) {\n scope.$activeIndex = [];\n } else {\n scope.$activeIndex = -1;\n }\n scope.$isMultiple = options.multiple;\n scope.$showAllNoneButtons = options.allNoneButtons && options.multiple;\n scope.$iconCheckmark = options.iconCheckmark;\n scope.$allText = options.allText;\n scope.$noneText = options.noneText;\n\n scope.$activate = function (index) {\n scope.$$postDigest(function () {\n $select.activate(index);\n });\n };\n\n scope.$select = function (index, evt) {\n scope.$$postDigest(function () {\n $select.select(index);\n });\n };\n\n scope.$isVisible = function () {\n return $select.$isVisible();\n };\n\n scope.$isActive = function (index) {\n return $select.$isActive(index);\n };\n\n scope.$selectAll = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (!scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n scope.$selectNone = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n // Public methods\n\n $select.update = function (matches) {\n scope.$matches = matches;\n $select.$updateActiveIndex();\n };\n\n $select.activate = function (index) {\n if (options.multiple) {\n if ($select.$isActive(index)) {\n scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1);\n } else {\n scope.$activeIndex.push(index);\n }\n if (options.sort) scope.$activeIndex.sort(function (a, b) { return a - b; }); // use numeric sort instead of default sort\n } else {\n scope.$activeIndex = index;\n }\n return scope.$activeIndex;\n };\n\n $select.select = function (index) {\n if (angular.isUndefined(index) || index < 0 || index >= scope.$matches.length) { return; }\n var value = scope.$matches[index].value;\n scope.$apply(function () {\n $select.activate(index);\n if (options.multiple) {\n controller.$setViewValue(scope.$activeIndex.map(function (index) {\n if (angular.isUndefined(scope.$matches[index])) {\n return null;\n }\n return scope.$matches[index].value;\n }));\n } else {\n if (options.toggle) {\n controller.$setViewValue((value === controller.$modelValue) ? undefined : value);\n } else {\n controller.$setViewValue(value);\n }\n // Hide if single select\n $select.hide();\n }\n });\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $select);\n if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) {\n options.onSelect(value, index, $select);\n }\n };\n\n // Protected methods\n\n $select.$updateActiveIndex = function () {\n if (options.multiple) {\n if (angular.isArray(controller.$modelValue)) {\n scope.$activeIndex = controller.$modelValue.map(function (value) {\n return $select.$getIndex(value);\n });\n } else {\n scope.$activeIndex = [];\n }\n } else {\n if (angular.isDefined(controller.$modelValue) && scope.$matches.length) {\n scope.$activeIndex = $select.$getIndex(controller.$modelValue);\n } else {\n scope.$activeIndex = -1;\n }\n }\n };\n\n $select.$isVisible = function () {\n if (!options.minLength || !controller) {\n return scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && controller.$viewValue.length >= options.minLength;\n };\n\n $select.$isActive = function (index) {\n if (options.multiple) {\n return scope.$activeIndex.indexOf(index) !== -1;\n }\n return scope.$activeIndex === index;\n };\n\n $select.$getIndex = function (value) {\n var index;\n for (index = scope.$matches.length; index--;) {\n if (angular.equals(scope.$matches[index].value, value)) break;\n }\n return index;\n };\n\n $select.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n var anchor;\n\n if (evt.target.nodeName !== 'A') {\n var anchorCandidate = targetEl.parent();\n while (!anchor && anchorCandidate.length > 0) {\n if (anchorCandidate[0].nodeName === 'A') {\n anchor = anchorCandidate;\n }\n anchorCandidate = anchorCandidate.parent();\n }\n }\n\n if (anchor) {\n angular.element(anchor).triggerHandler('click');\n } else {\n targetEl.triggerHandler('click');\n }\n }\n };\n\n $select.$onKeyDown = function (evt) {\n if (!/(9|13|38|40)/.test(evt.keyCode)) return;\n // Let tab propagate\n if (evt.keyCode !== 9) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // release focus on tab\n if (options.multiple && evt.keyCode === 9) {\n return $select.hide();\n }\n\n // Select with enter\n if (!options.multiple && (evt.keyCode === 13 || evt.keyCode === 9)) {\n return $select.select(scope.$activeIndex);\n }\n\n if (!options.multiple) {\n // Navigate with keyboard\n if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if (evt.keyCode === 38 && scope.$activeIndex < 0) scope.$activeIndex = scope.$matches.length - 1;\n else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n }\n };\n\n $select.$isIE = function () {\n var ua = $window.navigator.userAgent;\n return ua.indexOf('MSIE ') > 0 || ua.indexOf('Trident/') > 0 || ua.indexOf('Edge/') > 0;\n };\n\n $select.$selectScrollFix = function (e) {\n if ($document[0].activeElement.tagName === 'UL') {\n e.preventDefault();\n e.stopImmediatePropagation();\n e.target.focus();\n }\n };\n\n // Overrides\n\n var _show = $select.show;\n $select.show = function () {\n _show();\n if (options.multiple) {\n $select.$element.addClass('select-multiple');\n }\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n $select.$element.on(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $select.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $select.hide;\n $select.hide = function () {\n if (!options.multiple && angular.isUndefined(controller.$modelValue)) {\n scope.$activeIndex = -1;\n }\n $select.$element.off(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $select.$onKeyDown);\n }\n _hide(true);\n };\n\n return $select;\n\n }\n\n SelectFactory.defaults = defaults;\n return SelectFactory;\n\n };\n\n })\n\n .directive('bsSelect', function ($window, $parse, $q, $select, $parseOptions) {\n\n var defaults = $select.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope, placeholder: defaults.placeholder};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent', 'toggle'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'allNoneButtons', 'sort'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show, hide and select events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Only parse data-multiple. Angular sets existence attributes to true (multiple/required/etc), they apply this\n // to data-multiple as well for some reason, so we'll parse this ourselves and disregard multiple\n var dataMultiple = element.attr('data-multiple');\n if (angular.isDefined(dataMultiple)) {\n if (falseValueRegExp.test(dataMultiple)) {\n options.multiple = false;\n } else {\n options.multiple = dataMultiple;\n }\n }\n\n // Add support for select markup\n if (element[0].nodeName.toLowerCase() === 'select') {\n var inputEl = element;\n inputEl.css('display', 'none');\n element = angular.element('');\n inputEl.after(element);\n }\n\n // Build proper bsOptions\n var parsedOptions = $parseOptions(attr.bsOptions);\n\n // Initialize select\n var select = $select(element, controller, options);\n\n if (select.$isIE()) {\n element[0].addEventListener('blur', select.$selectScrollFix);\n }\n\n // Watch bsOptions values before filtering for changes\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').trim();\n scope.$watch(watchedOptions, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller)\n .then(function (values) {\n select.update(values);\n controller.$render();\n });\n }, true);\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue);\n select.$updateActiveIndex();\n controller.$render();\n }, true);\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var selected;\n var index;\n if (options.multiple && angular.isArray(controller.$modelValue)) {\n selected = controller.$modelValue.map(function (value) {\n index = select.$getIndex(value);\n return index !== -1 ? select.$scope.$matches[index].label : false;\n }).filter(angular.isDefined);\n if (selected.length > (options.maxLength || defaults.maxLength)) {\n selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml);\n } else {\n selected = selected.join(', ');\n }\n } else {\n index = select.$getIndex(controller.$modelValue);\n selected = index !== -1 ? select.$scope.$matches[index].label : false;\n }\n element.html((selected || options.placeholder) + (options.caretHtml || defaults.caretHtml));\n };\n\n if (options.multiple) {\n controller.$isEmpty = function (value) {\n return !value || value.length === 0;\n };\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (select) select.destroy();\n options = null;\n select = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.scrollspy', ['mgcrea.ngStrap.helpers.debounce', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$scrollspy', function () {\n\n // Pool of registered spies\n var spies = this.$$spies = {};\n\n var defaults = this.defaults = {\n debounce: 150,\n throttle: 100,\n offset: 100\n };\n\n this.$get = function ($window, $document, $rootScope, dimensions, debounce, throttle) {\n\n var windowEl = angular.element($window);\n var docEl = angular.element($document.prop('documentElement'));\n var bodyEl = angular.element($window.document.body);\n\n // Helper functions\n\n function nodeName (element, name) {\n return element[0].nodeName && element[0].nodeName.toLowerCase() === name.toLowerCase();\n }\n\n function ScrollSpyFactory (config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n if (!options.element) options.element = bodyEl;\n var isWindowSpy = nodeName(options.element, 'body');\n var scrollEl = isWindowSpy ? windowEl : options.element;\n var scrollId = isWindowSpy ? 'window' : options.id;\n\n // Use existing spy\n if (spies[scrollId]) {\n spies[scrollId].$$count++;\n return spies[scrollId];\n }\n\n var $scrollspy = {};\n\n // Private vars\n var unbindViewContentLoaded;\n var unbindIncludeContentLoaded;\n var trackedElements = $scrollspy.$trackedElements = [];\n var sortedElements = [];\n var activeTarget;\n var debouncedCheckPosition;\n var throttledCheckPosition;\n var debouncedCheckOffsets;\n /* eslint-disable no-unused-vars */\n var viewportHeight;\n /* eslint-enable no-unused-vars */\n var scrollTop;\n\n $scrollspy.init = function () {\n\n // Setup internal ref counter\n this.$$count = 1;\n\n // Bind events\n debouncedCheckPosition = debounce(this.checkPosition, options.debounce);\n throttledCheckPosition = throttle(this.checkPosition, options.throttle);\n scrollEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', debouncedCheckPosition);\n scrollEl.on('scroll', throttledCheckPosition);\n\n debouncedCheckOffsets = debounce(this.checkOffsets, options.debounce);\n unbindViewContentLoaded = $rootScope.$on('$viewContentLoaded', debouncedCheckOffsets);\n unbindIncludeContentLoaded = $rootScope.$on('$includeContentLoaded', debouncedCheckOffsets);\n debouncedCheckOffsets();\n\n // Register spy for reuse\n if (scrollId) {\n spies[scrollId] = $scrollspy;\n }\n\n };\n\n $scrollspy.destroy = function () {\n\n // Check internal ref counter\n this.$$count--;\n if (this.$$count > 0) {\n return;\n }\n\n // Unbind events\n scrollEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', debouncedCheckPosition);\n scrollEl.off('scroll', throttledCheckPosition);\n unbindViewContentLoaded();\n unbindIncludeContentLoaded();\n if (scrollId) {\n delete spies[scrollId];\n }\n };\n\n $scrollspy.checkPosition = function () {\n\n // Not ready yet\n if (!sortedElements.length) return;\n\n // Calculate the scroll position\n scrollTop = (isWindowSpy ? $window.pageYOffset : scrollEl.prop('scrollTop')) || 0;\n\n // Calculate the viewport height for use by the components\n viewportHeight = Math.max($window.innerHeight, docEl.prop('clientHeight'));\n\n // Activate first element if scroll is smaller\n if (scrollTop < sortedElements[0].offsetTop && activeTarget !== sortedElements[0].target) {\n return $scrollspy.$activateElement(sortedElements[0]);\n }\n\n // Activate proper element\n for (var i = sortedElements.length; i--;) {\n if (angular.isUndefined(sortedElements[i].offsetTop) || sortedElements[i].offsetTop === null) continue;\n if (activeTarget === sortedElements[i].target) continue;\n if (scrollTop < sortedElements[i].offsetTop) continue;\n if (sortedElements[i + 1] && scrollTop > sortedElements[i + 1].offsetTop) continue;\n return $scrollspy.$activateElement(sortedElements[i]);\n }\n\n };\n\n $scrollspy.checkPositionWithEventLoop = function () {\n // IE 9 throws an error if we use 'this' instead of '$scrollspy'\n // in this setTimeout call\n setTimeout($scrollspy.checkPosition, 1);\n };\n\n // Protected methods\n\n $scrollspy.$activateElement = function (element) {\n if (activeTarget) {\n var activeElement = $scrollspy.$getTrackedElement(activeTarget);\n if (activeElement) {\n activeElement.source.removeClass('active');\n if (nodeName(activeElement.source, 'li') && nodeName(activeElement.source.parent().parent(), 'li')) {\n activeElement.source.parent().parent().removeClass('active');\n }\n }\n }\n activeTarget = element.target;\n element.source.addClass('active');\n if (nodeName(element.source, 'li') && nodeName(element.source.parent().parent(), 'li')) {\n element.source.parent().parent().addClass('active');\n }\n };\n\n $scrollspy.$getTrackedElement = function (target) {\n return trackedElements.filter(function (obj) {\n return obj.target === target;\n })[0];\n };\n\n // Track offsets behavior\n\n $scrollspy.checkOffsets = function () {\n\n angular.forEach(trackedElements, function (trackedElement) {\n var targetElement = document.querySelector(trackedElement.target);\n trackedElement.offsetTop = targetElement ? dimensions.offset(targetElement).top : null;\n if (options.offset && trackedElement.offsetTop !== null) trackedElement.offsetTop -= options.offset * 1;\n });\n\n sortedElements = trackedElements\n .filter(function (el) {\n return el.offsetTop !== null;\n })\n .sort(function (a, b) {\n return a.offsetTop - b.offsetTop;\n });\n\n debouncedCheckPosition();\n\n };\n\n $scrollspy.trackElement = function (target, source) {\n trackedElements.push({target: target, source: source});\n };\n\n $scrollspy.untrackElement = function (target, source) {\n var toDelete;\n for (var i = trackedElements.length; i--;) {\n if (trackedElements[i].target === target && trackedElements[i].source === source) {\n toDelete = i;\n break;\n }\n }\n trackedElements.splice(toDelete, 1);\n };\n\n $scrollspy.activate = function (i) {\n trackedElements[i].addClass('active');\n };\n\n // Initialize plugin\n\n $scrollspy.init();\n return $scrollspy;\n\n }\n\n return ScrollSpyFactory;\n\n };\n\n })\n\n .directive('bsScrollspy', function ($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'EAC',\n link: function postLink (scope, element, attr) {\n\n var options = {scope: scope};\n angular.forEach(['offset', 'target'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n var scrollspy = $scrollspy(options);\n scrollspy.trackElement(options.target, element);\n\n scope.$on('$destroy', function () {\n if (scrollspy) {\n scrollspy.untrackElement(options.target, element);\n scrollspy.destroy();\n }\n options = null;\n scrollspy = null;\n });\n\n }\n };\n\n })\n\n\n .directive('bsScrollspyList', function ($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'A',\n compile: function postLink (element, attr) {\n var children = element[0].querySelectorAll('li > a[href]');\n angular.forEach(children, function (child) {\n var childEl = angular.element(child);\n childEl.parent().attr('bs-scrollspy', '').attr('data-target', childEl.attr('href'));\n });\n }\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.popover', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$popover', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n // uncommenting the next two lines will break backwards compatability\n // prefixClass: 'popover',\n // prefixEvent: 'popover',\n container: false,\n target: false,\n placement: 'right',\n templateUrl: 'popover/popover.tpl.html',\n contentTemplate: false,\n trigger: 'click',\n keyboard: true,\n html: false,\n title: '',\n content: '',\n delay: 0,\n autoClose: false\n };\n\n this.$get = function ($tooltip) {\n\n function PopoverFactory (element, config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n var $popover = $tooltip(element, options);\n\n // Support scope as string options [/*title, */content]\n if (options.content) {\n $popover.$scope.content = options.content;\n }\n\n return $popover;\n\n }\n\n return PopoverFactory;\n\n };\n\n })\n\n .directive('bsPopover', function ($window, $sce, $popover) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr) {\n\n var popover;\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent', 'bsEnabled'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoClose'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if (angular.isDefined(dataTarget)) {\n if (falseValueRegExp.test(dataTarget)) {\n options.target = false;\n } else {\n options.target = dataTarget;\n }\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n if (angular.isDefined(oldValue)) {\n requestAnimationFrame(function () {\n if (popover) popover.$applyPlacement();\n });\n }\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsPopover) {\n scope.$watch(attr.bsPopover, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n if (angular.isDefined(oldValue)) {\n requestAnimationFrame(function () {\n if (popover) popover.$applyPlacement();\n });\n }\n }, true);\n }\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i);\n if (newValue === true) {\n popover.show();\n } else {\n popover.hide();\n }\n });\n }\n\n // Enabled binding support\n if (attr.bsEnabled) {\n scope.$watch(attr.bsEnabled, function (newValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(popover),?/i);\n if (newValue === false) {\n popover.setEnabled(false);\n } else {\n popover.setEnabled(true);\n }\n });\n }\n\n // Viewport support\n if (attr.viewport) {\n scope.$watch(attr.viewport, function (newValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n popover.setViewport(newValue);\n });\n }\n\n // Initialize popover\n popover = $popover(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (popover) popover.destroy();\n options = null;\n popover = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.navbar', [])\n\n .provider('$navbar', function () {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n routeAttr: 'data-match-route',\n strict: false\n };\n\n this.$get = function () {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsNavbar', function ($window, $location, $navbar) {\n\n var defaults = $navbar.defaults;\n\n return {\n restrict: 'A',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = angular.copy(defaults);\n angular.forEach(Object.keys(defaults), function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Watch for the $location\n scope.$watch(function () {\n\n return $location.path();\n\n }, function (newValue, oldValue) {\n\n var liElements = element[0].querySelectorAll('li[' + options.routeAttr + ']');\n\n angular.forEach(liElements, function (li) {\n\n var liElement = angular.element(li);\n var pattern = liElement.attr(options.routeAttr).replace('/', '\\\\/');\n if (options.strict) {\n pattern = '^' + pattern + '$';\n }\n var regexp = new RegExp(pattern, 'i');\n\n if (regexp.test(newValue)) {\n liElement.addClass(options.activeClass);\n } else {\n liElement.removeClass(options.activeClass);\n }\n\n });\n\n });\n\n }\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.modal', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$modal', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n backdropAnimation: 'am-fade',\n customClass: '',\n prefixClass: 'modal',\n prefixEvent: 'modal',\n placement: 'top',\n templateUrl: 'modal/modal.tpl.html',\n template: '',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true,\n size: null,\n zIndex: null\n };\n\n this.$get = function ($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) {\n\n var forEach = angular.forEach;\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n var bodyElement = angular.element($window.document.body);\n\n var backdropCount = 0;\n var dialogBaseZindex = 1050;\n var backdropBaseZindex = 1040;\n\n var validSizes = {\n lg: 'modal-lg',\n sm: 'modal-sm'\n };\n\n function ModalFactory (config) {\n\n var $modal = {};\n\n // Common vars\n var options = $modal.$options = angular.extend({}, defaults, config);\n var promise = $modal.$promise = $bsCompiler.compile(options);\n var scope = $modal.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n if (!options.element && !options.container) {\n options.container = 'body';\n }\n\n if (options.zIndex) {\n dialogBaseZindex = parseInt(options.zIndex, 10);\n backdropBaseZindex = dialogBaseZindex - 10;\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $modal.$id = options.id || options.element && options.element.attr('id') || '';\n\n // Support scope as string options\n forEach(['title', 'content'], function (key) {\n if (options[key]) scope[key] = $sce.trustAsHtml(options[key]);\n });\n\n // Provide scope helpers\n scope.$hide = function () {\n scope.$$postDigest(function () {\n $modal.hide();\n });\n };\n scope.$show = function () {\n scope.$$postDigest(function () {\n $modal.show();\n });\n };\n scope.$toggle = function () {\n scope.$$postDigest(function () {\n $modal.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $modal.$isShown = scope.$isShown = false;\n\n // Fetch, compile then initialize modal\n var compileData;\n var modalElement;\n var modalScope;\n var backdropElement = angular.element('
');\n backdropElement.css({position: 'fixed', top: '0px', left: '0px', bottom: '0px', right: '0px'});\n promise.then(function (data) {\n compileData = data;\n $modal.init();\n });\n\n $modal.init = function () {\n\n // Options: show\n if (options.show) {\n scope.$$postDigest(function () {\n $modal.show();\n });\n }\n\n };\n\n $modal.destroy = function () {\n\n // Remove element\n destroyModalElement();\n\n // remove backdrop element\n if (backdropElement) {\n backdropElement.remove();\n backdropElement = null;\n }\n\n // Destroy scope\n scope.$destroy();\n };\n\n $modal.show = function () {\n if ($modal.$isShown) return;\n\n var parent;\n var after;\n if (angular.isElement(options.container)) {\n parent = options.container;\n after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null;\n } else {\n if (options.container) {\n parent = findElement(options.container);\n after = parent[0] && parent[0].lastChild ? angular.element(parent[0].lastChild) : null;\n } else {\n parent = null;\n after = options.element;\n }\n }\n\n // destroy any existing modal elements\n if (modalElement) destroyModalElement();\n\n // create a new scope, so we can destroy it and all child scopes\n // when destroying the modal element\n modalScope = $modal.$scope.$new();\n // Fetch a cloned element linked from template (noop callback is required)\n modalElement = $modal.$element = compileData.link(modalScope, function (clonedElement, scope) {});\n\n if (options.backdrop) {\n // set z-index\n modalElement.css({'z-index': dialogBaseZindex + (backdropCount * 20)});\n backdropElement.css({'z-index': backdropBaseZindex + (backdropCount * 20)});\n\n // increment number of backdrops\n backdropCount++;\n }\n\n if (scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) {\n return;\n }\n if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) {\n options.onBeforeShow($modal);\n }\n\n // Set the initial positioning.\n modalElement.css({display: 'block'}).addClass(options.placement);\n\n // Options: customClass\n if (options.customClass) {\n modalElement.addClass(options.customClass);\n }\n\n // Options: size\n if (options.size && validSizes[options.size]) {\n angular.element(findElement('.modal-dialog', modalElement[0])).addClass(validSizes[options.size]);\n }\n\n // Options: animation\n if (options.animation) {\n if (options.backdrop) {\n backdropElement.addClass(options.backdropAnimation);\n }\n modalElement.addClass(options.animation);\n }\n\n if (options.backdrop) {\n $animate.enter(backdropElement, bodyElement, null);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.enter(modalElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(modalElement, parent, after).then(enterAnimateCallback);\n }\n\n $modal.$isShown = scope.$isShown = true;\n safeDigest(scope);\n // Focus once the enter-animation has started\n // Weird PhantomJS bug hack\n var el = modalElement[0];\n requestAnimationFrame(function () {\n el.focus();\n });\n\n bodyElement.addClass(options.prefixClass + '-open');\n if (options.animation) {\n bodyElement.addClass(options.prefixClass + '-with-' + options.animation);\n }\n\n // Bind events\n bindBackdropEvents();\n bindKeyboardEvents();\n };\n\n function enterAnimateCallback () {\n scope.$emit(options.prefixEvent + '.show', $modal);\n if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) {\n options.onShow($modal);\n }\n }\n\n $modal.hide = function () {\n if (!$modal.$isShown) return;\n\n if (scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) {\n return;\n }\n if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) {\n options.onBeforeHide($modal);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.leave(modalElement, leaveAnimateCallback);\n } else {\n $animate.leave(modalElement).then(leaveAnimateCallback);\n }\n\n if (options.backdrop) {\n // decrement number of backdrops\n backdropCount--;\n $animate.leave(backdropElement);\n }\n $modal.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n };\n\n function leaveAnimateCallback () {\n scope.$emit(options.prefixEvent + '.hide', $modal);\n if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) {\n options.onHide($modal);\n }\n if (findElement('.modal').length <= 0) {\n bodyElement.removeClass(options.prefixClass + '-open');\n }\n if (options.animation) {\n bodyElement.removeClass(options.prefixClass + '-with-' + options.animation);\n }\n }\n\n $modal.toggle = function () {\n if ($modal.$isShown) {\n $modal.hide();\n } else {\n $modal.show();\n }\n };\n\n $modal.focus = function () {\n modalElement[0].focus();\n };\n\n // Protected methods\n\n $modal.$onKeyUp = function (evt) {\n\n if (evt.which === 27 && $modal.$isShown) {\n $modal.hide();\n evt.stopPropagation();\n }\n\n };\n\n function bindBackdropEvents () {\n if (options.backdrop) {\n modalElement.on('click', hideOnBackdropClick);\n backdropElement.on('click', hideOnBackdropClick);\n backdropElement.on('wheel', preventEventDefault);\n }\n }\n\n function unbindBackdropEvents () {\n if (options.backdrop) {\n modalElement.off('click', hideOnBackdropClick);\n backdropElement.off('click', hideOnBackdropClick);\n backdropElement.off('wheel', preventEventDefault);\n }\n }\n\n function bindKeyboardEvents () {\n if (options.keyboard) {\n modalElement.on('keyup', $modal.$onKeyUp);\n }\n }\n\n function unbindKeyboardEvents () {\n if (options.keyboard) {\n modalElement.off('keyup', $modal.$onKeyUp);\n }\n }\n\n // Private helpers\n\n function hideOnBackdropClick (evt) {\n if (evt.target !== evt.currentTarget) return;\n if (options.backdrop === 'static') {\n $modal.focus();\n } else {\n $modal.hide();\n }\n }\n\n function preventEventDefault (evt) {\n evt.preventDefault();\n }\n\n function destroyModalElement () {\n if ($modal.$isShown && modalElement !== null) {\n // un-bind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n }\n\n if (modalScope) {\n modalScope.$destroy();\n modalScope = null;\n }\n\n if (modalElement) {\n modalElement.remove();\n modalElement = $modal.$element = null;\n }\n }\n\n return $modal;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return ModalFactory;\n\n };\n\n })\n\n .directive('bsModal', function ($window, $sce, $parse, $modal) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass', 'customClass', 'modalClass', 'size', 'zIndex'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Options: alias modalClass to customClass\n if (options.modalClass) {\n options.customClass = options.modalClass;\n }\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsModal) {\n scope.$watch(attr.bsModal, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize modal\n var modal = $modal(options);\n\n // Trigger\n element.on(attr.trigger || 'click', modal.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (modal) modal.destroy();\n options = null;\n modal = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nif (angular.version.minor < 3 && angular.version.dot < 14) {\n angular.module('ng')\n\n .factory('$$rAF', function ($window, $timeout) {\n\n var requestAnimationFrame = $window.requestAnimationFrame ||\n $window.webkitRequestAnimationFrame ||\n $window.mozRequestAnimationFrame;\n\n var cancelAnimationFrame = $window.cancelAnimationFrame ||\n $window.webkitCancelAnimationFrame ||\n $window.mozCancelAnimationFrame ||\n $window.webkitCancelRequestAnimationFrame;\n\n var rafSupported = !!requestAnimationFrame;\n var raf = rafSupported ?\n function (fn) {\n var id = requestAnimationFrame(fn);\n return function () {\n cancelAnimationFrame(id);\n };\n } :\n function (fn) {\n var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666\n return function () {\n $timeout.cancel(timer);\n };\n };\n\n raf.supported = rafSupported;\n\n return raf;\n\n });\n}\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.parseOptions', [])\n\n .provider('$parseOptions', function () {\n\n var defaults = this.defaults = {\n regexp: /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n };\n\n this.$get = function ($parse, $q) {\n\n function ParseOptionsFactory (attr, config) {\n\n var $parseOptions = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n $parseOptions.$values = [];\n\n // Private vars\n var match;\n var displayFn;\n var valueName;\n /* eslint-disable no-unused-vars */\n var keyName;\n var groupByFn;\n /* eslint-enable no-unused-vars */\n var valueFn;\n var valuesFn;\n\n $parseOptions.init = function () {\n $parseOptions.$match = match = attr.match(options.regexp);\n displayFn = $parse(match[2] || match[1]);\n valueName = match[4] || match[6];\n keyName = match[5];\n groupByFn = $parse(match[3] || '');\n valueFn = $parse(match[2] ? match[1] : valueName);\n valuesFn = $parse(match[7]);\n };\n\n $parseOptions.valuesFn = function (scope, controller) {\n return $q.when(valuesFn(scope, controller))\n .then(function (values) {\n if (!angular.isArray(values)) {\n values = [];\n }\n $parseOptions.$values = values.length ? parseValues(values, scope) : [];\n return $parseOptions.$values;\n });\n };\n\n $parseOptions.displayValue = function (modelValue) {\n var scope = {};\n scope[valueName] = modelValue;\n return displayFn(scope);\n };\n\n // Private functions\n\n function parseValues (values, scope) {\n return values.map(function (match, index) {\n var locals = {};\n var label;\n var value;\n locals[valueName] = match;\n label = displayFn(scope, locals);\n value = valueFn(scope, locals);\n return {label: label, value: value, index: index};\n });\n }\n\n $parseOptions.init();\n return $parseOptions;\n\n }\n\n return ParseOptionsFactory;\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dimensions', [])\n\n .factory('dimensions', function () {\n\n var fn = {};\n\n /**\n * Test the element nodeName\n * @param element\n * @param name\n */\n var nodeName = fn.nodeName = function (element, name) {\n return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase();\n };\n\n /**\n * Returns the element computed style\n * @param element\n * @param prop\n * @param extra\n */\n fn.css = function (element, prop, extra) {\n var value;\n if (element.currentStyle) { // IE\n value = element.currentStyle[prop];\n } else if (window.getComputedStyle) {\n value = window.getComputedStyle(element)[prop];\n } else {\n value = element.style[prop];\n }\n return extra === true ? parseFloat(value) || 0 : value;\n };\n\n /**\n * Provides read-only equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.offset = function (element) {\n var boxRect = element.getBoundingClientRect();\n var docElement = element.ownerDocument;\n return {\n width: boxRect.width || element.offsetWidth,\n height: boxRect.height || element.offsetHeight,\n top: boxRect.top + (window.pageYOffset || docElement.documentElement.scrollTop) - (docElement.documentElement.clientTop || 0),\n left: boxRect.left + (window.pageXOffset || docElement.documentElement.scrollLeft) - (docElement.documentElement.clientLeft || 0)\n };\n };\n\n /**\n * Provides set equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip\n * @url http://api.jquery.com/offset/\n * @param element\n * @param options\n * @param i\n */\n fn.setOffset = function (element, options, i) {\n var curPosition;\n var curLeft;\n var curCSSTop;\n var curTop;\n var curOffset;\n var curCSSLeft;\n var calculatePosition;\n var position = fn.css(element, 'position');\n var curElem = angular.element(element);\n var props = {};\n\n // Set position first, in-case top/left are set even on static elem\n if (position === 'static') {\n element.style.position = 'relative';\n }\n\n curOffset = fn.offset(element);\n curCSSTop = fn.css(element, 'top');\n curCSSLeft = fn.css(element, 'left');\n calculatePosition = (position === 'absolute' || position === 'fixed') &&\n (curCSSTop + curCSSLeft).indexOf('auto') > -1;\n\n // Need to be able to calculate position if either\n // top or left is auto and position is either absolute or fixed\n if (calculatePosition) {\n curPosition = fn.position(element);\n curTop = curPosition.top;\n curLeft = curPosition.left;\n } else {\n curTop = parseFloat(curCSSTop) || 0;\n curLeft = parseFloat(curCSSLeft) || 0;\n }\n\n if (angular.isFunction(options)) {\n options = options.call(element, i, curOffset);\n }\n\n if (options.top !== null) {\n props.top = (options.top - curOffset.top) + curTop;\n }\n if (options.left !== null) {\n props.left = (options.left - curOffset.left) + curLeft;\n }\n\n if ('using' in options) {\n options.using.call(curElem, props);\n } else {\n curElem.css({\n top: props.top + 'px',\n left: props.left + 'px'\n });\n }\n };\n\n /**\n * Provides read-only equivalent of jQuery's position function\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.position = function (element) {\n\n var offsetParentRect = {top: 0, left: 0};\n var offsetParentEl;\n var offset;\n\n // Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is it's only offset parent\n if (fn.css(element, 'position') === 'fixed') {\n\n // We assume that getBoundingClientRect is available when computed position is fixed\n offset = element.getBoundingClientRect();\n\n } else {\n\n // Get *real* offsetParentEl\n offsetParentEl = offsetParentElement(element);\n\n // Get correct offsets\n offset = fn.offset(element);\n if (!nodeName(offsetParentEl, 'html')) {\n offsetParentRect = fn.offset(offsetParentEl);\n }\n\n // Add offsetParent borders\n offsetParentRect.top += fn.css(offsetParentEl, 'borderTopWidth', true);\n offsetParentRect.left += fn.css(offsetParentEl, 'borderLeftWidth', true);\n }\n\n // Subtract parent offsets and element margins\n return {\n width: element.offsetWidth,\n height: element.offsetHeight,\n top: offset.top - offsetParentRect.top - fn.css(element, 'marginTop', true),\n left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true)\n };\n\n };\n\n /**\n * Returns the closest, non-statically positioned offsetParent of a given element\n * @required-by fn.position\n * @param element\n */\n function offsetParentElement (element) {\n var docElement = element.ownerDocument;\n var offsetParent = element.offsetParent || docElement;\n if (nodeName(offsetParent, '#document')) return docElement.documentElement;\n while (offsetParent && !nodeName(offsetParent, 'html') && fn.css(offsetParent, 'position') === 'static') {\n offsetParent = offsetParent.offsetParent;\n }\n return offsetParent || docElement.documentElement;\n }\n\n /**\n * Provides equivalent of jQuery's height function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/height/\n * @param element\n * @param outer\n */\n fn.height = function (element, outer) {\n var value = element.offsetHeight;\n if (outer) {\n value += fn.css(element, 'marginTop', true) + fn.css(element, 'marginBottom', true);\n } else {\n value -= fn.css(element, 'paddingTop', true) + fn.css(element, 'paddingBottom', true) + fn.css(element, 'borderTopWidth', true) + fn.css(element, 'borderBottomWidth', true);\n }\n return value;\n };\n\n /**\n * Provides equivalent of jQuery's width function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/width/\n * @param element\n * @param outer\n */\n fn.width = function (element, outer) {\n var value = element.offsetWidth;\n if (outer) {\n value += fn.css(element, 'marginLeft', true) + fn.css(element, 'marginRight', true);\n } else {\n value -= fn.css(element, 'paddingLeft', true) + fn.css(element, 'paddingRight', true) + fn.css(element, 'borderLeftWidth', true) + fn.css(element, 'borderRightWidth', true);\n }\n return value;\n };\n\n return fn;\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.debounce', [])\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L693\n.factory('debounce', function ($timeout) {\n return function (func, wait, immediate) {\n var timeout = null;\n return function () {\n var context = this;\n var args = arguments;\n var callNow = immediate && !timeout;\n if (timeout) {\n $timeout.cancel(timeout);\n }\n timeout = $timeout(function later () {\n timeout = null;\n if (!immediate) {\n func.apply(context, args);\n }\n }, wait, false);\n if (callNow) {\n func.apply(context, args);\n }\n return timeout;\n };\n };\n})\n\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L661\n.factory('throttle', function ($timeout) {\n return function (func, wait, options) {\n var timeout = null;\n if (!options) options = {};\n return function () {\n var context = this;\n var args = arguments;\n if (!timeout) {\n if (options.leading !== false) {\n func.apply(context, args);\n }\n timeout = $timeout(function later () {\n timeout = null;\n if (options.trailing !== false) {\n func.apply(context, args);\n }\n }, wait, false);\n }\n };\n };\n});\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateParser', [])\n\n.provider('$dateParser', function ($localeProvider) {\n\n // define a custom ParseDate object to use instead of native Date\n // to avoid date values wrapping when setting date component values\n function ParseDate () {\n this.year = 1970;\n this.month = 0;\n this.day = 1;\n this.hours = 0;\n this.minutes = 0;\n this.seconds = 0;\n this.milliseconds = 0;\n }\n\n ParseDate.prototype.setMilliseconds = function (value) { this.milliseconds = value; };\n ParseDate.prototype.setSeconds = function (value) { this.seconds = value; };\n ParseDate.prototype.setMinutes = function (value) { this.minutes = value; };\n ParseDate.prototype.setHours = function (value) { this.hours = value; };\n ParseDate.prototype.getHours = function () { return this.hours; };\n ParseDate.prototype.setDate = function (value) { this.day = value; };\n ParseDate.prototype.setMonth = function (value) { this.month = value; };\n ParseDate.prototype.setFullYear = function (value) { this.year = value; };\n ParseDate.prototype.fromDate = function (value) {\n this.year = value.getFullYear();\n this.month = value.getMonth();\n this.day = value.getDate();\n this.hours = value.getHours();\n this.minutes = value.getMinutes();\n this.seconds = value.getSeconds();\n this.milliseconds = value.getMilliseconds();\n return this;\n };\n\n ParseDate.prototype.toDate = function () {\n return new Date(this.year, this.month, this.day, this.hours, this.minutes, this.seconds, this.milliseconds);\n };\n\n var proto = ParseDate.prototype;\n\n function noop () {\n }\n\n function isNumeric (n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n\n function indexOfCaseInsensitive (array, value) {\n var len = array.length;\n var str = value.toString().toLowerCase();\n for (var i = 0; i < len; i++) {\n if (array[i].toLowerCase() === str) { return i; }\n }\n return -1; // Return -1 per the \"Array.indexOf()\" method.\n }\n\n var defaults = this.defaults = {\n format: 'shortDate',\n strict: false\n };\n\n this.$get = function ($locale, dateFilter) {\n\n var DateParserFactory = function (config) {\n\n var options = angular.extend({}, defaults, config);\n\n var $dateParser = {};\n\n /* eslint-disable key-spacing, quote-props */\n var regExpMap = {\n 'sss' : '[0-9]{3}',\n 'ss' : '[0-5][0-9]',\n 's' : options.strict ? '[1-5]?[0-9]' : '[0-9]|[0-5][0-9]',\n 'mm' : '[0-5][0-9]',\n 'm' : options.strict ? '[1-5]?[0-9]' : '[0-9]|[0-5][0-9]',\n 'HH' : '[01][0-9]|2[0-3]',\n 'H' : options.strict ? '1?[0-9]|2[0-3]' : '[01]?[0-9]|2[0-3]',\n 'hh' : '[0][1-9]|[1][012]',\n 'h' : options.strict ? '[1-9]|1[012]' : '0?[1-9]|1[012]',\n 'a' : 'AM|PM',\n 'EEEE' : $locale.DATETIME_FORMATS.DAY.join('|'),\n 'EEE' : $locale.DATETIME_FORMATS.SHORTDAY.join('|'),\n 'dd' : '0[1-9]|[12][0-9]|3[01]',\n 'd' : options.strict ? '[1-9]|[1-2][0-9]|3[01]' : '0?[1-9]|[1-2][0-9]|3[01]',\n 'MMMM' : $locale.DATETIME_FORMATS.MONTH.join('|'),\n 'MMM' : $locale.DATETIME_FORMATS.SHORTMONTH.join('|'),\n 'MM' : '0[1-9]|1[012]',\n 'M' : options.strict ? '[1-9]|1[012]' : '0?[1-9]|1[012]',\n 'yyyy' : '[1]{1}[0-9]{3}|[2]{1}[0-9]{3}',\n 'yy' : '[0-9]{2}',\n 'y' : options.strict ? '-?(0|[1-9][0-9]{0,3})' : '-?0*[0-9]{1,4}'\n };\n\n var setFnMap = {\n 'sss' : proto.setMilliseconds,\n 'ss' : proto.setSeconds,\n 's' : proto.setSeconds,\n 'mm' : proto.setMinutes,\n 'm' : proto.setMinutes,\n 'HH' : proto.setHours,\n 'H' : proto.setHours,\n 'hh' : proto.setHours,\n 'h' : proto.setHours,\n 'EEEE' : noop,\n 'EEE' : noop,\n 'dd' : proto.setDate,\n 'd' : proto.setDate,\n 'a' : function (value) { var hours = this.getHours() % 12; return this.setHours(value.match(/pm/i) ? hours + 12 : hours); },\n 'MMMM' : function (value) { return this.setMonth(indexOfCaseInsensitive($locale.DATETIME_FORMATS.MONTH, value)); },\n 'MMM' : function (value) { return this.setMonth(indexOfCaseInsensitive($locale.DATETIME_FORMATS.SHORTMONTH, value)); },\n 'MM' : function (value) { return this.setMonth(1 * value - 1); },\n 'M' : function (value) { return this.setMonth(1 * value - 1); },\n 'yyyy' : proto.setFullYear,\n 'yy' : function (value) { return this.setFullYear(2000 + 1 * value); },\n 'y' : function (value) { return (1 * value <= 50 && value.length === 2) ? this.setFullYear(2000 + 1 * value) : this.setFullYear(1 * value); }\n };\n /* eslint-enable key-spacing, quote-props */\n\n var regex;\n var setMap;\n\n $dateParser.init = function () {\n $dateParser.$format = $locale.DATETIME_FORMATS[options.format] || options.format;\n regex = regExpForFormat($dateParser.$format);\n setMap = setMapForFormat($dateParser.$format);\n };\n\n $dateParser.isValid = function (date) {\n if (angular.isDate(date)) return !isNaN(date.getTime());\n return regex.test(date);\n };\n\n $dateParser.parse = function (value, baseDate, format, timezone) {\n // check for date format special names\n if (format) format = $locale.DATETIME_FORMATS[format] || format;\n if (angular.isDate(value)) value = dateFilter(value, format || $dateParser.$format, timezone);\n var formatRegex = format ? regExpForFormat(format) : regex;\n var formatSetMap = format ? setMapForFormat(format) : setMap;\n var matches = formatRegex.exec(value);\n if (!matches) return false;\n // use custom ParseDate object to set parsed values\n var date = baseDate && !isNaN(baseDate.getTime()) ? new ParseDate().fromDate(baseDate) : new ParseDate().fromDate(new Date(1970, 0, 1, 0));\n for (var i = 0; i < matches.length - 1; i++) {\n if (formatSetMap[i]) formatSetMap[i].call(date, matches[i + 1]);\n }\n // convert back to native Date object\n var newDate = date.toDate();\n\n // check new native Date object for day values overflow\n if (parseInt(date.day, 10) !== newDate.getDate()) {\n return false;\n }\n\n return newDate;\n };\n\n $dateParser.getDateForAttribute = function (key, value) {\n var date;\n\n if (value === 'today') {\n var today = new Date();\n date = new Date(today.getFullYear(), today.getMonth(), today.getDate() + (key === 'maxDate' ? 1 : 0), 0, 0, 0, (key === 'minDate' ? 0 : -1));\n } else if (angular.isString(value) && value.match(/^\".+\"$/)) { // Support {{ dateObj }}\n date = new Date(value.substr(1, value.length - 2));\n } else if (isNumeric(value)) {\n date = new Date(parseInt(value, 10));\n } else if (angular.isString(value) && value.length === 0) { // Reset date\n date = key === 'minDate' ? -Infinity : +Infinity;\n } else {\n date = new Date(value);\n }\n\n return date;\n };\n\n $dateParser.getTimeForAttribute = function (key, value) {\n var time;\n\n if (value === 'now') {\n time = new Date().setFullYear(1970, 0, 1);\n } else if (angular.isString(value) && value.match(/^\".+\"$/)) {\n time = new Date(value.substr(1, value.length - 2)).setFullYear(1970, 0, 1);\n } else if (isNumeric(value)) {\n time = new Date(parseInt(value, 10)).setFullYear(1970, 0, 1);\n } else if (angular.isString(value) && value.length === 0) { // Reset time\n time = key === 'minTime' ? -Infinity : +Infinity;\n } else {\n time = $dateParser.parse(value, new Date(1970, 0, 1, 0));\n }\n\n return time;\n };\n\n /* Handle switch to/from daylight saving.\n * Hours may be non-zero on daylight saving cut-over:\n * > 12 when midnight changeover, but then cannot generate\n * midnight datetime, so jump to 1AM, otherwise reset.\n * @param date (Date) the date to check\n * @return (Date) the corrected date\n *\n * __ copied from jquery ui datepicker __\n */\n $dateParser.daylightSavingAdjust = function (date) {\n if (!date) {\n return null;\n }\n date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);\n return date;\n };\n\n /* Correct the date for timezone offset.\n * @param date (Date) the date to adjust\n * @param timezone (string) the timezone to adjust for\n * @param undo (boolean) to add or subtract timezone offset\n * @return (Date) the corrected date\n */\n $dateParser.timezoneOffsetAdjust = function (date, timezone, undo) {\n if (!date) {\n return null;\n }\n // Right now, only 'UTC' is supported.\n if (timezone && timezone === 'UTC') {\n date = new Date(date.getTime());\n date.setMinutes(date.getMinutes() + (undo ? -1 : 1) * date.getTimezoneOffset());\n }\n return date;\n };\n\n // Private functions\n\n function regExpForFormat (format) {\n // `format` string can contain literal values.\n // These need to be escaped by surrounding with\n // single quotes (e.g. `\"h 'in the morning'\"`).\n // In order to output a single quote, escape it - i.e.,\n // two single quotes in a sequence (e.g. `\"h 'o''clock'\"`).\n\n var re = buildDateAbstractRegex(format);\n return buildDateParseRegex(re);\n }\n\n function buildDateAbstractRegex (format) {\n var escapedFormat = escapeReservedSymbols(format);\n var escapedLiteralFormat = escapedFormat.replace(/''/g, '\\\\\\'');\n var literalRegex = /('(?:\\\\'|.)*?')/;\n var formatParts = escapedLiteralFormat.split(literalRegex);\n var dateElements = Object.keys(regExpMap);\n var dateRegexParts = [];\n\n angular.forEach(formatParts, function (part) {\n if (isFormatStringLiteral(part)) {\n part = trimLiteralEscapeChars(part);\n } else {\n // Abstract replaces to avoid collisions\n for (var i = 0; i < dateElements.length; i++) {\n part = part.split(dateElements[i]).join('${' + i + '}');\n }\n }\n dateRegexParts.push(part);\n });\n\n return dateRegexParts.join('');\n }\n\n function escapeReservedSymbols (text) {\n return text.replace(/\\\\/g, '[\\\\\\\\]')\n .replace(/-/g, '[-]')\n .replace(/\\./g, '[.]')\n .replace(/\\*/g, '[*]')\n .replace(/\\+/g, '[+]')\n .replace(/\\?/g, '[?]')\n .replace(/\\$/g, '[$]')\n .replace(/\\^/g, '[^]')\n .replace(/\\//g, '[/]')\n .replace(/\\\\s/g, '[\\\\s]');\n }\n\n function isFormatStringLiteral (text) {\n return /^'.*'$/.test(text);\n }\n\n function trimLiteralEscapeChars (text) {\n return text.replace(/^'(.*)'$/, '$1');\n }\n\n function buildDateParseRegex (abstractRegex) {\n var dateElements = Object.keys(regExpMap);\n var re = abstractRegex;\n\n // Replace abstracted values\n for (var i = 0; i < dateElements.length; i++) {\n re = re.split('${' + i + '}').join('(' + regExpMap[dateElements[i]] + ')');\n }\n\n return new RegExp('^' + re + '$', ['i']);\n }\n\n function setMapForFormat (format) {\n var re = buildDateAbstractRegex(format);\n return buildDateParseValuesMap(re);\n }\n\n function buildDateParseValuesMap (abstractRegex) {\n var dateElements = Object.keys(regExpMap);\n var valuesRegex = new RegExp('\\\\${(\\\\d+)}', 'g');\n var valuesMatch;\n var keyIndex;\n var valueKey;\n var valueFunction;\n var valuesFunctionMap = [];\n\n /* eslint-disable no-cond-assign */\n while ((valuesMatch = valuesRegex.exec(abstractRegex)) !== null) {\n keyIndex = valuesMatch[1];\n valueKey = dateElements[keyIndex];\n valueFunction = setFnMap[valueKey];\n\n valuesFunctionMap.push(valueFunction);\n }\n\n return valuesFunctionMap;\n }\n\n $dateParser.init();\n return $dateParser;\n\n };\n\n return DateParserFactory;\n\n };\n\n});\n","'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateFormatter', [])\n\n .service('$dateFormatter', function ($locale, dateFilter) {\n\n // The unused `lang` arguments are on purpose. The default implementation does not\n // use them and it always uses the locale loaded into the `$locale` service.\n // Custom implementations might use it, thus allowing different directives to\n // have different languages.\n\n this.getDefaultLocale = function () {\n return $locale.id;\n };\n\n // Format is either a data format name, e.g. \"shortTime\" or \"fullDate\", or a date format\n // Return either the corresponding date format or the given date format.\n this.getDatetimeFormat = function (format, lang) {\n return $locale.DATETIME_FORMATS[format] || format;\n };\n\n this.weekdaysShort = function (lang) {\n return $locale.DATETIME_FORMATS.SHORTDAY;\n };\n\n function splitTimeFormat (format) {\n return /(h+)([:\\.])?(m+)([:\\.])?(s*)[ ]?(a?)/i.exec(format).slice(1);\n }\n\n // h:mm a => h\n this.hoursFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[0];\n };\n\n // h:mm a => mm\n this.minutesFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[2];\n };\n\n // h:mm:ss a => ss\n this.secondsFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => :\n this.timeSeparator = function (timeFormat) {\n return splitTimeFormat(timeFormat)[1];\n };\n\n // h:mm:ss a => true, h:mm a => false\n this.showSeconds = function (timeFormat) {\n return !!splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => true, H.mm => false\n this.showAM = function (timeFormat) {\n return !!splitTimeFormat(timeFormat)[5];\n };\n\n this.formatDate = function (date, format, lang, timezone) {\n return dateFilter(date, format, timezone);\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.button', [])\n\n .provider('$button', function () {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n toggleEvent: 'click'\n };\n\n this.$get = function () {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsCheckboxGroup', function () {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink (element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"checkbox\"]');\n angular.forEach(children, function (child) {\n var childEl = angular.element(child);\n childEl.attr('bs-checkbox', '');\n childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value'));\n });\n }\n\n };\n\n })\n\n .directive('bsCheckbox', function ($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support label > input[type=\"checkbox\"]\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true;\n if (constantValueRegExp.test(attr.trueValue)) {\n trueValue = scope.$eval(attr.trueValue);\n }\n var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false;\n if (constantValueRegExp.test(attr.falseValue)) {\n falseValue = scope.$eval(attr.falseValue);\n }\n\n // Parse exotic values\n var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean';\n if (hasExoticValues) {\n controller.$parsers.push(function (viewValue) {\n // console.warn('$parser', element.attr('ng-model'), 'viewValue', viewValue);\n return viewValue ? trueValue : falseValue;\n });\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n return angular.equals(modelValue, trueValue);\n });\n }\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = !!controller.$viewValue;\n $$rAF(function () {\n if (isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function () {\n scope.$apply(function () {\n // console.warn('!click', element.attr('ng-model'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n if (!isInput) {\n controller.$setViewValue(!activeElement.hasClass('active'));\n }\n controller.$render();\n });\n });\n\n }\n\n };\n\n })\n\n .directive('bsRadioGroup', function () {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink (element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"radio\"]');\n angular.forEach(children, function (child) {\n angular.element(child).attr('bs-radio', '');\n angular.element(child).attr('ng-model', attr.ngModel);\n });\n }\n\n };\n\n })\n\n .directive('bsRadio', function ($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support `label > input[type=\"radio\"]` markup\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var value;\n attr.$observe('value', function (v) {\n if (typeof v !== 'boolean' && constantValueRegExp.test(v)) {\n value = scope.$eval(v);\n } else {\n value = v;\n }\n controller.$render();\n });\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('value'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$viewValue, value);\n $$rAF(function () {\n if (isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function () {\n scope.$apply(function () {\n // console.warn('!click', element.attr('value'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n controller.$setViewValue(value);\n controller.$render();\n });\n });\n\n }\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.datepicker', [\n 'mgcrea.ngStrap.helpers.dateParser',\n 'mgcrea.ngStrap.helpers.dateFormatter',\n 'mgcrea.ngStrap.tooltip'])\n\n .provider('$datepicker', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n // Uncommenting the following line will break backwards compatability\n // prefixEvent: 'datepicker',\n prefixClass: 'datepicker',\n placement: 'bottom-left',\n templateUrl: 'datepicker/datepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: false,\n dateType: 'date',\n dateFormat: 'shortDate',\n timezone: null,\n modelDateFormat: null,\n dayFormat: 'dd',\n monthFormat: 'MMM',\n yearFormat: 'yyyy',\n monthTitleFormat: 'MMMM yyyy',\n yearTitleFormat: 'yyyy',\n strictFormat: false,\n autoclose: false,\n minDate: -Infinity,\n maxDate: +Infinity,\n startView: 0,\n minView: 0,\n startWeek: 0,\n daysOfWeekDisabled: '',\n hasToday: false,\n hasClear: false,\n iconLeft: 'glyphicon glyphicon-chevron-left',\n iconRight: 'glyphicon glyphicon-chevron-right'\n };\n\n this.$get = function ($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale();\n\n function DatepickerFactory (element, controller, config) {\n\n var $datepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $datepicker.$options;\n var scope = $datepicker.$scope;\n if (options.startView) options.startView -= options.minView;\n\n // View vars\n\n var pickerViews = datepickerViews($datepicker);\n $datepicker.$views = pickerViews.views;\n var viewDate = pickerViews.viewDate;\n scope.$mode = options.startView;\n scope.$iconLeft = options.iconLeft;\n scope.$iconRight = options.iconRight;\n scope.$hasToday = options.hasToday;\n scope.$hasClear = options.hasClear;\n var $picker = $datepicker.$views[scope.$mode];\n\n // Scope methods\n\n scope.$select = function (date, disabled) {\n if (disabled) return;\n $datepicker.select(date);\n };\n scope.$selectPane = function (value) {\n $datepicker.$selectPane(value);\n };\n scope.$toggleMode = function () {\n $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length);\n };\n scope.$setToday = function () {\n if (options.autoclose) {\n $datepicker.setMode(0);\n $datepicker.select(new Date());\n } else {\n $datepicker.select(new Date(), true);\n }\n };\n scope.$clear = function () {\n if (options.autoclose) {\n $datepicker.setMode(0);\n $datepicker.select(null);\n } else {\n $datepicker.select(null, true);\n }\n };\n\n // Public methods\n\n $datepicker.update = function (date) {\n // console.warn('$datepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $datepicker.$date = date;\n $picker.update.call($picker, date);\n }\n // Build only if pristine\n $datepicker.$build(true);\n };\n\n $datepicker.updateDisabledDates = function (dateRanges) {\n options.disabledDateRanges = dateRanges;\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], $datepicker.$setDisabledEl);\n }\n };\n\n $datepicker.select = function (date, keep) {\n // console.warn('$datepicker.select', date, scope.$mode);\n if (angular.isDate(date)) {\n if (!angular.isDate(controller.$dateValue) || isNaN(controller.$dateValue.getTime())) {\n controller.$dateValue = new Date(date);\n }\n } else {\n controller.$dateValue = null;\n }\n if (!scope.$mode || keep) {\n controller.$setViewValue(angular.copy(date));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function () { $datepicker.hide(true); });\n }\n } else {\n angular.extend(viewDate, {year: date.getFullYear(), month: date.getMonth(), date: date.getDate()});\n $datepicker.setMode(scope.$mode - 1);\n $datepicker.$build();\n }\n };\n\n $datepicker.setMode = function (mode) {\n // console.warn('$datepicker.setMode', mode);\n scope.$mode = mode;\n $picker = $datepicker.$views[scope.$mode];\n $datepicker.$build();\n };\n\n // Protected methods\n\n $datepicker.$build = function (pristine) {\n // console.warn('$datepicker.$build() viewDate=%o', viewDate);\n if (pristine === true && $picker.built) return;\n if (pristine === false && !$picker.built) return;\n $picker.build.call($picker);\n };\n\n $datepicker.$updateSelected = function () {\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], updateSelected);\n }\n };\n\n $datepicker.$isSelected = function (date) {\n return $picker.isSelected(date);\n };\n\n $datepicker.$setDisabledEl = function (el) {\n el.disabled = $picker.isDisabled(el.date);\n };\n\n $datepicker.$selectPane = function (value) {\n var steps = $picker.steps;\n // set targetDate to first day of month to avoid problems with\n // date values rollover. This assumes the viewDate does not\n // depend on the day of the month\n var targetDate = new Date(Date.UTC(viewDate.year + ((steps.year || 0) * value), viewDate.month + ((steps.month || 0) * value), 1));\n angular.extend(viewDate, {year: targetDate.getUTCFullYear(), month: targetDate.getUTCMonth(), date: targetDate.getUTCDate()});\n $datepicker.$build();\n };\n\n $datepicker.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $datepicker.$onKeyDown = function (evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n if (evt.keyCode === 13) {\n if (!scope.$mode) {\n $datepicker.hide(true);\n } else {\n scope.$apply(function () { $datepicker.setMode(scope.$mode - 1); });\n }\n return;\n }\n\n // Navigate with keyboard\n $picker.onKeyDown(evt);\n parentScope.$digest();\n };\n\n // Private\n\n function updateSelected (el) {\n el.selected = $datepicker.$isSelected(el.date);\n }\n\n function focusElement () {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $datepicker.init;\n $datepicker.init = function () {\n if (isNative && options.useNative) {\n element.prop('type', 'date');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $datepicker.destroy;\n $datepicker.destroy = function () {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $datepicker.show;\n $datepicker.show = function () {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n // if $datepicker is no longer showing, don't setup events\n if (!$datepicker.$isShown) return;\n $datepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $datepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $datepicker.hide;\n $datepicker.hide = function (blur) {\n if (!$datepicker.$isShown) return;\n $datepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $datepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $datepicker;\n\n }\n\n DatepickerFactory.defaults = defaults;\n return DatepickerFactory;\n\n };\n\n })\n\n .directive('bsDatepicker', function ($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) {\n\n // var defaults = $datepicker.defaults;\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent', 'hasToday', 'hasClear'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'hasToday', 'hasClear'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Initialize datepicker\n var datepicker = $datepicker(element, controller, options);\n options = datepicker.$options;\n // Set expected iOS format\n if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd';\n\n var lang = options.lang;\n\n var formatDate = function (date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!datepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i);\n if (newValue === true) {\n datepicker.show();\n } else {\n datepicker.hide();\n }\n });\n }\n\n // Observe attributes for changes\n angular.forEach(['minDate', 'maxDate'], function (key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n if (angular.isDefined(attr[key])) {\n attr.$observe(key, function (newValue) {\n // console.warn('attr.$observe(%s)=%o', key, newValue);\n datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue);\n // Build only if dirty\n if (!isNaN(datepicker.$options[key])) datepicker.$build(false);\n validateAgainstMinMaxDate(controller.$dateValue);\n });\n }\n });\n\n // Observe date format\n if (angular.isDefined(attr.dateFormat)) {\n attr.$observe('dateFormat', function (newValue) {\n datepicker.$options.dateFormat = newValue;\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n datepicker.update(controller.$dateValue);\n }, true);\n\n // Normalize undefined/null/empty array,\n // so that we don't treat changing from undefined->null as a change.\n function normalizeDateRanges (ranges) {\n if (!ranges || !ranges.length) return null;\n return ranges;\n }\n\n if (angular.isDefined(attr.disabledDates)) {\n scope.$watch(attr.disabledDates, function (disabledRanges, previousValue) {\n disabledRanges = normalizeDateRanges(disabledRanges);\n previousValue = normalizeDateRanges(previousValue);\n\n if (disabledRanges) {\n datepicker.updateDisabledDates(disabledRanges);\n }\n });\n }\n\n function validateAgainstMinMaxDate (parsedDate) {\n if (!angular.isDate(parsedDate)) return;\n var isMinValid = isNaN(datepicker.$options.minDate) || parsedDate.getTime() >= datepicker.$options.minDate;\n var isMaxValid = isNaN(datepicker.$options.maxDate) || parsedDate.getTime() <= datepicker.$options.maxDate;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (isValid) controller.$dateValue = parsedDate;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function (viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n controller.$setValidity('date', true);\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n return null;\n }\n var parsedDate = dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedDate || isNaN(parsedDate.getTime())) {\n controller.$setValidity('date', false);\n // return undefined, causes ngModelController to\n // invalidate model value\n return;\n }\n validateAgainstMinMaxDate(parsedDate);\n\n if (options.dateType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true);\n return formatDate(date, options.modelDateFormat || options.dateFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.dateType === 'number') {\n return date.getTime();\n } else if (options.dateType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.dateType === 'iso') {\n return date.toISOString();\n }\n return new Date(date);\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.dateType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelDateFormat);\n } else if (options.dateType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if (isNaN(date.getTime())) {\n // var today = new Date();\n // date = new Date(today.getFullYear(), today.getMonth(), today.getDate(), 0, 0, 0, 0);\n // }\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getDateFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function () {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getDateFormattedString());\n };\n\n function getDateFormattedString () {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.dateFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (datepicker) datepicker.destroy();\n options = null;\n datepicker = null;\n });\n\n }\n };\n\n })\n\n .provider('datepickerViews', function () {\n\n // var defaults = this.defaults = {\n // dayFormat: 'dd',\n // daySplit: 7\n // };\n\n // Split array into smaller arrays\n function split (arr, size) {\n var arrays = [];\n while (arr.length > 0) {\n arrays.push(arr.splice(0, size));\n }\n return arrays;\n }\n\n // Modulus operator\n function mod (n, m) {\n return ((n % m) + m) % m;\n }\n\n this.$get = function ($dateFormatter, $dateParser, $sce) {\n\n return function (picker) {\n\n var scope = picker.$scope;\n var options = picker.$options;\n\n var lang = options.lang;\n var formatDate = function (date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n var weekDaysMin = $dateFormatter.weekdaysShort(lang);\n var weekDaysLabels = weekDaysMin.slice(options.startWeek).concat(weekDaysMin.slice(0, options.startWeek));\n var weekDaysLabelsHtml = $sce.trustAsHtml('' + weekDaysLabels.join('') + '');\n\n var startDate = picker.$date || (options.startDate ? dateParser.getDateForAttribute('startDate', options.startDate) : new Date());\n var viewDate = {year: startDate.getFullYear(), month: startDate.getMonth(), date: startDate.getDate()};\n\n var views = [{\n format: options.dayFormat,\n split: 7,\n steps: {month: 1},\n update: function (date, force) {\n if (!this.built || force || date.getFullYear() !== viewDate.year || date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getDate() !== viewDate.date || date.getDate() === 1) {\n // chaging picker current month will cause viewDate.date to be set to first day of the month,\n // in $datepicker.$selectPane, so picker would not update selected day display if\n // user picks first day of the new month.\n // As a workaround, we are always forcing update when picked date is first day of month.\n viewDate.date = picker.$date.getDate();\n picker.$updateSelected();\n }\n },\n build: function () {\n var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1);\n var firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset();\n var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5);\n var firstDateOffset = firstDate.getTimezoneOffset();\n var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString();\n // Handle daylight time switch\n if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 60e3);\n var days = [];\n var day;\n for (var i = 0; i < 42; i++) { // < 7 * 6\n day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i));\n days.push({date: day, isToday: day.toDateString() === today, label: formatDate(day, this.format), selected: picker.$date && this.isSelected(day), muted: day.getMonth() !== viewDate.month, disabled: this.isDisabled(day)});\n }\n scope.title = formatDate(firstDayOfMonth, options.monthTitleFormat);\n scope.showLabels = true;\n scope.labels = weekDaysLabelsHtml;\n scope.rows = split(days, this.split);\n scope.isTodayDisabled = this.isDisabled(new Date());\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth() && date.getDate() === picker.$date.getDate();\n },\n isDisabled: function (date) {\n var time = date.getTime();\n\n // Disabled because of min/max date.\n if (time < options.minDate || time > options.maxDate) return true;\n\n // Disabled due to being a disabled day of the week\n if (options.daysOfWeekDisabled.indexOf(date.getDay()) !== -1) return true;\n\n // Disabled because of disabled date range.\n if (options.disabledDateRanges) {\n for (var i = 0; i < options.disabledDateRanges.length; i++) {\n if (time >= options.disabledDateRanges[i].start && time <= options.disabledDateRanges[i].end) {\n return true;\n }\n }\n }\n\n return false;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualTime = picker.$date.getTime();\n var newDate;\n\n if (evt.keyCode === 37) newDate = new Date(actualTime - 1 * 864e5);\n else if (evt.keyCode === 38) newDate = new Date(actualTime - 7 * 864e5);\n else if (evt.keyCode === 39) newDate = new Date(actualTime + 1 * 864e5);\n else if (evt.keyCode === 40) newDate = new Date(actualTime + 7 * 864e5);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'month',\n format: options.monthFormat,\n split: 4,\n steps: {year: 1},\n update: function (date, force) {\n if (!this.built || date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function () {\n // var firstMonth = new Date(viewDate.year, 0, 1);\n var months = [];\n var month;\n for (var i = 0; i < 12; i++) {\n month = new Date(viewDate.year, i, 1);\n months.push({date: month, label: formatDate(month, this.format), selected: picker.$isSelected(month), disabled: this.isDisabled(month)});\n }\n scope.title = formatDate(month, options.yearTitleFormat);\n scope.showLabels = false;\n scope.rows = split(months, this.split);\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth();\n },\n isDisabled: function (date) {\n var lastDate = +new Date(date.getFullYear(), date.getMonth() + 1, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualMonth = picker.$date.getMonth();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setMonth(actualMonth - 1);\n else if (evt.keyCode === 38) newDate.setMonth(actualMonth - 4);\n else if (evt.keyCode === 39) newDate.setMonth(actualMonth + 1);\n else if (evt.keyCode === 40) newDate.setMonth(actualMonth + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'year',\n format: options.yearFormat,\n split: 4,\n steps: {year: 12},\n update: function (date, force) {\n if (!this.built || force || parseInt(date.getFullYear() / 20, 10) !== parseInt(viewDate.year / 20, 10)) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function () {\n var firstYear = viewDate.year - viewDate.year % (this.split * 3);\n var years = [];\n var year;\n for (var i = 0; i < 12; i++) {\n year = new Date(firstYear + i, 0, 1);\n years.push({date: year, label: formatDate(year, this.format), selected: picker.$isSelected(year), disabled: this.isDisabled(year)});\n }\n scope.title = years[0].label + '-' + years[years.length - 1].label;\n scope.showLabels = false;\n scope.rows = split(years, this.split);\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear();\n },\n isDisabled: function (date) {\n var lastDate = +new Date(date.getFullYear() + 1, 0, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualYear = picker.$date.getFullYear();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setYear(actualYear - 1);\n else if (evt.keyCode === 38) newDate.setYear(actualYear - 4);\n else if (evt.keyCode === 39) newDate.setYear(actualYear + 1);\n else if (evt.keyCode === 40) newDate.setYear(actualYear + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }];\n\n return {\n views: options.minView ? Array.prototype.slice.call(views, options.minView) : views,\n viewDate: viewDate\n };\n\n };\n\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.collapse', [])\n\n .provider('$collapse', function () {\n\n var defaults = this.defaults = {\n animation: 'am-collapse',\n disallowToggle: false,\n activeClass: 'in',\n startCollapsed: false,\n allowMultiple: false\n };\n\n var controller = this.controller = function ($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'disallowToggle', 'activeClass', 'startCollapsed', 'allowMultiple'], function (key) {\n if (angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['disallowToggle', 'startCollapsed', 'allowMultiple'], function (key) {\n if (angular.isDefined($attrs[key]) && falseValueRegExp.test($attrs[key])) {\n self.$options[key] = false;\n }\n });\n\n self.$toggles = [];\n self.$targets = [];\n\n self.$viewChangeListeners = [];\n\n self.$registerToggle = function (element) {\n self.$toggles.push(element);\n };\n self.$registerTarget = function (element) {\n self.$targets.push(element);\n };\n\n self.$unregisterToggle = function (element) {\n var index = self.$toggles.indexOf(element);\n // remove toggle from $toggles array\n self.$toggles.splice(index, 1);\n };\n self.$unregisterTarget = function (element) {\n var index = self.$targets.indexOf(element);\n\n // remove element from $targets array\n self.$targets.splice(index, 1);\n\n if (self.$options.allowMultiple) {\n // remove target index from $active array values\n deactivateItem(element);\n }\n\n // fix active item indexes\n fixActiveItemIndexes(index);\n\n self.$viewChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n // use array to store all the currently open panels\n self.$targets.$active = !self.$options.startCollapsed ? [0] : [];\n self.$setActive = $scope.$setActive = function (value) {\n if (angular.isArray(value)) {\n self.$targets.$active = value;\n } else if (!self.$options.disallowToggle && isActive(value)) {\n deactivateItem(value);\n } else {\n activateItem(value);\n }\n\n self.$viewChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n self.$activeIndexes = function () {\n if (self.$options.allowMultiple) {\n return self.$targets.$active;\n }\n return self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1;\n };\n\n function fixActiveItemIndexes (index) {\n // item with index was removed, so we\n // need to adjust other items index values\n var activeIndexes = self.$targets.$active;\n for (var i = 0; i < activeIndexes.length; i++) {\n if (index < activeIndexes[i]) {\n activeIndexes[i] = activeIndexes[i] - 1;\n }\n\n // the last item is active, so we need to\n // adjust its index\n if (activeIndexes[i] === self.$targets.length) {\n activeIndexes[i] = self.$targets.length - 1;\n }\n }\n }\n\n function isActive (value) {\n var activeItems = self.$targets.$active;\n return activeItems.indexOf(value) !== -1;\n }\n\n function deactivateItem (value) {\n var index = self.$targets.$active.indexOf(value);\n if (index !== -1) {\n self.$targets.$active.splice(index, 1);\n }\n }\n\n function activateItem (value) {\n if (!self.$options.allowMultiple) {\n // remove current selected item\n self.$targets.$active.splice(0, 1);\n }\n\n if (self.$targets.$active.indexOf(value) === -1) {\n self.$targets.$active.push(value);\n }\n }\n\n };\n\n this.$get = function () {\n var $collapse = {};\n $collapse.defaults = defaults;\n $collapse.controller = controller;\n return $collapse;\n };\n\n })\n\n .directive('bsCollapse', function ($window, $animate, $collapse) {\n\n return {\n require: ['?ngModel', 'bsCollapse'],\n controller: ['$scope', '$element', '$attrs', $collapse.controller],\n link: function postLink (scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n if (ngModelCtrl) {\n\n // Update the modelValue following\n bsCollapseCtrl.$viewChangeListeners.push(function () {\n ngModelCtrl.$setViewValue(bsCollapseCtrl.$activeIndexes());\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n if (angular.isArray(modelValue)) {\n // model value is an array, so just replace\n // the active items directly\n bsCollapseCtrl.$setActive(modelValue);\n } else {\n var activeIndexes = bsCollapseCtrl.$activeIndexes();\n\n if (angular.isArray(activeIndexes)) {\n // we have an array of selected indexes\n if (activeIndexes.indexOf(modelValue * 1) === -1) {\n // item with modelValue index is not active\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n } else if (activeIndexes !== modelValue * 1) {\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n return modelValue;\n });\n\n }\n\n }\n };\n\n })\n\n .directive('bsCollapseToggle', function () {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base attr\n element.attr('data-toggle', 'collapse');\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerToggle(element);\n\n // remove toggle from collapse controller when toggle is destroyed\n scope.$on('$destroy', function () {\n bsCollapseCtrl.$unregisterToggle(element);\n });\n\n element.on('click', function () {\n if (!attrs.disabled) {\n var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element);\n bsCollapseCtrl.$setActive(index * 1);\n scope.$apply();\n }\n });\n\n }\n };\n\n })\n\n .directive('bsCollapseTarget', function ($animate) {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n // scope: true,\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base class\n element.addClass('collapse');\n\n // Add animation class\n if (bsCollapseCtrl.$options.animation) {\n element.addClass(bsCollapseCtrl.$options.animation);\n }\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerTarget(element);\n\n // remove pane target from collapse controller when target is destroyed\n scope.$on('$destroy', function () {\n bsCollapseCtrl.$unregisterTarget(element);\n });\n\n function render () {\n var index = bsCollapseCtrl.$targets.indexOf(element);\n var active = bsCollapseCtrl.$activeIndexes();\n var action = 'removeClass';\n if (angular.isArray(active)) {\n if (active.indexOf(index) !== -1) {\n action = 'addClass';\n }\n } else if (index === active) {\n action = 'addClass';\n }\n\n $animate[action](element, bsCollapseCtrl.$options.activeClass);\n }\n\n bsCollapseCtrl.$viewChangeListeners.push(function () {\n render();\n });\n render();\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.aside', ['mgcrea.ngStrap.modal'])\n\n .provider('$aside', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade-and-slide-right',\n prefixClass: 'aside',\n prefixEvent: 'aside',\n placement: 'right',\n templateUrl: 'aside/aside.tpl.html',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function ($modal) {\n\n function AsideFactory (config) {\n\n var $aside = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $aside = $modal(options);\n\n return $aside;\n\n }\n\n return AsideFactory;\n\n };\n\n })\n\n .directive('bsAside', function ($window, $sce, $aside) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsAside) {\n scope.$watch(attr.bsAside, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize aside\n var aside = $aside(options);\n\n // Trigger\n element.on(attr.trigger || 'click', aside.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (aside) aside.destroy();\n options = null;\n aside = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\n// @BUG: following snippet won't compile correctly\n\nangular.module('mgcrea.ngStrap.alert', ['mgcrea.ngStrap.modal'])\n\n .provider('$alert', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'alert',\n prefixEvent: 'alert',\n placement: null,\n templateUrl: 'alert/alert.tpl.html',\n container: false,\n element: null,\n backdrop: false,\n keyboard: true,\n show: true,\n // Specific options\n duration: false,\n type: false,\n dismissable: true\n };\n\n this.$get = function ($modal, $timeout) {\n\n function AlertFactory (config) {\n\n var $alert = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $alert = $modal(options);\n\n // Support scope as string options [/*title, content, */ type, dismissable]\n $alert.$scope.dismissable = !!options.dismissable;\n if (options.type) {\n $alert.$scope.type = options.type;\n }\n\n // Support auto-close duration\n var show = $alert.show;\n if (options.duration) {\n $alert.show = function () {\n show();\n $timeout(function () {\n $alert.hide();\n }, options.duration * 1000);\n };\n }\n\n return $alert;\n\n }\n\n return AlertFactory;\n\n };\n\n })\n\n .directive('bsAlert', function ($window, $sce, $alert) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'keyboard', 'html', 'container', 'animation', 'duration', 'dismissable'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['keyboard', 'html', 'container', 'dismissable'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')) {\n scope.title = '';\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content', 'type'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsAlert) {\n scope.$watch(attr.bsAlert, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize alert\n var alert = $alert(options);\n\n // Trigger\n element.on(attr.trigger || 'click', alert.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (alert) alert.destroy();\n options = null;\n alert = null;\n });\n\n }\n };\n\n });\n","'use strict';\n\nangular.module('mgcrea.ngStrap.affix', ['mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce'])\n\n .provider('$affix', function () {\n\n var defaults = this.defaults = {\n offsetTop: 'auto',\n inlineStyles: true,\n setWidth: true\n };\n\n this.$get = function ($window, debounce, dimensions) {\n\n var bodyEl = angular.element($window.document.body);\n var windowEl = angular.element($window);\n\n function AffixFactory (element, config) {\n\n var $affix = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var targetEl = options.target;\n\n // Initial private vars\n var reset = 'affix affix-top affix-bottom';\n var setWidth = false;\n var initialAffixTop = 0;\n var initialOffsetTop = 0;\n var offsetTop = 0;\n var offsetBottom = 0;\n var affixed = null;\n var unpin = null;\n\n var parent = element.parent();\n // Options: custom parent\n if (options.offsetParent) {\n if (options.offsetParent.match(/^\\d+$/)) {\n for (var i = 0; i < (options.offsetParent * 1) - 1; i++) {\n parent = parent.parent();\n }\n } else {\n parent = angular.element(options.offsetParent);\n }\n }\n\n $affix.init = function () {\n\n this.$parseOffsets();\n initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop;\n setWidth = options.setWidth && !element[0].style.width;\n\n // Bind events\n targetEl.on('scroll', this.checkPosition);\n targetEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', this.$debouncedOnResize);\n\n // Both of these checkPosition() calls are necessary for the case where\n // the user hits refresh after scrolling to the bottom of the page.\n this.checkPosition();\n this.checkPositionWithEventLoop();\n\n };\n\n $affix.destroy = function () {\n\n // Unbind events\n targetEl.off('scroll', this.checkPosition);\n targetEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', this.$debouncedOnResize);\n\n };\n\n $affix.checkPositionWithEventLoop = function () {\n\n // IE 9 throws an error if we use 'this' instead of '$affix'\n // in this setTimeout call\n setTimeout($affix.checkPosition, 1);\n\n };\n\n $affix.checkPosition = function () {\n // if (!this.$element.is(':visible')) return\n\n var scrollTop = getScrollTop();\n var position = dimensions.offset(element[0]);\n var elementHeight = dimensions.height(element[0]);\n\n // Get required affix class according to position\n var affix = getRequiredAffixClass(unpin, position, elementHeight);\n\n // Did affix status changed this last check?\n if (affixed === affix) return;\n affixed = affix;\n\n if (affix === 'top') {\n unpin = null;\n if (setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n } else if (affix === 'bottom') {\n if (options.offsetUnpin) {\n unpin = -(options.offsetUnpin * 1);\n } else {\n // Calculate unpin threshold when affixed to bottom.\n // Hopefully the browser scrolls pixel by pixel.\n unpin = position.top - scrollTop;\n }\n if (setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', (options.offsetParent) ? '' : ((bodyEl[0].offsetHeight - offsetBottom - elementHeight - initialOffsetTop) + 'px'));\n }\n } else { // affix === 'middle'\n unpin = null;\n if (setWidth) {\n element.css('width', element[0].offsetWidth + 'px');\n }\n if (options.inlineStyles) {\n element.css('position', 'fixed');\n element.css('top', initialAffixTop + 'px');\n }\n }\n\n // Add proper affix class\n element.removeClass(reset).addClass('affix' + ((affix !== 'middle') ? '-' + affix : ''));\n\n };\n\n $affix.$onResize = function () {\n $affix.$parseOffsets();\n $affix.checkPosition();\n };\n $affix.$debouncedOnResize = debounce($affix.$onResize, 50);\n\n $affix.$parseOffsets = function () {\n var initialPosition = element[0].style.position;\n var initialTop = element[0].style.top;\n // Reset position to calculate correct offsetTop\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n\n if (options.offsetTop) {\n if (options.offsetTop === 'auto') {\n options.offsetTop = '+0';\n }\n if (options.offsetTop.match(/^[-+]\\d+$/)) {\n initialAffixTop = - options.offsetTop * 1;\n if (options.offsetParent) {\n offsetTop = dimensions.offset(parent[0]).top + (options.offsetTop * 1);\n } else {\n offsetTop = dimensions.offset(element[0]).top - dimensions.css(element[0], 'marginTop', true) + (options.offsetTop * 1);\n }\n } else {\n offsetTop = options.offsetTop * 1;\n }\n }\n\n if (options.offsetBottom) {\n if (options.offsetParent && options.offsetBottom.match(/^[-+]\\d+$/)) {\n // add 1 pixel due to rounding problems...\n offsetBottom = getScrollHeight() - (dimensions.offset(parent[0]).top + dimensions.height(parent[0])) + (options.offsetBottom * 1) + 1;\n } else {\n offsetBottom = options.offsetBottom * 1;\n }\n }\n\n // Bring back the element's position after calculations\n if (options.inlineStyles) {\n element.css('position', initialPosition);\n element.css('top', initialTop);\n }\n };\n\n // Private methods\n\n function getRequiredAffixClass (_unpin, position, elementHeight) {\n var scrollTop = getScrollTop();\n var scrollHeight = getScrollHeight();\n\n if (scrollTop <= offsetTop) {\n return 'top';\n } else if (_unpin !== null) {\n return scrollTop + _unpin <= position.top ? 'middle' : 'bottom';\n } else if (offsetBottom !== null && (position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom)) {\n return 'bottom';\n }\n return 'middle';\n }\n\n function getScrollTop () {\n return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop;\n }\n\n function getScrollHeight () {\n return targetEl[0] === $window ? $window.document.body.scrollHeight : targetEl[0].scrollHeight;\n }\n\n $affix.init();\n return $affix;\n\n }\n\n return AffixFactory;\n\n };\n\n })\n\n .directive('bsAffix', function ($affix, $window, $timeout) {\n\n return {\n restrict: 'EAC',\n require: '^?bsAffixTarget',\n link: function postLink (scope, element, attr, affixTarget) {\n\n var options = {scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window)};\n angular.forEach(['offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles', 'setWidth'], function (key) {\n if (angular.isDefined(attr[key])) {\n var option = attr[key];\n if (/true/i.test(option)) option = true;\n if (/false/i.test(option)) option = false;\n options[key] = option;\n }\n });\n\n var affix;\n $timeout(function () { affix = $affix(element, options); });\n scope.$on('$destroy', function () {\n if (affix) affix.destroy();\n options = null;\n affix = null;\n });\n\n }\n };\n\n })\n\n .directive('bsAffixTarget', function () {\n return {\n controller: function ($element) {\n this.$element = $element;\n }\n };\n });\n","\nangular.module('mgcrea.ngStrap', [\n 'mgcrea.ngStrap.modal',\n 'mgcrea.ngStrap.aside',\n 'mgcrea.ngStrap.alert',\n 'mgcrea.ngStrap.button',\n 'mgcrea.ngStrap.select',\n 'mgcrea.ngStrap.datepicker',\n 'mgcrea.ngStrap.timepicker',\n 'mgcrea.ngStrap.navbar',\n 'mgcrea.ngStrap.tooltip',\n 'mgcrea.ngStrap.popover',\n 'mgcrea.ngStrap.dropdown',\n 'mgcrea.ngStrap.typeahead',\n 'mgcrea.ngStrap.scrollspy',\n 'mgcrea.ngStrap.affix',\n 'mgcrea.ngStrap.tab',\n 'mgcrea.ngStrap.collapse'\n]);\n"]} \ No newline at end of file diff --git a/dist/angular-strap.tpl.js b/dist/angular-strap.tpl.js index e2e5fccff..f38c2debc 100644 --- a/dist/angular-strap.tpl.js +++ b/dist/angular-strap.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -8,16 +8,16 @@ (function(window, document, undefined) { 'use strict'; angular.module('mgcrea.ngStrap.alert').run([ '$templateCache', function($templateCache) { - $templateCache.put('alert/alert.tpl.html', '
 
'); + $templateCache.put('alert/alert.tpl.html', '
 
'); } ]); angular.module('mgcrea.ngStrap.aside').run([ '$templateCache', function($templateCache) { $templateCache.put('aside/aside.tpl.html', ''); } ]); angular.module('mgcrea.ngStrap.datepicker').run([ '$templateCache', function($templateCache) { - $templateCache.put('datepicker/datepicker.tpl.html', ''); + $templateCache.put('datepicker/datepicker.tpl.html', ''); } ]); angular.module('mgcrea.ngStrap.dropdown').run([ '$templateCache', function($templateCache) { - $templateCache.put('dropdown/dropdown.tpl.html', ''); + $templateCache.put('dropdown/dropdown.tpl.html', ''); } ]); angular.module('mgcrea.ngStrap.modal').run([ '$templateCache', function($templateCache) { $templateCache.put('modal/modal.tpl.html', ''); @@ -29,10 +29,10 @@ $templateCache.put('select/select.tpl.html', ''); } ]); angular.module('mgcrea.ngStrap.tab').run([ '$templateCache', function($templateCache) { - $templateCache.put('tab/tab.tpl.html', '
'); + $templateCache.put('tab/tab.tpl.html', '
'); } ]); angular.module('mgcrea.ngStrap.timepicker').run([ '$templateCache', function($templateCache) { - $templateCache.put('timepicker/timepicker.tpl.html', ''); + $templateCache.put('timepicker/timepicker.tpl.html', ''); } ]); angular.module('mgcrea.ngStrap.tooltip').run([ '$templateCache', function($templateCache) { $templateCache.put('tooltip/tooltip.tpl.html', '
'); diff --git a/dist/angular-strap.tpl.min.js b/dist/angular-strap.tpl.min.js index c2c268fe5..8da7ea854 100644 --- a/dist/angular-strap.tpl.min.js +++ b/dist/angular-strap.tpl.min.js @@ -1,8 +1,8 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -!function(t,e,n){'use strict';angular.module('mgcrea.ngStrap.alert').run(['$templateCache',function(t){t.put('alert/alert.tpl.html','
 
')}]),angular.module('mgcrea.ngStrap.aside').run(['$templateCache',function(t){t.put('aside/aside.tpl.html','')}]),angular.module('mgcrea.ngStrap.datepicker').run(['$templateCache',function(t){t.put('datepicker/datepicker.tpl.html','')}]),angular.module('mgcrea.ngStrap.dropdown').run(['$templateCache',function(t){t.put('dropdown/dropdown.tpl.html','')}]),angular.module('mgcrea.ngStrap.modal').run(['$templateCache',function(t){t.put('modal/modal.tpl.html','')}]),angular.module('mgcrea.ngStrap.popover').run(['$templateCache',function(t){t.put('popover/popover.tpl.html','

')}]),angular.module('mgcrea.ngStrap.select').run(['$templateCache',function(t){t.put('select/select.tpl.html','')}]),angular.module('mgcrea.ngStrap.tab').run(['$templateCache',function(t){t.put('tab/tab.tpl.html','
')}]),angular.module('mgcrea.ngStrap.timepicker').run(['$templateCache',function(t){t.put('timepicker/timepicker.tpl.html','')}]),angular.module('mgcrea.ngStrap.tooltip').run(['$templateCache',function(t){t.put('tooltip/tooltip.tpl.html','
')}]),angular.module('mgcrea.ngStrap.typeahead').run(['$templateCache',function(t){t.put('typeahead/typeahead.tpl.html','')}])}(window,document); \ No newline at end of file +!function(t,e,n){'use strict';angular.module('mgcrea.ngStrap.alert').run(['$templateCache',function(t){t.put('alert/alert.tpl.html','
 
')}]),angular.module('mgcrea.ngStrap.aside').run(['$templateCache',function(t){t.put('aside/aside.tpl.html','')}]),angular.module('mgcrea.ngStrap.datepicker').run(['$templateCache',function(t){t.put('datepicker/datepicker.tpl.html','')}]),angular.module('mgcrea.ngStrap.dropdown').run(['$templateCache',function(t){t.put('dropdown/dropdown.tpl.html','')}]),angular.module('mgcrea.ngStrap.modal').run(['$templateCache',function(t){t.put('modal/modal.tpl.html','')}]),angular.module('mgcrea.ngStrap.popover').run(['$templateCache',function(t){t.put('popover/popover.tpl.html','

')}]),angular.module('mgcrea.ngStrap.select').run(['$templateCache',function(t){t.put('select/select.tpl.html','')}]),angular.module('mgcrea.ngStrap.tab').run(['$templateCache',function(t){t.put('tab/tab.tpl.html','
')}]),angular.module('mgcrea.ngStrap.timepicker').run(['$templateCache',function(t){t.put('timepicker/timepicker.tpl.html','')}]),angular.module('mgcrea.ngStrap.tooltip').run(['$templateCache',function(t){t.put('tooltip/tooltip.tpl.html','
')}]),angular.module('mgcrea.ngStrap.typeahead').run(['$templateCache',function(t){t.put('typeahead/typeahead.tpl.html','')}])}(window,document); \ No newline at end of file diff --git a/dist/modules/affix.js b/dist/modules/affix.js index df62e9d13..471cf1c26 100644 --- a/dist/modules/affix.js +++ b/dist/modules/affix.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -10,7 +10,8 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce' ]).provider('$affix', function() { var defaults = this.defaults = { offsetTop: 'auto', - inlineStyles: true + inlineStyles: true, + setWidth: true }; this.$get = [ '$window', 'debounce', 'dimensions', function($window, debounce, dimensions) { var bodyEl = angular.element($window.document.body); @@ -19,7 +20,14 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm var $affix = {}; var options = angular.extend({}, defaults, config); var targetEl = options.target; - var reset = 'affix affix-top affix-bottom', setWidth = false, initialAffixTop = 0, initialOffsetTop = 0, offsetTop = 0, offsetBottom = 0, affixed = null, unpin = null; + var reset = 'affix affix-top affix-bottom'; + var setWidth = false; + var initialAffixTop = 0; + var initialOffsetTop = 0; + var offsetTop = 0; + var offsetBottom = 0; + var affixed = null; + var unpin = null; var parent = element.parent(); if (options.offsetParent) { if (options.offsetParent.match(/^\d+$/)) { @@ -33,7 +41,7 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm $affix.init = function() { this.$parseOffsets(); initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop; - setWidth = !element[0].style.width; + setWidth = options.setWidth && !element[0].style.width; targetEl.on('scroll', this.checkPosition); targetEl.on('click', this.checkPositionWithEventLoop); windowEl.on('resize', this.$debouncedOnResize); @@ -95,9 +103,11 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm }; $affix.$debouncedOnResize = debounce($affix.$onResize, 50); $affix.$parseOffsets = function() { - var initialPosition = element.css('position'); + var initialPosition = element[0].style.position; + var initialTop = element[0].style.top; if (options.inlineStyles) { element.css('position', options.offsetParent ? '' : 'relative'); + element.css('top', ''); } if (options.offsetTop) { if (options.offsetTop === 'auto') { @@ -123,20 +133,20 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm } if (options.inlineStyles) { element.css('position', initialPosition); + element.css('top', initialTop); } }; - function getRequiredAffixClass(unpin, position, elementHeight) { + function getRequiredAffixClass(_unpin, position, elementHeight) { var scrollTop = getScrollTop(); var scrollHeight = getScrollHeight(); if (scrollTop <= offsetTop) { return 'top'; - } else if (unpin !== null && scrollTop + unpin <= position.top) { - return 'middle'; + } else if (_unpin !== null) { + return scrollTop + _unpin <= position.top ? 'middle' : 'bottom'; } else if (offsetBottom !== null && position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom) { return 'bottom'; - } else { - return 'middle'; } + return 'middle'; } function getScrollTop() { return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop; @@ -149,7 +159,7 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm } return AffixFactory; } ]; -}).directive('bsAffix', [ '$affix', '$window', function($affix, $window) { +}).directive('bsAffix', [ '$affix', '$window', '$timeout', function($affix, $window, $timeout) { return { restrict: 'EAC', require: '^?bsAffixTarget', @@ -158,7 +168,7 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window) }; - angular.forEach([ 'offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles' ], function(key) { + angular.forEach([ 'offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles', 'setWidth' ], function(key) { if (angular.isDefined(attr[key])) { var option = attr[key]; if (/true/i.test(option)) option = true; @@ -166,9 +176,12 @@ angular.module('mgcrea.ngStrap.affix', [ 'mgcrea.ngStrap.helpers.dimensions', 'm options[key] = option; } }); - var affix = $affix(element, options); + var affix; + $timeout(function() { + affix = $affix(element, options); + }); scope.$on('$destroy', function() { - affix && affix.destroy(); + if (affix) affix.destroy(); options = null; affix = null; }); diff --git a/dist/modules/affix.min.js b/dist/modules/affix.min.js index b6c210ca7..cf01b12a0 100644 --- a/dist/modules/affix.min.js +++ b/dist/modules/affix.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.affix',['mgcrea.ngStrap.helpers.dimensions','mgcrea.ngStrap.helpers.debounce']).provider('$affix',function(){var t=this.defaults={offsetTop:'auto',inlineStyles:!0};this.$get=['$window','debounce','dimensions',function(e,o,n){function i(i,r){function c(t,e,o){var n=a(),i=l();return P>=n?'top':null!==t&&n+t<=e.top?'middle':null!==$&&e.top+o+g>=i-$?'bottom':'middle'}function a(){return h[0]===e?e.pageYOffset:h[0].scrollTop}function l(){return h[0]===e?e.document.body.scrollHeight:h[0].scrollHeight}var u={},p=angular.extend({},t,r),h=p.target,d='affix affix-top affix-bottom',m=!1,g=0,v=0,P=0,$=0,x=null,T=null,b=i.parent();if(p.offsetParent)if(p.offsetParent.match(/^\d+$/))for(var k=0;k<1*p.offsetParent-1;k++)b=b.parent();else b=angular.element(p.offsetParent);return u.init=function(){this.$parseOffsets(),v=n.offset(i[0]).top+g,m=!i[0].style.width,h.on('scroll',this.checkPosition),h.on('click',this.checkPositionWithEventLoop),f.on('resize',this.$debouncedOnResize),this.checkPosition(),this.checkPositionWithEventLoop()},u.destroy=function(){h.off('scroll',this.checkPosition),h.off('click',this.checkPositionWithEventLoop),f.off('resize',this.$debouncedOnResize)},u.checkPositionWithEventLoop=function(){setTimeout(u.checkPosition,1)},u.checkPosition=function(){var t=a(),e=n.offset(i[0]),o=n.height(i[0]),f=c(T,e,o);x!==f&&(x=f,'top'===f?(T=null,m&&i.css('width',''),p.inlineStyles&&(i.css('position',p.offsetParent?'':'relative'),i.css('top',''))):'bottom'===f?(T=p.offsetUnpin?-(1*p.offsetUnpin):e.top-t,m&&i.css('width',''),p.inlineStyles&&(i.css('position',p.offsetParent?'':'relative'),i.css('top',p.offsetParent?'':s[0].offsetHeight-$-o-v+'px'))):(T=null,m&&i.css('width',i[0].offsetWidth+'px'),p.inlineStyles&&(i.css('position','fixed'),i.css('top',g+'px'))),i.removeClass(d).addClass('affix'+('middle'!==f?'-'+f:'')))},u.$onResize=function(){u.$parseOffsets(),u.checkPosition()},u.$debouncedOnResize=o(u.$onResize,50),u.$parseOffsets=function(){var t=i.css('position');p.inlineStyles&&i.css('position',p.offsetParent?'':'relative'),p.offsetTop&&('auto'===p.offsetTop&&(p.offsetTop='+0'),p.offsetTop.match(/^[-+]\d+$/)?(g=1*-p.offsetTop,P=p.offsetParent?n.offset(b[0]).top+1*p.offsetTop:n.offset(i[0]).top-n.css(i[0],'marginTop',!0)+1*p.offsetTop):P=1*p.offsetTop),p.offsetBottom&&($=p.offsetParent&&p.offsetBottom.match(/^[-+]\d+$/)?l()-(n.offset(b[0]).top+n.height(b[0]))+1*p.offsetBottom+1:1*p.offsetBottom),p.inlineStyles&&i.css('position',t)},u.init(),u}var s=angular.element(e.document.body),f=angular.element(e);return i}]}).directive('bsAffix',['$affix','$window',function(t,e){return{restrict:'EAC',require:'^?bsAffixTarget',link:function(o,n,i,s){var f={scope:o,target:s?s.$element:angular.element(e)};angular.forEach(['offsetTop','offsetBottom','offsetParent','offsetUnpin','inlineStyles'],function(t){if(angular.isDefined(i[t])){var e=i[t];/true/i.test(e)&&(e=!0),/false/i.test(e)&&(e=!1),f[t]=e}});var r=t(n,f);o.$on('$destroy',function(){r&&r.destroy(),f=null,r=null})}}}]).directive('bsAffixTarget',function(){return{controller:['$element',function(t){this.$element=t}]}}); +'use strict';angular.module('mgcrea.ngStrap.affix',['mgcrea.ngStrap.helpers.dimensions','mgcrea.ngStrap.helpers.debounce']).provider('$affix',function(){var t=this.defaults={offsetTop:'auto',inlineStyles:!0,setWidth:!0};this.$get=['$window','debounce','dimensions',function(e,o,s){function n(n,r){function c(t,e,o){var s=l(),n=a();return s<=v?'top':null!==t?s+t<=e.top?'middle':'bottom':null!==P&&e.top+o+g>=n-P?'bottom':'middle'}function l(){return h[0]===e?e.pageYOffset:h[0].scrollTop}function a(){return h[0]===e?e.document.body.scrollHeight:h[0].scrollHeight}var u={},p=angular.extend({},t,r),h=p.target,d='affix affix-top affix-bottom',m=!1,g=0,$=0,v=0,P=0,x=null,y=null,T=n.parent();if(p.offsetParent)if(p.offsetParent.match(/^\d+$/))for(var b=0;b<1*p.offsetParent-1;b++)T=T.parent();else T=angular.element(p.offsetParent);return u.init=function(){this.$parseOffsets(),$=s.offset(n[0]).top+g,m=p.setWidth&&!n[0].style.width,h.on('scroll',this.checkPosition),h.on('click',this.checkPositionWithEventLoop),f.on('resize',this.$debouncedOnResize),this.checkPosition(),this.checkPositionWithEventLoop()},u.destroy=function(){h.off('scroll',this.checkPosition),h.off('click',this.checkPositionWithEventLoop),f.off('resize',this.$debouncedOnResize)},u.checkPositionWithEventLoop=function(){setTimeout(u.checkPosition,1)},u.checkPosition=function(){var t=l(),e=s.offset(n[0]),o=s.height(n[0]),f=c(y,e,o);x!==f&&(x=f,'top'===f?(y=null,m&&n.css('width',''),p.inlineStyles&&(n.css('position',p.offsetParent?'':'relative'),n.css('top',''))):'bottom'===f?(y=p.offsetUnpin?-(1*p.offsetUnpin):e.top-t,m&&n.css('width',''),p.inlineStyles&&(n.css('position',p.offsetParent?'':'relative'),n.css('top',p.offsetParent?'':i[0].offsetHeight-P-o-$+'px'))):(y=null,m&&n.css('width',n[0].offsetWidth+'px'),p.inlineStyles&&(n.css('position','fixed'),n.css('top',g+'px'))),n.removeClass(d).addClass('affix'+('middle'!==f?'-'+f:'')))},u.$onResize=function(){u.$parseOffsets(),u.checkPosition()},u.$debouncedOnResize=o(u.$onResize,50),u.$parseOffsets=function(){var t=n[0].style.position,e=n[0].style.top;p.inlineStyles&&(n.css('position',p.offsetParent?'':'relative'),n.css('top','')),p.offsetTop&&('auto'===p.offsetTop&&(p.offsetTop='+0'),p.offsetTop.match(/^[-+]\d+$/)?(g=1*-p.offsetTop,v=p.offsetParent?s.offset(T[0]).top+1*p.offsetTop:s.offset(n[0]).top-s.css(n[0],'marginTop',!0)+1*p.offsetTop):v=1*p.offsetTop),p.offsetBottom&&(P=p.offsetParent&&p.offsetBottom.match(/^[-+]\d+$/)?a()-(s.offset(T[0]).top+s.height(T[0]))+1*p.offsetBottom+1:1*p.offsetBottom),p.inlineStyles&&(n.css('position',t),n.css('top',e))},u.init(),u}var i=angular.element(e.document.body),f=angular.element(e);return n}]}).directive('bsAffix',['$affix','$window','$timeout',function(t,e,o){return{restrict:'EAC',require:'^?bsAffixTarget',link:function(s,n,i,f){var r={scope:s,target:f?f.$element:angular.element(e)};angular.forEach(['offsetTop','offsetBottom','offsetParent','offsetUnpin','inlineStyles','setWidth'],function(t){if(angular.isDefined(i[t])){var e=i[t];/true/i.test(e)&&(e=!0),/false/i.test(e)&&(e=!1),r[t]=e}});var c;o(function(){c=t(n,r)}),s.$on('$destroy',function(){c&&c.destroy(),r=null,c=null})}}}]).directive('bsAffixTarget',function(){return{controller:['$element',function(t){this.$element=t}]}}); //# sourceMappingURL=affix.min.js.map diff --git a/dist/modules/affix.min.js.map b/dist/modules/affix.min.js.map index f01603f9d..d27e099f7 100644 --- a/dist/modules/affix.min.js.map +++ b/dist/modules/affix.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/affix.js"],"names":["angular","windowEl","$window","defaults","AffixFactory","offsetTop","$affix","inlineStyles","options","this","$get","reset","setWidth","initialAffixTop","offsetParent","match","getRequiredAffixClass","unpin","position","top","scrollTop","getScrollTop","scrollHeight","getScrollHeight","targetEl","pageYOffset","directive","document","parent","target","initialOffsetTop","offsetBottom","affixed","element","init","i","$parseOffsets","on","checkPosition","require","checkPositionWithEventLoop","dimensions","offset","destroy","style","width","off","affix","setTimeout","height","elementHeight","css","offsetUnpin","bodyEl","offsetHeight","removeClass","addClass","offsetWidth","$onResize","$debouncedOnResize","initialPosition","restrict","affixTarget","$element","forEach","link","key","scope","option","$on","attr","test","controller"],"mappings":"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","file":"modules/affix.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.affix', ['mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce'])\n\n .provider('$affix', function() {\n\n var defaults = this.defaults = {\n offsetTop: 'auto',\n inlineStyles: true\n };\n\n this.$get = function($window, debounce, dimensions) {\n\n var bodyEl = angular.element($window.document.body);\n var windowEl = angular.element($window);\n\n function AffixFactory(element, config) {\n\n var $affix = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var targetEl = options.target;\n\n // Initial private vars\n var reset = 'affix affix-top affix-bottom',\n setWidth = false,\n initialAffixTop = 0,\n initialOffsetTop = 0,\n offsetTop = 0,\n offsetBottom = 0,\n affixed = null,\n unpin = null;\n\n var parent = element.parent();\n // Options: custom parent\n if (options.offsetParent) {\n if (options.offsetParent.match(/^\\d+$/)) {\n for (var i = 0; i < (options.offsetParent * 1) - 1; i++) {\n parent = parent.parent();\n }\n }\n else {\n parent = angular.element(options.offsetParent);\n }\n }\n\n $affix.init = function() {\n\n this.$parseOffsets();\n initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop;\n setWidth = !element[0].style.width;\n\n // Bind events\n targetEl.on('scroll', this.checkPosition);\n targetEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', this.$debouncedOnResize);\n\n // Both of these checkPosition() calls are necessary for the case where\n // the user hits refresh after scrolling to the bottom of the page.\n this.checkPosition();\n this.checkPositionWithEventLoop();\n\n };\n\n $affix.destroy = function() {\n\n // Unbind events\n targetEl.off('scroll', this.checkPosition);\n targetEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', this.$debouncedOnResize);\n\n };\n\n $affix.checkPositionWithEventLoop = function() {\n\n // IE 9 throws an error if we use 'this' instead of '$affix'\n // in this setTimeout call\n setTimeout($affix.checkPosition, 1);\n\n };\n\n $affix.checkPosition = function() {\n // if (!this.$element.is(':visible')) return\n\n var scrollTop = getScrollTop();\n var position = dimensions.offset(element[0]);\n var elementHeight = dimensions.height(element[0]);\n\n // Get required affix class according to position\n var affix = getRequiredAffixClass(unpin, position, elementHeight);\n\n // Did affix status changed this last check?\n if(affixed === affix) return;\n affixed = affix;\n\n if(affix === 'top') {\n unpin = null;\n if(setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n } else if(affix === 'bottom') {\n if (options.offsetUnpin) {\n unpin = -(options.offsetUnpin * 1);\n }\n else {\n // Calculate unpin threshold when affixed to bottom.\n // Hopefully the browser scrolls pixel by pixel.\n unpin = position.top - scrollTop;\n }\n if(setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', (options.offsetParent) ? '' : ((bodyEl[0].offsetHeight - offsetBottom - elementHeight - initialOffsetTop) + 'px'));\n }\n } else { // affix === 'middle'\n unpin = null;\n if(setWidth) {\n element.css('width', element[0].offsetWidth + 'px');\n }\n if (options.inlineStyles) {\n element.css('position', 'fixed');\n element.css('top', initialAffixTop + 'px');\n }\n }\n\n // Add proper affix class\n element.removeClass(reset).addClass('affix' + ((affix !== 'middle') ? '-' + affix : ''));\n\n };\n\n $affix.$onResize = function() {\n $affix.$parseOffsets();\n $affix.checkPosition();\n };\n $affix.$debouncedOnResize = debounce($affix.$onResize, 50);\n\n $affix.$parseOffsets = function() {\n var initialPosition = element.css('position');\n // Reset position to calculate correct offsetTop\n if (options.inlineStyles){\n element.css('position', (options.offsetParent) ? '' : 'relative');\n }\n\n if(options.offsetTop) {\n if(options.offsetTop === 'auto') {\n options.offsetTop = '+0';\n }\n if(options.offsetTop.match(/^[-+]\\d+$/)) {\n initialAffixTop = - options.offsetTop * 1;\n if(options.offsetParent) {\n offsetTop = dimensions.offset(parent[0]).top + (options.offsetTop * 1);\n }\n else {\n offsetTop = dimensions.offset(element[0]).top - dimensions.css(element[0], 'marginTop', true) + (options.offsetTop * 1);\n }\n }\n else {\n offsetTop = options.offsetTop * 1;\n }\n }\n\n if(options.offsetBottom) {\n if(options.offsetParent && options.offsetBottom.match(/^[-+]\\d+$/)) {\n // add 1 pixel due to rounding problems...\n offsetBottom = getScrollHeight() - (dimensions.offset(parent[0]).top + dimensions.height(parent[0])) + (options.offsetBottom * 1) + 1;\n }\n else {\n offsetBottom = options.offsetBottom * 1;\n }\n }\n\n // Bring back the element's position after calculations\n if (options.inlineStyles){\n element.css('position', initialPosition);\n }\n };\n\n // Private methods\n\n function getRequiredAffixClass(unpin, position, elementHeight) {\n\n var scrollTop = getScrollTop();\n var scrollHeight = getScrollHeight();\n\n if(scrollTop <= offsetTop) {\n return 'top';\n } else if(unpin !== null && (scrollTop + unpin <= position.top)) {\n return 'middle';\n } else if(offsetBottom !== null && (position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom)) {\n return 'bottom';\n } else {\n return 'middle';\n }\n\n }\n\n function getScrollTop() {\n return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop;\n }\n\n function getScrollHeight() {\n return targetEl[0] === $window ? $window.document.body.scrollHeight : targetEl[0].scrollHeight;\n }\n\n $affix.init();\n return $affix;\n\n }\n\n return AffixFactory;\n\n };\n\n })\n\n .directive('bsAffix', function($affix, $window) {\n\n return {\n restrict: 'EAC',\n require: '^?bsAffixTarget',\n link: function postLink(scope, element, attr, affixTarget) {\n\n var options = {scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window)};\n angular.forEach(['offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles'], function(key) {\n if(angular.isDefined(attr[key])) {\n var option = attr[key];\n if (/true/i.test(option)) option = true;\n if (/false/i.test(option)) option = false;\n options[key] = option;\n }\n });\n\n var affix = $affix(element, options);\n scope.$on('$destroy', function() {\n affix && affix.destroy();\n options = null;\n affix = null;\n });\n\n }\n };\n\n })\n\n .directive('bsAffixTarget', function() {\n return {\n controller: function($element) {\n this.$element = $element;\n }\n };\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/affix.js"],"names":["angular","bodyEl","document","body","defaults","windowEl","offsetTop","AffixFactory","inlineStyles","$affix","setWidth","options","this","$get","reset","$window","debounce","dimensions","initialOffsetTop","config","getRequiredAffixClass","_unpin","position","top","scrollTop","getScrollTop","scrollHeight","getScrollHeight","targetEl","pageYOffset","offsetBottom","elementHeight","initialAffixTop","directive","$timeout","extend","affixed","target","unpin","parent","element","offsetParent","i","match","init","$parseOffsets","on","checkPosition","require","checkPositionWithEventLoop","offset","destroy","style","width","off","affix","setTimeout","height","css","offsetUnpin","offsetHeight","removeClass","addClass","offsetWidth","$onResize","initialTop","$debouncedOnResize","initialPosition","restrict","affixTarget","$element","forEach","link","key","scope","option","attr","test","controller"],"mappings":"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","file":"affix.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.affix', ['mgcrea.ngStrap.helpers.dimensions', 'mgcrea.ngStrap.helpers.debounce'])\n\n .provider('$affix', function () {\n\n var defaults = this.defaults = {\n offsetTop: 'auto',\n inlineStyles: true,\n setWidth: true\n };\n\n this.$get = function ($window, debounce, dimensions) {\n\n var bodyEl = angular.element($window.document.body);\n var windowEl = angular.element($window);\n\n function AffixFactory (element, config) {\n\n var $affix = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var targetEl = options.target;\n\n // Initial private vars\n var reset = 'affix affix-top affix-bottom';\n var setWidth = false;\n var initialAffixTop = 0;\n var initialOffsetTop = 0;\n var offsetTop = 0;\n var offsetBottom = 0;\n var affixed = null;\n var unpin = null;\n\n var parent = element.parent();\n // Options: custom parent\n if (options.offsetParent) {\n if (options.offsetParent.match(/^\\d+$/)) {\n for (var i = 0; i < (options.offsetParent * 1) - 1; i++) {\n parent = parent.parent();\n }\n } else {\n parent = angular.element(options.offsetParent);\n }\n }\n\n $affix.init = function () {\n\n this.$parseOffsets();\n initialOffsetTop = dimensions.offset(element[0]).top + initialAffixTop;\n setWidth = options.setWidth && !element[0].style.width;\n\n // Bind events\n targetEl.on('scroll', this.checkPosition);\n targetEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', this.$debouncedOnResize);\n\n // Both of these checkPosition() calls are necessary for the case where\n // the user hits refresh after scrolling to the bottom of the page.\n this.checkPosition();\n this.checkPositionWithEventLoop();\n\n };\n\n $affix.destroy = function () {\n\n // Unbind events\n targetEl.off('scroll', this.checkPosition);\n targetEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', this.$debouncedOnResize);\n\n };\n\n $affix.checkPositionWithEventLoop = function () {\n\n // IE 9 throws an error if we use 'this' instead of '$affix'\n // in this setTimeout call\n setTimeout($affix.checkPosition, 1);\n\n };\n\n $affix.checkPosition = function () {\n // if (!this.$element.is(':visible')) return\n\n var scrollTop = getScrollTop();\n var position = dimensions.offset(element[0]);\n var elementHeight = dimensions.height(element[0]);\n\n // Get required affix class according to position\n var affix = getRequiredAffixClass(unpin, position, elementHeight);\n\n // Did affix status changed this last check?\n if (affixed === affix) return;\n affixed = affix;\n\n if (affix === 'top') {\n unpin = null;\n if (setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n } else if (affix === 'bottom') {\n if (options.offsetUnpin) {\n unpin = -(options.offsetUnpin * 1);\n } else {\n // Calculate unpin threshold when affixed to bottom.\n // Hopefully the browser scrolls pixel by pixel.\n unpin = position.top - scrollTop;\n }\n if (setWidth) {\n element.css('width', '');\n }\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', (options.offsetParent) ? '' : ((bodyEl[0].offsetHeight - offsetBottom - elementHeight - initialOffsetTop) + 'px'));\n }\n } else { // affix === 'middle'\n unpin = null;\n if (setWidth) {\n element.css('width', element[0].offsetWidth + 'px');\n }\n if (options.inlineStyles) {\n element.css('position', 'fixed');\n element.css('top', initialAffixTop + 'px');\n }\n }\n\n // Add proper affix class\n element.removeClass(reset).addClass('affix' + ((affix !== 'middle') ? '-' + affix : ''));\n\n };\n\n $affix.$onResize = function () {\n $affix.$parseOffsets();\n $affix.checkPosition();\n };\n $affix.$debouncedOnResize = debounce($affix.$onResize, 50);\n\n $affix.$parseOffsets = function () {\n var initialPosition = element[0].style.position;\n var initialTop = element[0].style.top;\n // Reset position to calculate correct offsetTop\n if (options.inlineStyles) {\n element.css('position', (options.offsetParent) ? '' : 'relative');\n element.css('top', '');\n }\n\n if (options.offsetTop) {\n if (options.offsetTop === 'auto') {\n options.offsetTop = '+0';\n }\n if (options.offsetTop.match(/^[-+]\\d+$/)) {\n initialAffixTop = - options.offsetTop * 1;\n if (options.offsetParent) {\n offsetTop = dimensions.offset(parent[0]).top + (options.offsetTop * 1);\n } else {\n offsetTop = dimensions.offset(element[0]).top - dimensions.css(element[0], 'marginTop', true) + (options.offsetTop * 1);\n }\n } else {\n offsetTop = options.offsetTop * 1;\n }\n }\n\n if (options.offsetBottom) {\n if (options.offsetParent && options.offsetBottom.match(/^[-+]\\d+$/)) {\n // add 1 pixel due to rounding problems...\n offsetBottom = getScrollHeight() - (dimensions.offset(parent[0]).top + dimensions.height(parent[0])) + (options.offsetBottom * 1) + 1;\n } else {\n offsetBottom = options.offsetBottom * 1;\n }\n }\n\n // Bring back the element's position after calculations\n if (options.inlineStyles) {\n element.css('position', initialPosition);\n element.css('top', initialTop);\n }\n };\n\n // Private methods\n\n function getRequiredAffixClass (_unpin, position, elementHeight) {\n var scrollTop = getScrollTop();\n var scrollHeight = getScrollHeight();\n\n if (scrollTop <= offsetTop) {\n return 'top';\n } else if (_unpin !== null) {\n return scrollTop + _unpin <= position.top ? 'middle' : 'bottom';\n } else if (offsetBottom !== null && (position.top + elementHeight + initialAffixTop >= scrollHeight - offsetBottom)) {\n return 'bottom';\n }\n return 'middle';\n }\n\n function getScrollTop () {\n return targetEl[0] === $window ? $window.pageYOffset : targetEl[0].scrollTop;\n }\n\n function getScrollHeight () {\n return targetEl[0] === $window ? $window.document.body.scrollHeight : targetEl[0].scrollHeight;\n }\n\n $affix.init();\n return $affix;\n\n }\n\n return AffixFactory;\n\n };\n\n })\n\n .directive('bsAffix', function ($affix, $window, $timeout) {\n\n return {\n restrict: 'EAC',\n require: '^?bsAffixTarget',\n link: function postLink (scope, element, attr, affixTarget) {\n\n var options = {scope: scope, target: affixTarget ? affixTarget.$element : angular.element($window)};\n angular.forEach(['offsetTop', 'offsetBottom', 'offsetParent', 'offsetUnpin', 'inlineStyles', 'setWidth'], function (key) {\n if (angular.isDefined(attr[key])) {\n var option = attr[key];\n if (/true/i.test(option)) option = true;\n if (/false/i.test(option)) option = false;\n options[key] = option;\n }\n });\n\n var affix;\n $timeout(function () { affix = $affix(element, options); });\n scope.$on('$destroy', function () {\n if (affix) affix.destroy();\n options = null;\n affix = null;\n });\n\n }\n };\n\n })\n\n .directive('bsAffixTarget', function () {\n return {\n controller: function ($element) {\n this.$element = $element;\n }\n };\n });\n"]} \ No newline at end of file diff --git a/dist/modules/alert.js b/dist/modules/alert.js index 8d1313cba..3dae24718 100644 --- a/dist/modules/alert.js +++ b/dist/modules/alert.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -46,7 +46,6 @@ angular.module('mgcrea.ngStrap.alert', [ 'mgcrea.ngStrap.modal' ]).provider('$al return AlertFactory; } ]; }).directive('bsAlert', [ '$window', '$sce', '$alert', function($window, $sce, $alert) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; return { restrict: 'EAC', scope: true, @@ -63,21 +62,31 @@ angular.module('mgcrea.ngStrap.alert', [ 'mgcrea.ngStrap.modal' ]).provider('$al angular.forEach([ 'keyboard', 'html', 'container', 'dismissable' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); if (!scope.hasOwnProperty('title')) { scope.title = ''; } angular.forEach([ 'title', 'content', 'type' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); - }); - attr.bsAlert && scope.$watch(attr.bsAlert, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsAlert) { + scope.$watch(attr.bsAlert, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var alert = $alert(options); element.on(attr.trigger || 'click', alert.toggle); scope.$on('$destroy', function() { diff --git a/dist/modules/alert.min.js b/dist/modules/alert.min.js index 94ac8b608..47bbfd43c 100644 --- a/dist/modules/alert.min.js +++ b/dist/modules/alert.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.alert',['mgcrea.ngStrap.modal']).provider('$alert',function(){var t=this.defaults={animation:'am-fade',prefixClass:'alert',prefixEvent:'alert',placement:null,templateUrl:'alert/alert.tpl.html',container:!1,element:null,backdrop:!1,keyboard:!0,show:!0,duration:!1,type:!1,dismissable:!0};this.$get=['$modal','$timeout',function(e,n){function a(a){var r={},l=angular.extend({},t,a);r=e(l),r.$scope.dismissable=!!l.dismissable,l.type&&(r.$scope.type=l.type);var i=r.show;return l.duration&&(r.show=function(){i(),n(function(){r.hide()},1e3*l.duration)}),r}return a}]}).directive('bsAlert',['$window','$sce','$alert',function(t,e,n){t.requestAnimationFrame||t.setTimeout;return{restrict:'EAC',scope:!0,link:function(t,a,r,l){var i={scope:t,element:a,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','placement','keyboard','html','container','animation','duration','dismissable'],function(t){angular.isDefined(r[t])&&(i[t]=r[t])});var o=/^(false|0|)$/i;angular.forEach(['keyboard','html','container','dismissable'],function(t){angular.isDefined(r[t])&&o.test(r[t])&&(i[t]=!1)}),t.hasOwnProperty('title')||(t.title=''),angular.forEach(['title','content','type'],function(n){r[n]&&r.$observe(n,function(a,r){t[n]=e.trustAsHtml(a)})}),r.bsAlert&&t.$watch(r.bsAlert,function(e,n){angular.isObject(e)?angular.extend(t,e):t.content=e},!0);var s=n(i);a.on(r.trigger||'click',s.toggle),t.$on('$destroy',function(){s&&s.destroy(),i=null,s=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.alert',['mgcrea.ngStrap.modal']).provider('$alert',function(){var e=this.defaults={animation:'am-fade',prefixClass:'alert',prefixEvent:'alert',placement:null,templateUrl:'alert/alert.tpl.html',container:!1,element:null,backdrop:!1,keyboard:!0,show:!0,duration:!1,type:!1,dismissable:!0};this.$get=['$modal','$timeout',function(t,n){function a(a){var r={},o=angular.extend({},e,a);r=t(o),r.$scope.dismissable=!!o.dismissable,o.type&&(r.$scope.type=o.type);var l=r.show;return o.duration&&(r.show=function(){l(),n(function(){r.hide()},1e3*o.duration)}),r}return a}]}).directive('bsAlert',['$window','$sce','$alert',function(e,t,n){return{restrict:'EAC',scope:!0,link:function(e,a,r,o){var l={scope:e,element:a,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','placement','keyboard','html','container','animation','duration','dismissable'],function(e){angular.isDefined(r[e])&&(l[e]=r[e])});var i=/^(false|0|)$/i;angular.forEach(['keyboard','html','container','dismissable'],function(e){angular.isDefined(r[e])&&i.test(r[e])&&(l[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(r[n])&&(l[t]=e.$eval(r[n]))}),e.hasOwnProperty('title')||(e.title=''),angular.forEach(['title','content','type'],function(n){r[n]&&r.$observe(n,function(a,r){e[n]=t.trustAsHtml(a)})}),r.bsAlert&&e.$watch(r.bsAlert,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t},!0);var s=n(l);a.on(r.trigger||'click',s.toggle),e.$on('$destroy',function(){s&&s.destroy(),l=null,s=null})}}}]); //# sourceMappingURL=alert.min.js.map diff --git a/dist/modules/alert.min.js.map b/dist/modules/alert.min.js.map index a765be17c..ffb7d787c 100644 --- a/dist/modules/alert.min.js.map +++ b/dist/modules/alert.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/alert.js"],"names":["angular","container","provider","element","this","defaults","backdrop","keyboard","show","duration","type","dismissable","$get","AlertFactory","$alert","options","$scope","config","extend","$timeout","hide","restrict","link","directive","scope","isDefined","key","falseValueRegExp","forEach","hasOwnProperty","title","attr","newValue","oldValue","trustAsHtml","bsAlert","$observe","content","alert","trigger","on","toggle","$on","destroy"],"mappings":"AAOA,YAEAA,SAOMC,OAAW,wBAAA,yBAAAC,SAAA,SAAA,WANf,GAOIC,GAASC,KAAAC,UACTC,UAAU,UACVC,YAAU,QACVC,YAAM,QAENC,UAAU,KACVC,YAAM,uBACNC,WAAAA,EAPFR,QAAS,KAUTC,UAAKQ,EAEHL,UAASM,EATXL,MAWQM,EAVRL,UAaQM,EAZRL,MAcII,EAbJH,aAgBWK,EAdbZ,MAAKQ,MAgBUI,SAAON,WAAeA,SAAAA,EAAAA,GAfnC,QAASG,GAAaI,GAmBlB,GAAIT,MACDO,EAAQN,QAAUS,UAAAb,EAAAY,EAjBvBH,GAkBWN,EAAOO,GAjBlBD,EAkBMN,OAAAA,cAAAA,EAAAA,YACAW,EAAAA,OAjBJL,EAkBMA,OAAOM,KAAAA,EAAAA,KAhBf,IAAIZ,GAAOM,EAAON,IAoClBa,OAnCIN,GAAQN,WAoBVK,EAAOA,KAAAA,WAlBLN,IAsBJW,EAAON,WApBDC,EAAOM,QA0BN,IAAAL,EAAAN,YAKPY,EAEAC,MAAMT,OAzBTU,UA4BoBC,WAAOA,UAAAA,OAAAA,SAAAA,SAAAA,EAAAA,EAAAA,GAAgBrB,EAAAA,uBAAAA,EAAAA,UA1B5C,QACEkB,SAAU,MACVG,OAyBIxB,EAxBJsB,KAyBM,SAAWG,EAAeC,EAAOX,EAAQW,GAxB7C,GAAIX,IA4BFS,MAAIG,EACJ3B,QAAQ4B,EACNpB,MAAGR,EAMLA,SAAKwB,SAAMK,WAAe,cAAS,aAAA,eAAA,YAAA,WAAA,OAAA,YAAA,YAAA,WAAA,eAAA,SAAAH,GACjCF,QAAMM,UAAQC,EAAAL,MAAAX,EAAAW,GAAAK,EAAAL,KAIhB1B,IAAAA,GAAiB,eAhCnBA,SAiCS0B,SAAQK,WAAcL,OAAK,YAASM,eAAUC,SAAAA,GACjDT,QAAME,UAAYQ,EAAAA,KAAYF,EAAAA,KAAAA,EAAAA,MAAAA,EAAAA,IAAAA,KA/B/BR,EAAMK,eAAe,WAoCxBE,EAAKI,MAAWX,IAjClBxB,QAmCMA,SAAQkB,QAAcc,UAAAA,QAAAA,SAAAA,GAlC1BD,EAAKL,IAmCIK,EAAAK,SAAAV,EAAA,SAAAM,EAAAC,GACLT,EAAMa,GAAAA,EAAUL,YAAAA,OAKpBD,EAAIO,SAAQxB,EAAOC,OAAAA,EAAAA,QAAAA,SAAAA,EAAAA,GAGnBZ,QAAW4B,SAAKQ,GAGhBf,QAAUN,OAAAM,EAAYQ,GAEpBjB,EAAAA,QAAUiB,IAvCX,EACH,IAAIM,GAAQxB,EAAOC,EACnBZ,GAAQqC,GAAGT,EAAKQ,SAAW,QAASD,EAAMG,QAC1CjB,EAAMkB,IAAI,WAAY,WAChBJ,GAAOA,EAAMK,UACjB5B,EAAU,KACVuB,EAAQ","file":"modules/alert.min.js","sourcesContent":["'use strict';\n\n// @BUG: following snippet won't compile correctly\n// @TODO: submit issue to core\n// ' ' +\n\nangular.module('mgcrea.ngStrap.alert', ['mgcrea.ngStrap.modal'])\n\n .provider('$alert', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'alert',\n prefixEvent: 'alert',\n placement: null,\n templateUrl: 'alert/alert.tpl.html',\n container: false,\n element: null,\n backdrop: false,\n keyboard: true,\n show: true,\n // Specific options\n duration: false,\n type: false,\n dismissable: true\n };\n\n this.$get = function($modal, $timeout) {\n\n function AlertFactory(config) {\n\n var $alert = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $alert = $modal(options);\n\n // Support scope as string options [/*title, content, */ type, dismissable]\n $alert.$scope.dismissable = !!options.dismissable;\n if(options.type) {\n $alert.$scope.type = options.type;\n }\n\n // Support auto-close duration\n var show = $alert.show;\n if(options.duration) {\n $alert.show = function() {\n show();\n $timeout(function() {\n $alert.hide();\n }, options.duration * 1000);\n };\n }\n\n return $alert;\n\n }\n\n return AlertFactory;\n\n };\n\n })\n\n .directive('bsAlert', function($window, $sce, $alert) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'keyboard', 'html', 'container', 'animation', 'duration', 'dismissable'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['keyboard', 'html', 'container', 'dismissable'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')){\n scope.title = '';\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content', 'type'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsAlert && scope.$watch(attr.bsAlert, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize alert\n var alert = $alert(options);\n\n // Trigger\n element.on(attr.trigger || 'click', alert.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (alert) alert.destroy();\n options = null;\n alert = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/alert.js"],"names":["angular","container","provider","element","this","defaults","backdrop","keyboard","show","duration","type","dismissable","$get","AlertFactory","$alert","options","$scope","config","extend","$timeout","hide","scope","directive","restrict","link","isDefined","key","falseValueRegExp","forEach","bsKey","charAt","toUpperCase","slice","attr","test","title","$eval","hasOwnProperty","$watch","isObject","newValue","oldValue","bsAlert","alert","trigger","destroy","on","toggle","$on"],"mappings":"AAOA,YAEAA,SAKMC,OAAW,wBAAA,yBAAAC,SAAA,SAAA,WAJf,GAKIC,GAASC,KAAAC,UACTC,UAAU,UACVC,YAAU,QACVC,YAAM,QAENC,UAAU,KACVC,YAAM,uBACNC,WAAAA,EALFR,QAAS,KAQTC,UAAKQ,EAEHL,UAASM,EAPXL,MASQM,EARRL,UAWQM,EAVRL,MAYII,EAXJH,aAcWK,EAZbZ,MAAKQ,MAcUI,SAAON,WAAeA,SAAAA,EAAAA,GAbnC,QAASG,GAAaI,GAiBlB,GAAIT,MACAO,EAAQN,QAAUS,UAAAb,EAAAY,EAfxBH,GAgBWN,EAAOO,GAflBD,EAgBMN,OAAAA,cAAAA,EAAAA,YACAW,EAAAA,OAfJL,EAgBMA,OAAOM,KAAAA,EAAAA,KAdf,IAAIZ,GAAOM,EAAON,IAiClBa,OAhCIN,GAAQN,WAkBVK,EAAOA,KAAAA,WAhBLN,IAoBJW,EAAON,WAlBDC,EAAOM,QAwBN,IAAAL,EAAAN,YAIAK,EAtBT,MA0BQC,OAxBTO,UAwBkCnB,WAASA,UAAAA,OAAAA,SAAAA,SAAAA,EAAAA,EAAAA,GAvB5C,OACEoB,SAAU,MACVF,OAsBIrB,EArBJwB,KAsBM,SAAYC,EAAeC,EAAOX,EAAQW,GArB9C,GAAIX,IAyBFM,MAAIM,EACJ3B,QAAQ4B,EACNpB,MAAIR,EAINA,SAAQ4B,SAAS,WAAA,cAA0B,aAAA,eAA2B,YAAUF,WAAAA,OAAAA,YAAAA,YAAAA,WAAAA,eAAAA,SAAAA,GAC1EG,QAAQJ,UAAOC,EAAII,MAAUC,EAAAA,GAAgBL,EAAIM,KAxBzD,IA0BMjB,GAAeM,eAzBrBrB,SAAQ4B,SAAU,WAAY,OAAQ,YAAa,eAAiB,SAASF,GACvE1B,QAAQyB,UAAUQ,EAAKP,KAASC,EAAiBO,KAAKD,EAAKP,MAAOX,EAAQW,IAAO,KAEvF1B,QA6BUmC,SAAQ,eAAA,SAAA,eAAA,UAAA,SAAAT,GA5BhB,GAAIG,GAAQ,KAAOH,EAAII,OAAO,GAAGC,cAAgBL,EAAIM,MAAM,EAgC3DhC,SAAQ4B,UAASK,EAASJ,MACxBd,EAASW,GAAML,EAAAe,MAAAH,EAAAJ,OA5BdR,EAAMgB,eAAe,WACxBhB,EAAMc,MAAQ,IAmCdnC,QAAIiC,SAAc,QAAA,UAAA,QAAA,SAAAP,GAChBL,EAAMiB,IACJL,EAAIjC,SAAQuC,EAAAA,SAASC,EAAWC,GAC9BzC,EAAAA,GAAQkB,EAAOG,YAAOmB,OA7B1BP,EAiCGS,SAhCLrB,EAAMiB,OAAOL,EAAKS,QAAS,SAASF,EAAUC,GAoC1CE,QAAQ7B,SAAOC,GAGnBZ,QAAW8B,OAAKW,EAAWJ,GAIrBG,EAAAA,QAAaE,IAEjBF,EArCJ,IAAIA,GAAQ7B,EAAOC,EACnBZ,GAAQ2C,GAAGb,EAAKW,SAAW,QAASD,EAAMI,QAC1C1B,EAAM2B,IAAI,WAAY,WAChBL,GAAOA,EAAME,UACjB9B,EAAU,KACV4B,EAAQ","file":"alert.min.js","sourcesContent":["'use strict';\n\n// @BUG: following snippet won't compile correctly\n\nangular.module('mgcrea.ngStrap.alert', ['mgcrea.ngStrap.modal'])\n\n .provider('$alert', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'alert',\n prefixEvent: 'alert',\n placement: null,\n templateUrl: 'alert/alert.tpl.html',\n container: false,\n element: null,\n backdrop: false,\n keyboard: true,\n show: true,\n // Specific options\n duration: false,\n type: false,\n dismissable: true\n };\n\n this.$get = function ($modal, $timeout) {\n\n function AlertFactory (config) {\n\n var $alert = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $alert = $modal(options);\n\n // Support scope as string options [/*title, content, */ type, dismissable]\n $alert.$scope.dismissable = !!options.dismissable;\n if (options.type) {\n $alert.$scope.type = options.type;\n }\n\n // Support auto-close duration\n var show = $alert.show;\n if (options.duration) {\n $alert.show = function () {\n show();\n $timeout(function () {\n $alert.hide();\n }, options.duration * 1000);\n };\n }\n\n return $alert;\n\n }\n\n return AlertFactory;\n\n };\n\n })\n\n .directive('bsAlert', function ($window, $sce, $alert) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'keyboard', 'html', 'container', 'animation', 'duration', 'dismissable'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['keyboard', 'html', 'container', 'dismissable'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')) {\n scope.title = '';\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content', 'type'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsAlert) {\n scope.$watch(attr.bsAlert, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize alert\n var alert = $alert(options);\n\n // Trigger\n element.on(attr.trigger || 'click', alert.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (alert) alert.destroy();\n options = null;\n alert = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/alert.tpl.js b/dist/modules/alert.tpl.js index 861acbe6c..4ee6e0a26 100644 --- a/dist/modules/alert.tpl.js +++ b/dist/modules/alert.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -8,5 +8,5 @@ 'use strict'; angular.module('mgcrea.ngStrap.alert').run([ '$templateCache', function($templateCache) { - $templateCache.put('alert/alert.tpl.html', '
 
'); + $templateCache.put('alert/alert.tpl.html', '
 
'); } ]); \ No newline at end of file diff --git a/dist/modules/alert.tpl.min.js b/dist/modules/alert.tpl.min.js index b9f3710b0..62390d054 100644 --- a/dist/modules/alert.tpl.min.js +++ b/dist/modules/alert.tpl.min.js @@ -1,8 +1,8 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.alert').run(['$templateCache',function(t){t.put('alert/alert.tpl.html','
 
')}]); \ No newline at end of file +'use strict';angular.module('mgcrea.ngStrap.alert').run(['$templateCache',function(t){t.put('alert/alert.tpl.html','
 
')}]); \ No newline at end of file diff --git a/dist/modules/aside.js b/dist/modules/aside.js index 62c38447d..ac144ba77 100644 --- a/dist/modules/aside.js +++ b/dist/modules/aside.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -32,7 +32,6 @@ angular.module('mgcrea.ngStrap.aside', [ 'mgcrea.ngStrap.modal' ]).provider('$as return AsideFactory; } ]; }).directive('bsAside', [ '$window', '$sce', '$aside', function($window, $sce, $aside) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; return { restrict: 'EAC', scope: true, @@ -49,18 +48,28 @@ angular.module('mgcrea.ngStrap.aside', [ 'mgcrea.ngStrap.modal' ]).provider('$as angular.forEach([ 'backdrop', 'keyboard', 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); - attr.bsAside && scope.$watch(attr.bsAside, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + angular.forEach([ 'title', 'content' ], function(key) { + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsAside) { + scope.$watch(attr.bsAside, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var aside = $aside(options); element.on(attr.trigger || 'click', aside.toggle); scope.$on('$destroy', function() { diff --git a/dist/modules/aside.min.js b/dist/modules/aside.min.js index 6d7cd72df..892213c89 100644 --- a/dist/modules/aside.min.js +++ b/dist/modules/aside.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.aside',['mgcrea.ngStrap.modal']).provider('$aside',function(){var e=this.defaults={animation:'am-fade-and-slide-right',prefixClass:'aside',prefixEvent:'aside',placement:'right',templateUrl:'aside/aside.tpl.html',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0};this.$get=['$modal',function(t){function n(n){var a={},r=angular.extend({},e,n);return a=t(r)}return n}]}).directive('bsAside',['$window','$sce','$aside',function(e,t,n){e.requestAnimationFrame||e.setTimeout;return{restrict:'EAC',scope:!0,link:function(e,a,r,i){var o={scope:e,element:a,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation'],function(e){angular.isDefined(r[e])&&(o[e]=r[e])});var l=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(e){angular.isDefined(r[e])&&l.test(r[e])&&(o[e]=!1)}),angular.forEach(['title','content'],function(n){r[n]&&r.$observe(n,function(a,r){e[n]=t.trustAsHtml(a)})}),r.bsAside&&e.$watch(r.bsAside,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t},!0);var c=n(o);a.on(r.trigger||'click',c.toggle),e.$on('$destroy',function(){c&&c.destroy(),o=null,c=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.aside',['mgcrea.ngStrap.modal']).provider('$aside',function(){var e=this.defaults={animation:'am-fade-and-slide-right',prefixClass:'aside',prefixEvent:'aside',placement:'right',templateUrl:'aside/aside.tpl.html',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0};this.$get=['$modal',function(n){function t(t){var a={},r=angular.extend({},e,t);return a=n(r)}return t}]}).directive('bsAside',['$window','$sce','$aside',function(e,n,t){return{restrict:'EAC',scope:!0,link:function(e,a,r,o){var i={scope:e,element:a,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation'],function(e){angular.isDefined(r[e])&&(i[e]=r[e])});var l=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(e){angular.isDefined(r[e])&&l.test(r[e])&&(i[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(n){var t='bs'+n.charAt(0).toUpperCase()+n.slice(1);angular.isDefined(r[t])&&(i[n]=e.$eval(r[t]))}),angular.forEach(['title','content'],function(t){r[t]&&r.$observe(t,function(a,r){e[t]=n.trustAsHtml(a)})}),r.bsAside&&e.$watch(r.bsAside,function(n,t){angular.isObject(n)?angular.extend(e,n):e.content=n},!0);var c=t(i);a.on(r.trigger||'click',c.toggle),e.$on('$destroy',function(){c&&c.destroy(),i=null,c=null})}}}]); //# sourceMappingURL=aside.min.js.map diff --git a/dist/modules/aside.min.js.map b/dist/modules/aside.min.js.map index 6d595d34b..ce2c87093 100644 --- a/dist/modules/aside.min.js.map +++ b/dist/modules/aside.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/aside.js"],"names":["angular","contentTemplate","provider","container","this","defaults","element","backdrop","keyboard","html","show","$get","AsideFactory","$aside","options","config","directive","requestAnimationFrame","restrict","$modal","link","scope","isDefined","key","falseValueRegExp","forEach","attr","$observe","newValue","oldValue","bsAside","$watch","extend","content","aside","trigger","on","toggle","$on","destroy"],"mappings":"AAOA,YAEAA,SAGMC,OAAAA,wBAAiB,yBAAAC,SAAA,SAAA,WAFrB,GAGIC,GAAWC,KAAAC,UACXC,UAAS,0BACTC,YAAU,QACVC,YAAU,QACVC,UAAM,QACNC,YAAM,uBAFRT,iBAAiB,EAKjBG,WAAKO,EAEHL,QAASM,KAJXL,UAMQM,EALRL,UAQQM,EAPRL,MASII,EARJH,MAUI,EARNN,MAYIO,MAAOC,SAAAA,SAAAA,GAXT,QAASA,GAAaG,GAiBvBC,GAAAA,MAEKC,EAAAA,QAAAA,UAAgCA,EAAAA,EAGlCC,OADFL,GAAAM,EAAAL,GAGEM,MAAMR,OAhBTI,UAkBoBK,WAAOA,UAAAA,OAAAA,SAAAA,SAAAA,EAAAA,EAAAA,GAAgBf,EAAAA,uBAAAA,EAAAA,UAhB5C,QACEY,SAAU,MACVG,OAeIrB,EAdJoB,KAeM,SAAWE,EAAeC,EAAOT,EAAQS,GAd7C,GAAIT,IAkBFO,MAAIG,EACJxB,QAAQyB,EACNf,MAAGV,EAKLA,SAAQyB,SAAS,WAAS,cAAY,aAASF,eAAAA,kBAAAA,YAAAA,WAAAA,WAAAA,OAAAA,YAAAA,aAAAA,SAAAA,GAC7CG,QAAKH,UAAaI,EAASJ,MAAKT,EAASc,GAAAA,EAAUC,KAlBvD,IAAIL,GAAmB,eACvBxB,SAAQyB,SAAU,WAAY,WAAY,OAAQ,aAAe,SAASF,GAuBnEO,QAAAA,UAAiBC,EAAOL,KAAKI,EAAkBF,KAAAA,EAAUC,MAAAA,EAAAA,IAAAA,KApBhE7B,QAsBMA,SAAQgC,QAAcJ,WAAAA,SAAAA,GArB1BF,EAAKH,IAsBIG,EAAAC,SAAAJ,EAAA,SAAAK,EAAAC,GACLR,EAAMY,GAAAA,EAAUL,YAAAA,OAKpBF,EAAIQ,SAAQrB,EAAOC,OAAAA,EAAAA,QAAAA,SAAAA,EAAAA,GAGnBR,QAAWoB,SAAKS,GAGhBd,QAAUW,OAAAX,EAAYO,GAEpBd,EAAAA,QAAUc,IA1BX,EACH,IAAIM,GAAQrB,EAAOC,EACnBR,GAAQ8B,GAAGV,EAAKS,SAAW,QAASD,EAAMG,QAC1ChB,EAAMiB,IAAI,WAAY,WAChBJ,GAAOA,EAAMK,UACjBzB,EAAU,KACVoB,EAAQ","file":"modules/aside.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.aside', ['mgcrea.ngStrap.modal'])\n\n .provider('$aside', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade-and-slide-right',\n prefixClass: 'aside',\n prefixEvent: 'aside',\n placement: 'right',\n templateUrl: 'aside/aside.tpl.html',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function($modal) {\n\n function AsideFactory(config) {\n\n var $aside = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $aside = $modal(options);\n\n return $aside;\n\n }\n\n return AsideFactory;\n\n };\n\n })\n\n .directive('bsAside', function($window, $sce, $aside) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsAside && scope.$watch(attr.bsAside, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize aside\n var aside = $aside(options);\n\n // Trigger\n element.on(attr.trigger || 'click', aside.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (aside) aside.destroy();\n options = null;\n aside = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/aside.js"],"names":["angular","contentTemplate","provider","container","this","defaults","element","backdrop","keyboard","html","show","$get","AsideFactory","$aside","options","config","directive","extend","scope","restrict","$modal","link","isDefined","key","falseValueRegExp","forEach","bsKey","charAt","toUpperCase","slice","attr","test","$observe","newValue","oldValue","trustAsHtml","$eval","$watch","isObject","bsAside","aside","trigger","destroy","on","toggle","$on"],"mappings":"AAOA,YAEAA,SAGMC,OAAAA,wBAAiB,yBAAAC,SAAA,SAAA,WAFrB,GAGIC,GAAWC,KAAAC,UACXC,UAAS,0BACTC,YAAU,QACVC,YAAU,QACVC,UAAM,QACNC,YAAM,uBAFRT,iBAAiB,EAKjBG,WAAKO,EAEHL,QAASM,KAJXL,UAMQM,EALRL,UAQQM,EAPRL,MASII,EARJH,MAUI,EARNN,MAYIO,MAAOC,SAAAA,SAAAA,GAXT,QAASA,GAAaG,GAiBvBC,GAAAA,MAECF,EAAAd,QAAAiB,UAAAZ,EAAAU,EAEEG,OADAC,GAAUC,EAAAN,GAdZ,MAkBQA,OAhBTE,UAgBkCV,WAASA,UAAAA,OAAAA,SAAAA,SAAAA,EAAAA,EAAAA,GAf5C,OACEa,SAAU,MACVD,OAcIlB,EAbJqB,KAcM,SAAYC,EAAeC,EAAOT,EAAQS,GAb9C,GAAIT,IAiBFI,MAAIM,EACJxB,QAAQyB,EACNf,MAAIV,EAINA,SAAQyB,SAAS,WAAA,cAA0B,aAAA,eAA2B,kBAAUF,YAAAA,WAAAA,WAAAA,OAAAA,YAAAA,aAAAA,SAAAA,GAC1EG,QAAQJ,UAAOC,EAAII,MAAUC,EAAAA,GAAgBL,EAAIM,KAhBzD,IAkBMf,GAAeI,eAjBrBlB,SAAQyB,SAAU,WAAY,WAAY,OAAQ,aAAe,SAASF,GACpEvB,QAAQsB,UAAUQ,EAAKP,KAASC,EAAiBO,KAAKD,EAAKP,MAAOT,EAAQS,IAAO,KAEvFvB,QAoBQ8B,SAAW,eAAA,SAAA,eAAA,UAAA,SAAAP,GAnBjB,GAoBIO,GAAKE,KAAST,EAAKI,OAAUM,GAAAA,cAAUC,EAAAA,MAAAA,EACrChB,SAAMK,UAAYY,EAAAA,MAnBtBrB,EAAQS,GAAOL,EAAMkB,MAAMN,EAAKJ,OAyBlC1B,QAAI8B,SAAc,QAAA,WAAA,SAAAP,GAChBL,EAAMmB,IACJP,EAAI9B,SAAQsC,EAAAA,SAASL,EAAWC,GAC9BlC,EAAAA,GAAQiB,EAAOC,YAAOe,OAlB1BH,EAsBGS,SArBLrB,EAAMmB,OAAOP,EAAKS,QAAS,SAASN,EAAUC,GAyB1CM,QAAQ3B,SAAOC,GAGnBR,QAAWwB,OAAKW,EAAWR,GAIrBO,EAAAA,QAAaE,IAEjBF,EA1BJ,IAAIA,GAAQ3B,EAAOC,EACnBR,GAAQqC,GAAGb,EAAKW,SAAW,QAASD,EAAMI,QAC1C1B,EAAM2B,IAAI,WAAY,WAChBL,GAAOA,EAAME,UACjB5B,EAAU,KACV0B,EAAQ","file":"aside.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.aside', ['mgcrea.ngStrap.modal'])\n\n .provider('$aside', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade-and-slide-right',\n prefixClass: 'aside',\n prefixEvent: 'aside',\n placement: 'right',\n templateUrl: 'aside/aside.tpl.html',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function ($modal) {\n\n function AsideFactory (config) {\n\n var $aside = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $aside = $modal(options);\n\n return $aside;\n\n }\n\n return AsideFactory;\n\n };\n\n })\n\n .directive('bsAside', function ($window, $sce, $aside) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsAside) {\n scope.$watch(attr.bsAside, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize aside\n var aside = $aside(options);\n\n // Trigger\n element.on(attr.trigger || 'click', aside.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (aside) aside.destroy();\n options = null;\n aside = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/aside.tpl.js b/dist/modules/aside.tpl.js index 3460e50a7..cf64e22ce 100644 --- a/dist/modules/aside.tpl.js +++ b/dist/modules/aside.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/aside.tpl.min.js b/dist/modules/aside.tpl.min.js index 48725c076..b22e663e9 100644 --- a/dist/modules/aside.tpl.min.js +++ b/dist/modules/aside.tpl.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/button.js b/dist/modules/button.js index fadedd395..1b027394e 100644 --- a/dist/modules/button.js +++ b/dist/modules/button.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -58,12 +58,9 @@ angular.module('mgcrea.ngStrap.button', []).provider('$button', function() { controller.$formatters.push(function(modelValue) { return angular.equals(modelValue, trueValue); }); - scope.$watch(attr.ngModel, function(newValue, oldValue) { - controller.$render(); - }); } controller.$render = function() { - var isActive = angular.equals(controller.$modelValue, trueValue); + var isActive = !!controller.$viewValue; $$rAF(function() { if (isInput) element[0].checked = isActive; activeElement.toggleClass(options.activeClass, isActive); @@ -74,9 +71,7 @@ angular.module('mgcrea.ngStrap.button', []).provider('$button', function() { if (!isInput) { controller.$setViewValue(!activeElement.hasClass('active')); } - if (!hasExoticValues) { - controller.$render(); - } + controller.$render(); }); }); } @@ -107,11 +102,15 @@ angular.module('mgcrea.ngStrap.button', []).provider('$button', function() { var activeElement = isInput ? element.parent() : element; var value; attr.$observe('value', function(v) { - value = constantValueRegExp.test(v) ? scope.$eval(v) : v; + if (typeof v !== 'boolean' && constantValueRegExp.test(v)) { + value = scope.$eval(v); + } else { + value = v; + } controller.$render(); }); controller.$render = function() { - var isActive = angular.equals(controller.$modelValue, value); + var isActive = angular.equals(controller.$viewValue, value); $$rAF(function() { if (isInput) element[0].checked = isActive; activeElement.toggleClass(options.activeClass, isActive); diff --git a/dist/modules/button.min.js b/dist/modules/button.min.js index 7c5b1ba51..b36376e2a 100644 --- a/dist/modules/button.min.js +++ b/dist/modules/button.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.button',[]).provider('$button',function(){var e=this.defaults={activeClass:'active',toggleEvent:'click'};this.$get=function(){return{defaults:e}}}).directive('bsCheckboxGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var n=e[0].querySelectorAll('input[type="checkbox"]');angular.forEach(n,function(e){var n=angular.element(e);n.attr('bs-checkbox',''),n.attr('ng-model',t.ngModel+'.'+n.attr('value'))})}}}).directive('bsCheckbox',['$button','$$rAF',function(e,t){var n=e.defaults,r=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,a,u,l){var o=n,i='INPUT'===a[0].nodeName,c=i?a.parent():a,s=angular.isDefined(u.trueValue)?u.trueValue:!0;r.test(u.trueValue)&&(s=e.$eval(u.trueValue));var d=angular.isDefined(u.falseValue)?u.falseValue:!1;r.test(u.falseValue)&&(d=e.$eval(u.falseValue));var f='boolean'!=typeof s||'boolean'!=typeof d;f&&(l.$parsers.push(function(e){return e?s:d}),l.$formatters.push(function(e){return angular.equals(e,s)}),e.$watch(u.ngModel,function(e,t){l.$render()})),l.$render=function(){var e=angular.equals(l.$modelValue,s);t(function(){i&&(a[0].checked=e),c.toggleClass(o.activeClass,e)})},a.bind(o.toggleEvent,function(){e.$apply(function(){i||l.$setViewValue(!c.hasClass('active')),f||l.$render()})})}}}]).directive('bsRadioGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var n=e[0].querySelectorAll('input[type="radio"]');angular.forEach(n,function(e){angular.element(e).attr('bs-radio',''),angular.element(e).attr('ng-model',t.ngModel)})}}}).directive('bsRadio',['$button','$$rAF',function(e,t){var n=e.defaults,r=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,a,u,l){var o,i=n,c='INPUT'===a[0].nodeName,s=c?a.parent():a;u.$observe('value',function(t){o=r.test(t)?e.$eval(t):t,l.$render()}),l.$render=function(){var e=angular.equals(l.$modelValue,o);t(function(){c&&(a[0].checked=e),s.toggleClass(i.activeClass,e)})},a.bind(i.toggleEvent,function(){e.$apply(function(){l.$setViewValue(o),l.$render()})})}}}]); +'use strict';angular.module('mgcrea.ngStrap.button',[]).provider('$button',function(){var e=this.defaults={activeClass:'active',toggleEvent:'click'};this.$get=function(){return{defaults:e}}}).directive('bsCheckboxGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var r=e[0].querySelectorAll('input[type="checkbox"]');angular.forEach(r,function(e){var r=angular.element(e);r.attr('bs-checkbox',''),r.attr('ng-model',t.ngModel+'.'+r.attr('value'))})}}}).directive('bsCheckbox',['$button','$$rAF',function(e,t){var r=e.defaults,n=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,a,u,l){var o=r,i='INPUT'===a[0].nodeName,c=i?a.parent():a,s=!angular.isDefined(u.trueValue)||u.trueValue;n.test(u.trueValue)&&(s=e.$eval(u.trueValue));var d=!!angular.isDefined(u.falseValue)&&u.falseValue;n.test(u.falseValue)&&(d=e.$eval(u.falseValue));var f='boolean'!=typeof s||'boolean'!=typeof d;f&&(l.$parsers.push(function(e){return e?s:d}),l.$formatters.push(function(e){return angular.equals(e,s)})),l.$render=function(){var e=!!l.$viewValue;t(function(){i&&(a[0].checked=e),c.toggleClass(o.activeClass,e)})},a.bind(o.toggleEvent,function(){e.$apply(function(){i||l.$setViewValue(!c.hasClass('active')),l.$render()})})}}}]).directive('bsRadioGroup',function(){return{restrict:'A',require:'ngModel',compile:function(e,t){e.attr('data-toggle','buttons'),e.removeAttr('ng-model');var r=e[0].querySelectorAll('input[type="radio"]');angular.forEach(r,function(e){angular.element(e).attr('bs-radio',''),angular.element(e).attr('ng-model',t.ngModel)})}}}).directive('bsRadio',['$button','$$rAF',function(e,t){var r=e.defaults,n=/^(true|false|\d+)$/;return{restrict:'A',require:'ngModel',link:function(e,a,u,l){var o,i=r,c='INPUT'===a[0].nodeName,s=c?a.parent():a;u.$observe('value',function(t){o='boolean'!=typeof t&&n.test(t)?e.$eval(t):t,l.$render()}),l.$render=function(){var e=angular.equals(l.$viewValue,o);t(function(){c&&(a[0].checked=e),s.toggleClass(i.activeClass,e)})},a.bind(i.toggleEvent,function(){e.$apply(function(){l.$setViewValue(o),l.$render()})})}}}]); //# sourceMappingURL=button.min.js.map diff --git a/dist/modules/button.min.js.map b/dist/modules/button.min.js.map index be65ba9ab..909f60f0f 100644 --- a/dist/modules/button.min.js.map +++ b/dist/modules/button.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/button.js"],"names":["angular","defaults","this","activeClass","directive","restrict","require","compile","children","querySelectorAll","childEl","element","ngModel","attr","removeAttr","forEach","child","$button","constantValueRegExp","options","isInput","trueValue","link","scope","controller","falseValue","test","isDefined","hasExoticValues","viewValue","$formatters","push","$render","modelValue","isActive","equals","checked","activeElement","bind","toggleEvent","$modelValue","$$rAF","$setViewValue","toggleClass","$apply","hasClass","$observe","value","nodeName"],"mappings":"AAOA,YAEAA,SAGcC,OAAAA,4BAAUA,SAAAA,UAAAA,WAFtB,GAAIA,GAAWC,KAAKD,UAClBE,YAAa,SAMdC,YAAU,QAHXF,MAMIG,KAAU,WACVC,OACAC,SAASN,MAHZG,UAMSI,kBAAsBC,WALhC,OACEJ,SAMUK,IALVJ,QAMMI,UALNH,QAMMG,SAAaC,EAAiBC,GALlCD,EAAQE,KAAK,cAAe,WAC5BF,EAAQG,WAAW,WACnB,IAAIN,GAAWG,EAAQ,GAAGF,iBAAiB,yBAW9CL,SAAUW,QAAAP,EAAA,SAAAQ,GAELf,GAAAA,GAAWgB,QAAQhB,QAAAA,EACnBiB,GAAAA,KAAAA,cAAsB,IAE1BR,EAAAG,KAAA,WAAAA,EAAAD,QAAA,IAAAF,EAAAG,KAAA,gBARDT,UAaSe,cAAUlB,UAAAA,QAAAA,SAAAA,EAAAA,GAZpB,GAeMA,GAAImB,EAAUT,SACdO,EAAoBE,oBAd1B,QACEf,SAgBOa,IAfPZ,QAgBMe,UAfNC,KAAM,SAAkBC,EAAOZ,EAASE,EAAMW,GAiB1C,GAAIC,GAAAA,EACDP,EAA8BO,UAA9BP,EAAAA,GAAoBQ,SACrBD,EAAaF,EAAYV,EAAKY,SAAAA,EAf9BJ,EAAYrB,QAAQ2B,UAAUd,EAAKQ,WAAaR,EAAKQ,WAAY,CAmB/DO,GAAAA,KAAyBP,EAAAA,aAC7BA,EAAGO,EAAAA,MAAiBf,EAAAQ,WAhBtB,IAmBMI,GAAOI,QAAYR,UAAYI,EAAAA,YAAAA,EAAAA,YAAAA,CAlBjCP,GAAoBQ,KAAKb,EAAKY,cAqB9BD,EAAWM,EAAAA,MAAYC,EAAKN,YAlBhC,IAAIG,GAAuC,iBAAdP,IAAiD,iBAAfI,EAuB3DF,KArBFC,EAsBIA,SAAWQ,KAAAA,SAAAA,GArBb,MAAOH,GAAYR,EAAYI,IA0BjCD,EAAWQ,YAAUD,KAAA,SAAAE,GAEnB,MAAIC,SAAWlC,OAAQmC,EAAOX,KAvBhCD,EAyBOH,OAAAA,EAAST,QAAWyB,SAAUF,EAAAA,GACjCG,EAAAA,aAKJ1B,EAAQ2B,QAAaC,WACnBhB,GAAAA,GAAavB,QAAAmC,OAAAX,EAAAgB,YAAAnB,EA1BfoB,GA4BI,WACEjB,IAAWkB,EAAAA,GAAAA,QAAeL,GA3B9BA,EAAcM,YAAYxB,EAAQhB,YAAa+B,MAGnDvB,EAAQ2B,KAAKnB,EAAQoB,YAAa,WAChChB,EAAMqB,OAAO,WACNxB,GACHI,EAAWkB,eAAeL,EAAcQ,SAAS,WAmChDjB,GAETJ,EAAAQ,mBA5BC5B,UAkCOI,eAAmB,WAjC7B,OACEH,SAkCML,IAjCNM,QAkCMN,UAjCNO,QAAS,SAAkBI,EAASE,GAClCF,EAAQE,KAAK,cAAe,WAC5BF,EAAQG,WAAW,WAuCtBV,IAAAA,GAAUO,EAAA,GAAAF,iBAAW,sBAEhBR,SAAAA,QAAWgB,EAAQhB,SAAAA,GACnBiB,QAAAA,QAAAA,GAAsBL,KAAA,WAAA,IAE1Bb,QAAAW,QAAAK,GAAAH,KAAA,WAAAA,EAAAD,eApCDR,UAyCSe,WAAUlB,UAAAA,QAAAA,SAAAA,EAAAA,GAxCpB,GA2CMA,GAAImB,EAAUT,SACdO,EAAoBE,oBA1C1B,QACEf,SA4CSyC,IA3CTxC,QA4CMyC,UA3CNzB,KA4CME,SAAWQ,EAAAA,EAAAA,EAAAA,GA3Cf,GAkDIS,GAlDAtB,EAAUlB,EA+CZuB,EAAqB,UAAVQ,EAAU,GAAAgB,SAEnBX,EAAerC,EAAQmC,EAAOX,SAAWgB,CA7C7C3B,GA+CMiC,SAAG1B,QAAST,SAAWyB,GA9C3BW,EA+CIV,EAA0BlB,KAAAA,GAAQhB,EAAAA,MAAa+B,GAAAA,EA9CnDV,EAAWQ,YAmDXrB,EAAQ2B,QAAaC,WACnBhB,GAAAA,GAAavB,QAAAmC,OAAAX,EAAAgB,YAAAO,EAhDfN,GAkDIjB,WACAA,IAAWQ,EAAAA,GAAAA,QAAAA,GAjDbK,EAAcM,YAAYxB,EAAQhB,YAAa+B,MAGnDvB,EAAQ2B,KAAKnB,EAAQoB,YAAa,WAChChB,EAAMqB,OAAO,WACXpB,EAAWkB,cAAcK,GACzBvB,EAAWQ","file":"modules/button.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.button', [])\n\n .provider('$button', function() {\n\n var defaults = this.defaults = {\n activeClass:'active',\n toggleEvent:'click'\n };\n\n this.$get = function() {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsCheckboxGroup', function() {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink(element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"checkbox\"]');\n angular.forEach(children, function(child) {\n var childEl = angular.element(child);\n childEl.attr('bs-checkbox', '');\n childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value'));\n });\n }\n\n };\n\n })\n\n .directive('bsCheckbox', function($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support label > input[type=\"checkbox\"]\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true;\n if(constantValueRegExp.test(attr.trueValue)) {\n trueValue = scope.$eval(attr.trueValue);\n }\n var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false;\n if(constantValueRegExp.test(attr.falseValue)) {\n falseValue = scope.$eval(attr.falseValue);\n }\n\n // Parse exotic values\n var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean';\n if(hasExoticValues) {\n controller.$parsers.push(function(viewValue) {\n // console.warn('$parser', element.attr('ng-model'), 'viewValue', viewValue);\n return viewValue ? trueValue : falseValue;\n });\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n return angular.equals(modelValue, trueValue);\n });\n // Fix rendering for exotic values\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n controller.$render();\n });\n }\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$modelValue, trueValue);\n $$rAF(function() {\n if(isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function() {\n scope.$apply(function () {\n // console.warn('!click', element.attr('ng-model'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n if(!isInput) {\n controller.$setViewValue(!activeElement.hasClass('active'));\n }\n if(!hasExoticValues) {\n controller.$render();\n }\n });\n });\n\n }\n\n };\n\n })\n\n .directive('bsRadioGroup', function() {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink(element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"radio\"]');\n angular.forEach(children, function(child) {\n angular.element(child).attr('bs-radio', '');\n angular.element(child).attr('ng-model', attr.ngModel);\n });\n }\n\n };\n\n })\n\n .directive('bsRadio', function($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support `label > input[type=\"radio\"]` markup\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var value;\n attr.$observe('value', function(v) {\n value = constantValueRegExp.test(v) ? scope.$eval(v) : v;\n controller.$render();\n });\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('value'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$modelValue, value);\n $$rAF(function() {\n if(isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function() {\n scope.$apply(function () {\n // console.warn('!click', element.attr('value'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n controller.$setViewValue(value);\n controller.$render();\n });\n });\n\n }\n\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/button.js"],"names":["angular","defaults","this","activeClass","directive","restrict","require","compile","children","querySelectorAll","childEl","element","ngModel","attr","removeAttr","forEach","child","$button","constantValueRegExp","options","isInput","trueValue","link","scope","controller","falseValue","test","isDefined","hasExoticValues","viewValue","$formatters","push","$render","isActive","$viewValue","checked","activeElement","toggleClass","bind","toggleEvent","$$rAF","$setViewValue","$apply","hasClass","$observe","value","$eval","v","parent","equals"],"mappings":"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","file":"button.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.button', [])\n\n .provider('$button', function () {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n toggleEvent: 'click'\n };\n\n this.$get = function () {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsCheckboxGroup', function () {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink (element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"checkbox\"]');\n angular.forEach(children, function (child) {\n var childEl = angular.element(child);\n childEl.attr('bs-checkbox', '');\n childEl.attr('ng-model', attr.ngModel + '.' + childEl.attr('value'));\n });\n }\n\n };\n\n })\n\n .directive('bsCheckbox', function ($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support label > input[type=\"checkbox\"]\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var trueValue = angular.isDefined(attr.trueValue) ? attr.trueValue : true;\n if (constantValueRegExp.test(attr.trueValue)) {\n trueValue = scope.$eval(attr.trueValue);\n }\n var falseValue = angular.isDefined(attr.falseValue) ? attr.falseValue : false;\n if (constantValueRegExp.test(attr.falseValue)) {\n falseValue = scope.$eval(attr.falseValue);\n }\n\n // Parse exotic values\n var hasExoticValues = typeof trueValue !== 'boolean' || typeof falseValue !== 'boolean';\n if (hasExoticValues) {\n controller.$parsers.push(function (viewValue) {\n // console.warn('$parser', element.attr('ng-model'), 'viewValue', viewValue);\n return viewValue ? trueValue : falseValue;\n });\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n return angular.equals(modelValue, trueValue);\n });\n }\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = !!controller.$viewValue;\n $$rAF(function () {\n if (isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function () {\n scope.$apply(function () {\n // console.warn('!click', element.attr('ng-model'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n if (!isInput) {\n controller.$setViewValue(!activeElement.hasClass('active'));\n }\n controller.$render();\n });\n });\n\n }\n\n };\n\n })\n\n .directive('bsRadioGroup', function () {\n\n return {\n restrict: 'A',\n require: 'ngModel',\n compile: function postLink (element, attr) {\n element.attr('data-toggle', 'buttons');\n element.removeAttr('ng-model');\n var children = element[0].querySelectorAll('input[type=\"radio\"]');\n angular.forEach(children, function (child) {\n angular.element(child).attr('bs-radio', '');\n angular.element(child).attr('ng-model', attr.ngModel);\n });\n }\n\n };\n\n })\n\n .directive('bsRadio', function ($button, $$rAF) {\n\n var defaults = $button.defaults;\n var constantValueRegExp = /^(true|false|\\d+)$/;\n\n return {\n restrict: 'A',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n var options = defaults;\n\n // Support `label > input[type=\"radio\"]` markup\n var isInput = element[0].nodeName === 'INPUT';\n var activeElement = isInput ? element.parent() : element;\n\n var value;\n attr.$observe('value', function (v) {\n if (typeof v !== 'boolean' && constantValueRegExp.test(v)) {\n value = scope.$eval(v);\n } else {\n value = v;\n }\n controller.$render();\n });\n\n // model -> view\n controller.$render = function () {\n // console.warn('$render', element.attr('value'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var isActive = angular.equals(controller.$viewValue, value);\n $$rAF(function () {\n if (isInput) element[0].checked = isActive;\n activeElement.toggleClass(options.activeClass, isActive);\n });\n };\n\n // view -> model\n element.bind(options.toggleEvent, function () {\n scope.$apply(function () {\n // console.warn('!click', element.attr('value'), 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue, 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue);\n controller.$setViewValue(value);\n controller.$render();\n });\n });\n\n }\n\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/collapse.js b/dist/modules/collapse.js index 0131aea46..4a0328fa2 100644 --- a/dist/modules/collapse.js +++ b/dist/modules/collapse.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -55,8 +55,8 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { self.$setActive = $scope.$setActive = function(value) { if (angular.isArray(value)) { self.$targets.$active = value; - } else if (!self.$options.disallowToggle) { - isActive(value) ? deactivateItem(value) : activateItem(value); + } else if (!self.$options.disallowToggle && isActive(value)) { + deactivateItem(value); } else { activateItem(value); } @@ -65,7 +65,10 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { }); }; self.$activeIndexes = function() { - return self.$options.allowMultiple ? self.$targets.$active : self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1; + if (self.$options.allowMultiple) { + return self.$targets.$active; + } + return self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1; }; function fixActiveItemIndexes(index) { var activeIndexes = self.$targets.$active; @@ -80,7 +83,7 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { } function isActive(value) { var activeItems = self.$targets.$active; - return activeItems.indexOf(value) === -1 ? false : true; + return activeItems.indexOf(value) !== -1; } function deactivateItem(value) { var index = self.$targets.$active.indexOf(value); @@ -104,7 +107,6 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { return $collapse; }; }).directive('bsCollapse', [ '$window', '$animate', '$collapse', function($window, $animate, $collapse) { - var defaults = $collapse.defaults; return { require: [ '?ngModel', 'bsCollapse' ], controller: [ '$scope', '$element', '$attrs', $collapse.controller ], @@ -137,7 +139,6 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { return { require: [ '^?ngModel', '^bsCollapse' ], link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsCollapseCtrl = controllers[1]; element.attr('data-toggle', 'collapse'); bsCollapseCtrl.$registerToggle(element); @@ -145,9 +146,11 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { bsCollapseCtrl.$unregisterToggle(element); }); element.on('click', function() { - var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element); - bsCollapseCtrl.$setActive(index * 1); - scope.$apply(); + if (!attrs.disabled) { + var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element); + bsCollapseCtrl.$setActive(index * 1); + scope.$apply(); + } }); } }; @@ -155,7 +158,6 @@ angular.module('mgcrea.ngStrap.collapse', []).provider('$collapse', function() { return { require: [ '^?ngModel', '^bsCollapse' ], link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsCollapseCtrl = controllers[1]; element.addClass('collapse'); if (bsCollapseCtrl.$options.animation) { diff --git a/dist/modules/collapse.min.js b/dist/modules/collapse.min.js index d144c202f..98bf63a1b 100644 --- a/dist/modules/collapse.min.js +++ b/dist/modules/collapse.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.collapse',[]).provider('$collapse',function(){var e=this.defaults={animation:'am-collapse',disallowToggle:!1,activeClass:'in',startCollapsed:!1,allowMultiple:!1},t=this.controller=function(t,a,s){function i(e){for(var t=r.$targets.$active,a=0;a $formatters -> viewValue\n ngModelCtrl.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n if (angular.isArray(modelValue)) {\n // model value is an array, so just replace\n // the active items directly\n bsCollapseCtrl.$setActive(modelValue);\n }\n else {\n var activeIndexes = bsCollapseCtrl.$activeIndexes();\n\n if (angular.isArray(activeIndexes)) {\n // we have an array of selected indexes\n if (activeIndexes.indexOf(modelValue * 1) === -1) {\n // item with modelValue index is not active\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n else if (activeIndexes !== modelValue * 1) {\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n return modelValue;\n });\n\n }\n\n }\n };\n\n })\n\n .directive('bsCollapseToggle', function() {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base attr\n element.attr('data-toggle', 'collapse');\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerToggle(element);\n\n // remove toggle from collapse controller when toggle is destroyed\n scope.$on('$destroy', function() {\n bsCollapseCtrl.$unregisterToggle(element);\n });\n\n element.on('click', function() {\n var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element);\n bsCollapseCtrl.$setActive(index * 1);\n scope.$apply();\n });\n\n }\n };\n\n })\n\n .directive('bsCollapseTarget', function($animate) {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n // scope: true,\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base class\n element.addClass('collapse');\n\n // Add animation class\n if(bsCollapseCtrl.$options.animation) {\n element.addClass(bsCollapseCtrl.$options.animation);\n }\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerTarget(element);\n\n // remove pane target from collapse controller when target is destroyed\n scope.$on('$destroy', function() {\n bsCollapseCtrl.$unregisterTarget(element);\n });\n\n function render() {\n var index = bsCollapseCtrl.$targets.indexOf(element);\n var active = bsCollapseCtrl.$activeIndexes();\n var action = 'removeClass';\n if (angular.isArray(active)) {\n if (active.indexOf(index) !== -1) {\n action = 'addClass';\n }\n }\n else if (index === active) {\n action = 'addClass';\n }\n\n $animate[action](element, bsCollapseCtrl.$options.activeClass);\n }\n\n bsCollapseCtrl.$viewChangeListeners.push(function() {\n render();\n });\n render();\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/collapse.js"],"names":["angular","module","provider","controller","this","animation","self","forEach","startCollapsed","isDefined","allowMultiple","$scope","$element","$attrs","activeIndexes","$targets","$active","i","length","index","value","indexOf","activeItems","splice","activateItem","$options","$get","$collapse","defaults","key","$viewChangeListeners","$registerToggle","element","$toggles","push","$unregisterToggle","$unregisterTarget","deactivateItem","fn","fixActiveItemIndexes","$setActive","disallowToggle","isActive","$activeIndexes","require","link","postLink","ngModelCtrl","directive","$window","$animate","$setViewValue","$formatters","attrs","modelValue","isArray","bsCollapseCtrl","$on","scope","controllers","disabled","bsCollapseToggle","$apply","on","$registerTarget","render","active","action","activeClass","addClass"],"mappings":"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","file":"collapse.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.collapse', [])\n\n .provider('$collapse', function () {\n\n var defaults = this.defaults = {\n animation: 'am-collapse',\n disallowToggle: false,\n activeClass: 'in',\n startCollapsed: false,\n allowMultiple: false\n };\n\n var controller = this.controller = function ($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'disallowToggle', 'activeClass', 'startCollapsed', 'allowMultiple'], function (key) {\n if (angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['disallowToggle', 'startCollapsed', 'allowMultiple'], function (key) {\n if (angular.isDefined($attrs[key]) && falseValueRegExp.test($attrs[key])) {\n self.$options[key] = false;\n }\n });\n\n self.$toggles = [];\n self.$targets = [];\n\n self.$viewChangeListeners = [];\n\n self.$registerToggle = function (element) {\n self.$toggles.push(element);\n };\n self.$registerTarget = function (element) {\n self.$targets.push(element);\n };\n\n self.$unregisterToggle = function (element) {\n var index = self.$toggles.indexOf(element);\n // remove toggle from $toggles array\n self.$toggles.splice(index, 1);\n };\n self.$unregisterTarget = function (element) {\n var index = self.$targets.indexOf(element);\n\n // remove element from $targets array\n self.$targets.splice(index, 1);\n\n if (self.$options.allowMultiple) {\n // remove target index from $active array values\n deactivateItem(element);\n }\n\n // fix active item indexes\n fixActiveItemIndexes(index);\n\n self.$viewChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n // use array to store all the currently open panels\n self.$targets.$active = !self.$options.startCollapsed ? [0] : [];\n self.$setActive = $scope.$setActive = function (value) {\n if (angular.isArray(value)) {\n self.$targets.$active = value;\n } else if (!self.$options.disallowToggle && isActive(value)) {\n deactivateItem(value);\n } else {\n activateItem(value);\n }\n\n self.$viewChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n self.$activeIndexes = function () {\n if (self.$options.allowMultiple) {\n return self.$targets.$active;\n }\n return self.$targets.$active.length === 1 ? self.$targets.$active[0] : -1;\n };\n\n function fixActiveItemIndexes (index) {\n // item with index was removed, so we\n // need to adjust other items index values\n var activeIndexes = self.$targets.$active;\n for (var i = 0; i < activeIndexes.length; i++) {\n if (index < activeIndexes[i]) {\n activeIndexes[i] = activeIndexes[i] - 1;\n }\n\n // the last item is active, so we need to\n // adjust its index\n if (activeIndexes[i] === self.$targets.length) {\n activeIndexes[i] = self.$targets.length - 1;\n }\n }\n }\n\n function isActive (value) {\n var activeItems = self.$targets.$active;\n return activeItems.indexOf(value) !== -1;\n }\n\n function deactivateItem (value) {\n var index = self.$targets.$active.indexOf(value);\n if (index !== -1) {\n self.$targets.$active.splice(index, 1);\n }\n }\n\n function activateItem (value) {\n if (!self.$options.allowMultiple) {\n // remove current selected item\n self.$targets.$active.splice(0, 1);\n }\n\n if (self.$targets.$active.indexOf(value) === -1) {\n self.$targets.$active.push(value);\n }\n }\n\n };\n\n this.$get = function () {\n var $collapse = {};\n $collapse.defaults = defaults;\n $collapse.controller = controller;\n return $collapse;\n };\n\n })\n\n .directive('bsCollapse', function ($window, $animate, $collapse) {\n\n return {\n require: ['?ngModel', 'bsCollapse'],\n controller: ['$scope', '$element', '$attrs', $collapse.controller],\n link: function postLink (scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n if (ngModelCtrl) {\n\n // Update the modelValue following\n bsCollapseCtrl.$viewChangeListeners.push(function () {\n ngModelCtrl.$setViewValue(bsCollapseCtrl.$activeIndexes());\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n if (angular.isArray(modelValue)) {\n // model value is an array, so just replace\n // the active items directly\n bsCollapseCtrl.$setActive(modelValue);\n } else {\n var activeIndexes = bsCollapseCtrl.$activeIndexes();\n\n if (angular.isArray(activeIndexes)) {\n // we have an array of selected indexes\n if (activeIndexes.indexOf(modelValue * 1) === -1) {\n // item with modelValue index is not active\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n } else if (activeIndexes !== modelValue * 1) {\n bsCollapseCtrl.$setActive(modelValue * 1);\n }\n }\n return modelValue;\n });\n\n }\n\n }\n };\n\n })\n\n .directive('bsCollapseToggle', function () {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base attr\n element.attr('data-toggle', 'collapse');\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerToggle(element);\n\n // remove toggle from collapse controller when toggle is destroyed\n scope.$on('$destroy', function () {\n bsCollapseCtrl.$unregisterToggle(element);\n });\n\n element.on('click', function () {\n if (!attrs.disabled) {\n var index = attrs.bsCollapseToggle && attrs.bsCollapseToggle !== 'bs-collapse-toggle' ? attrs.bsCollapseToggle : bsCollapseCtrl.$toggles.indexOf(element);\n bsCollapseCtrl.$setActive(index * 1);\n scope.$apply();\n }\n });\n\n }\n };\n\n })\n\n .directive('bsCollapseTarget', function ($animate) {\n\n return {\n require: ['^?ngModel', '^bsCollapse'],\n // scope: true,\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsCollapseCtrl = controllers[1];\n\n // Add base class\n element.addClass('collapse');\n\n // Add animation class\n if (bsCollapseCtrl.$options.animation) {\n element.addClass(bsCollapseCtrl.$options.animation);\n }\n\n // Push pane to parent bsCollapse controller\n bsCollapseCtrl.$registerTarget(element);\n\n // remove pane target from collapse controller when target is destroyed\n scope.$on('$destroy', function () {\n bsCollapseCtrl.$unregisterTarget(element);\n });\n\n function render () {\n var index = bsCollapseCtrl.$targets.indexOf(element);\n var active = bsCollapseCtrl.$activeIndexes();\n var action = 'removeClass';\n if (angular.isArray(active)) {\n if (active.indexOf(index) !== -1) {\n action = 'addClass';\n }\n } else if (index === active) {\n action = 'addClass';\n }\n\n $animate[action](element, bsCollapseCtrl.$options.activeClass);\n }\n\n bsCollapseCtrl.$viewChangeListeners.push(function () {\n render();\n });\n render();\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/compiler.js b/dist/modules/compiler.js index 7f0728e3a..1e575629d 100644 --- a/dist/modules/compiler.js +++ b/dist/modules/compiler.js @@ -1,12 +1,14 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ 'use strict'; +bsCompilerService.$inject = [ '$q', '$http', '$injector', '$compile', '$controller', '$templateCache' ]; + angular.module('mgcrea.ngStrap.core', []).service('$bsCompiler', bsCompilerService); function bsCompilerService($q, $http, $injector, $compile, $controller, $templateCache) { @@ -20,8 +22,8 @@ function bsCompilerService($q, $http, $injector, $compile, $controller, $templat var template = options.template || ''; var controller = options.controller; var controllerAs = options.controllerAs; - var resolve = angular.copy(options.resolve || {}); - var locals = angular.copy(options.locals || {}); + var resolve = options.resolve || {}; + var locals = options.locals || {}; var transformTemplate = options.transformTemplate || angular.identity; var bindToController = options.bindToController; angular.forEach(resolve, function(value, key) { @@ -39,6 +41,13 @@ function bsCompilerService($q, $http, $injector, $compile, $controller, $templat } else { throw new Error('Missing `template` / `templateUrl` option.'); } + if (options.titleTemplate) { + resolve.$template = $q.all([ resolve.$template, fetchTemplate(options.titleTemplate) ]).then(function(templates) { + var templateEl = angular.element(templates[0]); + findElement('[ng-bind="title"]', templateEl[0]).removeAttr('ng-bind').html(templates[1]); + return templateEl[0].outerHTML; + }); + } if (options.contentTemplate) { resolve.$template = $q.all([ resolve.$template, fetchTemplate(options.contentTemplate) ]).then(function(templates) { var templateEl = angular.element(templates[0]); @@ -88,6 +97,4 @@ function bsCompilerService($q, $http, $injector, $compile, $controller, $templat return res.data; }); } -} - -bsCompilerService.$inject = [ '$q', '$http', '$injector', '$compile', '$controller', '$templateCache' ]; \ No newline at end of file +} \ No newline at end of file diff --git a/dist/modules/compiler.min.js b/dist/modules/compiler.min.js index a3b954d99..80937f493 100644 --- a/dist/modules/compiler.min.js +++ b/dist/modules/compiler.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';function bsCompilerService(e,t,n,l,r,a){function o(e,t){return angular.element((t||document).querySelectorAll(e))}function i(e){return c[e]?c[e]:c[e]=t.get(e,{cache:a}).then(function(e){return e.data})}this.compile=function(t){t.template&&/\.html$/.test(t.template)&&(console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.'),t.templateUrl=t.template,t.template='');var a=t.templateUrl,c=t.template||'',m=t.controller,p=t.controllerAs,u=angular.copy(t.resolve||{}),s=angular.copy(t.locals||{}),g=t.transformTemplate||angular.identity,d=t.bindToController;if(angular.forEach(u,function(e,t){angular.isString(e)?u[t]=n.get(e):u[t]=n.invoke(e)}),angular.extend(u,s),c)u.$template=e.when(c);else{if(!a)throw new Error('Missing `template` / `templateUrl` option.');u.$template=i(a)}return t.contentTemplate&&(u.$template=e.all([u.$template,i(t.contentTemplate)]).then(function(e){var n=angular.element(e[0]),l=o('[ng-bind="content"]',n[0]).removeAttr('ng-bind').html(e[1]);return t.templateUrl||l.next().remove(),n[0].outerHTML})),e.all(u).then(function(e){var n=g(e.$template);t.html&&(n=n.replace(/ng-bind="/gi,'ng-bind-html="'));var a=angular.element('
').html(n.trim()).contents(),o=l(a);return{locals:e,element:a,link:function(t){if(e.$scope=t,m){var n=r(m,e,!0);d&&angular.extend(n.instance,e);var l=angular.isObject(n)?n:n();a.data('$ngControllerController',l),a.children().data('$ngControllerController',l),p&&(t[p]=l)}return o.apply(null,arguments)}}})};var c={}}angular.module('mgcrea.ngStrap.core',[]).service('$bsCompiler',bsCompilerService),bsCompilerService.$inject=['$q','$http','$injector','$compile','$controller','$templateCache']; +'use strict';function bsCompilerService(e,t,n,l,r,a){function o(e,t){return angular.element((t||document).querySelectorAll(e))}function i(e){return m[e]?m[e]:m[e]=t.get(e,{cache:a}).then(function(e){return e.data})}this.compile=function(t){t.template&&/\.html$/.test(t.template)&&(console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.'),t.templateUrl=t.template,t.template='');var a=t.templateUrl,m=t.template||'',c=t.controller,p=t.controllerAs,u=t.resolve||{},s=t.locals||{},g=t.transformTemplate||angular.identity,d=t.bindToController;if(angular.forEach(u,function(e,t){angular.isString(e)?u[t]=n.get(e):u[t]=n.invoke(e)}),angular.extend(u,s),m)u.$template=e.when(m);else{if(!a)throw new Error('Missing `template` / `templateUrl` option.');u.$template=i(a)}return t.titleTemplate&&(u.$template=e.all([u.$template,i(t.titleTemplate)]).then(function(e){var t=angular.element(e[0]);return o('[ng-bind="title"]',t[0]).removeAttr('ng-bind').html(e[1]),t[0].outerHTML})),t.contentTemplate&&(u.$template=e.all([u.$template,i(t.contentTemplate)]).then(function(e){var n=angular.element(e[0]),l=o('[ng-bind="content"]',n[0]).removeAttr('ng-bind').html(e[1]);return t.templateUrl||l.next().remove(),n[0].outerHTML})),e.all(u).then(function(e){var n=g(e.$template);t.html&&(n=n.replace(/ng-bind="/gi,'ng-bind-html="'));var a=angular.element('
').html(n.trim()).contents(),o=l(a);return{locals:e,element:a,link:function(t){if(e.$scope=t,c){var n=r(c,e,!0);d&&angular.extend(n.instance,e);var l=angular.isObject(n)?n:n();a.data('$ngControllerController',l),a.children().data('$ngControllerController',l),p&&(t[p]=l)}return o.apply(null,arguments)}}})};var m={}}bsCompilerService.$inject=['$q','$http','$injector','$compile','$controller','$templateCache'],angular.module('mgcrea.ngStrap.core',[]).service('$bsCompiler',bsCompilerService); //# sourceMappingURL=compiler.min.js.map diff --git a/dist/modules/compiler.min.js.map b/dist/modules/compiler.min.js.map index 25a55f118..60a99efa3 100644 --- a/dist/modules/compiler.min.js.map +++ b/dist/modules/compiler.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/compiler.js"],"names":["templateUrl","options","cache","$templateCache","then","element","res","fetchTemplate","template","fetchPromises","$http","get","bsCompilerService","data","this","compile","controller","console","controllerAs","resolve","copy","locals","bindToController","angular","forEach","value","isString","$injector","key","invoke","transformTemplate","identity","extend","$template","when","$q","all","templateEl","contentEl","findElement","contentTemplate","removeAttr","html","templates","replace","next","remove","outerHTML","link","scope","trim","contents","linkFn","invokeCtrl","children","instance","ctrl","isObject","arguments","apply","service","$inject"],"mappings":"AAOA,YAIA,SAmEQA,mBAAcC,EAAQD,EAAAA,EAAAA,EAAAA,EAAAA,GAA5B,QA4FyDE,GAAOC,EAAAA,GA3F9D,MA4FGC,SAAKC,SAASC,GAAAA,UAAAA,iBAAAA,IAzFnB,QAASC,GAAcC,GACrB,MAAIC,GAAcD,GAAkBC,EAAcD,GAC3CC,EAAcD,GAAYE,EAAMC,IAAIH,GA6F/CI,MAAAA,IA3FOR,KAAK,SAASE,GACf,MAAOA,GAAIO,OA3EfC,KAmEEC,QAAIP,SAAWP,GACXe,EAAAA,UAAaf,UAAQe,KAAAA,EAAAA,YACzBC,QAAIC,KAAAA,oGACJjB,EAAIkB,YAAkBC,EAAKnB,SAC3BA,EAAIoB,SAAiBD,GAErB,IAAIE,GAAAA,EAAmBrB,YAKvBsB,EAAQC,EAAQL,UAAkBM,GAChCT,EAAYU,EAASD,WACnBN,EAAeQ,EAAAA,aAtEfR,EAuEKI,QAAAH,KAAAnB,EAAAkB,aACLA,EAAQS,QAAOD,KAAAA,EAAUE,YAtEzBC,EAAoB7B,EAAQ6B,mBAAqBP,QAAQQ,SACzDT,EAAmBrB,EAAQqB,gBAoF/B,IAVAC,QAAQS,QAAOb,EAASE,SAAAA,EAAAA,GAEpBb,QAAUkB,SAAAD,GACZN,EAAQc,GAAAA,EAAeC,IAAK1B,GAE5BW,EAAQc,GAAAA,EAAY1B,OAAcP,KAvEpCuB,QAAQS,OAAOb,EAASE,GA4EpBpB,EAEFkB,EAAQc,UAAYE,EAAGC,KAAAA,OAEnB,CAAA,IAAIC,EA1ER,KA8EI,IAAOA,OAAAA,6CAhFXlB,GA6EQmB,UAAYC,EAAYvC,GAjElC,MARIC,GAAQuC,kBAiFZrB,EAAUiB,UAAIjB,EAASf,KAAKe,EAASE,UAAAA,EAAAA,EAAAA,mBAAAA,KAAAA,SAAAA,GAEnC,GAAIb,GAAWsB,QAAAA,QAAkBT,EAAOY,IACpChC,EAAcsC,EAAA,sBAAAF,EAAA,IAAAI,WAAA,WAAAC,KAAAC,EAAA,GA/EhB,OAgFAnC,GAAWA,aAASoC,EAAQC,OAAeC,SAhFpCT,EAAW,GAAGU,aAuFvBZ,EAAAC,IAAAjB,GAAAf,KAAA,SAAAiB,GAnFA,GAoFEA,GAAQA,EAAAA,EAAAA,UACRhB,GAASA,OACT2C,EAAMxC,EAAcyC,QAAAA,cAAAA,kBAlFtB,IAsFI5C,GAAIW,QAAYX,QAAA,SAAAqC,KAAAlC,EAAA0C,QAAAC,WArFhBC,EAsFMC,EAAAA,EArFV,QACEhC,OAsFME,EArFNlB,QAASA,EACT2C,KAuFI,SAAWzB,GArFb,GADAF,EAwFEhB,OAAa4C,EACb5C,EAAQiD,CAER,GAAIpC,GAAAA,EAAcF,EAAAK,GAAA,EAChB4B,IAxFA1B,QAAQS,OAAOqB,EAAWE,SAAUlC,EA4FxC,IAAAmC,GAAOJ,QAAaK,SAAMC,GAAAA,EAAAA,GAzFxBrD,GAAQQ,KAAK,0BAA2B2C,GACxCnD,EAAQiD,WAAWzC,KAAK,0BAA2B2C,GAC/CtC,IACF+B,EAAM/B,GAAgBsC,GAG1B,MAAOJ,GAAOO,MAAM,KAAMD,eAQlC,IAAIjD,MAxENc,QAkEMtB,OAAQO,0BAAWoD,QAAA,cAAAhD,mBAiBzBA,kBAAkBiD,SAAY,KAAM,QAAS,YAAa,WAAY,cAAe","file":"modules/compiler.min.js","sourcesContent":["'use strict';\n\n// NOTICE: This file was forked from the angular-material project (github.com/angular/material)\n// MIT Licensed - Copyright (c) 2014-2015 Google, Inc. http://angularjs.org\n\nangular.module('mgcrea.ngStrap.core', [])\n .service('$bsCompiler', bsCompilerService);\n\nfunction bsCompilerService($q, $http, $injector, $compile, $controller, $templateCache) {\n /* jshint validthis: true */\n\n /*\n * @ngdoc service\n * @name $bsCompiler\n * @module material.core\n * @description\n * The $bsCompiler service is an abstraction of angular's compiler, that allows the developer\n * to easily compile an element with a templateUrl, controller, and locals.\n *\n * @usage\n * \n * $bsCompiler.compile({\n * templateUrl: 'modal.html',\n * controller: 'ModalCtrl',\n * locals: {\n * modal: myModalInstance;\n * }\n * }).then(function(compileData) {\n * compileData.element; // modal.html's template in an element\n * compileData.link(myScope); //attach controller & scope to element\n * });\n * \n */\n\n /*\n * @ngdoc method\n * @name $bsCompiler#compile\n * @description A helper to compile an HTML template/templateUrl with a given controller,\n * locals, and scope.\n * @param {object} options An options object, with the following properties:\n *\n * - `controller` - `{(string=|function()=}` Controller fn that should be associated with\n * newly created scope or the name of a registered controller if passed as a string.\n * - `controllerAs` - `{string=}` A controller alias name. If present the controller will be\n * published to scope under the `controllerAs` name.\n * - `template` - `{string=}` An html template as a string.\n * - `templateUrl` - `{string=}` A path to an html template.\n * - `transformTemplate` - `{function(template)=}` A function which transforms the template after\n * it is loaded. It will be given the template string as a parameter, and should\n * return a a new string representing the transformed template.\n * - `resolve` - `{Object.=}` - An optional map of dependencies which should\n * be injected into the controller. If any of these dependencies are promises, the compiler\n * will wait for them all to be resolved, or if one is rejected before the controller is\n * instantiated `compile()` will fail..\n * * `key` - `{string}`: a name of a dependency to be injected into the controller.\n * * `factory` - `{string|function}`: If `string` then it is an alias for a service.\n * Otherwise if function, then it is injected and the return value is treated as the\n * dependency. If the result is a promise, it is resolved before its value is\n * injected into the controller.\n *\n * @returns {object=} promise A promise, which will be resolved with a `compileData` object.\n * `compileData` has the following properties:\n *\n * - `element` - `{element}`: an uncompiled element matching the provided template.\n * - `link` - `{function(scope)}`: A link function, which, when called, will compile\n * the element and instantiate the provided controller (if given).\n * - `locals` - `{object}`: The locals which will be passed into the controller once `link` is\n * called. If `bindToController` is true, they will be coppied to the ctrl instead\n * - `bindToController` - `bool`: bind the locals to the controller, instead of passing them in.\n */\n this.compile = function(options) {\n\n if(options.template && /\\.html$/.test(options.template)) {\n console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.');\n options.templateUrl = options.template;\n options.template = '';\n }\n\n var templateUrl = options.templateUrl;\n var template = options.template || '';\n var controller = options.controller;\n var controllerAs = options.controllerAs;\n var resolve = angular.copy(options.resolve || {});\n var locals = angular.copy(options.locals || {});\n var transformTemplate = options.transformTemplate || angular.identity;\n var bindToController = options.bindToController;\n\n // Take resolve values and invoke them.\n // Resolves can either be a string (value: 'MyRegisteredAngularConst'),\n // or an invokable 'factory' of sorts: (value: function ValueGetter($dependency) {})\n angular.forEach(resolve, function(value, key) {\n if (angular.isString(value)) {\n resolve[key] = $injector.get(value);\n } else {\n resolve[key] = $injector.invoke(value);\n }\n });\n // Add the locals, which are just straight values to inject\n // eg locals: { three: 3 }, will inject three into the controller\n angular.extend(resolve, locals);\n\n if (template) {\n resolve.$template = $q.when(template);\n } else if (templateUrl) {\n resolve.$template = fetchTemplate(templateUrl);\n } else {\n throw new Error('Missing `template` / `templateUrl` option.');\n }\n\n if (options.contentTemplate) {\n // TODO(mgcrea): deprecate?\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.contentTemplate)])\n .then(function(templates) {\n var templateEl = angular.element(templates[0]);\n var contentEl = findElement('[ng-bind=\"content\"]', templateEl[0]).removeAttr('ng-bind').html(templates[1]);\n // Drop the default footer as you probably don't want it if you use a custom contentTemplate\n if(!options.templateUrl) contentEl.next().remove();\n return templateEl[0].outerHTML;\n });\n }\n\n // Wait for all the resolves to finish if they are promises\n return $q.all(resolve).then(function(locals) {\n\n var template = transformTemplate(locals.$template);\n if (options.html) {\n template = template.replace(/ng-bind=\"/ig, 'ng-bind-html=\"');\n }\n // var element = options.element || angular.element('
').html(template.trim()).contents();\n var element = angular.element('
').html(template.trim()).contents();\n var linkFn = $compile(element);\n\n // Return a linking function that can be used later when the element is ready\n return {\n locals: locals,\n element: element,\n link: function link(scope) {\n locals.$scope = scope;\n\n // Instantiate controller if it exists, because we have scope\n if (controller) {\n var invokeCtrl = $controller(controller, locals, true);\n if (bindToController) {\n angular.extend(invokeCtrl.instance, locals);\n }\n // Support angular@~1.2 invokeCtrl\n var ctrl = angular.isObject(invokeCtrl) ? invokeCtrl : invokeCtrl();\n // See angular-route source for this logic\n element.data('$ngControllerController', ctrl);\n element.children().data('$ngControllerController', ctrl);\n\n if (controllerAs) {\n scope[controllerAs] = ctrl;\n }\n }\n\n return linkFn.apply(null, arguments);\n }\n };\n });\n\n };\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate(template) {\n if(fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache})\n .then(function(res) {\n return res.data;\n }));\n }\n\n}\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/compiler.js"],"names":["templateUrl","options","cache","$templateCache","then","element","res","fetchTemplate","template","fetchPromises","$http","get","data","this","compile","controller","console","controllerAs","resolve","locals","bindToController","angular","forEach","value","isString","$injector","key","transformTemplate","identity","extend","$template","when","$q","all","templateEl","Error","findElement","contentTemplate","templates","contentEl","removeAttr","html","replace","next","remove","outerHTML","link","scope","trim","contents","linkFn","invokeCtrl","children","instance","ctrl","isObject","arguments","apply","bsCompilerService","warn","service"],"mappings":"AAOA,YAMA,SAgEQA,mBAAcC,EAAQD,EAAAA,EAAAA,EAAAA,EAAAA,GAU5B,QA+FyDE,GAAOC,EAAAA,GA9F9D,MA+FGC,SAAKC,SAAUC,GAAAA,UAAAA,iBAAAA,IA5FpB,QAASC,GAAcC,GACrB,MAAIC,GAAcD,GAAkBC,EAAcD,GAC3CC,EAAcD,GAAYE,EAAMC,IAAIH,GACzCN,MAAOC,IACNC,KAAK,SAASE,GACf,MAAOA,GAAIM,OAlFfC,KAgEEC,QAAIN,SAAWP,GACXc,EAAAA,UAAad,UAAQc,KAAAA,EAAAA,YACzBC,QAAIC,KAAAA,oGACJhB,EAAIiB,YAAkBA,EAAAA,SACtBjB,EAAIkB,SAAiBA,GAErB,IAAIC,GAAAA,EAAmBnB,YAKvBoB,EAAQC,EAAQJ,UAAmBK,GACjCR,EAAYS,EAASD,WACnBL,EAAeO,EAAAA,aAnEfP,EAoEKjB,EAAAiB,YACLA,EAAQQ,EAAOD,WAnEfE,EAAoB1B,EAAQ0B,mBAAqBN,QAAQO,SACzDR,EAAmBnB,EAAQmB,gBAiF/B,IAVAC,QAAQQ,QAAOX,EAASC,SAAAA,EAAAA,GAEpBX,QAAUgB,SAAAD,GACZL,EAAQY,GAAAA,EAAeC,IAAKvB,GAE5BU,EAAQY,GAAAA,EAAYvB,OAAcP,KApEpCqB,QAAQQ,OAAOX,EAASC,GAyEpBlB,EACFiB,EAAQY,UAAYE,EAAGC,KAAAA,OAEnB,CAAA,IAAIC,EAtER,KAAM,IAAIC,OAAM,6CAFhBjB,GAyEIkB,UAAY7B,EAAAP,GAtDlB,MA6DIC,GAAQoC,gBAEVnB,EAAQY,UAAYE,EAAGC,KAAKf,EAAQY,UAAWvB,EAAcN,EAAQoC,iBAAAA,KAClEjC,SAAKkC,GACJ,GAAIJ,GAAab,QAAQhB,QAAQiC,EAAU,GAK3C,OAJAF,GAAIG,oBAAwBL,EAAA,IAAAM,WAAuBN,WAChDM,KAAAA,EAAW,IAGTvC,EAAQD,GAAAA,aA9EfC,EAAQoC,kBAoFZnB,EAAUe,UAAIf,EAASd,KAAKc,EAAUC,UAAAA,EAAAA,EAAAA,mBAAAA,KAAAA,SAAAA,GAEpC,GAAIX,GAAWmB,QAAAA,QAAkBR,EAAOW,IACpC7B,EAAcmC,EAAA,sBAAAF,EAAA,IAAAM,WAAA,WAAAC,KAAAH,EAAA,GAlFhB,OAmFA9B,GAAWA,aAASkC,EAAQC,OAAeC,SAnFpCV,EAAW,GAAGW,aA0FvBb,EAAAC,IAAAf,GAAAd,KAAA,SAAAe,GAtFA,GAuFEA,GAAQA,EAAAA,EAAAA,UACRd,GAASA,OACTyC,EAAMtC,EAAeuC,QAAAA,cAAAA,kBArFvB,IAyFI1C,GAAIU,QAAYV,QAAA,SAAAoC,KAAAjC,EAAAwC,QAAAC,WAxFhBC,EAyFMC,EAAAA,EAxFV,QACEhC,OAyFME,EAxFNhB,QAASA,EACTyC,KA0FI,SAAWzB,GAxFb,GADAF,EA2FEd,OAAa0C,EACb1C,EAAQ+C,CAER,GAAInC,GAAAA,EAAcF,EAAAI,GAAA,EAChB4B,IA3FA1B,QAAQQ,OAAOsB,EAAWE,SAAUlC,EA+FxC,IAAAmC,GAAOJ,QAAaK,SAAMC,GAAAA,EAAAA,GA5FxBnD,GAAQO,KAAK,0BAA2B0C,GACxCjD,EAAQ+C,WAAWxC,KAAK,0BAA2B0C,GAC/CrC,IACF8B,EAAM9B,GAAgBqC,GAG1B,MAAOJ,GAAOO,MAAM,KAAMD,eAQlC,IAAI/C,MAjFNiD,kBA+DcC,SAAK,KAAA,QAAA,YAAA,WAAA,cAAA,kBA7DnBtC,QA+DMpB,OAAQO,0BAAWoD,QAAA,cAAAF","file":"compiler.min.js","sourcesContent":["'use strict';\n\n// NOTICE: This file was forked from the angular-material project (github.com/angular/material)\n// MIT Licensed - Copyright (c) 2014-2015 Google, Inc. http://angularjs.org\n\nangular.module('mgcrea.ngStrap.core', [])\n .service('$bsCompiler', bsCompilerService);\n\nfunction bsCompilerService ($q, $http, $injector, $compile, $controller, $templateCache) {\n\n /*\n * @ngdoc service\n * @name $bsCompiler\n * @module material.core\n * @description\n * The $bsCompiler service is an abstraction of angular's compiler, that allows the developer\n * to easily compile an element with a templateUrl, controller, and locals.\n *\n * @usage\n * \n * $bsCompiler.compile({\n * templateUrl: 'modal.html',\n * controller: 'ModalCtrl',\n * locals: {\n * modal: myModalInstance;\n * }\n * }).then(function(compileData) {\n * compileData.element; // modal.html's template in an element\n * compileData.link(myScope); //attach controller & scope to element\n * });\n * \n */\n\n /*\n * @ngdoc method\n * @name $bsCompiler#compile\n * @description A helper to compile an HTML template/templateUrl with a given controller,\n * locals, and scope.\n * @param {object} options An options object, with the following properties:\n *\n * - `controller` - `{(string=|function()=}` Controller fn that should be associated with\n * newly created scope or the name of a registered controller if passed as a string.\n * - `controllerAs` - `{string=}` A controller alias name. If present the controller will be\n * published to scope under the `controllerAs` name.\n * - `template` - `{string=}` An html template as a string.\n * - `templateUrl` - `{string=}` A path to an html template.\n * - `transformTemplate` - `{function(template)=}` A function which transforms the template after\n * it is loaded. It will be given the template string as a parameter, and should\n * return a a new string representing the transformed template.\n * - `resolve` - `{Object.=}` - An optional map of dependencies which should\n * be injected into the controller. If any of these dependencies are promises, the compiler\n * will wait for them all to be resolved, or if one is rejected before the controller is\n * instantiated `compile()` will fail..\n * * `key` - `{string}`: a name of a dependency to be injected into the controller.\n * * `factory` - `{string|function}`: If `string` then it is an alias for a service.\n * Otherwise if function, then it is injected and the return value is treated as the\n * dependency. If the result is a promise, it is resolved before its value is\n * injected into the controller.\n *\n * @returns {object=} promise A promise, which will be resolved with a `compileData` object.\n * `compileData` has the following properties:\n *\n * - `element` - `{element}`: an uncompiled element matching the provided template.\n * - `link` - `{function(scope)}`: A link function, which, when called, will compile\n * the element and instantiate the provided controller (if given).\n * - `locals` - `{object}`: The locals which will be passed into the controller once `link` is\n * called. If `bindToController` is true, they will be coppied to the ctrl instead\n * - `bindToController` - `bool`: bind the locals to the controller, instead of passing them in.\n */\n this.compile = function (options) {\n\n if (options.template && /\\.html$/.test(options.template)) {\n console.warn('Deprecated use of `template` option to pass a file. Please use the `templateUrl` option instead.');\n options.templateUrl = options.template;\n options.template = '';\n }\n\n var templateUrl = options.templateUrl;\n var template = options.template || '';\n var controller = options.controller;\n var controllerAs = options.controllerAs;\n var resolve = options.resolve || {};\n var locals = options.locals || {};\n var transformTemplate = options.transformTemplate || angular.identity;\n var bindToController = options.bindToController;\n\n // Take resolve values and invoke them.\n // Resolves can either be a string (value: 'MyRegisteredAngularConst'),\n // or an invokable 'factory' of sorts: (value: function ValueGetter($dependency) {})\n angular.forEach(resolve, function (value, key) {\n if (angular.isString(value)) {\n resolve[key] = $injector.get(value);\n } else {\n resolve[key] = $injector.invoke(value);\n }\n });\n // Add the locals, which are just straight values to inject\n // eg locals: { three: 3 }, will inject three into the controller\n angular.extend(resolve, locals);\n\n if (template) {\n resolve.$template = $q.when(template);\n } else if (templateUrl) {\n resolve.$template = fetchTemplate(templateUrl);\n } else {\n throw new Error('Missing `template` / `templateUrl` option.');\n }\n\n if (options.titleTemplate) {\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.titleTemplate)])\n .then(function (templates) {\n var templateEl = angular.element(templates[0]);\n findElement('[ng-bind=\"title\"]', templateEl[0])\n .removeAttr('ng-bind')\n .html(templates[1]);\n return templateEl[0].outerHTML;\n });\n }\n\n if (options.contentTemplate) {\n // TODO(mgcrea): deprecate?\n resolve.$template = $q.all([resolve.$template, fetchTemplate(options.contentTemplate)])\n .then(function (templates) {\n var templateEl = angular.element(templates[0]);\n var contentEl = findElement('[ng-bind=\"content\"]', templateEl[0])\n .removeAttr('ng-bind')\n .html(templates[1]);\n // Drop the default footer as you probably don't want it if you use a custom contentTemplate\n if (!options.templateUrl) contentEl.next().remove();\n return templateEl[0].outerHTML;\n });\n }\n\n // Wait for all the resolves to finish if they are promises\n return $q.all(resolve).then(function (locals) {\n\n var template = transformTemplate(locals.$template);\n if (options.html) {\n template = template.replace(/ng-bind=\"/ig, 'ng-bind-html=\"');\n }\n // var element = options.element || angular.element('
').html(template.trim()).contents();\n var element = angular.element('
').html(template.trim()).contents();\n var linkFn = $compile(element);\n\n // Return a linking function that can be used later when the element is ready\n return {\n locals: locals,\n element: element,\n link: function link (scope) {\n locals.$scope = scope;\n\n // Instantiate controller if it exists, because we have scope\n if (controller) {\n var invokeCtrl = $controller(controller, locals, true);\n if (bindToController) {\n angular.extend(invokeCtrl.instance, locals);\n }\n // Support angular@~1.2 invokeCtrl\n var ctrl = angular.isObject(invokeCtrl) ? invokeCtrl : invokeCtrl();\n // See angular-route source for this logic\n element.data('$ngControllerController', ctrl);\n element.children().data('$ngControllerController', ctrl);\n\n if (controllerAs) {\n scope[controllerAs] = ctrl;\n }\n }\n\n return linkFn.apply(null, arguments);\n }\n };\n });\n\n };\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate (template) {\n if (fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache})\n .then(function (res) {\n return res.data;\n }));\n }\n\n}\n"]} \ No newline at end of file diff --git a/dist/modules/date-formatter.js b/dist/modules/date-formatter.js index bea988200..403d67690 100644 --- a/dist/modules/date-formatter.js +++ b/dist/modules/date-formatter.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/date-formatter.min.js b/dist/modules/date-formatter.min.js index 017c353df..ab47c8c84 100644 --- a/dist/modules/date-formatter.min.js +++ b/dist/modules/date-formatter.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/date-formatter.min.js.map b/dist/modules/date-formatter.min.js.map index 5b3fbc399..751fd2d91 100644 --- a/dist/modules/date-formatter.min.js.map +++ b/dist/modules/date-formatter.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/date-formatter.js"],"names":["angular","weekdaysShort","lang","splitTimeFormat","timeFormat","exec","format","slice","this","getDefaultLocale","DATETIME_FORMATS","$locale","id","getDatetimeFormat","SHORTDAY","hoursFormat","minutesFormat","secondsFormat","timeSeparator","showSeconds","dateFilter","showAM","formatDate","date","timezone"],"mappings":"AAOA,YAEAA,SAYSC,OAAAA,2CAAyBC,QAAAA,kBAAAA,UAAAA,aAAAA,SAAAA,EAAAA,GAFhC,QAiBWC,GAAgBC,GAhBzB,MAAO,wCAAwCC,KAAKC,GAAQC,MAAM,GAVpEC,KAYIC,iBAAeC,WAXjB,MAAOC,GAAQC,IAEjBJ,KAaIK,kBAAO,SAAAP,EAAAJ,GAZT,MAAOS,GAAQD,iBAAiBJ,IAAWA,GAE7CE,KAeIP,cAAOE,SAAgBC,GAdzB,MAAOO,GAAQD,iBAAiBI,UAKlCN,KAmBIO,YAAOZ,SAAgBC,GAlBzB,MAAOD,GAAgBC,GAAY,IAErCI,KAqBIQ,cAAOb,SAAgBC,GApBzB,MAAOD,GAAgBC,GAAY,IAErCI,KAuBIS,cAASd,SAAgBC,GAtB3B,MAAOD,GAAgBC,GAAY,IAErCI,KAyBIU,cAASf,SAAgBC,GAxB3B,MAAOD,GAAgBC,GAAY,IAErCI,KA0BIW,YAAOC,SAAiBd,GAzB1B,QAASH,EAAgBC,GAAY,IAEvCI,KAAKa,OAAS,SAASjB,GACrB,QAASD,EAAgBC,GAAY,IAEvCI,KAAKc,WAAa,SAASC,EAAMjB,EAAQJ,EAAMsB,GAC7C,MAAOJ,GAAWG,EAAMjB,EAAQkB","file":"modules/date-formatter.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateFormatter', [])\n\n .service('$dateFormatter', function($locale, dateFilter) {\n\n // The unused `lang` arguments are on purpose. The default implementation does not\n // use them and it always uses the locale loaded into the `$locale` service.\n // Custom implementations might use it, thus allowing different directives to\n // have different languages.\n\n this.getDefaultLocale = function() {\n return $locale.id;\n };\n\n // Format is either a data format name, e.g. \"shortTime\" or \"fullDate\", or a date format\n // Return either the corresponding date format or the given date format.\n this.getDatetimeFormat = function(format, lang) {\n return $locale.DATETIME_FORMATS[format] || format;\n };\n\n this.weekdaysShort = function(lang) {\n return $locale.DATETIME_FORMATS.SHORTDAY;\n };\n\n function splitTimeFormat(format) {\n return /(h+)([:\\.])?(m+)([:\\.])?(s*)[ ]?(a?)/i.exec(format).slice(1);\n }\n\n // h:mm a => h\n this.hoursFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[0];\n };\n\n // h:mm a => mm\n this.minutesFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[2];\n };\n\n // h:mm:ss a => ss\n this.secondsFormat = function(timeFormat) {\n return splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => :\n this.timeSeparator = function(timeFormat) {\n return splitTimeFormat(timeFormat)[1];\n };\n\n // h:mm:ss a => true, h:mm a => false\n this.showSeconds = function(timeFormat) {\n return !!splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => true, H.mm => false\n this.showAM = function(timeFormat) {\n return !!splitTimeFormat(timeFormat)[5];\n };\n\n this.formatDate = function(date, format, lang, timezone){\n return dateFilter(date, format, timezone);\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/date-formatter.js"],"names":["angular","weekdaysShort","lang","splitTimeFormat","timeFormat","exec","format","slice","this","getDefaultLocale","DATETIME_FORMATS","$locale","id","getDatetimeFormat","SHORTDAY","hoursFormat","minutesFormat","secondsFormat","timeSeparator","showSeconds","dateFilter","showAM","formatDate","date","timezone"],"mappings":"AAOA,YAEAA,SAYSC,OAAAA,2CAA0BC,QAAAA,kBAAAA,UAAAA,aAAAA,SAAAA,EAAAA,GAFjC,QAiBWC,GAAgBC,GAhBzB,MAAO,wCAAwCC,KAAKC,GAAQC,MAAM,GAVpEC,KAYIC,iBAAeC,WAXjB,MAAOC,GAAQC,IAEjBJ,KAaIK,kBAAO,SAAAP,EAAAJ,GAZT,MAAOS,GAAQD,iBAAiBJ,IAAWA,GAE7CE,KAeIP,cAAOE,SAAgBC,GAdzB,MAAOO,GAAQD,iBAAiBI,UAKlCN,KAmBIO,YAAOZ,SAAgBC,GAlBzB,MAAOD,GAAgBC,GAAY,IAErCI,KAqBIQ,cAAOb,SAAgBC,GApBzB,MAAOD,GAAgBC,GAAY,IAErCI,KAuBIS,cAASd,SAAgBC,GAtB3B,MAAOD,GAAgBC,GAAY,IAErCI,KAyBIU,cAASf,SAAgBC,GAxB3B,MAAOD,GAAgBC,GAAY,IAErCI,KA0BIW,YAAOC,SAAiBd,GAzB1B,QAASH,EAAgBC,GAAY,IAEvCI,KAAKa,OAAS,SAASjB,GACrB,QAASD,EAAgBC,GAAY,IAEvCI,KAAKc,WAAa,SAASC,EAAMjB,EAAQJ,EAAMsB,GAC7C,MAAOJ,GAAWG,EAAMjB,EAAQkB","file":"date-formatter.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateFormatter', [])\n\n .service('$dateFormatter', function ($locale, dateFilter) {\n\n // The unused `lang` arguments are on purpose. The default implementation does not\n // use them and it always uses the locale loaded into the `$locale` service.\n // Custom implementations might use it, thus allowing different directives to\n // have different languages.\n\n this.getDefaultLocale = function () {\n return $locale.id;\n };\n\n // Format is either a data format name, e.g. \"shortTime\" or \"fullDate\", or a date format\n // Return either the corresponding date format or the given date format.\n this.getDatetimeFormat = function (format, lang) {\n return $locale.DATETIME_FORMATS[format] || format;\n };\n\n this.weekdaysShort = function (lang) {\n return $locale.DATETIME_FORMATS.SHORTDAY;\n };\n\n function splitTimeFormat (format) {\n return /(h+)([:\\.])?(m+)([:\\.])?(s*)[ ]?(a?)/i.exec(format).slice(1);\n }\n\n // h:mm a => h\n this.hoursFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[0];\n };\n\n // h:mm a => mm\n this.minutesFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[2];\n };\n\n // h:mm:ss a => ss\n this.secondsFormat = function (timeFormat) {\n return splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => :\n this.timeSeparator = function (timeFormat) {\n return splitTimeFormat(timeFormat)[1];\n };\n\n // h:mm:ss a => true, h:mm a => false\n this.showSeconds = function (timeFormat) {\n return !!splitTimeFormat(timeFormat)[4];\n };\n\n // h:mm a => true, H.mm => false\n this.showAM = function (timeFormat) {\n return !!splitTimeFormat(timeFormat)[5];\n };\n\n this.formatDate = function (date, format, lang, timezone) {\n return dateFilter(date, format, timezone);\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/date-parser.js b/dist/modules/date-parser.js index 00740771d..5a7f9c91d 100644 --- a/dist/modules/date-parser.js +++ b/dist/modules/date-parser.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -60,7 +60,8 @@ angular.module('mgcrea.ngStrap.helpers.dateParser', []).provider('$dateParser', return !isNaN(parseFloat(n)) && isFinite(n); } function indexOfCaseInsensitive(array, value) { - var len = array.length, str = value.toString().toLowerCase(); + var len = array.length; + var str = value.toString().toLowerCase(); for (var i = 0; i < len; i++) { if (array[i].toLowerCase() === str) { return i; @@ -137,7 +138,8 @@ angular.module('mgcrea.ngStrap.helpers.dateParser', []).provider('$dateParser', return 1 * value <= 50 && value.length === 2 ? this.setFullYear(2e3 + 1 * value) : this.setFullYear(1 * value); } }; - var regex, setMap; + var regex; + var setMap; $dateParser.init = function() { $dateParser.$format = $locale.DATETIME_FORMATS[options.format] || options.format; regex = regExpForFormat($dateParser.$format); @@ -156,7 +158,7 @@ angular.module('mgcrea.ngStrap.helpers.dateParser', []).provider('$dateParser', if (!matches) return false; var date = baseDate && !isNaN(baseDate.getTime()) ? new ParseDate().fromDate(baseDate) : new ParseDate().fromDate(new Date(1970, 0, 1, 0)); for (var i = 0; i < matches.length - 1; i++) { - formatSetMap[i] && formatSetMap[i].call(date, matches[i + 1]); + if (formatSetMap[i]) formatSetMap[i].call(date, matches[i + 1]); } var newDate = date.toDate(); if (parseInt(date.day, 10) !== newDate.getDate()) { @@ -173,7 +175,7 @@ angular.module('mgcrea.ngStrap.helpers.dateParser', []).provider('$dateParser', date = new Date(value.substr(1, value.length - 2)); } else if (isNumeric(value)) { date = new Date(parseInt(value, 10)); - } else if (angular.isString(value) && 0 === value.length) { + } else if (angular.isString(value) && value.length === 0) { date = key === 'minDate' ? -Infinity : +Infinity; } else { date = new Date(value); @@ -188,7 +190,7 @@ angular.module('mgcrea.ngStrap.helpers.dateParser', []).provider('$dateParser', time = new Date(value.substr(1, value.length - 2)).setFullYear(1970, 0, 1); } else if (isNumeric(value)) { time = new Date(parseInt(value, 10)).setFullYear(1970, 0, 1); - } else if (angular.isString(value) && 0 === value.length) { + } else if (angular.isString(value) && value.length === 0) { time = key === 'minTime' ? -Infinity : +Infinity; } else { time = $dateParser.parse(value, new Date(1970, 0, 1, 0)); @@ -212,39 +214,66 @@ angular.module('mgcrea.ngStrap.helpers.dateParser', []).provider('$dateParser', } return date; }; - function setMapForFormat(format) { - var keys = Object.keys(setFnMap), i; - var map = [], sortedMap = []; - var clonedFormat = format; - for (i = 0; i < keys.length; i++) { - if (format.split(keys[i]).length > 1) { - var index = clonedFormat.search(keys[i]); - format = format.split(keys[i]).join(''); - if (setFnMap[keys[i]]) { - map[index] = setFnMap[keys[i]]; + function regExpForFormat(format) { + var re = buildDateAbstractRegex(format); + return buildDateParseRegex(re); + } + function buildDateAbstractRegex(format) { + var escapedFormat = escapeReservedSymbols(format); + var escapedLiteralFormat = escapedFormat.replace(/''/g, '\\\''); + var literalRegex = /('(?:\\'|.)*?')/; + var formatParts = escapedLiteralFormat.split(literalRegex); + var dateElements = Object.keys(regExpMap); + var dateRegexParts = []; + angular.forEach(formatParts, function(part) { + if (isFormatStringLiteral(part)) { + part = trimLiteralEscapeChars(part); + } else { + for (var i = 0; i < dateElements.length; i++) { + part = part.split(dateElements[i]).join('${' + i + '}'); } } - } - angular.forEach(map, function(v) { - if (v) sortedMap.push(v); + dateRegexParts.push(part); }); - return sortedMap; + return dateRegexParts.join(''); } function escapeReservedSymbols(text) { - return text.replace(/\//g, '[\\/]').replace('/-/g', '[-]').replace(/\./g, '[.]').replace(/\\s/g, '[\\s]'); + return text.replace(/\\/g, '[\\\\]').replace(/-/g, '[-]').replace(/\./g, '[.]').replace(/\*/g, '[*]').replace(/\+/g, '[+]').replace(/\?/g, '[?]').replace(/\$/g, '[$]').replace(/\^/g, '[^]').replace(/\//g, '[/]').replace(/\\s/g, '[\\s]'); } - function regExpForFormat(format) { - var keys = Object.keys(regExpMap), i; - var re = format; - for (i = 0; i < keys.length; i++) { - re = re.split(keys[i]).join('${' + i + '}'); - } - for (i = 0; i < keys.length; i++) { - re = re.split('${' + i + '}').join('(' + regExpMap[keys[i]] + ')'); + function isFormatStringLiteral(text) { + return /^'.*'$/.test(text); + } + function trimLiteralEscapeChars(text) { + return text.replace(/^'(.*)'$/, '$1'); + } + function buildDateParseRegex(abstractRegex) { + var dateElements = Object.keys(regExpMap); + var re = abstractRegex; + for (var i = 0; i < dateElements.length; i++) { + re = re.split('${' + i + '}').join('(' + regExpMap[dateElements[i]] + ')'); } - format = escapeReservedSymbols(format); return new RegExp('^' + re + '$', [ 'i' ]); } + function setMapForFormat(format) { + var re = buildDateAbstractRegex(format); + return buildDateParseValuesMap(re); + } + function buildDateParseValuesMap(abstractRegex) { + var dateElements = Object.keys(regExpMap); + var valuesRegex = new RegExp('\\${(\\d+)}', 'g'); + var valuesMatch; + var keyIndex; + var valueKey; + var valueFunction; + var valuesFunctionMap = []; + while ((valuesMatch = valuesRegex.exec(abstractRegex)) !== null) { + keyIndex = valuesMatch[1]; + valueKey = dateElements[keyIndex]; + valueFunction = setFnMap[valueKey]; + valuesFunctionMap.push(valueFunction); + } + return valuesFunctionMap; + } $dateParser.init(); return $dateParser; }; diff --git a/dist/modules/date-parser.min.js b/dist/modules/date-parser.min.js index 40d2c1e62..8651e041d 100644 --- a/dist/modules/date-parser.min.js +++ b/dist/modules/date-parser.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.helpers.dateParser',[]).provider('$dateParser',['$localeProvider',function(t){function e(){this.year=1970,this.month=0,this.day=1,this.hours=0,this.minutes=0,this.seconds=0,this.milliseconds=0}function n(){}function s(t){return!isNaN(parseFloat(t))&&isFinite(t)}function r(t,e){for(var n=t.length,s=e.toString().toLowerCase(),r=0;n>r;r++)if(t[r].toLowerCase()===s)return r;return-1}e.prototype.setMilliseconds=function(t){this.milliseconds=t},e.prototype.setSeconds=function(t){this.seconds=t},e.prototype.setMinutes=function(t){this.minutes=t},e.prototype.setHours=function(t){this.hours=t},e.prototype.getHours=function(){return this.hours},e.prototype.setDate=function(t){this.day=t},e.prototype.setMonth=function(t){this.month=t},e.prototype.setFullYear=function(t){this.year=t},e.prototype.fromDate=function(t){return this.year=t.getFullYear(),this.month=t.getMonth(),this.day=t.getDate(),this.hours=t.getHours(),this.minutes=t.getMinutes(),this.seconds=t.getSeconds(),this.milliseconds=t.getMilliseconds(),this},e.prototype.toDate=function(){return new Date(this.year,this.month,this.day,this.hours,this.minutes,this.seconds,this.milliseconds)};var i=e.prototype,o=this.defaults={format:'shortDate',strict:!1};this.$get=['$locale','dateFilter',function(t,a){var u=function(u){function h(t){var e,n=Object.keys(D),s=[],r=[],i=t;for(e=0;e1){var o=i.search(n[e]);t=t.split(n[e]).join(''),D[n[e]]&&(s[o]=D[n[e]])}return angular.forEach(s,function(t){t&&r.push(t)}),r}function l(t){return t.replace(/\//g,'[\\/]').replace('/-/g','[-]').replace(/\./g,'[.]').replace(/\\s/g,'[\\s]')}function c(t){var e,n=Object.keys(p),s=t;for(e=0;e=1*t&&2===t.length?this.setFullYear(2e3+1*t):this.setFullYear(1*t)}};return m.init=function(){m.$format=t.DATETIME_FORMATS[g.format]||g.format,f=c(m.$format),M=h(m.$format)},m.isValid=function(t){return angular.isDate(t)?!isNaN(t.getTime()):f.test(t)},m.parse=function(n,s,r,i){r&&(r=t.DATETIME_FORMATS[r]||r),angular.isDate(n)&&(n=a(n,r||m.$format,i));var o=r?c(r):f,u=r?h(r):M,l=o.exec(n);if(!l)return!1;for(var g=s&&!isNaN(s.getTime())?(new e).fromDate(s):(new e).fromDate(new Date(1970,0,1,0)),p=0;p12?t.getHours()+2:0),t):null},m.timezoneOffsetAdjust=function(t,e,n){return t?(e&&'UTC'===e&&(t=new Date(t.getTime()),t.setMinutes(t.getMinutes()+(n?-1:1)*t.getTimezoneOffset())),t):null},m.init(),m};return u}]}]); +'use strict';angular.module('mgcrea.ngStrap.helpers.dateParser',[]).provider('$dateParser',['$localeProvider',function(t){function e(){this.year=1970,this.month=0,this.day=1,this.hours=0,this.minutes=0,this.seconds=0,this.milliseconds=0}function n(){}function r(t){return!isNaN(parseFloat(t))&&isFinite(t)}function s(t,e){for(var n=t.length,r=e.toString().toLowerCase(),s=0;s12?t.getHours()+2:0),t):null},T.timezoneOffsetAdjust=function(t,e,n){return t?(e&&'UTC'===e&&(t=new Date(t.getTime()),t.setMinutes(t.getMinutes()+(n?-1:1)*t.getTimezoneOffset())),t):null},T.init(),T};return u}]}]); //# sourceMappingURL=date-parser.min.js.map diff --git a/dist/modules/date-parser.min.js.map b/dist/modules/date-parser.min.js.map index adb84a6bc..773a55769 100644 --- a/dist/modules/date-parser.min.js.map +++ b/dist/modules/date-parser.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/date-parser.js"],"names":["angular","this","seconds","provider","$localeProvider","milliseconds","year","ParseDate","prototype","hours","value","array","isNumeric","i","isNaN","parseFloat","n","isFinite","indexOfCaseInsensitive","defaults","len","length","str","toString","toLowerCase","format","strict","DateParserFactory","minutes","day","month","getFullYear","getHours","getMilliseconds","getMonth","Date","getMinutes","proto","noop","toDate","$dateParser","regExpMap","sss","$get","options","$locale","dateFilter","mm","config","split","keys","setFnMap","map","index","clonedFormat","forEach","search","v","sortedMap","push","regExpForFormat","re","join","text","replace","Object","escapeReservedSymbols","RegExp","regex","m","extend","HH","H","hh","h","a","EEEE","EEE","DATETIME_FORMATS","SHORTDAY","dd","d","MMMM","MMM","SHORTMONTH","MM","M","DAY","yyyy","yy","y","MONTH","ss","setSeconds","s","setMinutes","setHours","setDate","setMonth","match","setFullYear","setMap","init","setMapForFormat","$format","test","baseDate","formatRegex","formatSetMap","matches","date","getTime","timezone","newDate","parseInt","exec","fromDate","call","getDate","today","getDateForAttribute","key","isString","substr","getTimeForAttribute","time","daylightSavingAdjust","Infinity","parse","timezoneOffsetAdjust","undo"],"mappings":"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","file":"modules/date-parser.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateParser', [])\n\n.provider('$dateParser', function($localeProvider) {\n\n // define a custom ParseDate object to use instead of native Date\n // to avoid date values wrapping when setting date component values\n function ParseDate() {\n this.year = 1970;\n this.month = 0;\n this.day = 1;\n this.hours = 0;\n this.minutes = 0;\n this.seconds = 0;\n this.milliseconds = 0;\n }\n\n ParseDate.prototype.setMilliseconds = function(value) { this.milliseconds = value; };\n ParseDate.prototype.setSeconds = function(value) { this.seconds = value; };\n ParseDate.prototype.setMinutes = function(value) { this.minutes = value; };\n ParseDate.prototype.setHours = function(value) { this.hours = value; };\n ParseDate.prototype.getHours = function() { return this.hours; };\n ParseDate.prototype.setDate = function(value) { this.day = value; };\n ParseDate.prototype.setMonth = function(value) { this.month = value; };\n ParseDate.prototype.setFullYear = function(value) { this.year = value; };\n ParseDate.prototype.fromDate = function(value) {\n this.year = value.getFullYear();\n this.month = value.getMonth();\n this.day = value.getDate();\n this.hours = value.getHours();\n this.minutes = value.getMinutes();\n this.seconds = value.getSeconds();\n this.milliseconds = value.getMilliseconds();\n return this;\n };\n\n ParseDate.prototype.toDate = function() {\n return new Date(this.year, this.month, this.day, this.hours, this.minutes, this.seconds, this.milliseconds);\n };\n\n var proto = ParseDate.prototype;\n\n function noop() {\n }\n\n function isNumeric(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n\n function indexOfCaseInsensitive(array, value) {\n var len = array.length, str=value.toString().toLowerCase();\n for (var i=0; i 12 when midnight changeover, but then cannot generate\n * midnight datetime, so jump to 1AM, otherwise reset.\n * @param date (Date) the date to check\n * @return (Date) the corrected date\n *\n * __ copied from jquery ui datepicker __\n */\n $dateParser.daylightSavingAdjust = function(date) {\n if (!date) {\n return null;\n }\n date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);\n return date;\n };\n\n /* Correct the date for timezone offset.\n * @param date (Date) the date to adjust\n * @param timezone (string) the timezone to adjust for\n * @param undo (boolean) to add or subtract timezone offset\n * @return (Date) the corrected date\n */\n $dateParser.timezoneOffsetAdjust = function(date, timezone, undo) {\n if (!date) {\n return null;\n }\n // Right now, only 'UTC' is supported.\n if (timezone && timezone === 'UTC') {\n date = new Date(date.getTime());\n date.setMinutes(date.getMinutes() + (undo?-1:1)*date.getTimezoneOffset());\n }\n return date;\n };\n\n // Private functions\n\n function setMapForFormat(format) {\n var keys = Object.keys(setFnMap), i;\n var map = [], sortedMap = [];\n // Map to setFn\n var clonedFormat = format;\n for(i = 0; i < keys.length; i++) {\n if(format.split(keys[i]).length > 1) {\n var index = clonedFormat.search(keys[i]);\n format = format.split(keys[i]).join('');\n if(setFnMap[keys[i]]) {\n map[index] = setFnMap[keys[i]];\n }\n }\n }\n // Sort result map\n angular.forEach(map, function(v) {\n // conditional required since angular.forEach broke around v1.2.21\n // related pr: https://github.com/angular/angular.js/pull/8525\n if(v) sortedMap.push(v);\n });\n return sortedMap;\n }\n\n function escapeReservedSymbols(text) {\n return text.replace(/\\//g, '[\\\\/]').replace('/-/g', '[-]').replace(/\\./g, '[.]').replace(/\\\\s/g, '[\\\\s]');\n }\n\n function regExpForFormat(format) {\n var keys = Object.keys(regExpMap), i;\n\n var re = format;\n // Abstract replaces to avoid collisions\n for(i = 0; i < keys.length; i++) {\n re = re.split(keys[i]).join('${' + i + '}');\n }\n // Replace abstracted values\n for(i = 0; i < keys.length; i++) {\n re = re.split('${' + i + '}').join('(' + regExpMap[keys[i]] + ')');\n }\n format = escapeReservedSymbols(format);\n\n return new RegExp('^' + re + '$', ['i']);\n }\n\n $dateParser.init();\n return $dateParser;\n\n };\n\n return DateParserFactory;\n\n };\n\n});\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/date-parser.js"],"names":["angular","this","seconds","provider","$localeProvider","milliseconds","year","ParseDate","prototype","hours","value","noop","array","toLowerCase","isNaN","i","indexOfCaseInsensitive","defaults","len","length","str","toString","format","strict","minutes","day","month","getFullYear","getHours","getMilliseconds","getMonth","Date","getMinutes","proto","isNumeric","toDate","$dateParser","regExpMap","sss","$get","options","$locale","dateFilter","mm","config","literalRegex","formatParts","escapedLiteralFormat","dateElements","keys","escapedFormat","isFormatStringLiteral","part","trimLiteralEscapeChars","split","dateRegexParts","forEach","escapeReservedSymbols","text","join","replace","re","abstractRegex","buildDateParseRegex","setMapForFormat","buildDateAbstractRegex","buildDateParseValuesMap","Object","valuesMatch","keyIndex","valueKey","valuesFunctionMap","valueFunction","valuesRegex","exec","init","setFnMap","DateParserFactory","setMap","m","extend","HH","H","hh","h","a","EEEE","EEE","DATETIME_FORMATS","SHORTDAY","dd","d","MMMM","MMM","SHORTMONTH","MM","M","DAY","yyyy","yy","y","MONTH","ss","setSeconds","s","setMinutes","setHours","setDate","setMonth","match","setFullYear","regex","$format","regExpForFormat","test","baseDate","formatRegex","formatSetMap","matches","date","getTime","timezone","newDate","parseInt","fromDate","call","getDate","getDateForAttribute","key","isString","today","substr","getTimeForAttribute","time","daylightSavingAdjust","Infinity","parse","timezoneOffsetAdjust","undo","getTimezoneOffset"],"mappings":"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","file":"date-parser.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dateParser', [])\n\n.provider('$dateParser', function ($localeProvider) {\n\n // define a custom ParseDate object to use instead of native Date\n // to avoid date values wrapping when setting date component values\n function ParseDate () {\n this.year = 1970;\n this.month = 0;\n this.day = 1;\n this.hours = 0;\n this.minutes = 0;\n this.seconds = 0;\n this.milliseconds = 0;\n }\n\n ParseDate.prototype.setMilliseconds = function (value) { this.milliseconds = value; };\n ParseDate.prototype.setSeconds = function (value) { this.seconds = value; };\n ParseDate.prototype.setMinutes = function (value) { this.minutes = value; };\n ParseDate.prototype.setHours = function (value) { this.hours = value; };\n ParseDate.prototype.getHours = function () { return this.hours; };\n ParseDate.prototype.setDate = function (value) { this.day = value; };\n ParseDate.prototype.setMonth = function (value) { this.month = value; };\n ParseDate.prototype.setFullYear = function (value) { this.year = value; };\n ParseDate.prototype.fromDate = function (value) {\n this.year = value.getFullYear();\n this.month = value.getMonth();\n this.day = value.getDate();\n this.hours = value.getHours();\n this.minutes = value.getMinutes();\n this.seconds = value.getSeconds();\n this.milliseconds = value.getMilliseconds();\n return this;\n };\n\n ParseDate.prototype.toDate = function () {\n return new Date(this.year, this.month, this.day, this.hours, this.minutes, this.seconds, this.milliseconds);\n };\n\n var proto = ParseDate.prototype;\n\n function noop () {\n }\n\n function isNumeric (n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n\n function indexOfCaseInsensitive (array, value) {\n var len = array.length;\n var str = value.toString().toLowerCase();\n for (var i = 0; i < len; i++) {\n if (array[i].toLowerCase() === str) { return i; }\n }\n return -1; // Return -1 per the \"Array.indexOf()\" method.\n }\n\n var defaults = this.defaults = {\n format: 'shortDate',\n strict: false\n };\n\n this.$get = function ($locale, dateFilter) {\n\n var DateParserFactory = function (config) {\n\n var options = angular.extend({}, defaults, config);\n\n var $dateParser = {};\n\n /* eslint-disable key-spacing, quote-props */\n var regExpMap = {\n 'sss' : '[0-9]{3}',\n 'ss' : '[0-5][0-9]',\n 's' : options.strict ? '[1-5]?[0-9]' : '[0-9]|[0-5][0-9]',\n 'mm' : '[0-5][0-9]',\n 'm' : options.strict ? '[1-5]?[0-9]' : '[0-9]|[0-5][0-9]',\n 'HH' : '[01][0-9]|2[0-3]',\n 'H' : options.strict ? '1?[0-9]|2[0-3]' : '[01]?[0-9]|2[0-3]',\n 'hh' : '[0][1-9]|[1][012]',\n 'h' : options.strict ? '[1-9]|1[012]' : '0?[1-9]|1[012]',\n 'a' : 'AM|PM',\n 'EEEE' : $locale.DATETIME_FORMATS.DAY.join('|'),\n 'EEE' : $locale.DATETIME_FORMATS.SHORTDAY.join('|'),\n 'dd' : '0[1-9]|[12][0-9]|3[01]',\n 'd' : options.strict ? '[1-9]|[1-2][0-9]|3[01]' : '0?[1-9]|[1-2][0-9]|3[01]',\n 'MMMM' : $locale.DATETIME_FORMATS.MONTH.join('|'),\n 'MMM' : $locale.DATETIME_FORMATS.SHORTMONTH.join('|'),\n 'MM' : '0[1-9]|1[012]',\n 'M' : options.strict ? '[1-9]|1[012]' : '0?[1-9]|1[012]',\n 'yyyy' : '[1]{1}[0-9]{3}|[2]{1}[0-9]{3}',\n 'yy' : '[0-9]{2}',\n 'y' : options.strict ? '-?(0|[1-9][0-9]{0,3})' : '-?0*[0-9]{1,4}'\n };\n\n var setFnMap = {\n 'sss' : proto.setMilliseconds,\n 'ss' : proto.setSeconds,\n 's' : proto.setSeconds,\n 'mm' : proto.setMinutes,\n 'm' : proto.setMinutes,\n 'HH' : proto.setHours,\n 'H' : proto.setHours,\n 'hh' : proto.setHours,\n 'h' : proto.setHours,\n 'EEEE' : noop,\n 'EEE' : noop,\n 'dd' : proto.setDate,\n 'd' : proto.setDate,\n 'a' : function (value) { var hours = this.getHours() % 12; return this.setHours(value.match(/pm/i) ? hours + 12 : hours); },\n 'MMMM' : function (value) { return this.setMonth(indexOfCaseInsensitive($locale.DATETIME_FORMATS.MONTH, value)); },\n 'MMM' : function (value) { return this.setMonth(indexOfCaseInsensitive($locale.DATETIME_FORMATS.SHORTMONTH, value)); },\n 'MM' : function (value) { return this.setMonth(1 * value - 1); },\n 'M' : function (value) { return this.setMonth(1 * value - 1); },\n 'yyyy' : proto.setFullYear,\n 'yy' : function (value) { return this.setFullYear(2000 + 1 * value); },\n 'y' : function (value) { return (1 * value <= 50 && value.length === 2) ? this.setFullYear(2000 + 1 * value) : this.setFullYear(1 * value); }\n };\n /* eslint-enable key-spacing, quote-props */\n\n var regex;\n var setMap;\n\n $dateParser.init = function () {\n $dateParser.$format = $locale.DATETIME_FORMATS[options.format] || options.format;\n regex = regExpForFormat($dateParser.$format);\n setMap = setMapForFormat($dateParser.$format);\n };\n\n $dateParser.isValid = function (date) {\n if (angular.isDate(date)) return !isNaN(date.getTime());\n return regex.test(date);\n };\n\n $dateParser.parse = function (value, baseDate, format, timezone) {\n // check for date format special names\n if (format) format = $locale.DATETIME_FORMATS[format] || format;\n if (angular.isDate(value)) value = dateFilter(value, format || $dateParser.$format, timezone);\n var formatRegex = format ? regExpForFormat(format) : regex;\n var formatSetMap = format ? setMapForFormat(format) : setMap;\n var matches = formatRegex.exec(value);\n if (!matches) return false;\n // use custom ParseDate object to set parsed values\n var date = baseDate && !isNaN(baseDate.getTime()) ? new ParseDate().fromDate(baseDate) : new ParseDate().fromDate(new Date(1970, 0, 1, 0));\n for (var i = 0; i < matches.length - 1; i++) {\n if (formatSetMap[i]) formatSetMap[i].call(date, matches[i + 1]);\n }\n // convert back to native Date object\n var newDate = date.toDate();\n\n // check new native Date object for day values overflow\n if (parseInt(date.day, 10) !== newDate.getDate()) {\n return false;\n }\n\n return newDate;\n };\n\n $dateParser.getDateForAttribute = function (key, value) {\n var date;\n\n if (value === 'today') {\n var today = new Date();\n date = new Date(today.getFullYear(), today.getMonth(), today.getDate() + (key === 'maxDate' ? 1 : 0), 0, 0, 0, (key === 'minDate' ? 0 : -1));\n } else if (angular.isString(value) && value.match(/^\".+\"$/)) { // Support {{ dateObj }}\n date = new Date(value.substr(1, value.length - 2));\n } else if (isNumeric(value)) {\n date = new Date(parseInt(value, 10));\n } else if (angular.isString(value) && value.length === 0) { // Reset date\n date = key === 'minDate' ? -Infinity : +Infinity;\n } else {\n date = new Date(value);\n }\n\n return date;\n };\n\n $dateParser.getTimeForAttribute = function (key, value) {\n var time;\n\n if (value === 'now') {\n time = new Date().setFullYear(1970, 0, 1);\n } else if (angular.isString(value) && value.match(/^\".+\"$/)) {\n time = new Date(value.substr(1, value.length - 2)).setFullYear(1970, 0, 1);\n } else if (isNumeric(value)) {\n time = new Date(parseInt(value, 10)).setFullYear(1970, 0, 1);\n } else if (angular.isString(value) && value.length === 0) { // Reset time\n time = key === 'minTime' ? -Infinity : +Infinity;\n } else {\n time = $dateParser.parse(value, new Date(1970, 0, 1, 0));\n }\n\n return time;\n };\n\n /* Handle switch to/from daylight saving.\n * Hours may be non-zero on daylight saving cut-over:\n * > 12 when midnight changeover, but then cannot generate\n * midnight datetime, so jump to 1AM, otherwise reset.\n * @param date (Date) the date to check\n * @return (Date) the corrected date\n *\n * __ copied from jquery ui datepicker __\n */\n $dateParser.daylightSavingAdjust = function (date) {\n if (!date) {\n return null;\n }\n date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);\n return date;\n };\n\n /* Correct the date for timezone offset.\n * @param date (Date) the date to adjust\n * @param timezone (string) the timezone to adjust for\n * @param undo (boolean) to add or subtract timezone offset\n * @return (Date) the corrected date\n */\n $dateParser.timezoneOffsetAdjust = function (date, timezone, undo) {\n if (!date) {\n return null;\n }\n // Right now, only 'UTC' is supported.\n if (timezone && timezone === 'UTC') {\n date = new Date(date.getTime());\n date.setMinutes(date.getMinutes() + (undo ? -1 : 1) * date.getTimezoneOffset());\n }\n return date;\n };\n\n // Private functions\n\n function regExpForFormat (format) {\n // `format` string can contain literal values.\n // These need to be escaped by surrounding with\n // single quotes (e.g. `\"h 'in the morning'\"`).\n // In order to output a single quote, escape it - i.e.,\n // two single quotes in a sequence (e.g. `\"h 'o''clock'\"`).\n\n var re = buildDateAbstractRegex(format);\n return buildDateParseRegex(re);\n }\n\n function buildDateAbstractRegex (format) {\n var escapedFormat = escapeReservedSymbols(format);\n var escapedLiteralFormat = escapedFormat.replace(/''/g, '\\\\\\'');\n var literalRegex = /('(?:\\\\'|.)*?')/;\n var formatParts = escapedLiteralFormat.split(literalRegex);\n var dateElements = Object.keys(regExpMap);\n var dateRegexParts = [];\n\n angular.forEach(formatParts, function (part) {\n if (isFormatStringLiteral(part)) {\n part = trimLiteralEscapeChars(part);\n } else {\n // Abstract replaces to avoid collisions\n for (var i = 0; i < dateElements.length; i++) {\n part = part.split(dateElements[i]).join('${' + i + '}');\n }\n }\n dateRegexParts.push(part);\n });\n\n return dateRegexParts.join('');\n }\n\n function escapeReservedSymbols (text) {\n return text.replace(/\\\\/g, '[\\\\\\\\]')\n .replace(/-/g, '[-]')\n .replace(/\\./g, '[.]')\n .replace(/\\*/g, '[*]')\n .replace(/\\+/g, '[+]')\n .replace(/\\?/g, '[?]')\n .replace(/\\$/g, '[$]')\n .replace(/\\^/g, '[^]')\n .replace(/\\//g, '[/]')\n .replace(/\\\\s/g, '[\\\\s]');\n }\n\n function isFormatStringLiteral (text) {\n return /^'.*'$/.test(text);\n }\n\n function trimLiteralEscapeChars (text) {\n return text.replace(/^'(.*)'$/, '$1');\n }\n\n function buildDateParseRegex (abstractRegex) {\n var dateElements = Object.keys(regExpMap);\n var re = abstractRegex;\n\n // Replace abstracted values\n for (var i = 0; i < dateElements.length; i++) {\n re = re.split('${' + i + '}').join('(' + regExpMap[dateElements[i]] + ')');\n }\n\n return new RegExp('^' + re + '$', ['i']);\n }\n\n function setMapForFormat (format) {\n var re = buildDateAbstractRegex(format);\n return buildDateParseValuesMap(re);\n }\n\n function buildDateParseValuesMap (abstractRegex) {\n var dateElements = Object.keys(regExpMap);\n var valuesRegex = new RegExp('\\\\${(\\\\d+)}', 'g');\n var valuesMatch;\n var keyIndex;\n var valueKey;\n var valueFunction;\n var valuesFunctionMap = [];\n\n /* eslint-disable no-cond-assign */\n while ((valuesMatch = valuesRegex.exec(abstractRegex)) !== null) {\n keyIndex = valuesMatch[1];\n valueKey = dateElements[keyIndex];\n valueFunction = setFnMap[valueKey];\n\n valuesFunctionMap.push(valueFunction);\n }\n\n return valuesFunctionMap;\n }\n\n $dateParser.init();\n return $dateParser;\n\n };\n\n return DateParserFactory;\n\n };\n\n});\n"]} \ No newline at end of file diff --git a/dist/modules/datepicker.js b/dist/modules/datepicker.js index 975e6b190..bc9ef9580 100644 --- a/dist/modules/datepicker.js +++ b/dist/modules/datepicker.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -36,11 +36,13 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser minView: 0, startWeek: 0, daysOfWeekDisabled: '', + hasToday: false, + hasClear: false, iconLeft: 'glyphicon glyphicon-chevron-left', iconRight: 'glyphicon glyphicon-chevron-right' }; this.$get = [ '$window', '$document', '$rootScope', '$sce', '$dateFormatter', 'datepickerViews', '$tooltip', '$timeout', function($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) { - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale(); function DatepickerFactory(element, controller, config) { @@ -55,8 +57,11 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser scope.$mode = options.startView; scope.$iconLeft = options.iconLeft; scope.$iconRight = options.iconRight; + scope.$hasToday = options.hasToday; + scope.$hasClear = options.hasClear; var $picker = $datepicker.$views[scope.$mode]; - scope.$select = function(date) { + scope.$select = function(date, disabled) { + if (disabled) return; $datepicker.select(date); }; scope.$selectPane = function(value) { @@ -65,6 +70,22 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser scope.$toggleMode = function() { $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length); }; + scope.$setToday = function() { + if (options.autoclose) { + $datepicker.setMode(0); + $datepicker.select(new Date()); + } else { + $datepicker.select(new Date(), true); + } + }; + scope.$clear = function() { + if (options.autoclose) { + $datepicker.setMode(0); + $datepicker.select(null); + } else { + $datepicker.select(null, true); + } + }; $datepicker.update = function(date) { if (angular.isDate(date) && !isNaN(date.getTime())) { $datepicker.$date = date; @@ -79,7 +100,13 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser } }; $datepicker.select = function(date, keep) { - if (!angular.isDate(controller.$dateValue)) controller.$dateValue = new Date(date); + if (angular.isDate(date)) { + if (!angular.isDate(controller.$dateValue) || isNaN(controller.$dateValue.getTime())) { + controller.$dateValue = new Date(date); + } + } else { + controller.$dateValue = null; + } if (!scope.$mode || keep) { controller.$setViewValue(angular.copy(date)); controller.$render(); @@ -210,8 +237,7 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser return DatepickerFactory; } ]; }).directive('bsDatepicker', [ '$window', '$parse', '$q', '$dateFormatter', '$dateParser', '$datepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) { - var defaults = $datepicker.defaults; - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); return { restrict: 'EAC', require: 'ngModel', @@ -219,15 +245,21 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent', 'hasToday', 'hasClear' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; - angular.forEach([ 'html', 'container', 'autoclose', 'useNative' ], function(key) { + angular.forEach([ 'html', 'container', 'autoclose', 'useNative', 'hasToday', 'hasClear' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { options[key] = false; } }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); var datepicker = $datepicker(element, controller, options); options = datepicker.$options; if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd'; @@ -240,21 +272,31 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser lang: lang, strict: options.strictFormat }); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!datepicker || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i); - newValue === true ? datepicker.show() : datepicker.hide(); - }); - angular.forEach([ 'minDate', 'maxDate' ], function(key) { - angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) { - datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue); - !isNaN(datepicker.$options[key]) && datepicker.$build(false); - validateAgainstMinMaxDate(controller.$dateValue); + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!datepicker || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i); + if (newValue === true) { + datepicker.show(); + } else { + datepicker.hide(); + } }); + } + angular.forEach([ 'minDate', 'maxDate' ], function(key) { + if (angular.isDefined(attr[key])) { + attr.$observe(key, function(newValue) { + datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue); + if (!isNaN(datepicker.$options[key])) datepicker.$build(false); + validateAgainstMinMaxDate(controller.$dateValue); + }); + } }); - angular.isDefined(attr.dateFormat) && attr.$observe('dateFormat', function(newValue) { - datepicker.$options.dateFormat = newValue; - }); + if (angular.isDefined(attr.dateFormat)) { + attr.$observe('dateFormat', function(newValue) { + datepicker.$options.dateFormat = newValue; + }); + } scope.$watch(attr.ngModel, function(newValue, oldValue) { datepicker.update(controller.$dateValue); }, true); @@ -291,9 +333,8 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser if (!parsedDate || isNaN(parsedDate.getTime())) { controller.$setValidity('date', false); return; - } else { - validateAgainstMinMaxDate(parsedDate); } + validateAgainstMinMaxDate(parsedDate); if (options.dateType === 'string') { date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true); return formatDate(date, options.modelDateFormat || options.dateFormat); @@ -305,9 +346,8 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser return date.getTime() / 1e3; } else if (options.dateType === 'iso') { return date.toISOString(); - } else { - return new Date(date); } + return new Date(date); }); controller.$formatters.push(function(modelValue) { var date; @@ -339,10 +379,6 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser } }; } ]).provider('datepickerViews', function() { - var defaults = this.defaults = { - dayFormat: 'dd', - daySplit: 7 - }; function split(arr, size) { var arrays = []; while (arr.length > 0) { @@ -395,11 +431,14 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser } }, build: function() { - var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1), firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset(); - var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5), firstDateOffset = firstDate.getTimezoneOffset(); + var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1); + var firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset(); + var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5); + var firstDateOffset = firstDate.getTimezoneOffset(); var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString(); if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 6e4); - var days = [], day; + var days = []; + var day; for (var i = 0; i < 42; i++) { day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i)); days.push({ @@ -415,6 +454,7 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser scope.showLabels = true; scope.labels = weekDaysLabelsHtml; scope.rows = split(days, this.split); + scope.isTodayDisabled = this.isDisabled(new Date()); this.built = true; }, isSelected: function(date) { @@ -466,8 +506,8 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser } }, build: function() { - var firstMonth = new Date(viewDate.year, 0, 1); - var months = [], month; + var months = []; + var month; for (var i = 0; i < 12; i++) { month = new Date(viewDate.year, i, 1); months.push({ @@ -524,7 +564,8 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser }, build: function() { var firstYear = viewDate.year - viewDate.year % (this.split * 3); - var years = [], year; + var years = []; + var year; for (var i = 0; i < 12; i++) { year = new Date(firstYear + i, 0, 1); years.push({ @@ -550,7 +591,8 @@ angular.module('mgcrea.ngStrap.datepicker', [ 'mgcrea.ngStrap.helpers.dateParser if (!picker.$date) { return; } - var actualYear = picker.$date.getFullYear(), newDate = new Date(picker.$date); + var actualYear = picker.$date.getFullYear(); + var newDate = new Date(picker.$date); if (evt.keyCode === 37) newDate.setYear(actualYear - 1); else if (evt.keyCode === 38) newDate.setYear(actualYear - 4); else if (evt.keyCode === 39) newDate.setYear(actualYear + 1); else if (evt.keyCode === 40) newDate.setYear(actualYear + 4); if (!this.isDisabled(newDate)) picker.select(newDate, true); } diff --git a/dist/modules/datepicker.min.js b/dist/modules/datepicker.min.js index 53fce7b73..f509d6f6d 100644 --- a/dist/modules/datepicker.min.js +++ b/dist/modules/datepicker.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.datepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$datepicker',function(){var e=this.defaults={animation:'am-fade',prefixClass:'datepicker',placement:'bottom-left',templateUrl:'datepicker/datepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!1,dateType:'date',dateFormat:'shortDate',timezone:null,modelDateFormat:null,dayFormat:'dd',monthFormat:'MMM',yearFormat:'yyyy',monthTitleFormat:'MMMM yyyy',yearTitleFormat:'yyyy',strictFormat:!1,autoclose:!1,minDate:-(1/0),maxDate:+(1/0),startView:0,minView:0,startWeek:0,daysOfWeekDisabled:'',iconLeft:'glyphicon glyphicon-chevron-left',iconRight:'glyphicon glyphicon-chevron-right'};this.$get=['$window','$document','$rootScope','$sce','$dateFormatter','datepickerViews','$tooltip','$timeout',function(t,a,n,i,o,r,d,s){function l(t,a,n){function i(e){e.selected=l.$isSelected(e.date)}function o(){t[0].focus()}var l=d(t,angular.extend({},e,n)),g=n.scope,m=l.$options,f=l.$scope;m.startView&&(m.startView-=m.minView);var $=r(l);l.$views=$.views;var h=$.viewDate;f.$mode=m.startView,f.$iconLeft=m.iconLeft,f.$iconRight=m.iconRight;var D=l.$views[f.$mode];f.$select=function(e){l.select(e)},f.$selectPane=function(e){l.$selectPane(e)},f.$toggleMode=function(){l.setMode((f.$mode+1)%l.$views.length)},l.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())&&(l.$date=e,D.update.call(D,e)),l.$build(!0)},l.updateDisabledDates=function(e){m.disabledDateRanges=e;for(var t=0,a=f.rows.length;a>t;t++)angular.forEach(f.rows[t],l.$setDisabledEl)},l.select=function(e,t){angular.isDate(a.$dateValue)||(a.$dateValue=new Date(e)),!f.$mode||t?(a.$setViewValue(angular.copy(e)),a.$render(),m.autoclose&&!t&&s(function(){l.hide(!0)})):(angular.extend(h,{year:e.getFullYear(),month:e.getMonth(),date:e.getDate()}),l.setMode(f.$mode-1),l.$build())},l.setMode=function(e){f.$mode=e,D=l.$views[f.$mode],l.$build()},l.$build=function(e){e===!0&&D.built||(e!==!1||D.built)&&D.build.call(D)},l.$updateSelected=function(){for(var e=0,t=f.rows.length;t>e;e++)angular.forEach(f.rows[e],i)},l.$isSelected=function(e){return D.isSelected(e)},l.$setDisabledEl=function(e){e.disabled=D.isDisabled(e.date)},l.$selectPane=function(e){var t=D.steps,a=new Date(Date.UTC(h.year+(t.year||0)*e,h.month+(t.month||0)*e,1));angular.extend(h,{year:a.getUTCFullYear(),month:a.getUTCMonth(),date:a.getUTCDate()}),l.$build()},l.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),c){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},l.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void(f.$mode?f.$apply(function(){l.setMode(f.$mode-1)}):l.hide(!0));D.onKeyDown(e),g.$digest()}};var p=l.init;l.init=function(){return u&&m.useNative?(t.prop('type','date'),void t.css('-webkit-appearance','textfield')):(c&&(t.prop('type','text'),t.attr('readonly','true'),t.on('click',o)),void p())};var y=l.destroy;l.destroy=function(){u&&m.useNative&&t.off('click',o),y()};var w=l.show;l.show=function(){!c&&t.attr('readonly')||t.attr('disabled')||(w(),s(function(){l.$isShown&&(l.$element.on(c?'touchstart':'mousedown',l.$onMouseDown),m.keyboard&&t.on('keydown',l.$onKeyDown))},0,!1))};var b=l.hide;return l.hide=function(e){l.$isShown&&(l.$element.off(c?'touchstart':'mousedown',l.$onMouseDown),m.keyboard&&t.off('keydown',l.$onKeyDown),b(e))},l}var u=/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent),c='createTouch'in t.document&&u;return e.lang||(e.lang=o.getDefaultLocale()),l.defaults=e,l}]}).directive('bsDatepicker',['$window','$parse','$q','$dateFormatter','$dateParser','$datepicker',function(e,t,a,n,i,o){var r=(o.defaults,/(ip(a|o)d|iphone|android)/gi.test(e.navigator.userAgent));return{restrict:'EAC',require:'ngModel',link:function(e,t,a,d){function s(e){return e&&e.length?e:null}function l(e){if(angular.isDate(e)){var t=isNaN(m.$options.minDate)||e.getTime()>=m.$options.minDate,a=isNaN(m.$options.maxDate)||e.getTime()<=m.$options.maxDate,n=t&&a;d.$setValidity('date',n),d.$setValidity('min',t),d.$setValidity('max',a),n&&(d.$dateValue=e)}}function u(){return!d.$dateValue||isNaN(d.$dateValue.getTime())?'':$(d.$dateValue,c.dateFormat)}var c={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','html','animation','autoclose','dateType','dateFormat','timezone','modelDateFormat','dayFormat','strictFormat','startWeek','startDate','useNative','lang','startView','minView','iconLeft','iconRight','daysOfWeekDisabled','id','prefixClass','prefixEvent'],function(e){angular.isDefined(a[e])&&(c[e]=a[e])});var g=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative'],function(e){angular.isDefined(a[e])&&g.test(a[e])&&(c[e]=!1)});var m=o(t,d,c);c=m.$options,r&&c.useNative&&(c.dateFormat='yyyy-MM-dd');var f=c.lang,$=function(e,t){return n.formatDate(e,t,f)},h=i({format:c.dateFormat,lang:f,strict:c.strictFormat});a.bsShow&&e.$watch(a.bsShow,function(e,t){m&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(datepicker),?/i)),e===!0?m.show():m.hide())}),angular.forEach(['minDate','maxDate'],function(e){angular.isDefined(a[e])&&a.$observe(e,function(t){m.$options[e]=h.getDateForAttribute(e,t),!isNaN(m.$options[e])&&m.$build(!1),l(d.$dateValue)})}),angular.isDefined(a.dateFormat)&&a.$observe('dateFormat',function(e){m.$options.dateFormat=e}),e.$watch(a.ngModel,function(e,t){m.update(d.$dateValue)},!0),angular.isDefined(a.disabledDates)&&e.$watch(a.disabledDates,function(e,t){e=s(e),t=s(t),e&&m.updateDisabledDates(e)}),d.$parsers.unshift(function(e){var t;if(!e)return d.$setValidity('date',!0),null;var a=h.parse(e,d.$dateValue);return!a||isNaN(a.getTime())?void d.$setValidity('date',!1):(l(a),'string'===c.dateType?(t=h.timezoneOffsetAdjust(a,c.timezone,!0),$(t,c.modelDateFormat||c.dateFormat)):(t=h.timezoneOffsetAdjust(d.$dateValue,c.timezone,!0),'number'===c.dateType?t.getTime():'unix'===c.dateType?t.getTime()/1e3:'iso'===c.dateType?t.toISOString():new Date(t)))}),d.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===c.dateType?h.parse(e,null,c.modelDateFormat):'unix'===c.dateType?new Date(1e3*e):new Date(e),d.$dateValue=h.timezoneOffsetAdjust(t,c.timezone),u()}),d.$render=function(){t.val(u())},e.$on('$destroy',function(){m&&m.destroy(),c=null,m=null})}}}]).provider('datepickerViews',function(){function e(e,t){for(var a=[];e.length>0;)a.push(e.splice(0,t));return a}function t(e,t){return(e%t+t)%t}this.defaults={dayFormat:'dd',daySplit:7};this.$get=['$dateFormatter','$dateParser','$sce',function(a,n,i){return function(o){var r=o.$scope,d=o.$options,s=d.lang,l=function(e,t){return a.formatDate(e,t,s)},u=n({format:d.dateFormat,lang:s,strict:d.strictFormat}),c=a.weekdaysShort(s),g=c.slice(d.startWeek).concat(c.slice(0,d.startWeek)),m=i.trustAsHtml(''+g.join('')+''),f=o.$date||(d.startDate?u.getDateForAttribute('startDate',d.startDate):new Date),$={year:f.getFullYear(),month:f.getMonth(),date:f.getDate()},h=[{format:d.dayFormat,split:7,steps:{month:1},update:function(e,t){!this.built||t||e.getFullYear()!==$.year||e.getMonth()!==$.month?(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$build()):(e.getDate()!==$.date||1===e.getDate())&&($.date=o.$date.getDate(),o.$updateSelected())},build:function(){var a=new Date($.year,$.month,1),n=a.getTimezoneOffset(),i=new Date(+a-864e5*t(a.getDay()-d.startWeek,7)),s=i.getTimezoneOffset(),c=u.timezoneOffsetAdjust(new Date,d.timezone).toDateString();s!==n&&(i=new Date(+i+6e4*(s-n)));for(var g,f=[],h=0;42>h;h++)g=u.daylightSavingAdjust(new Date(i.getFullYear(),i.getMonth(),i.getDate()+h)),f.push({date:g,isToday:g.toDateString()===c,label:l(g,this.format),selected:o.$date&&this.isSelected(g),muted:g.getMonth()!==$.month,disabled:this.isDisabled(g)});r.title=l(a,d.monthTitleFormat),r.showLabels=!0,r.labels=m,r.rows=e(f,this.split),this.built=!0},isSelected:function(e){return o.$date&&e.getFullYear()===o.$date.getFullYear()&&e.getMonth()===o.$date.getMonth()&&e.getDate()===o.$date.getDate()},isDisabled:function(e){var t=e.getTime();if(td.maxDate)return!0;if(-1!==d.daysOfWeekDisabled.indexOf(e.getDay()))return!0;if(d.disabledDateRanges)for(var a=0;a=d.disabledDateRanges[a].start&&t<=d.disabledDateRanges[a].end)return!0;return!1},onKeyDown:function(e){if(o.$date){var t,a=o.$date.getTime();37===e.keyCode?t=new Date(a-864e5):38===e.keyCode?t=new Date(a-6048e5):39===e.keyCode?t=new Date(a+864e5):40===e.keyCode&&(t=new Date(a+6048e5)),this.isDisabled(t)||o.select(t,!0)}}},{name:'month',format:d.monthFormat,split:4,steps:{year:1},update:function(e,t){this.built&&e.getFullYear()===$.year?e.getMonth()!==$.month&&(angular.extend($,{month:o.$date.getMonth(),date:o.$date.getDate()}),o.$updateSelected()):(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$build())},build:function(){for(var t,a=(new Date($.year,0,1),[]),n=0;12>n;n++)t=new Date($.year,n,1),a.push({date:t,label:l(t,this.format),selected:o.$isSelected(t),disabled:this.isDisabled(t)});r.title=l(t,d.yearTitleFormat),r.showLabels=!1,r.rows=e(a,this.split),this.built=!0},isSelected:function(e){return o.$date&&e.getFullYear()===o.$date.getFullYear()&&e.getMonth()===o.$date.getMonth()},isDisabled:function(e){var t=+new Date(e.getFullYear(),e.getMonth()+1,0);return td.maxDate},onKeyDown:function(e){if(o.$date){var t=o.$date.getMonth(),a=new Date(o.$date);37===e.keyCode?a.setMonth(t-1):38===e.keyCode?a.setMonth(t-4):39===e.keyCode?a.setMonth(t+1):40===e.keyCode&&a.setMonth(t+4),this.isDisabled(a)||o.select(a,!0)}}},{name:'year',format:d.yearFormat,split:4,steps:{year:12},update:function(e,t){!this.built||t||parseInt(e.getFullYear()/20,10)!==parseInt($.year/20,10)?(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$build()):e.getFullYear()!==$.year&&(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$updateSelected())},build:function(){for(var t,a=$.year-$.year%(3*this.split),n=[],i=0;12>i;i++)t=new Date(a+i,0,1),n.push({date:t,label:l(t,this.format),selected:o.$isSelected(t),disabled:this.isDisabled(t)});r.title=n[0].label+'-'+n[n.length-1].label,r.showLabels=!1,r.rows=e(n,this.split),this.built=!0},isSelected:function(e){return o.$date&&e.getFullYear()===o.$date.getFullYear()},isDisabled:function(e){var t=+new Date(e.getFullYear()+1,0,0);return td.maxDate},onKeyDown:function(e){if(o.$date){var t=o.$date.getFullYear(),a=new Date(o.$date);37===e.keyCode?a.setYear(t-1):38===e.keyCode?a.setYear(t-4):39===e.keyCode?a.setYear(t+1):40===e.keyCode&&a.setYear(t+4),this.isDisabled(a)||o.select(a,!0)}}}];return{views:d.minView?Array.prototype.slice.call(h,d.minView):h,viewDate:$}}}]}); +'use strict';angular.module('mgcrea.ngStrap.datepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$datepicker',function(){var e=this.defaults={animation:'am-fade',prefixClass:'datepicker',placement:'bottom-left',templateUrl:'datepicker/datepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!1,dateType:'date',dateFormat:'shortDate',timezone:null,modelDateFormat:null,dayFormat:'dd',monthFormat:'MMM',yearFormat:'yyyy',monthTitleFormat:'MMMM yyyy',yearTitleFormat:'yyyy',strictFormat:!1,autoclose:!1,minDate:-(1/0),maxDate:+(1/0),startView:0,minView:0,startWeek:0,daysOfWeekDisabled:'',hasToday:!1,hasClear:!1,iconLeft:'glyphicon glyphicon-chevron-left',iconRight:'glyphicon glyphicon-chevron-right'};this.$get=['$window','$document','$rootScope','$sce','$dateFormatter','datepickerViews','$tooltip','$timeout',function(t,a,n,i,o,r,s,l){function d(t,a,n){function i(e){e.selected=d.$isSelected(e.date)}function o(){t[0].focus()}var d=s(t,angular.extend({},e,n)),g=n.scope,f=d.$options,m=d.$scope;f.startView&&(f.startView-=f.minView);var $=r(d);d.$views=$.views;var h=$.viewDate;m.$mode=f.startView,m.$iconLeft=f.iconLeft,m.$iconRight=f.iconRight,m.$hasToday=f.hasToday,m.$hasClear=f.hasClear;var D=d.$views[m.$mode];m.$select=function(e,t){t||d.select(e)},m.$selectPane=function(e){d.$selectPane(e)},m.$toggleMode=function(){d.setMode((m.$mode+1)%d.$views.length)},m.$setToday=function(){f.autoclose?(d.setMode(0),d.select(new Date)):d.select(new Date,!0)},m.$clear=function(){f.autoclose?(d.setMode(0),d.select(null)):d.select(null,!0)},d.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())&&(d.$date=e,D.update.call(D,e)),d.$build(!0)},d.updateDisabledDates=function(e){f.disabledDateRanges=e;for(var t=0,a=m.rows.length;t=f.$options.minDate,a=isNaN(f.$options.maxDate)||e.getTime()<=f.$options.maxDate,n=t&&a;s.$setValidity('date',n),s.$setValidity('min',t),s.$setValidity('max',a),n&&(s.$dateValue=e)}}function u(){return!s.$dateValue||isNaN(s.$dateValue.getTime())?'':$(s.$dateValue,c.dateFormat)}var c={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','html','animation','autoclose','dateType','dateFormat','timezone','modelDateFormat','dayFormat','strictFormat','startWeek','startDate','useNative','lang','startView','minView','iconLeft','iconRight','daysOfWeekDisabled','id','prefixClass','prefixEvent','hasToday','hasClear'],function(e){angular.isDefined(a[e])&&(c[e]=a[e])});var g=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','hasToday','hasClear'],function(e){angular.isDefined(a[e])&&g.test(a[e])&&(c[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(a[n])&&(c[t]=e.$eval(a[n]))});var f=o(t,s,c);c=f.$options,r&&c.useNative&&(c.dateFormat='yyyy-MM-dd');var m=c.lang,$=function(e,t){return n.formatDate(e,t,m)},h=i({format:c.dateFormat,lang:m,strict:c.strictFormat});a.bsShow&&e.$watch(a.bsShow,function(e,t){f&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(datepicker),?/i)),e===!0?f.show():f.hide())}),angular.forEach(['minDate','maxDate'],function(e){angular.isDefined(a[e])&&a.$observe(e,function(t){f.$options[e]=h.getDateForAttribute(e,t),isNaN(f.$options[e])||f.$build(!1),d(s.$dateValue)})}),angular.isDefined(a.dateFormat)&&a.$observe('dateFormat',function(e){f.$options.dateFormat=e}),e.$watch(a.ngModel,function(e,t){f.update(s.$dateValue)},!0),angular.isDefined(a.disabledDates)&&e.$watch(a.disabledDates,function(e,t){e=l(e),t=l(t),e&&f.updateDisabledDates(e)}),s.$parsers.unshift(function(e){var t;if(!e)return s.$setValidity('date',!0),null;var a=h.parse(e,s.$dateValue);return!a||isNaN(a.getTime())?void s.$setValidity('date',!1):(d(a),'string'===c.dateType?(t=h.timezoneOffsetAdjust(a,c.timezone,!0),$(t,c.modelDateFormat||c.dateFormat)):(t=h.timezoneOffsetAdjust(s.$dateValue,c.timezone,!0),'number'===c.dateType?t.getTime():'unix'===c.dateType?t.getTime()/1e3:'iso'===c.dateType?t.toISOString():new Date(t)))}),s.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===c.dateType?h.parse(e,null,c.modelDateFormat):'unix'===c.dateType?new Date(1e3*e):new Date(e),s.$dateValue=h.timezoneOffsetAdjust(t,c.timezone),u()}),s.$render=function(){t.val(u())},e.$on('$destroy',function(){f&&f.destroy(),c=null,f=null})}}}]).provider('datepickerViews',function(){function e(e,t){for(var a=[];e.length>0;)a.push(e.splice(0,t));return a}function t(e,t){return(e%t+t)%t}this.$get=['$dateFormatter','$dateParser','$sce',function(a,n,i){return function(o){var r=o.$scope,s=o.$options,l=s.lang,d=function(e,t){return a.formatDate(e,t,l)},u=n({format:s.dateFormat,lang:l,strict:s.strictFormat}),c=a.weekdaysShort(l),g=c.slice(s.startWeek).concat(c.slice(0,s.startWeek)),f=i.trustAsHtml(''+g.join('')+''),m=o.$date||(s.startDate?u.getDateForAttribute('startDate',s.startDate):new Date),$={year:m.getFullYear(),month:m.getMonth(),date:m.getDate()},h=[{format:s.dayFormat,split:7,steps:{month:1},update:function(e,t){!this.built||t||e.getFullYear()!==$.year||e.getMonth()!==$.month?(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$build()):e.getDate()===$.date&&1!==e.getDate()||($.date=o.$date.getDate(),o.$updateSelected())},build:function(){var a=new Date($.year,$.month,1),n=a.getTimezoneOffset(),i=new Date(+a-864e5*t(a.getDay()-s.startWeek,7)),l=i.getTimezoneOffset(),c=u.timezoneOffsetAdjust(new Date,s.timezone).toDateString();l!==n&&(i=new Date(+i+6e4*(l-n)));for(var g,m=[],h=0;h<42;h++)g=u.daylightSavingAdjust(new Date(i.getFullYear(),i.getMonth(),i.getDate()+h)),m.push({date:g,isToday:g.toDateString()===c,label:d(g,this.format),selected:o.$date&&this.isSelected(g),muted:g.getMonth()!==$.month,disabled:this.isDisabled(g)});r.title=d(a,s.monthTitleFormat),r.showLabels=!0,r.labels=f,r.rows=e(m,this.split),r.isTodayDisabled=this.isDisabled(new Date),this.built=!0},isSelected:function(e){return o.$date&&e.getFullYear()===o.$date.getFullYear()&&e.getMonth()===o.$date.getMonth()&&e.getDate()===o.$date.getDate()},isDisabled:function(e){var t=e.getTime();if(ts.maxDate)return!0;if(s.daysOfWeekDisabled.indexOf(e.getDay())!==-1)return!0;if(s.disabledDateRanges)for(var a=0;a=s.disabledDateRanges[a].start&&t<=s.disabledDateRanges[a].end)return!0;return!1},onKeyDown:function(e){if(o.$date){var t,a=o.$date.getTime();37===e.keyCode?t=new Date(a-864e5):38===e.keyCode?t=new Date(a-6048e5):39===e.keyCode?t=new Date(a+864e5):40===e.keyCode&&(t=new Date(a+6048e5)),this.isDisabled(t)||o.select(t,!0)}}},{name:'month',format:s.monthFormat,split:4,steps:{year:1},update:function(e,t){this.built&&e.getFullYear()===$.year?e.getMonth()!==$.month&&(angular.extend($,{month:o.$date.getMonth(),date:o.$date.getDate()}),o.$updateSelected()):(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$build())},build:function(){for(var t,a=[],n=0;n<12;n++)t=new Date($.year,n,1),a.push({date:t,label:d(t,this.format),selected:o.$isSelected(t),disabled:this.isDisabled(t)});r.title=d(t,s.yearTitleFormat),r.showLabels=!1,r.rows=e(a,this.split),this.built=!0},isSelected:function(e){return o.$date&&e.getFullYear()===o.$date.getFullYear()&&e.getMonth()===o.$date.getMonth()},isDisabled:function(e){var t=+new Date(e.getFullYear(),e.getMonth()+1,0);return ts.maxDate},onKeyDown:function(e){if(o.$date){var t=o.$date.getMonth(),a=new Date(o.$date);37===e.keyCode?a.setMonth(t-1):38===e.keyCode?a.setMonth(t-4):39===e.keyCode?a.setMonth(t+1):40===e.keyCode&&a.setMonth(t+4),this.isDisabled(a)||o.select(a,!0)}}},{name:'year',format:s.yearFormat,split:4,steps:{year:12},update:function(e,t){!this.built||t||parseInt(e.getFullYear()/20,10)!==parseInt($.year/20,10)?(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$build()):e.getFullYear()!==$.year&&(angular.extend($,{year:o.$date.getFullYear(),month:o.$date.getMonth(),date:o.$date.getDate()}),o.$updateSelected())},build:function(){for(var t,a=$.year-$.year%(3*this.split),n=[],i=0;i<12;i++)t=new Date(a+i,0,1),n.push({date:t,label:d(t,this.format),selected:o.$isSelected(t),disabled:this.isDisabled(t)});r.title=n[0].label+'-'+n[n.length-1].label,r.showLabels=!1,r.rows=e(n,this.split),this.built=!0},isSelected:function(e){return o.$date&&e.getFullYear()===o.$date.getFullYear()},isDisabled:function(e){var t=+new Date(e.getFullYear()+1,0,0);return ts.maxDate},onKeyDown:function(e){if(o.$date){var t=o.$date.getFullYear(),a=new Date(o.$date);37===e.keyCode?a.setYear(t-1):38===e.keyCode?a.setYear(t-4):39===e.keyCode?a.setYear(t+1):40===e.keyCode&&a.setYear(t+4),this.isDisabled(a)||o.select(a,!0)}}}];return{views:s.minView?Array.prototype.slice.call(h,s.minView):h,viewDate:$}}}]}); //# sourceMappingURL=datepicker.min.js.map diff --git a/dist/modules/datepicker.min.js.map b/dist/modules/datepicker.min.js.map index 82a6c8909..2c2c582c3 100644 --- a/dist/modules/datepicker.min.js.map +++ b/dist/modules/datepicker.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/datepicker.js"],"names":["angular","container","provider","keyboard","this","defaults","html","delay","useNative","dateType","dateFormat","timezone","modelDateFormat","dayFormat","monthFormat","yearFormat","monthTitleFormat","yearTitleFormat","strictFormat","autoclose","minDate","Infinity","maxDate","startView","minView","startWeek","daysOfWeekDisabled","iconLeft","iconRight","$get","isNative","isTouch","DatepickerFactory","$datepicker","options","$options","pickerViews","views","el","selected","date","element","focus","viewDate","scope","$iconLeft","$iconRight","$picker","$views","$mode","datepickerViews","$selectPane","value","$toggleMode","setMode","select","isDate","$build","length","updateDisabledDates","disabledDateRanges","dateRanges","$date","forEach","i","$dateValue","controller","$setViewValue","copy","$render","hide","keep","Date","extend","year","getFullYear","month","getDate","mode","pristine","call","$updateSelected","rows","built","$isSelected","$setDisabledEl","disabled","isDisabled","steps","targetDate","getUTCFullYear","getUTCMonth","UTC","$onMouseDown","evt","preventDefault","stopPropagation","getUTCDate","targetEl","$onKeyDown","nodeName","test","keyCode","shiftKey","altKey","parentScope","$digest","onKeyDown","prop","init","attr","focusElement","_init","_destroy","destroy","off","_show","show","$timeout","_hide","on","blur","$element","directive","$scope","require","restrict","$window","navigator","userAgent","link","falseValueRegExp","previousValue","normalizeDateRanges","ranges","disabledRanges","datepicker","isMaxValid","isValid","isMinValid","isNaN","parsedDate","getTime","$parsers","unshift","viewValue","$setValidity","getDateFormattedString","formatDate","key","isDefined","lang","dateParser","$dateParser","strict","format","bsShow","$watch","isString","newValue","getDateForAttribute","validateAgainstMinMaxDate","$observe","ngModel","oldValue","disabledDates","parse","timezoneOffsetAdjust","$formatters","isUndefined","modelValue","NaN","daySplit","arr","mod","n","m","arrays","splice","size","push","$dateFormatter","$sce","weekDaysMin","weekdaysShort","weekDaysLabelsHtml","startDate","picker","weekDaysLabels","slice","concat","split","getMonth","update","firstDayOfMonth","firstDate","getDay","today","firstDateOffset","build","day","days","isToday","toDateString","label","muted","title","showLabels","labels","time","isSelected","indexOf","newDate","name","firstMonth","months","lastDate","actualMonth","parseInt","firstYear","years","actualYear","setYear","Array","prototype"],"mappings":"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","file":"modules/datepicker.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.datepicker', [\n 'mgcrea.ngStrap.helpers.dateParser',\n 'mgcrea.ngStrap.helpers.dateFormatter',\n 'mgcrea.ngStrap.tooltip'])\n\n .provider('$datepicker', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n // Uncommenting the following line will break backwards compatability\n // prefixEvent: 'datepicker',\n prefixClass: 'datepicker',\n placement: 'bottom-left',\n templateUrl: 'datepicker/datepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: false,\n dateType: 'date',\n dateFormat: 'shortDate',\n timezone: null,\n modelDateFormat: null,\n dayFormat: 'dd',\n monthFormat: 'MMM',\n yearFormat: 'yyyy',\n monthTitleFormat: 'MMMM yyyy',\n yearTitleFormat: 'yyyy',\n strictFormat: false,\n autoclose: false,\n minDate: -Infinity,\n maxDate: +Infinity,\n startView: 0,\n minView: 0,\n startWeek: 0,\n daysOfWeekDisabled: '',\n iconLeft: 'glyphicon glyphicon-chevron-left',\n iconRight: 'glyphicon glyphicon-chevron-right'\n };\n\n this.$get = function($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) {\n\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale();\n\n function DatepickerFactory(element, controller, config) {\n\n var $datepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $datepicker.$options;\n var scope = $datepicker.$scope;\n if (options.startView) options.startView -= options.minView;\n\n // View vars\n\n var pickerViews = datepickerViews($datepicker);\n $datepicker.$views = pickerViews.views;\n var viewDate = pickerViews.viewDate;\n scope.$mode = options.startView;\n scope.$iconLeft = options.iconLeft;\n scope.$iconRight = options.iconRight;\n var $picker = $datepicker.$views[scope.$mode];\n\n // Scope methods\n\n scope.$select = function(date) {\n $datepicker.select(date);\n };\n scope.$selectPane = function(value) {\n $datepicker.$selectPane(value);\n };\n scope.$toggleMode = function() {\n $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length);\n };\n\n // Public methods\n\n $datepicker.update = function(date) {\n // console.warn('$datepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $datepicker.$date = date;\n $picker.update.call($picker, date);\n }\n // Build only if pristine\n $datepicker.$build(true);\n };\n\n $datepicker.updateDisabledDates = function(dateRanges) {\n options.disabledDateRanges = dateRanges;\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], $datepicker.$setDisabledEl);\n }\n };\n\n $datepicker.select = function(date, keep) {\n // console.warn('$datepicker.select', date, scope.$mode);\n if (!angular.isDate(controller.$dateValue)) controller.$dateValue = new Date(date);\n if (!scope.$mode || keep) {\n controller.$setViewValue(angular.copy(date));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function() { $datepicker.hide(true); });\n }\n } else {\n angular.extend(viewDate, {year: date.getFullYear(), month: date.getMonth(), date: date.getDate()});\n $datepicker.setMode(scope.$mode - 1);\n $datepicker.$build();\n }\n };\n\n $datepicker.setMode = function(mode) {\n // console.warn('$datepicker.setMode', mode);\n scope.$mode = mode;\n $picker = $datepicker.$views[scope.$mode];\n $datepicker.$build();\n };\n\n // Protected methods\n\n $datepicker.$build = function(pristine) {\n // console.warn('$datepicker.$build() viewDate=%o', viewDate);\n if (pristine === true && $picker.built) return;\n if (pristine === false && !$picker.built) return;\n $picker.build.call($picker);\n };\n\n $datepicker.$updateSelected = function() {\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], updateSelected);\n }\n };\n\n $datepicker.$isSelected = function(date) {\n return $picker.isSelected(date);\n };\n\n $datepicker.$setDisabledEl = function(el) {\n el.disabled = $picker.isDisabled(el.date);\n };\n\n $datepicker.$selectPane = function(value) {\n var steps = $picker.steps;\n // set targetDate to first day of month to avoid problems with\n // date values rollover. This assumes the viewDate does not\n // depend on the day of the month\n var targetDate = new Date(Date.UTC(viewDate.year + ((steps.year || 0) * value), viewDate.month + ((steps.month || 0) * value), 1));\n angular.extend(viewDate, {year: targetDate.getUTCFullYear(), month: targetDate.getUTCMonth(), date: targetDate.getUTCDate()});\n $datepicker.$build();\n };\n\n $datepicker.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $datepicker.$onKeyDown = function(evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n if (evt.keyCode === 13) {\n if (!scope.$mode) {\n $datepicker.hide(true);\n } else {\n scope.$apply(function() { $datepicker.setMode(scope.$mode - 1); });\n }\n return;\n }\n\n // Navigate with keyboard\n $picker.onKeyDown(evt);\n parentScope.$digest();\n };\n\n // Private\n\n function updateSelected(el) {\n el.selected = $datepicker.$isSelected(el.date);\n }\n\n function focusElement() {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $datepicker.init;\n $datepicker.init = function() {\n if (isNative && options.useNative) {\n element.prop('type', 'date');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $datepicker.destroy;\n $datepicker.destroy = function() {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $datepicker.show;\n $datepicker.show = function() {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n // if $datepicker is no longer showing, don't setup events\n if (!$datepicker.$isShown) return;\n $datepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $datepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $datepicker.hide;\n $datepicker.hide = function(blur) {\n if (!$datepicker.$isShown) return;\n $datepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $datepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $datepicker;\n\n }\n\n DatepickerFactory.defaults = defaults;\n return DatepickerFactory;\n\n };\n\n })\n\n .directive('bsDatepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) {\n\n var defaults = $datepicker.defaults;\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // Initialize datepicker\n var datepicker = $datepicker(element, controller, options);\n options = datepicker.$options;\n // Set expected iOS format\n if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd';\n\n var lang = options.lang;\n\n var formatDate = function(date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!datepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i);\n newValue === true ? datepicker.show() : datepicker.hide();\n });\n\n // Observe attributes for changes\n angular.forEach(['minDate', 'maxDate'], function(key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) {\n // console.warn('attr.$observe(%s)=%o', key, newValue);\n datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue);\n // Build only if dirty\n !isNaN(datepicker.$options[key]) && datepicker.$build(false);\n validateAgainstMinMaxDate(controller.$dateValue);\n });\n });\n\n // Observe date format\n angular.isDefined(attr.dateFormat) && attr.$observe('dateFormat', function(newValue) {\n datepicker.$options.dateFormat = newValue;\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n datepicker.update(controller.$dateValue);\n }, true);\n\n // Normalize undefined/null/empty array,\n // so that we don't treat changing from undefined->null as a change.\n function normalizeDateRanges(ranges) {\n if (!ranges || !ranges.length) return null;\n return ranges;\n }\n\n if (angular.isDefined(attr.disabledDates)) {\n scope.$watch(attr.disabledDates, function(disabledRanges, previousValue) {\n disabledRanges = normalizeDateRanges(disabledRanges);\n previousValue = normalizeDateRanges(previousValue);\n\n if (disabledRanges) {\n datepicker.updateDisabledDates(disabledRanges);\n }\n });\n }\n\n function validateAgainstMinMaxDate(parsedDate) {\n if (!angular.isDate(parsedDate)) return;\n var isMinValid = isNaN(datepicker.$options.minDate) || parsedDate.getTime() >= datepicker.$options.minDate;\n var isMaxValid = isNaN(datepicker.$options.maxDate) || parsedDate.getTime() <= datepicker.$options.maxDate;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (isValid) controller.$dateValue = parsedDate;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function(viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n controller.$setValidity('date', true);\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n return null;\n }\n var parsedDate = dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedDate || isNaN(parsedDate.getTime())) {\n controller.$setValidity('date', false);\n // return undefined, causes ngModelController to\n // invalidate model value\n return;\n } else {\n validateAgainstMinMaxDate(parsedDate);\n }\n\n if (options.dateType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true);\n return formatDate(date, options.modelDateFormat || options.dateFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.dateType === 'number') {\n return date.getTime();\n } else if (options.dateType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.dateType === 'iso') {\n return date.toISOString();\n } else {\n return new Date(date);\n }\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.dateType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelDateFormat);\n } else if (options.dateType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if (isNaN(date.getTime())) {\n // var today = new Date();\n // date = new Date(today.getFullYear(), today.getMonth(), today.getDate(), 0, 0, 0, 0);\n // }\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getDateFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function() {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getDateFormattedString());\n };\n\n function getDateFormattedString() {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.dateFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (datepicker) datepicker.destroy();\n options = null;\n datepicker = null;\n });\n\n }\n };\n\n })\n\n .provider('datepickerViews', function() {\n\n var defaults = this.defaults = {\n dayFormat: 'dd',\n daySplit: 7\n };\n\n // Split array into smaller arrays\n function split(arr, size) {\n var arrays = [];\n while(arr.length > 0) {\n arrays.push(arr.splice(0, size));\n }\n return arrays;\n }\n\n // Modulus operator\n function mod(n, m) {\n return ((n % m) + m) % m;\n }\n\n this.$get = function($dateFormatter, $dateParser, $sce) {\n\n return function(picker) {\n\n var scope = picker.$scope;\n var options = picker.$options;\n\n var lang = options.lang;\n var formatDate = function(date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n var weekDaysMin = $dateFormatter.weekdaysShort(lang);\n var weekDaysLabels = weekDaysMin.slice(options.startWeek).concat(weekDaysMin.slice(0, options.startWeek));\n var weekDaysLabelsHtml = $sce.trustAsHtml('' + weekDaysLabels.join('') + '');\n\n var startDate = picker.$date || (options.startDate ? dateParser.getDateForAttribute('startDate', options.startDate) : new Date());\n var viewDate = {year: startDate.getFullYear(), month: startDate.getMonth(), date: startDate.getDate()};\n\n var views = [{\n format: options.dayFormat,\n split: 7,\n steps: { month: 1 },\n update: function(date, force) {\n if (!this.built || force || date.getFullYear() !== viewDate.year || date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getDate() !== viewDate.date || date.getDate() === 1) {\n // chaging picker current month will cause viewDate.date to be set to first day of the month,\n // in $datepicker.$selectPane, so picker would not update selected day display if\n // user picks first day of the new month.\n // As a workaround, we are always forcing update when picked date is first day of month.\n viewDate.date = picker.$date.getDate();\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1), firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset();\n var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5), firstDateOffset = firstDate.getTimezoneOffset();\n var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString();\n // Handle daylight time switch\n if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 60e3);\n var days = [], day;\n for(var i = 0; i < 42; i++) { // < 7 * 6\n day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i));\n days.push({date: day, isToday: day.toDateString() === today, label: formatDate(day, this.format), selected: picker.$date && this.isSelected(day), muted: day.getMonth() !== viewDate.month, disabled: this.isDisabled(day)});\n }\n scope.title = formatDate(firstDayOfMonth, options.monthTitleFormat);\n scope.showLabels = true;\n scope.labels = weekDaysLabelsHtml;\n scope.rows = split(days, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth() && date.getDate() === picker.$date.getDate();\n },\n isDisabled: function(date) {\n var time = date.getTime();\n\n // Disabled because of min/max date.\n if (time < options.minDate || time > options.maxDate) return true;\n\n // Disabled due to being a disabled day of the week\n if (options.daysOfWeekDisabled.indexOf(date.getDay()) !== -1) return true;\n\n // Disabled because of disabled date range.\n if (options.disabledDateRanges) {\n for (var i = 0; i < options.disabledDateRanges.length; i++) {\n if (time >= options.disabledDateRanges[i].start && time <= options.disabledDateRanges[i].end) {\n return true;\n }\n }\n }\n\n return false;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualTime = picker.$date.getTime();\n var newDate;\n\n if (evt.keyCode === 37) newDate = new Date(actualTime - 1 * 864e5);\n else if (evt.keyCode === 38) newDate = new Date(actualTime - 7 * 864e5);\n else if (evt.keyCode === 39) newDate = new Date(actualTime + 1 * 864e5);\n else if (evt.keyCode === 40) newDate = new Date(actualTime + 7 * 864e5);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'month',\n format: options.monthFormat,\n split: 4,\n steps: { year: 1 },\n update: function(date, force) {\n if (!this.built || date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstMonth = new Date(viewDate.year, 0, 1);\n var months = [], month;\n for (var i = 0; i < 12; i++) {\n month = new Date(viewDate.year, i, 1);\n months.push({date: month, label: formatDate(month, this.format), selected: picker.$isSelected(month), disabled: this.isDisabled(month)});\n }\n scope.title = formatDate(month, options.yearTitleFormat);\n scope.showLabels = false;\n scope.rows = split(months, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth();\n },\n isDisabled: function(date) {\n var lastDate = +new Date(date.getFullYear(), date.getMonth() + 1, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualMonth = picker.$date.getMonth();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setMonth(actualMonth - 1);\n else if (evt.keyCode === 38) newDate.setMonth(actualMonth - 4);\n else if (evt.keyCode === 39) newDate.setMonth(actualMonth + 1);\n else if (evt.keyCode === 40) newDate.setMonth(actualMonth + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'year',\n format: options.yearFormat,\n split: 4,\n steps: { year: 12 },\n update: function(date, force) {\n if (!this.built || force || parseInt(date.getFullYear()/20, 10) !== parseInt(viewDate.year/20, 10)) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function() {\n var firstYear = viewDate.year - viewDate.year % (this.split * 3);\n var years = [], year;\n for (var i = 0; i < 12; i++) {\n year = new Date(firstYear + i, 0, 1);\n years.push({date: year, label: formatDate(year, this.format), selected: picker.$isSelected(year), disabled: this.isDisabled(year)});\n }\n scope.title = years[0].label + '-' + years[years.length - 1].label;\n scope.showLabels = false;\n scope.rows = split(years, this.split);\n this.built = true;\n },\n isSelected: function(date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear();\n },\n isDisabled: function(date) {\n var lastDate = +new Date(date.getFullYear() + 1, 0, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function(evt) {\n if (!picker.$date) {\n return;\n }\n var actualYear = picker.$date.getFullYear(),\n newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setYear(actualYear - 1);\n else if (evt.keyCode === 38) newDate.setYear(actualYear - 4);\n else if (evt.keyCode === 39) newDate.setYear(actualYear + 1);\n else if (evt.keyCode === 40) newDate.setYear(actualYear + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }];\n\n return {\n views: options.minView ? Array.prototype.slice.call(views, options.minView) : views,\n viewDate: viewDate\n };\n\n };\n\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/datepicker.js"],"names":["angular","container","provider","keyboard","this","defaults","html","delay","useNative","dateType","dateFormat","timezone","modelDateFormat","dayFormat","monthFormat","yearFormat","monthTitleFormat","yearTitleFormat","strictFormat","autoclose","minDate","Infinity","maxDate","startView","minView","startWeek","daysOfWeekDisabled","hasToday","hasClear","iconLeft","iconRight","$get","isTouch","DatepickerFactory","$datepicker","options","$options","pickerViews","views","el","selected","isNative","date","element","focus","viewDate","scope","$iconLeft","$iconRight","$hasToday","$picker","$select","disabled","select","$selectPane","value","$views","$mode","$toggleMode","Date","length","$setToday","$clear","setMode","update","isDate","$date","updateDisabledDates","disabledDateRanges","dateRanges","forEach","i","$build","controller","$dateValue","rows","$setDisabledEl","keep","$setViewValue","$render","copy","extend","year","getFullYear","month","getDate","mode","pristine","call","$updateSelected","built","$isSelected","isDisabled","steps","targetDate","getUTCFullYear","getUTCMonth","UTC","$onMouseDown","evt","preventDefault","stopPropagation","getUTCDate","targetEl","$onKeyDown","nodeName","test","keyCode","shiftKey","altKey","parentScope","$digest","hide","onKeyDown","prop","init","attr","focusElement","_init","_destroy","destroy","off","_show","show","$timeout","_hide","on","blur","$element","directive","$scope","link","datepickerViews","require","$window","navigator","userAgent","restrict","falseValueRegExp","previousValue","normalizeDateRanges","ranges","disabledRanges","datepicker","isMaxValid","isValid","isMinValid","isNaN","parsedDate","getTime","$parsers","unshift","viewValue","$setValidity","getDateFormattedString","formatDate","key","isDefined","$eval","bsKey","lang","format","dateParser","$dateParser","strict","bsShow","isString","newValue","$watch","oldValue","match","validateAgainstMinMaxDate","$observe","disabledDates","timezoneOffsetAdjust","$formatters","modelValue","isUndefined","NaN","parse","split","size","arrays","mod","n","m","arr","splice","$dateFormatter","$sce","weekDaysMin","weekdaysShort","weekDaysLabelsHtml","startDate","picker","weekDaysLabels","slice","concat","getMonth","firstDayOfMonth","firstDayOfMonthOffset","getTimezoneOffset","firstDate","firstDateOffset","build","days","day","daylightSavingAdjust","push","isToday","toDateString","today","muted","label","title","showLabels","labels","isTodayDisabled","isSelected","time","indexOf","getDay","newDate","name","months","lastDate","actualMonth","parseInt","firstYear","years","actualYear","Array","prototype","setYear"],"mappings":"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","file":"datepicker.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.datepicker', [\n 'mgcrea.ngStrap.helpers.dateParser',\n 'mgcrea.ngStrap.helpers.dateFormatter',\n 'mgcrea.ngStrap.tooltip'])\n\n .provider('$datepicker', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n // Uncommenting the following line will break backwards compatability\n // prefixEvent: 'datepicker',\n prefixClass: 'datepicker',\n placement: 'bottom-left',\n templateUrl: 'datepicker/datepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: false,\n dateType: 'date',\n dateFormat: 'shortDate',\n timezone: null,\n modelDateFormat: null,\n dayFormat: 'dd',\n monthFormat: 'MMM',\n yearFormat: 'yyyy',\n monthTitleFormat: 'MMMM yyyy',\n yearTitleFormat: 'yyyy',\n strictFormat: false,\n autoclose: false,\n minDate: -Infinity,\n maxDate: +Infinity,\n startView: 0,\n minView: 0,\n startWeek: 0,\n daysOfWeekDisabled: '',\n hasToday: false,\n hasClear: false,\n iconLeft: 'glyphicon glyphicon-chevron-left',\n iconRight: 'glyphicon glyphicon-chevron-right'\n };\n\n this.$get = function ($window, $document, $rootScope, $sce, $dateFormatter, datepickerViews, $tooltip, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) defaults.lang = $dateFormatter.getDefaultLocale();\n\n function DatepickerFactory (element, controller, config) {\n\n var $datepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $datepicker.$options;\n var scope = $datepicker.$scope;\n if (options.startView) options.startView -= options.minView;\n\n // View vars\n\n var pickerViews = datepickerViews($datepicker);\n $datepicker.$views = pickerViews.views;\n var viewDate = pickerViews.viewDate;\n scope.$mode = options.startView;\n scope.$iconLeft = options.iconLeft;\n scope.$iconRight = options.iconRight;\n scope.$hasToday = options.hasToday;\n scope.$hasClear = options.hasClear;\n var $picker = $datepicker.$views[scope.$mode];\n\n // Scope methods\n\n scope.$select = function (date, disabled) {\n if (disabled) return;\n $datepicker.select(date);\n };\n scope.$selectPane = function (value) {\n $datepicker.$selectPane(value);\n };\n scope.$toggleMode = function () {\n $datepicker.setMode((scope.$mode + 1) % $datepicker.$views.length);\n };\n scope.$setToday = function () {\n if (options.autoclose) {\n $datepicker.setMode(0);\n $datepicker.select(new Date());\n } else {\n $datepicker.select(new Date(), true);\n }\n };\n scope.$clear = function () {\n if (options.autoclose) {\n $datepicker.setMode(0);\n $datepicker.select(null);\n } else {\n $datepicker.select(null, true);\n }\n };\n\n // Public methods\n\n $datepicker.update = function (date) {\n // console.warn('$datepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $datepicker.$date = date;\n $picker.update.call($picker, date);\n }\n // Build only if pristine\n $datepicker.$build(true);\n };\n\n $datepicker.updateDisabledDates = function (dateRanges) {\n options.disabledDateRanges = dateRanges;\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], $datepicker.$setDisabledEl);\n }\n };\n\n $datepicker.select = function (date, keep) {\n // console.warn('$datepicker.select', date, scope.$mode);\n if (angular.isDate(date)) {\n if (!angular.isDate(controller.$dateValue) || isNaN(controller.$dateValue.getTime())) {\n controller.$dateValue = new Date(date);\n }\n } else {\n controller.$dateValue = null;\n }\n if (!scope.$mode || keep) {\n controller.$setViewValue(angular.copy(date));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function () { $datepicker.hide(true); });\n }\n } else {\n angular.extend(viewDate, {year: date.getFullYear(), month: date.getMonth(), date: date.getDate()});\n $datepicker.setMode(scope.$mode - 1);\n $datepicker.$build();\n }\n };\n\n $datepicker.setMode = function (mode) {\n // console.warn('$datepicker.setMode', mode);\n scope.$mode = mode;\n $picker = $datepicker.$views[scope.$mode];\n $datepicker.$build();\n };\n\n // Protected methods\n\n $datepicker.$build = function (pristine) {\n // console.warn('$datepicker.$build() viewDate=%o', viewDate);\n if (pristine === true && $picker.built) return;\n if (pristine === false && !$picker.built) return;\n $picker.build.call($picker);\n };\n\n $datepicker.$updateSelected = function () {\n for (var i = 0, l = scope.rows.length; i < l; i++) {\n angular.forEach(scope.rows[i], updateSelected);\n }\n };\n\n $datepicker.$isSelected = function (date) {\n return $picker.isSelected(date);\n };\n\n $datepicker.$setDisabledEl = function (el) {\n el.disabled = $picker.isDisabled(el.date);\n };\n\n $datepicker.$selectPane = function (value) {\n var steps = $picker.steps;\n // set targetDate to first day of month to avoid problems with\n // date values rollover. This assumes the viewDate does not\n // depend on the day of the month\n var targetDate = new Date(Date.UTC(viewDate.year + ((steps.year || 0) * value), viewDate.month + ((steps.month || 0) * value), 1));\n angular.extend(viewDate, {year: targetDate.getUTCFullYear(), month: targetDate.getUTCMonth(), date: targetDate.getUTCDate()});\n $datepicker.$build();\n };\n\n $datepicker.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $datepicker.$onKeyDown = function (evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n if (evt.keyCode === 13) {\n if (!scope.$mode) {\n $datepicker.hide(true);\n } else {\n scope.$apply(function () { $datepicker.setMode(scope.$mode - 1); });\n }\n return;\n }\n\n // Navigate with keyboard\n $picker.onKeyDown(evt);\n parentScope.$digest();\n };\n\n // Private\n\n function updateSelected (el) {\n el.selected = $datepicker.$isSelected(el.date);\n }\n\n function focusElement () {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $datepicker.init;\n $datepicker.init = function () {\n if (isNative && options.useNative) {\n element.prop('type', 'date');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $datepicker.destroy;\n $datepicker.destroy = function () {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $datepicker.show;\n $datepicker.show = function () {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n // if $datepicker is no longer showing, don't setup events\n if (!$datepicker.$isShown) return;\n $datepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $datepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $datepicker.hide;\n $datepicker.hide = function (blur) {\n if (!$datepicker.$isShown) return;\n $datepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $datepicker.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $datepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $datepicker;\n\n }\n\n DatepickerFactory.defaults = defaults;\n return DatepickerFactory;\n\n };\n\n })\n\n .directive('bsDatepicker', function ($window, $parse, $q, $dateFormatter, $dateParser, $datepicker) {\n\n // var defaults = $datepicker.defaults;\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'autoclose', 'dateType', 'dateFormat', 'timezone', 'modelDateFormat', 'dayFormat', 'strictFormat', 'startWeek', 'startDate', 'useNative', 'lang', 'startView', 'minView', 'iconLeft', 'iconRight', 'daysOfWeekDisabled', 'id', 'prefixClass', 'prefixEvent', 'hasToday', 'hasClear'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'hasToday', 'hasClear'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Initialize datepicker\n var datepicker = $datepicker(element, controller, options);\n options = datepicker.$options;\n // Set expected iOS format\n if (isNative && options.useNative) options.dateFormat = 'yyyy-MM-dd';\n\n var lang = options.lang;\n\n var formatDate = function (date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!datepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(datepicker),?/i);\n if (newValue === true) {\n datepicker.show();\n } else {\n datepicker.hide();\n }\n });\n }\n\n // Observe attributes for changes\n angular.forEach(['minDate', 'maxDate'], function (key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n if (angular.isDefined(attr[key])) {\n attr.$observe(key, function (newValue) {\n // console.warn('attr.$observe(%s)=%o', key, newValue);\n datepicker.$options[key] = dateParser.getDateForAttribute(key, newValue);\n // Build only if dirty\n if (!isNaN(datepicker.$options[key])) datepicker.$build(false);\n validateAgainstMinMaxDate(controller.$dateValue);\n });\n }\n });\n\n // Observe date format\n if (angular.isDefined(attr.dateFormat)) {\n attr.$observe('dateFormat', function (newValue) {\n datepicker.$options.dateFormat = newValue;\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n datepicker.update(controller.$dateValue);\n }, true);\n\n // Normalize undefined/null/empty array,\n // so that we don't treat changing from undefined->null as a change.\n function normalizeDateRanges (ranges) {\n if (!ranges || !ranges.length) return null;\n return ranges;\n }\n\n if (angular.isDefined(attr.disabledDates)) {\n scope.$watch(attr.disabledDates, function (disabledRanges, previousValue) {\n disabledRanges = normalizeDateRanges(disabledRanges);\n previousValue = normalizeDateRanges(previousValue);\n\n if (disabledRanges) {\n datepicker.updateDisabledDates(disabledRanges);\n }\n });\n }\n\n function validateAgainstMinMaxDate (parsedDate) {\n if (!angular.isDate(parsedDate)) return;\n var isMinValid = isNaN(datepicker.$options.minDate) || parsedDate.getTime() >= datepicker.$options.minDate;\n var isMaxValid = isNaN(datepicker.$options.maxDate) || parsedDate.getTime() <= datepicker.$options.maxDate;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (isValid) controller.$dateValue = parsedDate;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function (viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n controller.$setValidity('date', true);\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n return null;\n }\n var parsedDate = dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedDate || isNaN(parsedDate.getTime())) {\n controller.$setValidity('date', false);\n // return undefined, causes ngModelController to\n // invalidate model value\n return;\n }\n validateAgainstMinMaxDate(parsedDate);\n\n if (options.dateType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedDate, options.timezone, true);\n return formatDate(date, options.modelDateFormat || options.dateFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.dateType === 'number') {\n return date.getTime();\n } else if (options.dateType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.dateType === 'iso') {\n return date.toISOString();\n }\n return new Date(date);\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.dateType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelDateFormat);\n } else if (options.dateType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if (isNaN(date.getTime())) {\n // var today = new Date();\n // date = new Date(today.getFullYear(), today.getMonth(), today.getDate(), 0, 0, 0, 0);\n // }\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getDateFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function () {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getDateFormattedString());\n };\n\n function getDateFormattedString () {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.dateFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (datepicker) datepicker.destroy();\n options = null;\n datepicker = null;\n });\n\n }\n };\n\n })\n\n .provider('datepickerViews', function () {\n\n // var defaults = this.defaults = {\n // dayFormat: 'dd',\n // daySplit: 7\n // };\n\n // Split array into smaller arrays\n function split (arr, size) {\n var arrays = [];\n while (arr.length > 0) {\n arrays.push(arr.splice(0, size));\n }\n return arrays;\n }\n\n // Modulus operator\n function mod (n, m) {\n return ((n % m) + m) % m;\n }\n\n this.$get = function ($dateFormatter, $dateParser, $sce) {\n\n return function (picker) {\n\n var scope = picker.$scope;\n var options = picker.$options;\n\n var lang = options.lang;\n var formatDate = function (date, format) {\n return $dateFormatter.formatDate(date, format, lang);\n };\n var dateParser = $dateParser({format: options.dateFormat, lang: lang, strict: options.strictFormat});\n\n var weekDaysMin = $dateFormatter.weekdaysShort(lang);\n var weekDaysLabels = weekDaysMin.slice(options.startWeek).concat(weekDaysMin.slice(0, options.startWeek));\n var weekDaysLabelsHtml = $sce.trustAsHtml('' + weekDaysLabels.join('') + '');\n\n var startDate = picker.$date || (options.startDate ? dateParser.getDateForAttribute('startDate', options.startDate) : new Date());\n var viewDate = {year: startDate.getFullYear(), month: startDate.getMonth(), date: startDate.getDate()};\n\n var views = [{\n format: options.dayFormat,\n split: 7,\n steps: {month: 1},\n update: function (date, force) {\n if (!this.built || force || date.getFullYear() !== viewDate.year || date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getDate() !== viewDate.date || date.getDate() === 1) {\n // chaging picker current month will cause viewDate.date to be set to first day of the month,\n // in $datepicker.$selectPane, so picker would not update selected day display if\n // user picks first day of the new month.\n // As a workaround, we are always forcing update when picked date is first day of month.\n viewDate.date = picker.$date.getDate();\n picker.$updateSelected();\n }\n },\n build: function () {\n var firstDayOfMonth = new Date(viewDate.year, viewDate.month, 1);\n var firstDayOfMonthOffset = firstDayOfMonth.getTimezoneOffset();\n var firstDate = new Date(+firstDayOfMonth - mod(firstDayOfMonth.getDay() - options.startWeek, 7) * 864e5);\n var firstDateOffset = firstDate.getTimezoneOffset();\n var today = dateParser.timezoneOffsetAdjust(new Date(), options.timezone).toDateString();\n // Handle daylight time switch\n if (firstDateOffset !== firstDayOfMonthOffset) firstDate = new Date(+firstDate + (firstDateOffset - firstDayOfMonthOffset) * 60e3);\n var days = [];\n var day;\n for (var i = 0; i < 42; i++) { // < 7 * 6\n day = dateParser.daylightSavingAdjust(new Date(firstDate.getFullYear(), firstDate.getMonth(), firstDate.getDate() + i));\n days.push({date: day, isToday: day.toDateString() === today, label: formatDate(day, this.format), selected: picker.$date && this.isSelected(day), muted: day.getMonth() !== viewDate.month, disabled: this.isDisabled(day)});\n }\n scope.title = formatDate(firstDayOfMonth, options.monthTitleFormat);\n scope.showLabels = true;\n scope.labels = weekDaysLabelsHtml;\n scope.rows = split(days, this.split);\n scope.isTodayDisabled = this.isDisabled(new Date());\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth() && date.getDate() === picker.$date.getDate();\n },\n isDisabled: function (date) {\n var time = date.getTime();\n\n // Disabled because of min/max date.\n if (time < options.minDate || time > options.maxDate) return true;\n\n // Disabled due to being a disabled day of the week\n if (options.daysOfWeekDisabled.indexOf(date.getDay()) !== -1) return true;\n\n // Disabled because of disabled date range.\n if (options.disabledDateRanges) {\n for (var i = 0; i < options.disabledDateRanges.length; i++) {\n if (time >= options.disabledDateRanges[i].start && time <= options.disabledDateRanges[i].end) {\n return true;\n }\n }\n }\n\n return false;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualTime = picker.$date.getTime();\n var newDate;\n\n if (evt.keyCode === 37) newDate = new Date(actualTime - 1 * 864e5);\n else if (evt.keyCode === 38) newDate = new Date(actualTime - 7 * 864e5);\n else if (evt.keyCode === 39) newDate = new Date(actualTime + 1 * 864e5);\n else if (evt.keyCode === 40) newDate = new Date(actualTime + 7 * 864e5);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'month',\n format: options.monthFormat,\n split: 4,\n steps: {year: 1},\n update: function (date, force) {\n if (!this.built || date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getMonth() !== viewDate.month) {\n angular.extend(viewDate, {month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function () {\n // var firstMonth = new Date(viewDate.year, 0, 1);\n var months = [];\n var month;\n for (var i = 0; i < 12; i++) {\n month = new Date(viewDate.year, i, 1);\n months.push({date: month, label: formatDate(month, this.format), selected: picker.$isSelected(month), disabled: this.isDisabled(month)});\n }\n scope.title = formatDate(month, options.yearTitleFormat);\n scope.showLabels = false;\n scope.rows = split(months, this.split);\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear() && date.getMonth() === picker.$date.getMonth();\n },\n isDisabled: function (date) {\n var lastDate = +new Date(date.getFullYear(), date.getMonth() + 1, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualMonth = picker.$date.getMonth();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setMonth(actualMonth - 1);\n else if (evt.keyCode === 38) newDate.setMonth(actualMonth - 4);\n else if (evt.keyCode === 39) newDate.setMonth(actualMonth + 1);\n else if (evt.keyCode === 40) newDate.setMonth(actualMonth + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }, {\n name: 'year',\n format: options.yearFormat,\n split: 4,\n steps: {year: 12},\n update: function (date, force) {\n if (!this.built || force || parseInt(date.getFullYear() / 20, 10) !== parseInt(viewDate.year / 20, 10)) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$build();\n } else if (date.getFullYear() !== viewDate.year) {\n angular.extend(viewDate, {year: picker.$date.getFullYear(), month: picker.$date.getMonth(), date: picker.$date.getDate()});\n picker.$updateSelected();\n }\n },\n build: function () {\n var firstYear = viewDate.year - viewDate.year % (this.split * 3);\n var years = [];\n var year;\n for (var i = 0; i < 12; i++) {\n year = new Date(firstYear + i, 0, 1);\n years.push({date: year, label: formatDate(year, this.format), selected: picker.$isSelected(year), disabled: this.isDisabled(year)});\n }\n scope.title = years[0].label + '-' + years[years.length - 1].label;\n scope.showLabels = false;\n scope.rows = split(years, this.split);\n this.built = true;\n },\n isSelected: function (date) {\n return picker.$date && date.getFullYear() === picker.$date.getFullYear();\n },\n isDisabled: function (date) {\n var lastDate = +new Date(date.getFullYear() + 1, 0, 0);\n return lastDate < options.minDate || date.getTime() > options.maxDate;\n },\n onKeyDown: function (evt) {\n if (!picker.$date) {\n return;\n }\n var actualYear = picker.$date.getFullYear();\n var newDate = new Date(picker.$date);\n\n if (evt.keyCode === 37) newDate.setYear(actualYear - 1);\n else if (evt.keyCode === 38) newDate.setYear(actualYear - 4);\n else if (evt.keyCode === 39) newDate.setYear(actualYear + 1);\n else if (evt.keyCode === 40) newDate.setYear(actualYear + 4);\n\n if (!this.isDisabled(newDate)) picker.select(newDate, true);\n }\n }];\n\n return {\n views: options.minView ? Array.prototype.slice.call(views, options.minView) : views,\n viewDate: viewDate\n };\n\n };\n\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/datepicker.tpl.js b/dist/modules/datepicker.tpl.js index 468c859b6..e9a14c223 100644 --- a/dist/modules/datepicker.tpl.js +++ b/dist/modules/datepicker.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -8,5 +8,5 @@ 'use strict'; angular.module('mgcrea.ngStrap.datepicker').run([ '$templateCache', function($templateCache) { - $templateCache.put('datepicker/datepicker.tpl.html', ''); + $templateCache.put('datepicker/datepicker.tpl.html', ''); } ]); \ No newline at end of file diff --git a/dist/modules/datepicker.tpl.min.js b/dist/modules/datepicker.tpl.min.js index 31bba4e2d..9ed184c64 100644 --- a/dist/modules/datepicker.tpl.min.js +++ b/dist/modules/datepicker.tpl.min.js @@ -1,8 +1,8 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.datepicker').run(['$templateCache',function(t){t.put('datepicker/datepicker.tpl.html','')}]); \ No newline at end of file +'use strict';angular.module('mgcrea.ngStrap.datepicker').run(['$templateCache',function(t){t.put('datepicker/datepicker.tpl.html','')}]); \ No newline at end of file diff --git a/dist/modules/debounce.js b/dist/modules/debounce.js index 146873025..382ff15fc 100644 --- a/dist/modules/debounce.js +++ b/dist/modules/debounce.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -11,7 +11,9 @@ angular.module('mgcrea.ngStrap.helpers.debounce', []).factory('debounce', [ '$ti return function(func, wait, immediate) { var timeout = null; return function() { - var context = this, args = arguments, callNow = immediate && !timeout; + var context = this; + var args = arguments; + var callNow = immediate && !timeout; if (timeout) { $timeout.cancel(timeout); } @@ -30,9 +32,10 @@ angular.module('mgcrea.ngStrap.helpers.debounce', []).factory('debounce', [ '$ti } ]).factory('throttle', [ '$timeout', function($timeout) { return function(func, wait, options) { var timeout = null; - options || (options = {}); + if (!options) options = {}; return function() { - var context = this, args = arguments; + var context = this; + var args = arguments; if (!timeout) { if (options.leading !== false) { func.apply(context, args); diff --git a/dist/modules/debounce.min.js b/dist/modules/debounce.min.js index eebad6c27..ceb758ecf 100644 --- a/dist/modules/debounce.min.js +++ b/dist/modules/debounce.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.helpers.debounce',[]).factory('debounce',['$timeout',function(n){return function(t,u,r){var e=null;return function(){var a=this,l=arguments,c=r&&!e;return e&&n.cancel(e),e=n(function(){e=null,r||t.apply(a,l)},u,!1),c&&t.apply(a,l),e}}}]).factory('throttle',['$timeout',function(n){return function(t,u,r){var e=null;return r||(r={}),function(){var a=this,l=arguments;e||(r.leading!==!1&&t.apply(a,l),e=n(function(){e=null,r.trailing!==!1&&t.apply(a,l)},u,!1))}}}]); +'use strict';angular.module('mgcrea.ngStrap.helpers.debounce',[]).factory('debounce',['$timeout',function(n){return function(t,u,r){var e=null;return function(){var l=this,a=arguments,c=r&&!e;return e&&n.cancel(e),e=n(function(){e=null,r||t.apply(l,a)},u,!1),c&&t.apply(l,a),e}}}]).factory('throttle',['$timeout',function(n){return function(t,u,r){var e=null;return r||(r={}),function(){var l=this,a=arguments;e||(r.leading!==!1&&t.apply(l,a),e=n(function(){e=null,r.trailing!==!1&&t.apply(l,a)},u,!1))}}}]); //# sourceMappingURL=debounce.min.js.map diff --git a/dist/modules/debounce.min.js.map b/dist/modules/debounce.min.js.map index 2bb753a66..ccfa7b883 100644 --- a/dist/modules/debounce.min.js.map +++ b/dist/modules/debounce.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/debounce.js"],"names":["angular","module","factory","$timeout","timeout","immediate","func","apply","context","args","cancel","callNow","options","leading","trailing","wait"],"mappings":"AAOA,YAEAA,SAAQC,OAAO,sCAAuCC,QAAQ,YAAc,WAAY,SAASC,GAC/F,MAMIC,UAAUD,EAAAA,EAASE,GALrB,GAMID,GAAU,IALd,OAMI,YALF,GAMIE,GAAKC,KAAMC,EAASC,UAAAA,EAAAA,IAAAA,CAkB1B,OAvBML,IACFD,EAMOO,OAAAN,GAJTA,EAMOG,EAAMC,WALXJ,EAAU,KAOLA,GALHE,EAAKC,MAAMC,EAASC,IAa7BP,GAAQ,GACAS,GACDP,EAAAA,MAAUI,EAAAC,GAEPL,OARNF,QAYMU,YAAQC,WAAmB,SAAAV,GAXpC,MAAO,UAYMI,EAAMC,EAASC,GAX1B,GAAIL,GAAU,IAEd,OADAQ,KAYIR,MACEA,WAXJ,GAYII,GAAGI,KAAQE,EAAAA,SAXVV,KACCQ,EAAQC,WAAY,GACtBP,EAYCS,MAAMP,EAAAC,GAVTL,EAAUD,EAAS,WACjBC,EAAU,KACNQ,EAAQE,YAAa,GACvBR,EAAKC,MAAMC,EAASC,IAErBM,GAAM","file":"modules/debounce.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.debounce', [])\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L693\n.factory('debounce', function($timeout) {\n return function(func, wait, immediate) {\n var timeout = null;\n return function() {\n var context = this,\n args = arguments,\n callNow = immediate && !timeout;\n if(timeout) {\n $timeout.cancel(timeout);\n }\n timeout = $timeout(function later() {\n timeout = null;\n if(!immediate) {\n func.apply(context, args);\n }\n }, wait, false);\n if(callNow) {\n func.apply(context, args);\n }\n return timeout;\n };\n };\n})\n\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L661\n.factory('throttle', function($timeout) {\n return function(func, wait, options) {\n var timeout = null;\n options || (options = {});\n return function() {\n var context = this,\n args = arguments;\n if(!timeout) {\n if(options.leading !== false) {\n func.apply(context, args);\n }\n timeout = $timeout(function later() {\n timeout = null;\n if(options.trailing !== false) {\n func.apply(context, args);\n }\n }, wait, false);\n }\n };\n };\n});\n\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/debounce.js"],"names":["angular","module","timeout","factory","$timeout","cancel","this","args","immediate","func","apply","context","callNow","options","leading","wait","trailing"],"mappings":"AAOA,YAEAA,SAIMC,OAAIC,sCAASC,QAAA,YAAA,WAAA,SAAAC,GAHjB,MAIMA,UAASC,EAAOH,EAAAA,GAHpB,GAAIA,GAAU,IACd,OAIEA,YAHA,GAIEA,GAAUI,KACVC,EAAKC,UACHC,EAAKC,IAAeH,CAkB1B,OArBML,IACFE,EAIOC,OAAAH,GAFTA,EAIOQ,EAAMC,WAHXT,EAAU,KAKLA,GAHHO,EAAKC,MAAMC,EAASJ,IAW7BJ,GAAQ,GACAS,GACDV,EAAAA,MAAUS,EAAAJ,GAEPL,OANNC,QASMD,YAAS,WAAA,SAAAE,GARlB,MASM,UAAIS,EAAQC,EAAAA,GARhB,GAAIZ,GASOQ,IAPX,OADKG,KAASA,MAUVX,WARF,GASIA,GAAUI,KACVC,EAAIM,SARHX,KACCW,EAAQC,WAAY,GACtBL,EASCM,MAAMJ,EAAAJ,GAPTL,EAAUE,EAAS,WACjBF,EAAU,KACNW,EAAQG,YAAa,GACvBP,EAAKC,MAAMC,EAASJ,IAErBQ,GAAM","file":"debounce.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.debounce', [])\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L693\n.factory('debounce', function ($timeout) {\n return function (func, wait, immediate) {\n var timeout = null;\n return function () {\n var context = this;\n var args = arguments;\n var callNow = immediate && !timeout;\n if (timeout) {\n $timeout.cancel(timeout);\n }\n timeout = $timeout(function later () {\n timeout = null;\n if (!immediate) {\n func.apply(context, args);\n }\n }, wait, false);\n if (callNow) {\n func.apply(context, args);\n }\n return timeout;\n };\n };\n})\n\n\n// @source jashkenas/underscore\n// @url https://github.com/jashkenas/underscore/blob/1.5.2/underscore.js#L661\n.factory('throttle', function ($timeout) {\n return function (func, wait, options) {\n var timeout = null;\n if (!options) options = {};\n return function () {\n var context = this;\n var args = arguments;\n if (!timeout) {\n if (options.leading !== false) {\n func.apply(context, args);\n }\n timeout = $timeout(function later () {\n timeout = null;\n if (options.trailing !== false) {\n func.apply(context, args);\n }\n }, wait, false);\n }\n };\n };\n});\n"]} \ No newline at end of file diff --git a/dist/modules/dimensions.js b/dist/modules/dimensions.js index 5850c9ac7..5daf91607 100644 --- a/dist/modules/dimensions.js +++ b/dist/modules/dimensions.js @@ -1,14 +1,13 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ 'use strict'; -angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ '$document', '$window', function($document, $window) { - var jqLite = angular.element; +angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', function() { var fn = {}; var nodeName = fn.nodeName = function(element, name) { return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase(); @@ -35,7 +34,16 @@ angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ }; }; fn.setOffset = function(element, options, i) { - var curPosition, curLeft, curCSSTop, curTop, curOffset, curCSSLeft, calculatePosition, position = fn.css(element, 'position'), curElem = angular.element(element), props = {}; + var curPosition; + var curLeft; + var curCSSTop; + var curTop; + var curOffset; + var curCSSLeft; + var calculatePosition; + var position = fn.css(element, 'position'); + var curElem = angular.element(element); + var props = {}; if (position === 'static') { element.style.position = 'relative'; } @@ -73,17 +81,19 @@ angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ var offsetParentRect = { top: 0, left: 0 - }, offsetParentElement, offset; + }; + var offsetParentEl; + var offset; if (fn.css(element, 'position') === 'fixed') { offset = element.getBoundingClientRect(); } else { - offsetParentElement = offsetParent(element); + offsetParentEl = offsetParentElement(element); offset = fn.offset(element); - if (!nodeName(offsetParentElement, 'html')) { - offsetParentRect = fn.offset(offsetParentElement); + if (!nodeName(offsetParentEl, 'html')) { + offsetParentRect = fn.offset(offsetParentEl); } - offsetParentRect.top += fn.css(offsetParentElement, 'borderTopWidth', true); - offsetParentRect.left += fn.css(offsetParentElement, 'borderLeftWidth', true); + offsetParentRect.top += fn.css(offsetParentEl, 'borderTopWidth', true); + offsetParentRect.left += fn.css(offsetParentEl, 'borderLeftWidth', true); } return { width: element.offsetWidth, @@ -92,7 +102,7 @@ angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true) }; }; - var offsetParent = function offsetParentElement(element) { + function offsetParentElement(element) { var docElement = element.ownerDocument; var offsetParent = element.offsetParent || docElement; if (nodeName(offsetParent, '#document')) return docElement.documentElement; @@ -100,7 +110,7 @@ angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ offsetParent = offsetParent.offsetParent; } return offsetParent || docElement.documentElement; - }; + } fn.height = function(element, outer) { var value = element.offsetHeight; if (outer) { @@ -120,4 +130,4 @@ angular.module('mgcrea.ngStrap.helpers.dimensions', []).factory('dimensions', [ return value; }; return fn; -} ]); \ No newline at end of file +}); \ No newline at end of file diff --git a/dist/modules/dimensions.min.js b/dist/modules/dimensions.min.js index 552e446a0..a2c491950 100644 --- a/dist/modules/dimensions.min.js +++ b/dist/modules/dimensions.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.helpers.dimensions',[]).factory('dimensions',['$document','$window',function(t,e){var o=(angular.element,{}),n=o.nodeName=function(t,e){return t.nodeName&&t.nodeName.toLowerCase()===e.toLowerCase()};o.css=function(t,e,o){var n;return n=t.currentStyle?t.currentStyle[e]:window.getComputedStyle?window.getComputedStyle(t)[e]:t.style[e],o===!0?parseFloat(n)||0:n},o.offset=function(t){var e=t.getBoundingClientRect(),o=t.ownerDocument;return{width:e.width||t.offsetWidth,height:e.height||t.offsetHeight,top:e.top+(window.pageYOffset||o.documentElement.scrollTop)-(o.documentElement.clientTop||0),left:e.left+(window.pageXOffset||o.documentElement.scrollLeft)-(o.documentElement.clientLeft||0)}},o.setOffset=function(t,e,n){var s,i,r,f,c,l,d,a=o.css(t,'position'),u=angular.element(t),p={};'static'===a&&(t.style.position='relative'),c=o.offset(t),r=o.css(t,'top'),l=o.css(t,'left'),d=('absolute'===a||'fixed'===a)&&(r+l).indexOf('auto')>-1,d?(s=o.position(t),f=s.top,i=s.left):(f=parseFloat(r)||0,i=parseFloat(l)||0),angular.isFunction(e)&&(e=e.call(t,n,c)),null!==e.top&&(p.top=e.top-c.top+f),null!==e.left&&(p.left=e.left-c.left+i),'using'in e?e.using.call(u,p):u.css({top:p.top+'px',left:p.left+'px'})},o.position=function(t){var e,i,r={top:0,left:0};return'fixed'===o.css(t,'position')?i=t.getBoundingClientRect():(e=s(t),i=o.offset(t),n(e,'html')||(r=o.offset(e)),r.top+=o.css(e,'borderTopWidth',!0),r.left+=o.css(e,'borderLeftWidth',!0)),{width:t.offsetWidth,height:t.offsetHeight,top:i.top-r.top-o.css(t,'marginTop',!0),left:i.left-r.left-o.css(t,'marginLeft',!0)}};var s=function(t){var e=t.ownerDocument,s=t.offsetParent||e;if(n(s,'#document'))return e.documentElement;for(;s&&!n(s,'html')&&'static'===o.css(s,'position');)s=s.offsetParent;return s||e.documentElement};return o.height=function(t,e){var n=t.offsetHeight;return e?n+=o.css(t,'marginTop',!0)+o.css(t,'marginBottom',!0):n-=o.css(t,'paddingTop',!0)+o.css(t,'paddingBottom',!0)+o.css(t,'borderTopWidth',!0)+o.css(t,'borderBottomWidth',!0),n},o.width=function(t,e){var n=t.offsetWidth;return e?n+=o.css(t,'marginLeft',!0)+o.css(t,'marginRight',!0):n-=o.css(t,'paddingLeft',!0)+o.css(t,'paddingRight',!0)+o.css(t,'borderLeftWidth',!0)+o.css(t,'borderRightWidth',!0),n},o}]); +'use strict';angular.module('mgcrea.ngStrap.helpers.dimensions',[]).factory('dimensions',function(){function t(t){var n=t.ownerDocument,s=t.offsetParent||n;if(o(s,'#document'))return n.documentElement;for(;s&&!o(s,'html')&&'static'===e.css(s,'position');)s=s.offsetParent;return s||n.documentElement}var e={},o=e.nodeName=function(t,e){return t.nodeName&&t.nodeName.toLowerCase()===e.toLowerCase()};return e.css=function(t,e,o){var n;return n=t.currentStyle?t.currentStyle[e]:window.getComputedStyle?window.getComputedStyle(t)[e]:t.style[e],o===!0?parseFloat(n)||0:n},e.offset=function(t){var e=t.getBoundingClientRect(),o=t.ownerDocument;return{width:e.width||t.offsetWidth,height:e.height||t.offsetHeight,top:e.top+(window.pageYOffset||o.documentElement.scrollTop)-(o.documentElement.clientTop||0),left:e.left+(window.pageXOffset||o.documentElement.scrollLeft)-(o.documentElement.clientLeft||0)}},e.setOffset=function(t,o,n){var s,i,r,f,c,l,d,a=e.css(t,'position'),u=angular.element(t),p={};'static'===a&&(t.style.position='relative'),c=e.offset(t),r=e.css(t,'top'),l=e.css(t,'left'),d=('absolute'===a||'fixed'===a)&&(r+l).indexOf('auto')>-1,d?(s=e.position(t),f=s.top,i=s.left):(f=parseFloat(r)||0,i=parseFloat(l)||0),angular.isFunction(o)&&(o=o.call(t,n,c)),null!==o.top&&(p.top=o.top-c.top+f),null!==o.left&&(p.left=o.left-c.left+i),'using'in o?o.using.call(u,p):u.css({top:p.top+'px',left:p.left+'px'})},e.position=function(n){var s,i,r={top:0,left:0};return'fixed'===e.css(n,'position')?i=n.getBoundingClientRect():(s=t(n),i=e.offset(n),o(s,'html')||(r=e.offset(s)),r.top+=e.css(s,'borderTopWidth',!0),r.left+=e.css(s,'borderLeftWidth',!0)),{width:n.offsetWidth,height:n.offsetHeight,top:i.top-r.top-e.css(n,'marginTop',!0),left:i.left-r.left-e.css(n,'marginLeft',!0)}},e.height=function(t,o){var n=t.offsetHeight;return o?n+=e.css(t,'marginTop',!0)+e.css(t,'marginBottom',!0):n-=e.css(t,'paddingTop',!0)+e.css(t,'paddingBottom',!0)+e.css(t,'borderTopWidth',!0)+e.css(t,'borderBottomWidth',!0),n},e.width=function(t,o){var n=t.offsetWidth;return o?n+=e.css(t,'marginLeft',!0)+e.css(t,'marginRight',!0):n-=e.css(t,'paddingLeft',!0)+e.css(t,'paddingRight',!0)+e.css(t,'borderLeftWidth',!0)+e.css(t,'borderRightWidth',!0),n},e}); //# sourceMappingURL=dimensions.min.js.map diff --git a/dist/modules/dimensions.min.js.map b/dist/modules/dimensions.min.js.map index 1a0fa8f2d..68957ccf0 100644 --- a/dist/modules/dimensions.min.js.map +++ b/dist/modules/dimensions.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/dimensions.js"],"names":["angular","value","element","currentStyle","nodeName","window","getComputedStyle","name","prop","fn","css","extra","boxRect","getBoundingClientRect","style","height","offsetHeight","offset","left","docElement","ownerDocument","width","curPosition","curLeft","curCSSTop","position","pageYOffset","documentElement","scrollTop","clientTop","pageXOffset","scrollLeft","clientLeft","curCSSLeft","options","i","calculatePosition","curTop","curOffset","indexOf","curElem","props","parseFloat","isFunction","call","top","using","offsetParentRect","offsetParentElement","offsetParent","offsetWidth","outer"],"mappings":"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","file":"modules/dimensions.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dimensions', [])\n\n .factory('dimensions', function($document, $window) {\n\n var jqLite = angular.element;\n var fn = {};\n\n /**\n * Test the element nodeName\n * @param element\n * @param name\n */\n var nodeName = fn.nodeName = function(element, name) {\n return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase();\n };\n\n /**\n * Returns the element computed style\n * @param element\n * @param prop\n * @param extra\n */\n fn.css = function(element, prop, extra) {\n var value;\n if (element.currentStyle) { //IE\n value = element.currentStyle[prop];\n } else if (window.getComputedStyle) {\n value = window.getComputedStyle(element)[prop];\n } else {\n value = element.style[prop];\n }\n return extra === true ? parseFloat(value) || 0 : value;\n };\n\n /**\n * Provides read-only equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.offset = function(element) {\n var boxRect = element.getBoundingClientRect();\n var docElement = element.ownerDocument;\n return {\n width: boxRect.width || element.offsetWidth,\n height: boxRect.height || element.offsetHeight,\n top: boxRect.top + (window.pageYOffset || docElement.documentElement.scrollTop) - (docElement.documentElement.clientTop || 0),\n left: boxRect.left + (window.pageXOffset || docElement.documentElement.scrollLeft) - (docElement.documentElement.clientLeft || 0)\n };\n };\n \n /**\n * Provides set equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip\n * @url http://api.jquery.com/offset/\n * @param element\n * @param options\n * @param i\n */\n fn.setOffset = function (element, options, i) {\n var curPosition,\n curLeft,\n curCSSTop,\n curTop,\n curOffset,\n curCSSLeft,\n calculatePosition,\n position = fn.css(element, 'position'),\n curElem = angular.element(element),\n props = {};\n \n // Set position first, in-case top/left are set even on static elem\n if (position === 'static') {\n element.style.position = 'relative';\n }\n \n curOffset = fn.offset(element);\n curCSSTop = fn.css(element, 'top');\n curCSSLeft = fn.css(element, 'left');\n calculatePosition = (position === 'absolute' || position === 'fixed') && \n (curCSSTop + curCSSLeft).indexOf('auto') > -1;\n \n // Need to be able to calculate position if either\n // top or left is auto and position is either absolute or fixed\n if (calculatePosition) {\n curPosition = fn.position(element);\n curTop = curPosition.top;\n curLeft = curPosition.left;\n } else {\n curTop = parseFloat(curCSSTop) || 0;\n curLeft = parseFloat(curCSSLeft) || 0;\n }\n \n if (angular.isFunction(options)) {\n options = options.call(element, i, curOffset);\n }\n \n if (options.top !== null ) {\n props.top = (options.top - curOffset.top) + curTop;\n }\n if ( options.left !== null ) {\n props.left = (options.left - curOffset.left) + curLeft;\n }\n\n if ('using' in options) {\n options.using.call(curElem, props);\n } else {\n curElem.css({\n top: props.top + 'px',\n left: props.left + 'px'\n });\n }\n };\n\n /**\n * Provides read-only equivalent of jQuery's position function\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.position = function(element) {\n\n var offsetParentRect = {top: 0, left: 0},\n offsetParentElement,\n offset;\n\n // Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is it's only offset parent\n if (fn.css(element, 'position') === 'fixed') {\n\n // We assume that getBoundingClientRect is available when computed position is fixed\n offset = element.getBoundingClientRect();\n\n } else {\n\n // Get *real* offsetParentElement\n offsetParentElement = offsetParent(element);\n\n // Get correct offsets\n offset = fn.offset(element);\n if (!nodeName(offsetParentElement, 'html')) {\n offsetParentRect = fn.offset(offsetParentElement);\n }\n\n // Add offsetParent borders\n offsetParentRect.top += fn.css(offsetParentElement, 'borderTopWidth', true);\n offsetParentRect.left += fn.css(offsetParentElement, 'borderLeftWidth', true);\n }\n\n // Subtract parent offsets and element margins\n return {\n width: element.offsetWidth,\n height: element.offsetHeight,\n top: offset.top - offsetParentRect.top - fn.css(element, 'marginTop', true),\n left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true)\n };\n\n };\n\n /**\n * Returns the closest, non-statically positioned offsetParent of a given element\n * @required-by fn.position\n * @param element\n */\n var offsetParent = function offsetParentElement(element) {\n var docElement = element.ownerDocument;\n var offsetParent = element.offsetParent || docElement;\n if(nodeName(offsetParent, '#document')) return docElement.documentElement;\n while(offsetParent && !nodeName(offsetParent, 'html') && fn.css(offsetParent, 'position') === 'static') {\n offsetParent = offsetParent.offsetParent;\n }\n return offsetParent || docElement.documentElement;\n };\n\n /**\n * Provides equivalent of jQuery's height function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/height/\n * @param element\n * @param outer\n */\n fn.height = function(element, outer) {\n var value = element.offsetHeight;\n if(outer) {\n value += fn.css(element, 'marginTop', true) + fn.css(element, 'marginBottom', true);\n } else {\n value -= fn.css(element, 'paddingTop', true) + fn.css(element, 'paddingBottom', true) + fn.css(element, 'borderTopWidth', true) + fn.css(element, 'borderBottomWidth', true);\n }\n return value;\n };\n\n /**\n * Provides equivalent of jQuery's width function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/width/\n * @param element\n * @param outer\n */\n fn.width = function(element, outer) {\n var value = element.offsetWidth;\n if(outer) {\n value += fn.css(element, 'marginLeft', true) + fn.css(element, 'marginRight', true);\n } else {\n value -= fn.css(element, 'paddingLeft', true) + fn.css(element, 'paddingRight', true) + fn.css(element, 'borderLeftWidth', true) + fn.css(element, 'borderRightWidth', true);\n }\n return value;\n };\n\n return fn;\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/dimensions.js"],"names":["angular","element","factory","offsetParent","docElement","ownerDocument","height","outer","value","offsetHeight","nodeName","fn","css","documentElement","window","getComputedStyle","name","prop","extra","currentStyle","boxRect","getBoundingClientRect","style","offset","left","width","curPosition","curLeft","top","curCSSTop","curTop","setOffset","calculatePosition","position","curElem","props","curCSSLeft","curOffset","parseFloat","indexOf","isFunction","options","call","i","using","offsetParentRect","offsetParentEl","offsetParentElement","offsetWidth"],"mappings":"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","file":"dimensions.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.dimensions', [])\n\n .factory('dimensions', function () {\n\n var fn = {};\n\n /**\n * Test the element nodeName\n * @param element\n * @param name\n */\n var nodeName = fn.nodeName = function (element, name) {\n return element.nodeName && element.nodeName.toLowerCase() === name.toLowerCase();\n };\n\n /**\n * Returns the element computed style\n * @param element\n * @param prop\n * @param extra\n */\n fn.css = function (element, prop, extra) {\n var value;\n if (element.currentStyle) { // IE\n value = element.currentStyle[prop];\n } else if (window.getComputedStyle) {\n value = window.getComputedStyle(element)[prop];\n } else {\n value = element.style[prop];\n }\n return extra === true ? parseFloat(value) || 0 : value;\n };\n\n /**\n * Provides read-only equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.offset = function (element) {\n var boxRect = element.getBoundingClientRect();\n var docElement = element.ownerDocument;\n return {\n width: boxRect.width || element.offsetWidth,\n height: boxRect.height || element.offsetHeight,\n top: boxRect.top + (window.pageYOffset || docElement.documentElement.scrollTop) - (docElement.documentElement.clientTop || 0),\n left: boxRect.left + (window.pageXOffset || docElement.documentElement.scrollLeft) - (docElement.documentElement.clientLeft || 0)\n };\n };\n\n /**\n * Provides set equivalent of jQuery's offset function:\n * @required-by bootstrap-tooltip\n * @url http://api.jquery.com/offset/\n * @param element\n * @param options\n * @param i\n */\n fn.setOffset = function (element, options, i) {\n var curPosition;\n var curLeft;\n var curCSSTop;\n var curTop;\n var curOffset;\n var curCSSLeft;\n var calculatePosition;\n var position = fn.css(element, 'position');\n var curElem = angular.element(element);\n var props = {};\n\n // Set position first, in-case top/left are set even on static elem\n if (position === 'static') {\n element.style.position = 'relative';\n }\n\n curOffset = fn.offset(element);\n curCSSTop = fn.css(element, 'top');\n curCSSLeft = fn.css(element, 'left');\n calculatePosition = (position === 'absolute' || position === 'fixed') &&\n (curCSSTop + curCSSLeft).indexOf('auto') > -1;\n\n // Need to be able to calculate position if either\n // top or left is auto and position is either absolute or fixed\n if (calculatePosition) {\n curPosition = fn.position(element);\n curTop = curPosition.top;\n curLeft = curPosition.left;\n } else {\n curTop = parseFloat(curCSSTop) || 0;\n curLeft = parseFloat(curCSSLeft) || 0;\n }\n\n if (angular.isFunction(options)) {\n options = options.call(element, i, curOffset);\n }\n\n if (options.top !== null) {\n props.top = (options.top - curOffset.top) + curTop;\n }\n if (options.left !== null) {\n props.left = (options.left - curOffset.left) + curLeft;\n }\n\n if ('using' in options) {\n options.using.call(curElem, props);\n } else {\n curElem.css({\n top: props.top + 'px',\n left: props.left + 'px'\n });\n }\n };\n\n /**\n * Provides read-only equivalent of jQuery's position function\n * @required-by bootstrap-tooltip, bootstrap-affix\n * @url http://api.jquery.com/offset/\n * @param element\n */\n fn.position = function (element) {\n\n var offsetParentRect = {top: 0, left: 0};\n var offsetParentEl;\n var offset;\n\n // Fixed elements are offset from window (parentOffset = {top:0, left: 0}, because it is it's only offset parent\n if (fn.css(element, 'position') === 'fixed') {\n\n // We assume that getBoundingClientRect is available when computed position is fixed\n offset = element.getBoundingClientRect();\n\n } else {\n\n // Get *real* offsetParentEl\n offsetParentEl = offsetParentElement(element);\n\n // Get correct offsets\n offset = fn.offset(element);\n if (!nodeName(offsetParentEl, 'html')) {\n offsetParentRect = fn.offset(offsetParentEl);\n }\n\n // Add offsetParent borders\n offsetParentRect.top += fn.css(offsetParentEl, 'borderTopWidth', true);\n offsetParentRect.left += fn.css(offsetParentEl, 'borderLeftWidth', true);\n }\n\n // Subtract parent offsets and element margins\n return {\n width: element.offsetWidth,\n height: element.offsetHeight,\n top: offset.top - offsetParentRect.top - fn.css(element, 'marginTop', true),\n left: offset.left - offsetParentRect.left - fn.css(element, 'marginLeft', true)\n };\n\n };\n\n /**\n * Returns the closest, non-statically positioned offsetParent of a given element\n * @required-by fn.position\n * @param element\n */\n function offsetParentElement (element) {\n var docElement = element.ownerDocument;\n var offsetParent = element.offsetParent || docElement;\n if (nodeName(offsetParent, '#document')) return docElement.documentElement;\n while (offsetParent && !nodeName(offsetParent, 'html') && fn.css(offsetParent, 'position') === 'static') {\n offsetParent = offsetParent.offsetParent;\n }\n return offsetParent || docElement.documentElement;\n }\n\n /**\n * Provides equivalent of jQuery's height function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/height/\n * @param element\n * @param outer\n */\n fn.height = function (element, outer) {\n var value = element.offsetHeight;\n if (outer) {\n value += fn.css(element, 'marginTop', true) + fn.css(element, 'marginBottom', true);\n } else {\n value -= fn.css(element, 'paddingTop', true) + fn.css(element, 'paddingBottom', true) + fn.css(element, 'borderTopWidth', true) + fn.css(element, 'borderBottomWidth', true);\n }\n return value;\n };\n\n /**\n * Provides equivalent of jQuery's width function\n * @required-by bootstrap-affix\n * @url http://api.jquery.com/width/\n * @param element\n * @param outer\n */\n fn.width = function (element, outer) {\n var value = element.offsetWidth;\n if (outer) {\n value += fn.css(element, 'marginLeft', true) + fn.css(element, 'marginRight', true);\n } else {\n value -= fn.css(element, 'paddingLeft', true) + fn.css(element, 'paddingRight', true) + fn.css(element, 'borderLeftWidth', true) + fn.css(element, 'borderRightWidth', true);\n }\n return value;\n };\n\n return fn;\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/dropdown.js b/dist/modules/dropdown.js index cbdac3662..f6659ee57 100644 --- a/dist/modules/dropdown.js +++ b/dist/modules/dropdown.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -26,7 +26,7 @@ angular.module('mgcrea.ngStrap.dropdown', [ 'mgcrea.ngStrap.tooltip' ]).provider function DropdownFactory(element, config) { var $dropdown = {}; var options = angular.extend({}, defaults, config); - var scope = $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new(); + $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new(); $dropdown = $tooltip(element, options); var parentEl = element.parent(); $dropdown.$onKeyDown = function(evt) { @@ -46,17 +46,17 @@ angular.module('mgcrea.ngStrap.dropdown', [ 'mgcrea.ngStrap.tooltip' ]).provider $dropdown.show = function() { show(); $timeout(function() { - options.keyboard && $dropdown.$element && $dropdown.$element.on('keydown', $dropdown.$onKeyDown); + if (options.keyboard && $dropdown.$element) $dropdown.$element.on('keydown', $dropdown.$onKeyDown); bodyEl.on('click', onBodyClick); }, 0, false); - parentEl.hasClass('dropdown') && parentEl.addClass('open'); + if (parentEl.hasClass('dropdown')) parentEl.addClass('open'); }; var hide = $dropdown.hide; $dropdown.hide = function() { if (!$dropdown.$isShown) return; - options.keyboard && $dropdown.$element && $dropdown.$element.off('keydown', $dropdown.$onKeyDown); + if (options.keyboard && $dropdown.$element) $dropdown.$element.off('keydown', $dropdown.$onKeyDown); bodyEl.off('click', onBodyClick); - parentEl.hasClass('dropdown') && parentEl.removeClass('open'); + if (parentEl.hasClass('dropdown')) parentEl.removeClass('open'); hide(); }; var destroy = $dropdown.destroy; @@ -77,36 +77,51 @@ angular.module('mgcrea.ngStrap.dropdown', [ 'mgcrea.ngStrap.tooltip' ]).provider restrict: 'EAC', scope: true, compile: function(tElement, tAttrs) { - var options = {}; if (!tAttrs.bsDropdown) { var nextSibling = tElement[0].nextSibling; while (nextSibling && nextSibling.nodeType !== 1) { nextSibling = nextSibling.nextSibling; } - if (nextSibling.classList.contains('dropdown-menu')) { - options.template = nextSibling.outerHTML; - options.templateUrl = undefined; + if (nextSibling && nextSibling.className.split(' ').indexOf('dropdown-menu') >= 0) { + tAttrs.template = nextSibling.outerHTML; + tAttrs.templateUrl = undefined; nextSibling.parentNode.removeChild(nextSibling); } } return function postLink(scope, element, attr) { - options.scope = scope; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id' ], function(key) { + var options = { + scope: scope + }; + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id', 'autoClose' ], function(key) { if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - attr.bsDropdown && scope.$watch(attr.bsDropdown, function(newValue, oldValue) { - scope.content = newValue; - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!dropdown || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i); - newValue === true ? dropdown.show() : dropdown.hide(); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); + if (attr.bsDropdown) { + scope.$watch(attr.bsDropdown, function(newValue, oldValue) { + scope.content = newValue; + }, true); + } var dropdown = $dropdown(element, options); + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!dropdown || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i); + if (newValue === true) { + dropdown.show(); + } else { + dropdown.hide(); + } + }); + } scope.$on('$destroy', function() { if (dropdown) dropdown.destroy(); options = null; diff --git a/dist/modules/dropdown.min.js b/dist/modules/dropdown.min.js index fe84c8a60..1c518182b 100644 --- a/dist/modules/dropdown.min.js +++ b/dist/modules/dropdown.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.dropdown',['mgcrea.ngStrap.tooltip']).provider('$dropdown',function(){var e=this.defaults={animation:'am-fade',prefixClass:'dropdown',prefixEvent:'dropdown',placement:'bottom-left',templateUrl:'dropdown/dropdown.tpl.html',trigger:'click',container:!1,keyboard:!0,html:!1,delay:0};this.$get=['$window','$rootScope','$tooltip','$timeout',function(o,n,t,r){function a(o,a){function c(e){return e.target!==o[0]?e.target!==o[0]&&d.hide():void 0}var d={},s=angular.extend({},e,a);d.$scope=s.scope&&s.scope.$new()||n.$new();d=t(o,s);var p=o.parent();d.$onKeyDown=function(e){if(/(38|40)/.test(e.keyCode)){e.preventDefault(),e.stopPropagation();var o=angular.element(d.$element[0].querySelectorAll('li:not(.divider) a'));if(o.length){var n;angular.forEach(o,function(e,o){i&&i.call(e,':focus')&&(n=o)}),38===e.keyCode&&n>0?n--:40===e.keyCode&&n0?n--:40===e.keyCode&&n=0&&(o.template=t.outerHTML,o.templateUrl=void 0,t.parentNode.removeChild(t))}return function(e,t,r){var a={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','id','autoClose'],function(e){angular.isDefined(o[e])&&(a[e]=o[e])});var l=/^(false|0|)$/i;angular.forEach(['html','container'],function(e){angular.isDefined(r[e])&&l.test(r[e])&&(a[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(o){var n='bs'+o.charAt(0).toUpperCase()+o.slice(1);angular.isDefined(r[n])&&(a[o]=e.$eval(r[n]))}),r.bsDropdown&&e.$watch(r.bsDropdown,function(o,n){e.content=o},!0);var i=n(t,a);r.bsShow&&e.$watch(r.bsShow,function(e,o){i&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(dropdown),?/i)),e===!0?i.show():i.hide())}),e.$on('$destroy',function(){i&&i.destroy(),a=null,i=null})}}}}]); //# sourceMappingURL=dropdown.min.js.map diff --git a/dist/modules/dropdown.min.js.map b/dist/modules/dropdown.min.js.map index 712381154..bd4ee8d89 100644 --- a/dist/modules/dropdown.min.js.map +++ b/dist/modules/dropdown.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/dropdown.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","placement","bodyEl","matchesSelector","DropdownFactory","$dropdown","options","element","evt","keyCode","onBodyClick","directive","target","hide","stopPropagation","items","$element","querySelectorAll","$rootScope","$new","parentEl","index","forEach","preventDefault","focus","length","show","el","i","$onKeyDown","on","hasClass","$isShown","removeClass","destroy","off","scope","parent","prototype","nextSibling","tElement","restrict","compile","classList","templateUrl","parentNode","removeChild","nodeType","outerHTML","tAttrs","isDefined","key","falseValueRegExp","attr","bsDropdown","$watch","newValue","oldValue","content","dropdown","isString","match","bsShow","$on"],"mappings":"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","file":"modules/dropdown.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.dropdown', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$dropdown', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'dropdown',\n prefixEvent: 'dropdown',\n placement: 'bottom-left',\n templateUrl: 'dropdown/dropdown.tpl.html',\n trigger: 'click',\n container: false,\n keyboard: true,\n html: false,\n delay: 0\n };\n\n this.$get = function($window, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var matchesSelector = Element.prototype.matchesSelector || Element.prototype.webkitMatchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector;\n\n function DropdownFactory(element, config) {\n\n var $dropdown = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n var scope = $dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n $dropdown = $tooltip(element, options);\n var parentEl = element.parent();\n\n // Protected methods\n\n $dropdown.$onKeyDown = function(evt) {\n if (!/(38|40)/.test(evt.keyCode)) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Retrieve focused index\n var items = angular.element($dropdown.$element[0].querySelectorAll('li:not(.divider) a'));\n if (!items.length) return;\n var index;\n angular.forEach(items, function(el, i) {\n if (matchesSelector && matchesSelector.call(el, ':focus')) index = i;\n });\n\n // Navigate with keyboard\n if (evt.keyCode === 38 && index > 0) index--;\n else if (evt.keyCode === 40 && index < items.length - 1) index++;\n else if (angular.isUndefined(index)) index = 0;\n items.eq(index)[0].focus();\n\n };\n\n // Overrides\n\n var show = $dropdown.show;\n $dropdown.show = function() {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n options.keyboard && $dropdown.$element && $dropdown.$element.on('keydown', $dropdown.$onKeyDown);\n bodyEl.on('click', onBodyClick);\n }, 0, false);\n parentEl.hasClass('dropdown') && parentEl.addClass('open');\n };\n\n var hide = $dropdown.hide;\n $dropdown.hide = function() {\n if (!$dropdown.$isShown) return;\n options.keyboard && $dropdown.$element && $dropdown.$element.off('keydown', $dropdown.$onKeyDown);\n bodyEl.off('click', onBodyClick);\n parentEl.hasClass('dropdown') && parentEl.removeClass('open');\n hide();\n };\n\n var destroy = $dropdown.destroy;\n $dropdown.destroy = function() {\n bodyEl.off('click', onBodyClick);\n destroy();\n };\n\n // Private functions\n\n function onBodyClick(evt) {\n if (evt.target === element[0]) return;\n return evt.target !== element[0] && $dropdown.hide();\n }\n\n return $dropdown;\n\n }\n\n return DropdownFactory;\n\n };\n\n })\n\n .directive('bsDropdown', function($window, $sce, $dropdown) {\n\n return {\n restrict: 'EAC',\n scope: true,\n compile: function(tElement, tAttrs) {\n\n // Directive options\n var options = {};\n\n // Support for inlined template (next sibling)\n // It must be fetched before compilation\n if (!tAttrs.bsDropdown) {\n var nextSibling = tElement[0].nextSibling;\n while (nextSibling && nextSibling.nodeType !== 1) {\n nextSibling = nextSibling.nextSibling;\n }\n if (nextSibling.classList.contains('dropdown-menu')) {\n options.template = nextSibling.outerHTML;\n options.templateUrl = undefined;\n nextSibling.parentNode.removeChild(nextSibling);\n }\n }\n\n return function postLink(scope, element, attr) {\n\n // Directive options\n options.scope = scope;\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id'], function(key) {\n if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as an object\n attr.bsDropdown && scope.$watch(attr.bsDropdown, function(newValue, oldValue) {\n scope.content = newValue;\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!dropdown || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i);\n newValue === true ? dropdown.show() : dropdown.hide();\n });\n\n // Initialize dropdown\n var dropdown = $dropdown(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (dropdown) dropdown.destroy();\n options = null;\n dropdown = null;\n });\n\n };\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/dropdown.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","placement","bodyEl","matchesSelector","DropdownFactory","$dropdown","options","element","evt","keyCode","onBodyClick","directive","target","hide","stopPropagation","items","$element","$new","querySelectorAll","parentEl","index","forEach","preventDefault","focus","length","show","el","i","$onKeyDown","on","hasClass","removeClass","destroy","scope","parent","prototype","tAttrs","nextSibling","nodeType","$window","$sce","restrict","compile","template","templateUrl","parentNode","removeChild","undefined","falseValueRegExp","attr","key","bsKey","isDefined","charAt","toUpperCase","slice","test","$watch","newValue","oldValue","content","$eval","dropdown","isString","bsShow","match","$on"],"mappings":"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","file":"dropdown.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.dropdown', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$dropdown', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'dropdown',\n prefixEvent: 'dropdown',\n placement: 'bottom-left',\n templateUrl: 'dropdown/dropdown.tpl.html',\n trigger: 'click',\n container: false,\n keyboard: true,\n html: false,\n delay: 0\n };\n\n this.$get = function ($window, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var matchesSelector = Element.prototype.matchesSelector || Element.prototype.webkitMatchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector;\n\n function DropdownFactory (element, config) {\n\n var $dropdown = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n /* var scope = */$dropdown.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n $dropdown = $tooltip(element, options);\n var parentEl = element.parent();\n\n // Protected methods\n\n $dropdown.$onKeyDown = function (evt) {\n if (!/(38|40)/.test(evt.keyCode)) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Retrieve focused index\n var items = angular.element($dropdown.$element[0].querySelectorAll('li:not(.divider) a'));\n if (!items.length) return;\n var index;\n angular.forEach(items, function (el, i) {\n if (matchesSelector && matchesSelector.call(el, ':focus')) index = i;\n });\n\n // Navigate with keyboard\n if (evt.keyCode === 38 && index > 0) index--;\n else if (evt.keyCode === 40 && index < items.length - 1) index++;\n else if (angular.isUndefined(index)) index = 0;\n items.eq(index)[0].focus();\n\n };\n\n // Overrides\n\n var show = $dropdown.show;\n $dropdown.show = function () {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n if (options.keyboard && $dropdown.$element) $dropdown.$element.on('keydown', $dropdown.$onKeyDown);\n bodyEl.on('click', onBodyClick);\n }, 0, false);\n if (parentEl.hasClass('dropdown')) parentEl.addClass('open');\n };\n\n var hide = $dropdown.hide;\n $dropdown.hide = function () {\n if (!$dropdown.$isShown) return;\n if (options.keyboard && $dropdown.$element) $dropdown.$element.off('keydown', $dropdown.$onKeyDown);\n bodyEl.off('click', onBodyClick);\n if (parentEl.hasClass('dropdown')) parentEl.removeClass('open');\n hide();\n };\n\n var destroy = $dropdown.destroy;\n $dropdown.destroy = function () {\n bodyEl.off('click', onBodyClick);\n destroy();\n };\n\n // Private functions\n\n function onBodyClick (evt) {\n if (evt.target === element[0]) return;\n return evt.target !== element[0] && $dropdown.hide();\n }\n\n return $dropdown;\n\n }\n\n return DropdownFactory;\n\n };\n\n })\n\n .directive('bsDropdown', function ($window, $sce, $dropdown) {\n\n return {\n restrict: 'EAC',\n scope: true,\n compile: function (tElement, tAttrs) {\n\n // Support for inlined template (next sibling)\n // It must be fetched before compilation\n if (!tAttrs.bsDropdown) {\n var nextSibling = tElement[0].nextSibling;\n while (nextSibling && nextSibling.nodeType !== 1) {\n nextSibling = nextSibling.nextSibling;\n }\n if (nextSibling && nextSibling.className.split(' ').indexOf('dropdown-menu') >= 0) {\n tAttrs.template = nextSibling.outerHTML;\n tAttrs.templateUrl = undefined;\n nextSibling.parentNode.removeChild(nextSibling);\n }\n }\n\n return function postLink (scope, element, attr) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'id', 'autoClose'], function (key) {\n if (angular.isDefined(tAttrs[key])) options[key] = tAttrs[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as an object\n if (attr.bsDropdown) {\n scope.$watch(attr.bsDropdown, function (newValue, oldValue) {\n scope.content = newValue;\n }, true);\n }\n\n // Initialize dropdown\n var dropdown = $dropdown(element, options);\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!dropdown || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(dropdown),?/i);\n if (newValue === true) {\n dropdown.show();\n } else {\n dropdown.hide();\n }\n });\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (dropdown) dropdown.destroy();\n options = null;\n dropdown = null;\n });\n\n };\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/dropdown.tpl.js b/dist/modules/dropdown.tpl.js index 7b4e83ac8..62889390b 100644 --- a/dist/modules/dropdown.tpl.js +++ b/dist/modules/dropdown.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -8,5 +8,5 @@ 'use strict'; angular.module('mgcrea.ngStrap.dropdown').run([ '$templateCache', function($templateCache) { - $templateCache.put('dropdown/dropdown.tpl.html', ''); + $templateCache.put('dropdown/dropdown.tpl.html', ''); } ]); \ No newline at end of file diff --git a/dist/modules/dropdown.tpl.min.js b/dist/modules/dropdown.tpl.min.js index 792779f3f..a3953e297 100644 --- a/dist/modules/dropdown.tpl.min.js +++ b/dist/modules/dropdown.tpl.min.js @@ -1,8 +1,8 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.dropdown').run(['$templateCache',function(e){e.put('dropdown/dropdown.tpl.html','')}]); \ No newline at end of file +'use strict';angular.module('mgcrea.ngStrap.dropdown').run(['$templateCache',function(e){e.put('dropdown/dropdown.tpl.html','')}]); \ No newline at end of file diff --git a/dist/modules/modal.js b/dist/modules/modal.js index 63a100eff..f4898f0cd 100644 --- a/dist/modules/modal.js +++ b/dist/modules/modal.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -11,6 +11,7 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. var defaults = this.defaults = { animation: 'am-fade', backdropAnimation: 'am-fade', + customClass: '', prefixClass: 'modal', prefixEvent: 'modal', placement: 'top', @@ -22,13 +23,21 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. backdrop: true, keyboard: true, html: false, - show: true + show: true, + size: null, + zIndex: null }; this.$get = [ '$window', '$rootScope', '$bsCompiler', '$animate', '$timeout', '$sce', 'dimensions', function($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) { var forEach = angular.forEach; - var trim = String.prototype.trim; var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout; var bodyElement = angular.element($window.document.body); + var backdropCount = 0; + var dialogBaseZindex = 1050; + var backdropBaseZindex = 1040; + var validSizes = { + lg: 'modal-lg', + sm: 'modal-sm' + }; function ModalFactory(config) { var $modal = {}; var options = $modal.$options = angular.extend({}, defaults, config); @@ -37,6 +46,10 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. if (!options.element && !options.container) { options.container = 'body'; } + if (options.zIndex) { + dialogBaseZindex = parseInt(options.zIndex, 10); + backdropBaseZindex = dialogBaseZindex - 10; + } $modal.$id = options.id || options.element && options.element.attr('id') || ''; forEach([ 'title', 'content' ], function(key) { if (options[key]) scope[key] = $sce.trustAsHtml(options[key]); @@ -57,15 +70,16 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. }); }; $modal.$isShown = scope.$isShown = false; - var compileData, modalElement, modalScope; + var compileData; + var modalElement; + var modalScope; var backdropElement = angular.element('
'); backdropElement.css({ position: 'fixed', top: '0px', left: '0px', bottom: '0px', - right: '0px', - 'z-index': 1038 + right: '0px' }); promise.then(function(data) { compileData = data; @@ -88,7 +102,8 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. }; $modal.show = function() { if ($modal.$isShown) return; - var parent, after; + var parent; + var after; if (angular.isElement(options.container)) { parent = options.container; after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null; @@ -104,12 +119,30 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. if (modalElement) destroyModalElement(); modalScope = $modal.$scope.$new(); modalElement = $modal.$element = compileData.link(modalScope, function(clonedElement, scope) {}); + if (options.backdrop) { + modalElement.css({ + 'z-index': dialogBaseZindex + backdropCount * 20 + }); + backdropElement.css({ + 'z-index': backdropBaseZindex + backdropCount * 20 + }); + backdropCount++; + } if (scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) { return; } + if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) { + options.onBeforeShow($modal); + } modalElement.css({ display: 'block' }).addClass(options.placement); + if (options.customClass) { + modalElement.addClass(options.customClass); + } + if (options.size && validSizes[options.size]) { + angular.element(findElement('.modal-dialog', modalElement[0])).addClass(validSizes[options.size]); + } if (options.animation) { if (options.backdrop) { backdropElement.addClass(options.backdropAnimation); @@ -139,18 +172,25 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. }; function enterAnimateCallback() { scope.$emit(options.prefixEvent + '.show', $modal); + if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) { + options.onShow($modal); + } } $modal.hide = function() { if (!$modal.$isShown) return; if (scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) { return; } + if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) { + options.onBeforeHide($modal); + } if (angular.version.minor <= 2) { $animate.leave(modalElement, leaveAnimateCallback); } else { $animate.leave(modalElement).then(leaveAnimateCallback); } if (options.backdrop) { + backdropCount--; $animate.leave(backdropElement); } $modal.$isShown = scope.$isShown = false; @@ -160,13 +200,22 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. }; function leaveAnimateCallback() { scope.$emit(options.prefixEvent + '.hide', $modal); - bodyElement.removeClass(options.prefixClass + '-open'); + if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) { + options.onHide($modal); + } + if (findElement('.modal').length <= 0) { + bodyElement.removeClass(options.prefixClass + '-open'); + } if (options.animation) { bodyElement.removeClass(options.prefixClass + '-with-' + options.animation); } } $modal.toggle = function() { - $modal.$isShown ? $modal.hide() : $modal.show(); + if ($modal.$isShown) { + $modal.hide(); + } else { + $modal.show(); + } }; $modal.focus = function() { modalElement[0].focus(); @@ -203,7 +252,11 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. } function hideOnBackdropClick(evt) { if (evt.target !== evt.currentTarget) return; - options.backdrop === 'static' ? $modal.focus() : $modal.hide(); + if (options.backdrop === 'static') { + $modal.focus(); + } else { + $modal.hide(); + } } function preventEventDefault(evt) { evt.preventDefault(); @@ -232,7 +285,7 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. } return ModalFactory; } ]; -}).directive('bsModal', [ '$window', '$sce', '$modal', function($window, $sce, $modal) { +}).directive('bsModal', [ '$window', '$sce', '$parse', '$modal', function($window, $sce, $parse, $modal) { return { restrict: 'EAC', scope: true, @@ -242,25 +295,38 @@ angular.module('mgcrea.ngStrap.modal', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStrap. element: element, show: false }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass', 'customClass', 'modalClass', 'size', 'zIndex' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); + if (options.modalClass) { + options.customClass = options.modalClass; + } var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'backdrop', 'keyboard', 'html', 'container' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); - angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); - attr.bsModal && scope.$watch(attr.bsModal, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; + angular.forEach([ 'title', 'content' ], function(key) { + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + }); } - }, true); + }); + if (attr.bsModal) { + scope.$watch(attr.bsModal, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + }, true); + } var modal = $modal(options); element.on(attr.trigger || 'click', modal.toggle); scope.$on('$destroy', function() { diff --git a/dist/modules/modal.min.js b/dist/modules/modal.min.js index f37555d03..5e40301cb 100644 --- a/dist/modules/modal.min.js +++ b/dist/modules/modal.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.modal',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$modal',function(){var n=this.defaults={animation:'am-fade',backdropAnimation:'am-fade',prefixClass:'modal',prefixEvent:'modal',placement:'top',templateUrl:'modal/modal.tpl.html',template:'',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0};this.$get=['$window','$rootScope','$bsCompiler','$animate','$timeout','$sce','dimensions',function(e,t,o,i,a,r,l){function s(e){function a(){y.$emit(k.prefixEvent+'.show',b)}function l(){y.$emit(k.prefixEvent+'.hide',b),p.removeClass(k.prefixClass+'-open'),k.animation&&p.removeClass(k.prefixClass+'-with-'+k.animation)}function s(){k.backdrop&&(S.on('click',g),A.on('click',g),A.on('wheel',w))}function m(){k.backdrop&&(S.off('click',g),A.off('click',g),A.off('wheel',w))}function $(){k.keyboard&&S.on('keyup',b.$onKeyUp)}function h(){k.keyboard&&S.off('keyup',b.$onKeyUp)}function g(n){n.target===n.currentTarget&&('static'===k.backdrop?b.focus():b.hide())}function w(n){n.preventDefault()}function v(){b.$isShown&&null!==S&&(m(),h()),E&&(E.$destroy(),E=null),S&&(S.remove(),S=b.$element=null)}var b={},k=b.$options=angular.extend({},n,e),x=b.$promise=o.compile(k),y=b.$scope=k.scope&&k.scope.$new()||t.$new();k.element||k.container||(k.container='body'),b.$id=k.id||k.element&&k.element.attr('id')||'',d(['title','content'],function(n){k[n]&&(y[n]=r.trustAsHtml(k[n]))}),y.$hide=function(){y.$$postDigest(function(){b.hide()})},y.$show=function(){y.$$postDigest(function(){b.show()})},y.$toggle=function(){y.$$postDigest(function(){b.toggle()})},b.$isShown=y.$isShown=!1;var C,S,E,A=angular.element('
');return A.css({position:'fixed',top:'0px',left:'0px',bottom:'0px',right:'0px','z-index':1038}),x.then(function(n){C=n,b.init()}),b.init=function(){k.show&&y.$$postDigest(function(){b.show()})},b.destroy=function(){v(),A&&(A.remove(),A=null),y.$destroy()},b.show=function(){if(!b.$isShown){var n,e;if(angular.isElement(k.container)?(n=k.container,e=k.container[0].lastChild?angular.element(k.container[0].lastChild):null):k.container?(n=u(k.container),e=n[0]&&n[0].lastChild?angular.element(n[0].lastChild):null):(n=null,e=k.element),S&&v(),E=b.$scope.$new(),S=b.$element=C.link(E,function(n,e){}),!y.$emit(k.prefixEvent+'.show.before',b).defaultPrevented){S.css({display:'block'}).addClass(k.placement),k.animation&&(k.backdrop&&A.addClass(k.backdropAnimation),S.addClass(k.animation)),k.backdrop&&i.enter(A,p,null),angular.version.minor<=2?i.enter(S,n,e,a):i.enter(S,n,e).then(a),b.$isShown=y.$isShown=!0,c(y);var t=S[0];f(function(){t.focus()}),p.addClass(k.prefixClass+'-open'),k.animation&&p.addClass(k.prefixClass+'-with-'+k.animation),s(),$()}}},b.hide=function(){b.$isShown&&(y.$emit(k.prefixEvent+'.hide.before',b).defaultPrevented||(angular.version.minor<=2?i.leave(S,l):i.leave(S).then(l),k.backdrop&&i.leave(A),b.$isShown=y.$isShown=!1,c(y),m(),h()))},b.toggle=function(){b.$isShown?b.hide():b.show()},b.focus=function(){S[0].focus()},b.$onKeyUp=function(n){27===n.which&&b.$isShown&&(b.hide(),n.stopPropagation())},b}function c(n){n.$$phase||n.$root&&n.$root.$$phase||n.$digest()}function u(n,e){return angular.element((e||document).querySelectorAll(n))}var d=angular.forEach,f=(String.prototype.trim,e.requestAnimationFrame||e.setTimeout),p=angular.element(e.document.body);return s}]}).directive('bsModal',['$window','$sce','$modal',function(n,e,t){return{restrict:'EAC',scope:!0,link:function(n,o,i,a){var r={scope:n,element:o,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation','backdropAnimation','id','prefixEvent','prefixClass'],function(n){angular.isDefined(i[n])&&(r[n]=i[n])});var l=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(n){angular.isDefined(i[n])&&l.test(i[n])&&(r[n]=!1)}),angular.forEach(['title','content'],function(t){i[t]&&i.$observe(t,function(o,i){n[t]=e.trustAsHtml(o)})}),i.bsModal&&n.$watch(i.bsModal,function(e,t){angular.isObject(e)?angular.extend(n,e):n.content=e},!0);var s=t(r);o.on(i.trigger||'click',s.toggle),n.$on('$destroy',function(){s&&s.destroy(),r=null,s=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.modal',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$modal',function(){var n=this.defaults={animation:'am-fade',backdropAnimation:'am-fade',customClass:'',prefixClass:'modal',prefixEvent:'modal',placement:'top',templateUrl:'modal/modal.tpl.html',template:'',contentTemplate:!1,container:!1,element:null,backdrop:!0,keyboard:!0,html:!1,show:!0,size:null,zIndex:null};this.$get=['$window','$rootScope','$bsCompiler','$animate','$timeout','$sce','dimensions',function(e,o,t,a,i,l,s){function r(e){function i(){D.$emit(y.prefixEvent+'.show',S),angular.isDefined(y.onShow)&&angular.isFunction(y.onShow)&&y.onShow(S)}function s(){D.$emit(y.prefixEvent+'.hide',S),angular.isDefined(y.onHide)&&angular.isFunction(y.onHide)&&y.onHide(S),d('.modal').length<=0&&m.removeClass(y.prefixClass+'-open'),y.animation&&m.removeClass(y.prefixClass+'-with-'+y.animation)}function r(){y.backdrop&&(A.on('click',C),B.on('click',C),B.on('wheel',k))}function w(){y.backdrop&&(A.off('click',C),B.off('click',C),B.off('wheel',k))}function v(){y.keyboard&&A.on('keyup',S.$onKeyUp)}function b(){y.keyboard&&A.off('keyup',S.$onKeyUp)}function C(n){n.target===n.currentTarget&&('static'===y.backdrop?S.focus():S.hide())}function k(n){n.preventDefault()}function x(){S.$isShown&&null!==A&&(w(),b()),H&&(H.$destroy(),H=null),A&&(A.remove(),A=S.$element=null)}var S={},y=S.$options=angular.extend({},n,e),E=S.$promise=t.compile(y),D=S.$scope=y.scope&&y.scope.$new()||o.$new();y.element||y.container||(y.container='body'),y.zIndex&&(h=parseInt(y.zIndex,10),$=h-10),S.$id=y.id||y.element&&y.element.attr('id')||'',u(['title','content'],function(n){y[n]&&(D[n]=l.trustAsHtml(y[n]))}),D.$hide=function(){D.$$postDigest(function(){S.hide()})},D.$show=function(){D.$$postDigest(function(){S.show()})},D.$toggle=function(){D.$$postDigest(function(){S.toggle()})},S.$isShown=D.$isShown=!1;var z,A,H,B=angular.element('
');return B.css({position:'fixed',top:'0px',left:'0px',bottom:'0px',right:'0px'}),E.then(function(n){z=n,S.init()}),S.init=function(){y.show&&D.$$postDigest(function(){S.show()})},S.destroy=function(){x(),B&&(B.remove(),B=null),D.$destroy()},S.show=function(){if(!S.$isShown){var n,e;if(angular.isElement(y.container)?(n=y.container,e=y.container[0].lastChild?angular.element(y.container[0].lastChild):null):y.container?(n=d(y.container),e=n[0]&&n[0].lastChild?angular.element(n[0].lastChild):null):(n=null,e=y.element),A&&x(),H=S.$scope.$new(),A=S.$element=z.link(H,function(n,e){}),y.backdrop&&(A.css({'z-index':h+20*p}),B.css({'z-index':$+20*p}),p++),!D.$emit(y.prefixEvent+'.show.before',S).defaultPrevented){angular.isDefined(y.onBeforeShow)&&angular.isFunction(y.onBeforeShow)&&y.onBeforeShow(S),A.css({display:'block'}).addClass(y.placement),y.customClass&&A.addClass(y.customClass),y.size&&g[y.size]&&angular.element(d('.modal-dialog',A[0])).addClass(g[y.size]),y.animation&&(y.backdrop&&B.addClass(y.backdropAnimation),A.addClass(y.animation)),y.backdrop&&a.enter(B,m,null),angular.version.minor<=2?a.enter(A,n,e,i):a.enter(A,n,e).then(i),S.$isShown=D.$isShown=!0,c(D);var o=A[0];f(function(){o.focus()}),m.addClass(y.prefixClass+'-open'),y.animation&&m.addClass(y.prefixClass+'-with-'+y.animation),r(),v()}}},S.hide=function(){S.$isShown&&(D.$emit(y.prefixEvent+'.hide.before',S).defaultPrevented||(angular.isDefined(y.onBeforeHide)&&angular.isFunction(y.onBeforeHide)&&y.onBeforeHide(S),angular.version.minor<=2?a.leave(A,s):a.leave(A).then(s),y.backdrop&&(p--,a.leave(B)),S.$isShown=D.$isShown=!1,c(D),w(),b()))},S.toggle=function(){S.$isShown?S.hide():S.show()},S.focus=function(){A[0].focus()},S.$onKeyUp=function(n){27===n.which&&S.$isShown&&(S.hide(),n.stopPropagation())},S}function c(n){n.$$phase||n.$root&&n.$root.$$phase||n.$digest()}function d(n,e){return angular.element((e||document).querySelectorAll(n))}var u=angular.forEach,f=e.requestAnimationFrame||e.setTimeout,m=angular.element(e.document.body),p=0,h=1050,$=1040,g={lg:'modal-lg',sm:'modal-sm'};return r}]}).directive('bsModal',['$window','$sce','$parse','$modal',function(n,e,o,t){return{restrict:'EAC',scope:!0,link:function(n,o,a,i){var l={scope:n,element:o,show:!1};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','backdrop','keyboard','html','container','animation','backdropAnimation','id','prefixEvent','prefixClass','customClass','modalClass','size','zIndex'],function(n){angular.isDefined(a[n])&&(l[n]=a[n])}),l.modalClass&&(l.customClass=l.modalClass);var s=/^(false|0|)$/i;angular.forEach(['backdrop','keyboard','html','container'],function(n){angular.isDefined(a[n])&&s.test(a[n])&&(l[n]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(e){var o='bs'+e.charAt(0).toUpperCase()+e.slice(1);angular.isDefined(a[o])&&(l[e]=n.$eval(a[o]))}),angular.forEach(['title','content'],function(o){a[o]&&a.$observe(o,function(t,a){n[o]=e.trustAsHtml(t)})}),a.bsModal&&n.$watch(a.bsModal,function(e,o){angular.isObject(e)?angular.extend(n,e):n.content=e},!0);var r=t(l);o.on(a.trigger||'click',r.toggle),n.$on('$destroy',function(){r&&r.destroy(),l=null,r=null})}}}]); //# sourceMappingURL=modal.min.js.map diff --git a/dist/modules/modal.min.js.map b/dist/modules/modal.min.js.map index b5407cf1e..e293c5d50 100644 --- a/dist/modules/modal.min.js.map +++ b/dist/modules/modal.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/modal.js"],"names":["angular","templateUrl","provider","template","this","defaults","contentTemplate","container","element","backdrop","keyboard","html","show","$get","forEach","trim","requestAnimationFrame","bodyElement","options","$options","extend","config","ModalFactory","enterAnimateCallback","scope","version","minor","$modal","toggle","$isShown","removeClass","prefixClass","focus","modalElement","unbindBackdropEvents","off","hideOnBackdropClick","backdropElement","preventEventDefault","bindKeyboardEvents","on","$onKeyUp","evt","destroyModalElement","modalScope","$destroy","$element","key","promise","trustAsHtml","$scope","$new","$rootScope","$hide","$$postDigest","$id","id","attr","$show","$sce","hide","css","position","bottom","right","z-index","then","compileData","init","data","destroy","after","isElement","parent","$emit","display","placement","animation","link","clonedElement","prefixEvent","defaultPrevented","addClass","$animate","enter","backdropAnimation","el","safeDigest","bindBackdropEvents","leave","leaveAnimateCallback","unbindKeyboardEvents","which","directive","$root","$$phase","$window","restrict","query","document","querySelectorAll","setTimeout","body","isDefined","falseValueRegExp","$observe","newValue","oldValue","bsModal","$watch","content","modal","trigger","$on"],"mappings":"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","file":"modules/modal.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.modal', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$modal', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n backdropAnimation: 'am-fade',\n prefixClass: 'modal',\n prefixEvent: 'modal',\n placement: 'top',\n templateUrl: 'modal/modal.tpl.html',\n template: '',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true\n };\n\n this.$get = function($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) {\n\n var forEach = angular.forEach;\n var trim = String.prototype.trim;\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n var bodyElement = angular.element($window.document.body);\n\n function ModalFactory(config) {\n\n var $modal = {};\n\n // Common vars\n var options = $modal.$options = angular.extend({}, defaults, config);\n var promise = $modal.$promise = $bsCompiler.compile(options);\n var scope = $modal.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n if(!options.element && !options.container) {\n options.container = 'body';\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $modal.$id = options.id || options.element && options.element.attr('id') || '';\n\n // Support scope as string options\n forEach(['title', 'content'], function(key) {\n if(options[key]) scope[key] = $sce.trustAsHtml(options[key]);\n });\n\n // Provide scope helpers\n scope.$hide = function() {\n scope.$$postDigest(function() {\n $modal.hide();\n });\n };\n scope.$show = function() {\n scope.$$postDigest(function() {\n $modal.show();\n });\n };\n scope.$toggle = function() {\n scope.$$postDigest(function() {\n $modal.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $modal.$isShown = scope.$isShown = false;\n\n // Fetch, compile then initialize modal\n var compileData, modalElement, modalScope;\n var backdropElement = angular.element('
');\n backdropElement.css({position:'fixed', top:'0px', left:'0px', bottom:'0px', right:'0px', 'z-index': 1038});\n promise.then(function(data) {\n compileData = data;\n $modal.init();\n });\n\n $modal.init = function() {\n\n // Options: show\n if(options.show) {\n scope.$$postDigest(function() {\n $modal.show();\n });\n }\n\n };\n\n $modal.destroy = function() {\n\n // Remove element\n destroyModalElement();\n\n // remove backdrop element\n if(backdropElement) {\n backdropElement.remove();\n backdropElement = null;\n }\n\n // Destroy scope\n scope.$destroy();\n };\n\n $modal.show = function() {\n if($modal.$isShown) return;\n\n var parent, after;\n if(angular.isElement(options.container)) {\n parent = options.container;\n after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null;\n } else {\n if (options.container) {\n parent = findElement(options.container);\n after = parent[0] && parent[0].lastChild ? angular.element(parent[0].lastChild) : null;\n } else {\n parent = null;\n after = options.element;\n }\n }\n\n // destroy any existing modal elements\n if(modalElement) destroyModalElement();\n\n // create a new scope, so we can destroy it and all child scopes\n // when destroying the modal element\n modalScope = $modal.$scope.$new();\n // Fetch a cloned element linked from template (noop callback is required)\n modalElement = $modal.$element = compileData.link(modalScope, function(clonedElement, scope) {});\n\n if(scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) {\n return;\n }\n\n // Set the initial positioning.\n modalElement.css({display: 'block'}).addClass(options.placement);\n\n // Options: animation\n if(options.animation) {\n if(options.backdrop) {\n backdropElement.addClass(options.backdropAnimation);\n }\n modalElement.addClass(options.animation);\n }\n\n if(options.backdrop) {\n $animate.enter(backdropElement, bodyElement, null);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.enter(modalElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(modalElement, parent, after).then(enterAnimateCallback);\n }\n\n $modal.$isShown = scope.$isShown = true;\n safeDigest(scope);\n // Focus once the enter-animation has started\n // Weird PhantomJS bug hack\n var el = modalElement[0];\n requestAnimationFrame(function() {\n el.focus();\n });\n\n bodyElement.addClass(options.prefixClass + '-open');\n if(options.animation) {\n bodyElement.addClass(options.prefixClass + '-with-' + options.animation);\n }\n\n // Bind events\n bindBackdropEvents();\n bindKeyboardEvents();\n };\n\n function enterAnimateCallback() {\n scope.$emit(options.prefixEvent + '.show', $modal);\n }\n\n $modal.hide = function() {\n if(!$modal.$isShown) return;\n\n if(scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) {\n return;\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.leave(modalElement, leaveAnimateCallback);\n } else {\n $animate.leave(modalElement).then(leaveAnimateCallback);\n }\n\n if(options.backdrop) {\n $animate.leave(backdropElement);\n }\n $modal.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n };\n\n function leaveAnimateCallback() {\n scope.$emit(options.prefixEvent + '.hide', $modal);\n bodyElement.removeClass(options.prefixClass + '-open');\n if(options.animation) {\n bodyElement.removeClass(options.prefixClass + '-with-' + options.animation);\n }\n }\n\n $modal.toggle = function() {\n\n $modal.$isShown ? $modal.hide() : $modal.show();\n\n };\n\n $modal.focus = function() {\n modalElement[0].focus();\n };\n\n // Protected methods\n\n $modal.$onKeyUp = function(evt) {\n\n if (evt.which === 27 && $modal.$isShown) {\n $modal.hide();\n evt.stopPropagation();\n }\n\n };\n\n function bindBackdropEvents() {\n if(options.backdrop) {\n modalElement.on('click', hideOnBackdropClick);\n backdropElement.on('click', hideOnBackdropClick);\n backdropElement.on('wheel', preventEventDefault);\n }\n }\n\n function unbindBackdropEvents() {\n if(options.backdrop) {\n modalElement.off('click', hideOnBackdropClick);\n backdropElement.off('click', hideOnBackdropClick);\n backdropElement.off('wheel', preventEventDefault);\n }\n }\n\n function bindKeyboardEvents() {\n if(options.keyboard) {\n modalElement.on('keyup', $modal.$onKeyUp);\n }\n }\n\n function unbindKeyboardEvents() {\n if(options.keyboard) {\n modalElement.off('keyup', $modal.$onKeyUp);\n }\n }\n\n // Private helpers\n\n function hideOnBackdropClick(evt) {\n if(evt.target !== evt.currentTarget) return;\n options.backdrop === 'static' ? $modal.focus() : $modal.hide();\n }\n\n function preventEventDefault(evt) {\n evt.preventDefault();\n }\n\n function destroyModalElement() {\n if($modal.$isShown && modalElement !== null) {\n // un-bind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n }\n\n if(modalScope) {\n modalScope.$destroy();\n modalScope = null;\n }\n\n if(modalElement) {\n modalElement.remove();\n modalElement = $modal.$element = null;\n }\n }\n\n return $modal;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return ModalFactory;\n\n };\n\n })\n\n .directive('bsModal', function($window, $sce, $modal) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n });\n\n // Support scope as an object\n attr.bsModal && scope.$watch(attr.bsModal, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n\n // Initialize modal\n var modal = $modal(options);\n\n // Trigger\n element.on(attr.trigger || 'click', modal.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (modal) modal.destroy();\n options = null;\n modal = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/modal.js"],"names":["angular","placement","provider","templateUrl","this","defaults","template","contentTemplate","container","element","backdrop","keyboard","html","show","size","zIndex","$get","forEach","requestAnimationFrame","bodyElement","backdropCount","dialogBaseZindex","validSizes","ModalFactory","config","$modal","scope","$emit","options","prefixEvent","isDefined","onShow","isFunction","leaveAnimateCallback","animation","removeClass","prefixClass","onHide","length","$isShown","unbindBackdropEvents","modalElement","off","hideOnBackdropClick","backdropElement","preventEventDefault","bindKeyboardEvents","on","$onKeyUp","focus","evt","preventDefault","modalScope","$destroy","$element","parseInt","backdropBaseZindex","compile","$scope","$new","$rootScope","id","$id","attr","key","$show","$sce","trustAsHtml","$$postDigest","hide","toggle","left","bottom","right","promise","css","compileData","init","data","destroy","destroyModalElement","parent","after","isElement","z-index","link","clonedElement","onBeforeShow","display","defaultPrevented","addClass","customClass","findElement","$animate","enter","minor","backdropAnimation","el","enterAnimateCallback","then","safeDigest","bindBackdropEvents","version","leave","onBeforeHide","unbindKeyboardEvents","which","directive","$root","$$phase","$digest","restrict","query","document","querySelectorAll","lg","sm","$window","setTimeout","body","$options","falseValueRegExp","test","bsKey","modalClass","$observe","charAt","newValue","oldValue","$eval","$watch","isObject","extend","bsModal","modal","trigger","$on"],"mappings":"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","file":"modal.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.modal', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$modal', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n backdropAnimation: 'am-fade',\n customClass: '',\n prefixClass: 'modal',\n prefixEvent: 'modal',\n placement: 'top',\n templateUrl: 'modal/modal.tpl.html',\n template: '',\n contentTemplate: false,\n container: false,\n element: null,\n backdrop: true,\n keyboard: true,\n html: false,\n show: true,\n size: null,\n zIndex: null\n };\n\n this.$get = function ($window, $rootScope, $bsCompiler, $animate, $timeout, $sce, dimensions) {\n\n var forEach = angular.forEach;\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n var bodyElement = angular.element($window.document.body);\n\n var backdropCount = 0;\n var dialogBaseZindex = 1050;\n var backdropBaseZindex = 1040;\n\n var validSizes = {\n lg: 'modal-lg',\n sm: 'modal-sm'\n };\n\n function ModalFactory (config) {\n\n var $modal = {};\n\n // Common vars\n var options = $modal.$options = angular.extend({}, defaults, config);\n var promise = $modal.$promise = $bsCompiler.compile(options);\n var scope = $modal.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n if (!options.element && !options.container) {\n options.container = 'body';\n }\n\n if (options.zIndex) {\n dialogBaseZindex = parseInt(options.zIndex, 10);\n backdropBaseZindex = dialogBaseZindex - 10;\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $modal.$id = options.id || options.element && options.element.attr('id') || '';\n\n // Support scope as string options\n forEach(['title', 'content'], function (key) {\n if (options[key]) scope[key] = $sce.trustAsHtml(options[key]);\n });\n\n // Provide scope helpers\n scope.$hide = function () {\n scope.$$postDigest(function () {\n $modal.hide();\n });\n };\n scope.$show = function () {\n scope.$$postDigest(function () {\n $modal.show();\n });\n };\n scope.$toggle = function () {\n scope.$$postDigest(function () {\n $modal.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $modal.$isShown = scope.$isShown = false;\n\n // Fetch, compile then initialize modal\n var compileData;\n var modalElement;\n var modalScope;\n var backdropElement = angular.element('
');\n backdropElement.css({position: 'fixed', top: '0px', left: '0px', bottom: '0px', right: '0px'});\n promise.then(function (data) {\n compileData = data;\n $modal.init();\n });\n\n $modal.init = function () {\n\n // Options: show\n if (options.show) {\n scope.$$postDigest(function () {\n $modal.show();\n });\n }\n\n };\n\n $modal.destroy = function () {\n\n // Remove element\n destroyModalElement();\n\n // remove backdrop element\n if (backdropElement) {\n backdropElement.remove();\n backdropElement = null;\n }\n\n // Destroy scope\n scope.$destroy();\n };\n\n $modal.show = function () {\n if ($modal.$isShown) return;\n\n var parent;\n var after;\n if (angular.isElement(options.container)) {\n parent = options.container;\n after = options.container[0].lastChild ? angular.element(options.container[0].lastChild) : null;\n } else {\n if (options.container) {\n parent = findElement(options.container);\n after = parent[0] && parent[0].lastChild ? angular.element(parent[0].lastChild) : null;\n } else {\n parent = null;\n after = options.element;\n }\n }\n\n // destroy any existing modal elements\n if (modalElement) destroyModalElement();\n\n // create a new scope, so we can destroy it and all child scopes\n // when destroying the modal element\n modalScope = $modal.$scope.$new();\n // Fetch a cloned element linked from template (noop callback is required)\n modalElement = $modal.$element = compileData.link(modalScope, function (clonedElement, scope) {});\n\n if (options.backdrop) {\n // set z-index\n modalElement.css({'z-index': dialogBaseZindex + (backdropCount * 20)});\n backdropElement.css({'z-index': backdropBaseZindex + (backdropCount * 20)});\n\n // increment number of backdrops\n backdropCount++;\n }\n\n if (scope.$emit(options.prefixEvent + '.show.before', $modal).defaultPrevented) {\n return;\n }\n if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) {\n options.onBeforeShow($modal);\n }\n\n // Set the initial positioning.\n modalElement.css({display: 'block'}).addClass(options.placement);\n\n // Options: customClass\n if (options.customClass) {\n modalElement.addClass(options.customClass);\n }\n\n // Options: size\n if (options.size && validSizes[options.size]) {\n angular.element(findElement('.modal-dialog', modalElement[0])).addClass(validSizes[options.size]);\n }\n\n // Options: animation\n if (options.animation) {\n if (options.backdrop) {\n backdropElement.addClass(options.backdropAnimation);\n }\n modalElement.addClass(options.animation);\n }\n\n if (options.backdrop) {\n $animate.enter(backdropElement, bodyElement, null);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.enter(modalElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(modalElement, parent, after).then(enterAnimateCallback);\n }\n\n $modal.$isShown = scope.$isShown = true;\n safeDigest(scope);\n // Focus once the enter-animation has started\n // Weird PhantomJS bug hack\n var el = modalElement[0];\n requestAnimationFrame(function () {\n el.focus();\n });\n\n bodyElement.addClass(options.prefixClass + '-open');\n if (options.animation) {\n bodyElement.addClass(options.prefixClass + '-with-' + options.animation);\n }\n\n // Bind events\n bindBackdropEvents();\n bindKeyboardEvents();\n };\n\n function enterAnimateCallback () {\n scope.$emit(options.prefixEvent + '.show', $modal);\n if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) {\n options.onShow($modal);\n }\n }\n\n $modal.hide = function () {\n if (!$modal.$isShown) return;\n\n if (scope.$emit(options.prefixEvent + '.hide.before', $modal).defaultPrevented) {\n return;\n }\n if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) {\n options.onBeforeHide($modal);\n }\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.leave(modalElement, leaveAnimateCallback);\n } else {\n $animate.leave(modalElement).then(leaveAnimateCallback);\n }\n\n if (options.backdrop) {\n // decrement number of backdrops\n backdropCount--;\n $animate.leave(backdropElement);\n }\n $modal.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n };\n\n function leaveAnimateCallback () {\n scope.$emit(options.prefixEvent + '.hide', $modal);\n if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) {\n options.onHide($modal);\n }\n if (findElement('.modal').length <= 0) {\n bodyElement.removeClass(options.prefixClass + '-open');\n }\n if (options.animation) {\n bodyElement.removeClass(options.prefixClass + '-with-' + options.animation);\n }\n }\n\n $modal.toggle = function () {\n if ($modal.$isShown) {\n $modal.hide();\n } else {\n $modal.show();\n }\n };\n\n $modal.focus = function () {\n modalElement[0].focus();\n };\n\n // Protected methods\n\n $modal.$onKeyUp = function (evt) {\n\n if (evt.which === 27 && $modal.$isShown) {\n $modal.hide();\n evt.stopPropagation();\n }\n\n };\n\n function bindBackdropEvents () {\n if (options.backdrop) {\n modalElement.on('click', hideOnBackdropClick);\n backdropElement.on('click', hideOnBackdropClick);\n backdropElement.on('wheel', preventEventDefault);\n }\n }\n\n function unbindBackdropEvents () {\n if (options.backdrop) {\n modalElement.off('click', hideOnBackdropClick);\n backdropElement.off('click', hideOnBackdropClick);\n backdropElement.off('wheel', preventEventDefault);\n }\n }\n\n function bindKeyboardEvents () {\n if (options.keyboard) {\n modalElement.on('keyup', $modal.$onKeyUp);\n }\n }\n\n function unbindKeyboardEvents () {\n if (options.keyboard) {\n modalElement.off('keyup', $modal.$onKeyUp);\n }\n }\n\n // Private helpers\n\n function hideOnBackdropClick (evt) {\n if (evt.target !== evt.currentTarget) return;\n if (options.backdrop === 'static') {\n $modal.focus();\n } else {\n $modal.hide();\n }\n }\n\n function preventEventDefault (evt) {\n evt.preventDefault();\n }\n\n function destroyModalElement () {\n if ($modal.$isShown && modalElement !== null) {\n // un-bind events\n unbindBackdropEvents();\n unbindKeyboardEvents();\n }\n\n if (modalScope) {\n modalScope.$destroy();\n modalScope = null;\n }\n\n if (modalElement) {\n modalElement.remove();\n modalElement = $modal.$element = null;\n }\n }\n\n return $modal;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return ModalFactory;\n\n };\n\n })\n\n .directive('bsModal', function ($window, $sce, $parse, $modal) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope, element: element, show: false};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'backdrop', 'keyboard', 'html', 'container', 'animation', 'backdropAnimation', 'id', 'prefixEvent', 'prefixClass', 'customClass', 'modalClass', 'size', 'zIndex'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Options: alias modalClass to customClass\n if (options.modalClass) {\n options.customClass = options.modalClass;\n }\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['backdrop', 'keyboard', 'html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsModal) {\n scope.$watch(attr.bsModal, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n }, true);\n }\n\n // Initialize modal\n var modal = $modal(options);\n\n // Trigger\n element.on(attr.trigger || 'click', modal.toggle);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (modal) modal.destroy();\n options = null;\n modal = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/modal.tpl.js b/dist/modules/modal.tpl.js index e123a7620..2efacf967 100644 --- a/dist/modules/modal.tpl.js +++ b/dist/modules/modal.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/modal.tpl.min.js b/dist/modules/modal.tpl.min.js index 79e5d38fa..7e3c82807 100644 --- a/dist/modules/modal.tpl.min.js +++ b/dist/modules/modal.tpl.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/navbar.js b/dist/modules/navbar.js index b5c2543a2..dc849a692 100644 --- a/dist/modules/navbar.js +++ b/dist/modules/navbar.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/navbar.min.js b/dist/modules/navbar.min.js index 76fb644fc..b7ba9a6c5 100644 --- a/dist/modules/navbar.min.js +++ b/dist/modules/navbar.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/navbar.min.js.map b/dist/modules/navbar.min.js.map index 12f5fad65..f7dd7daa0 100644 --- a/dist/modules/navbar.min.js.map +++ b/dist/modules/navbar.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/navbar.js"],"names":["angular","module","provider","defaults","activeClass","routeAttr","directive","this","$get","restrict","link","isDefined","key","$navbar","oldValue","options","liElements","element","forEach","li","liElement","scope","$watch","pattern","path","newValue","RegExp","regexp","test","addClass","attr","replace","removeClass"],"mappings":"AAOA,YAEAA,SAIMC,OAAA,4BAAAC,SAAA,UAAA,WAHJ,GAGYC,GAAUA,KAAAA,UAFpBC,YAAa,SACbC,UAAW,mBAMZC,QAAAA,EAICC,MAAAC,KAAA,WACEC,OACAC,SAAMP,MAJTG,UASUN,YAAQW,UAAeC,YAAeA,UAAYA,SAAAA,EAAAA,EAAAA,GAR7D,GAAIT,GAAWU,EAAQV,QACvB,QACEM,SAYM,IAXNC,KAaO,SAAmBI,EAAAA,EAAAA,EAAAA,GAZxB,GAcIC,GAAIC,QAAaC,KAAQd,EAb7BH,SAeIA,QAAQkB,OAAQF,KAAAA,GAAqBG,SAAAA,GAEnCnB,QAAIoB,UAAYpB,EAAQiB,MAAQE,EAAAA,GAAAA,EAAAA,MAdtCE,EAgBMC,OAAGP,WAfP,MAgBMQ,GAAUC,QAff,SAASC,EAAUX,GACpB,GAgBIE,GAAaC,EAAIS,GAAOH,iBAAS,MAAAR,EAAAV,UAAA,IAfrCL,SAiBO2B,QAAOC,EAAKH,SAAWN,GAhB5B,GAiBIC,GAAUS,QAASd,QAAQX,GAhB3BmB,EAiBKH,EAAAU,KAAAf,EAAAV,WAAA0B,QAAA,IAAA,MACLX,GAAUY,SAhBZT,EAAU,IAAMA,EAAU,IAE5B,IAAII,GAAS,GAAID,QAAOH,EAAS,IAC7BI,GAAOC,KAAKH,GACdL,EAAUS,SAASd,EAAQX,aAE3BgB,EAAUY,YAAYjB,EAAQX","file":"modules/navbar.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.navbar', [])\n\n .provider('$navbar', function() {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n routeAttr: 'data-match-route',\n strict: false\n };\n\n this.$get = function() {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsNavbar', function($window, $location, $navbar) {\n\n var defaults = $navbar.defaults;\n\n return {\n restrict: 'A',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = angular.copy(defaults);\n angular.forEach(Object.keys(defaults), function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Watch for the $location\n scope.$watch(function() {\n\n return $location.path();\n\n }, function(newValue, oldValue) {\n\n var liElements = element[0].querySelectorAll('li[' + options.routeAttr + ']');\n\n angular.forEach(liElements, function(li) {\n\n var liElement = angular.element(li);\n var pattern = liElement.attr(options.routeAttr).replace('/', '\\\\/');\n if(options.strict) {\n pattern = '^' + pattern + '$';\n }\n var regexp = new RegExp(pattern, 'i');\n\n if(regexp.test(newValue)) {\n liElement.addClass(options.activeClass);\n } else {\n liElement.removeClass(options.activeClass);\n }\n\n });\n\n });\n\n }\n\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/navbar.js"],"names":["angular","module","provider","defaults","activeClass","routeAttr","directive","this","$get","restrict","link","isDefined","key","$navbar","oldValue","options","liElements","element","forEach","li","liElement","scope","$watch","pattern","path","newValue","RegExp","regexp","test","addClass","attr","replace","removeClass"],"mappings":"AAOA,YAEAA,SAIMC,OAAA,4BAAAC,SAAA,UAAA,WAHJ,GAGYC,GAAUA,KAAAA,UAFpBC,YAAa,SACbC,UAAW,mBAMZC,QAAAA,EAICC,MAAAC,KAAA,WACEC,OACAC,SAAMP,MAJTG,UASWN,YAAQW,UAAeC,YAAeA,UAAYA,SAAAA,EAAAA,EAAAA,GAR9D,GAAIT,GAAWU,EAAQV,QACvB,QACEM,SAYM,IAXNC,KAaO,SAAoBI,EAAAA,EAAAA,EAAAA,GAZzB,GAcIC,GAAIC,QAAaC,KAAQd,EAb7BH,SAeIA,QAAQkB,OAAQF,KAAAA,GAAsBG,SAAAA,GAEpCnB,QAAIoB,UAAYpB,EAAQiB,MAAQE,EAAAA,GAAAA,EAAAA,MAdtCE,EAgBMC,OAAIP,WAfR,MAgBMQ,GAAUC,QAff,SAASC,EAAUX,GACpB,GAgBIE,GAAaC,EAAIS,GAAOH,iBAAS,MAAAR,EAAAV,UAAA,IAfrCL,SAiBQ2B,QAAOC,EAAKH,SAAWN,GAhB7B,GAiBIC,GAAUS,QAASd,QAAQX,GAhB3BmB,EAiBKH,EAAAU,KAAAf,EAAAV,WAAA0B,QAAA,IAAA,MACLX,GAAUY,SAhBZT,EAAU,IAAMA,EAAU,IAE5B,IAAII,GAAS,GAAID,QAAOH,EAAS,IAC7BI,GAAOC,KAAKH,GACdL,EAAUS,SAASd,EAAQX,aAE3BgB,EAAUY,YAAYjB,EAAQX","file":"navbar.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.navbar', [])\n\n .provider('$navbar', function () {\n\n var defaults = this.defaults = {\n activeClass: 'active',\n routeAttr: 'data-match-route',\n strict: false\n };\n\n this.$get = function () {\n return {defaults: defaults};\n };\n\n })\n\n .directive('bsNavbar', function ($window, $location, $navbar) {\n\n var defaults = $navbar.defaults;\n\n return {\n restrict: 'A',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = angular.copy(defaults);\n angular.forEach(Object.keys(defaults), function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // Watch for the $location\n scope.$watch(function () {\n\n return $location.path();\n\n }, function (newValue, oldValue) {\n\n var liElements = element[0].querySelectorAll('li[' + options.routeAttr + ']');\n\n angular.forEach(liElements, function (li) {\n\n var liElement = angular.element(li);\n var pattern = liElement.attr(options.routeAttr).replace('/', '\\\\/');\n if (options.strict) {\n pattern = '^' + pattern + '$';\n }\n var regexp = new RegExp(pattern, 'i');\n\n if (regexp.test(newValue)) {\n liElement.addClass(options.activeClass);\n } else {\n liElement.removeClass(options.activeClass);\n }\n\n });\n\n });\n\n }\n\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/parse-options.js b/dist/modules/parse-options.js index 907568c3b..4109e81d0 100644 --- a/dist/modules/parse-options.js +++ b/dist/modules/parse-options.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -16,11 +16,20 @@ angular.module('mgcrea.ngStrap.helpers.parseOptions', []).provider('$parseOption var $parseOptions = {}; var options = angular.extend({}, defaults, config); $parseOptions.$values = []; - var match, displayFn, valueName, keyName, groupByFn, valueFn, valuesFn; + var match; + var displayFn; + var valueName; + var keyName; + var groupByFn; + var valueFn; + var valuesFn; $parseOptions.init = function() { $parseOptions.$match = match = attr.match(options.regexp); - displayFn = $parse(match[2] || match[1]), valueName = match[4] || match[6], keyName = match[5], - groupByFn = $parse(match[3] || ''), valueFn = $parse(match[2] ? match[1] : valueName), + displayFn = $parse(match[2] || match[1]); + valueName = match[4] || match[6]; + keyName = match[5]; + groupByFn = $parse(match[3] || ''); + valueFn = $parse(match[2] ? match[1] : valueName); valuesFn = $parse(match[7]); }; $parseOptions.valuesFn = function(scope, controller) { @@ -39,7 +48,9 @@ angular.module('mgcrea.ngStrap.helpers.parseOptions', []).provider('$parseOption }; function parseValues(values, scope) { return values.map(function(match, index) { - var locals = {}, label, value; + var locals = {}; + var label; + var value; locals[valueName] = match; label = displayFn(scope, locals); value = valueFn(scope, locals); diff --git a/dist/modules/parse-options.min.js b/dist/modules/parse-options.min.js index b3ad138bc..2c875d165 100644 --- a/dist/modules/parse-options.min.js +++ b/dist/modules/parse-options.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/parse-options.min.js.map b/dist/modules/parse-options.min.js.map index 8b21e9d06..a32f0f2fe 100644 --- a/dist/modules/parse-options.min.js.map +++ b/dist/modules/parse-options.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/parse-options.js"],"names":["angular","module","options","config","$parseOptions","$values","regexp","match","this","$get","$match","attr","displayFn","label","value","index","locals","valueName","scope","valueFn","ParseOptionsFactory","groupByFn","valuesFn","$parse","defaults","keyName","controller","when","values","$q","then","displayValue","init"],"mappings":"AAOA,YAEAA,SAQQC,OAAIC,0CAAuCC,SAAAA,gBAAAA,WAPjD,GAQMC,GAAAA,KAAcC,UAPlBC,OAUQC,+KARVC,MAAKC,MAWGL,SAAcM,KAASH,SAAQI,EAAKJ,GAV1C,QAWMK,GAAyBD,EAAMJ,GAcnC,QAmBcM,GAAOA,EAAAA,GAlBnB,MAkB0BC,GAAOA,IAAAA,SAAAA,EAAAA,GAjB/B,GAiB6CC,GAAAA,EAAPA,IAIxCX,OApBEY,GAAOC,GAAaV,EACpBM,EAAQD,EAAUM,EAAOF,GACzBF,EAAQK,EAAQD,EAAOF,IAmBzBH,MAAOT,EAhBHU,MAAOA,EAoBNM,MAAAA,KApDP,GAaIC,MAEAC,EAAWC,QAAOhB,UAAMiB,EAAArB,EAb5BC,GAAcC,UAgBZD,IAAAA,GAAAA,EAAckB,EAAWG,EAAgBC,EAAAA,EAAAA,CAoB3C,OAlCAtB,GAecuB,KAAKL,WAdjBlB,EAgBQJ,OAAgB4B,EAAAA,EAASrB,MAAAL,EAAAI,QAfjCM,EAgBMgB,EAAAA,EAAAA,IAAAA,EAAAA,IAAAA,EAAAA,EAAAA,IAAAA,EAAAA,GAAAA,EAAAA,EAAAA,GAfNP,EAAYE,EAAOhB,EAAM,IAAM,IAAKY,EAAUI,EAAOhB,EAAM,GAAKA,EAAM,GAAKU,GAC3EK,EAgBIlB,EAAcC,EAAAA,KAdpBD,EAAckB,SAAW,SAASJ,EAAOQ,GACvC,MAAOG,GAAGF,KAAKL,EAASJ,EAAOQ,IAAaI,KAAK,SAASF,GAKxD,MAaFxB,SAAc2B,QAAAA,KACRb,MAEJd,EAAOQ,QAAUM,EAAAA,OAAAA,EAAAA,EAAAA,MAhBVd,EAAcC,WAGzBD,EAoBUY,aAAoBF,SAAAA,GAnB5B,GAoBIE,KAlBJ,OADAE,GAoBIL,GAAQD,EACRE,EAAQK,IALdf,EAAc4B,OACP5B,EAET,MAAOgB","file":"modules/parse-options.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.parseOptions', [])\n\n .provider('$parseOptions', function() {\n\n var defaults = this.defaults = {\n regexp: /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n };\n\n this.$get = function($parse, $q) {\n\n function ParseOptionsFactory(attr, config) {\n\n var $parseOptions = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n $parseOptions.$values = [];\n\n // Private vars\n var match, displayFn, valueName, keyName, groupByFn, valueFn, valuesFn;\n\n $parseOptions.init = function() {\n $parseOptions.$match = match = attr.match(options.regexp);\n displayFn = $parse(match[2] || match[1]),\n valueName = match[4] || match[6],\n keyName = match[5],\n groupByFn = $parse(match[3] || ''),\n valueFn = $parse(match[2] ? match[1] : valueName),\n valuesFn = $parse(match[7]);\n };\n\n $parseOptions.valuesFn = function(scope, controller) {\n return $q.when(valuesFn(scope, controller))\n .then(function(values) {\n if(!angular.isArray(values)) {\n values = [];\n }\n $parseOptions.$values = values.length ? parseValues(values, scope) : [];\n return $parseOptions.$values;\n });\n };\n\n $parseOptions.displayValue = function(modelValue) {\n var scope = {};\n scope[valueName] = modelValue;\n return displayFn(scope);\n };\n\n // Private functions\n\n function parseValues(values, scope) {\n return values.map(function(match, index) {\n var locals = {}, label, value;\n locals[valueName] = match;\n label = displayFn(scope, locals);\n value = valueFn(scope, locals);\n return {label: label, value: value, index: index};\n });\n }\n\n $parseOptions.init();\n return $parseOptions;\n\n }\n\n return ParseOptionsFactory;\n\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/parse-options.js"],"names":["angular","module","options","config","$parseOptions","$values","regexp","match","this","$get","valueName","keyName","value","scope","locals","values","map","index","label","displayFn","valueFn","ParseOptionsFactory","groupByFn","valuesFn","$parse","$match","attr","when","isArray","controller","$q","then","displayValue","init"],"mappings":"AAOA,YAEAA,SAQQC,OAAIC,0CAAuCC,SAAAA,gBAAAA,WAPjD,GAQMC,GAAAA,KAAcC,UAPlBC,OAUQC,+KARVC,MAUMC,MAAIC,SAAAA,KAAAA,SAAAA,EAAAA,GATR,QAWQC,GAAAA,EAAAA,GAuBN,QAmBMC,GAAgBC,EAAOC,GAlB3B,MAmBIC,GAAAC,IAAA,SAAAT,EAAAU,GAlBF,GAkBwBL,GAAcK,EAA5BC,IAIZd,OAnBEU,GAAOJ,GAAaH,EACpBW,EAAQC,EAAUN,EAAOC,GACzBF,EAAQQ,EAAQP,EAAOC,IAkBzBI,MAAOd,EAfHQ,MAAOA,EAmBNS,MAAAA,KAnDL,GAAIC,MAEAF,EAAAA,QAAAA,UAAAA,EAAAA,EACJhB,GAAImB,UAEJnB,IAAAA,GACEA,EACAe,EACAT,EACAC,EACAW,EACAF,CA2BJ,OAvCAhB,GAaeoB,KAAOjB,WAZpBH,EAAcqB,OAASlB,EAAQmB,EAAKnB,MAAML,EAAQI,QAelDF,EAAcmB,EAAAA,EAAW,IAAAhB,EAAUM,IACjCH,EAAUiB,EAAKJ,IAASV,EAAAA,GAb1BF,EAeSX,EAAQ4B,GAdjBN,EAeMP,EAAAA,EAAAA,IAAAA,IAdNK,EAAUI,EAAOjB,EAAM,GAAKA,EAAM,GAAKG,GACvCa,EAeInB,EAAcC,EAAAA,KAbpBD,EAAcmB,SAAW,SAASV,EAAOgB,GACvC,MAAOC,GAAGH,KAAKJ,EAASV,EAAOgB,IAAaE,KAAK,SAAShB,GAKxD,MAYFX,SAAc4B,QAAAA,KACRnB,MAEJT,EAAOe,QAAUN,EAAAA,OAAAA,EAAAA,EAAAA,MAfVT,EAAcC,WAGzBD,EAmBUU,aAAAA,SAAAA,GAlBR,GAmBID,KAjBJ,OADAA,GAmBQD,GAAAA,EACJE,EAAOJ,IAFbN,EAAc6B,OACP7B,EAET,MAAOiB","file":"parse-options.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.helpers.parseOptions', [])\n\n .provider('$parseOptions', function () {\n\n var defaults = this.defaults = {\n regexp: /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n };\n\n this.$get = function ($parse, $q) {\n\n function ParseOptionsFactory (attr, config) {\n\n var $parseOptions = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n $parseOptions.$values = [];\n\n // Private vars\n var match;\n var displayFn;\n var valueName;\n /* eslint-disable no-unused-vars */\n var keyName;\n var groupByFn;\n /* eslint-enable no-unused-vars */\n var valueFn;\n var valuesFn;\n\n $parseOptions.init = function () {\n $parseOptions.$match = match = attr.match(options.regexp);\n displayFn = $parse(match[2] || match[1]);\n valueName = match[4] || match[6];\n keyName = match[5];\n groupByFn = $parse(match[3] || '');\n valueFn = $parse(match[2] ? match[1] : valueName);\n valuesFn = $parse(match[7]);\n };\n\n $parseOptions.valuesFn = function (scope, controller) {\n return $q.when(valuesFn(scope, controller))\n .then(function (values) {\n if (!angular.isArray(values)) {\n values = [];\n }\n $parseOptions.$values = values.length ? parseValues(values, scope) : [];\n return $parseOptions.$values;\n });\n };\n\n $parseOptions.displayValue = function (modelValue) {\n var scope = {};\n scope[valueName] = modelValue;\n return displayFn(scope);\n };\n\n // Private functions\n\n function parseValues (values, scope) {\n return values.map(function (match, index) {\n var locals = {};\n var label;\n var value;\n locals[valueName] = match;\n label = displayFn(scope, locals);\n value = valueFn(scope, locals);\n return {label: label, value: value, index: index};\n });\n }\n\n $parseOptions.init();\n return $parseOptions;\n\n }\n\n return ParseOptionsFactory;\n\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/popover.js b/dist/modules/popover.js index 96c71d25b..61f315549 100644 --- a/dist/modules/popover.js +++ b/dist/modules/popover.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -41,48 +41,86 @@ angular.module('mgcrea.ngStrap.popover', [ 'mgcrea.ngStrap.tooltip' ]).provider( restrict: 'EAC', scope: true, link: function postLink(scope, element, attr) { + var popover; var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent', 'bsEnabled' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'autoClose' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); var dataTarget = element.attr('data-target'); if (angular.isDefined(dataTarget)) { - if (falseValueRegExp.test(dataTarget)) options.target = false; else options.target = dataTarget; + if (falseValueRegExp.test(dataTarget)) { + options.target = false; + } else { + options.target = dataTarget; + } } angular.forEach([ 'title', 'content' ], function(key) { - attr[key] && attr.$observe(key, function(newValue, oldValue) { - scope[key] = $sce.trustAsHtml(newValue); - angular.isDefined(oldValue) && requestAnimationFrame(function() { - popover && popover.$applyPlacement(); + if (attr[key]) { + attr.$observe(key, function(newValue, oldValue) { + scope[key] = $sce.trustAsHtml(newValue); + if (angular.isDefined(oldValue)) { + requestAnimationFrame(function() { + if (popover) popover.$applyPlacement(); + }); + } }); - }); - }); - attr.bsPopover && scope.$watch(attr.bsPopover, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.content = newValue; } - angular.isDefined(oldValue) && requestAnimationFrame(function() { - popover && popover.$applyPlacement(); - }); - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!popover || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i); - newValue === true ? popover.show() : popover.hide(); }); - attr.viewport && scope.$watch(attr.viewport, function(newValue) { - if (!popover || !angular.isDefined(newValue)) return; - popover.setViewport(newValue); - }); - var popover = $popover(element, options); + if (attr.bsPopover) { + scope.$watch(attr.bsPopover, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.content = newValue; + } + if (angular.isDefined(oldValue)) { + requestAnimationFrame(function() { + if (popover) popover.$applyPlacement(); + }); + } + }, true); + } + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!popover || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i); + if (newValue === true) { + popover.show(); + } else { + popover.hide(); + } + }); + } + if (attr.bsEnabled) { + scope.$watch(attr.bsEnabled, function(newValue) { + if (!popover || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(popover),?/i); + if (newValue === false) { + popover.setEnabled(false); + } else { + popover.setEnabled(true); + } + }); + } + if (attr.viewport) { + scope.$watch(attr.viewport, function(newValue) { + if (!popover || !angular.isDefined(newValue)) return; + popover.setViewport(newValue); + }); + } + popover = $popover(element, options); scope.$on('$destroy', function() { if (popover) popover.destroy(); options = null; diff --git a/dist/modules/popover.min.js b/dist/modules/popover.min.js index a6663608f..5d383ceb9 100644 --- a/dist/modules/popover.min.js +++ b/dist/modules/popover.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.popover',['mgcrea.ngStrap.tooltip']).provider('$popover',function(){var t=this.defaults={animation:'am-fade',customClass:'',container:!1,target:!1,placement:'right',templateUrl:'popover/popover.tpl.html',contentTemplate:!1,trigger:'click',keyboard:!0,html:!1,title:'',content:'',delay:0,autoClose:!1};this.$get=['$tooltip',function(e){function n(n,a){var o=angular.extend({},t,a),r=e(n,o);return o.content&&(r.$scope.content=o.content),r}return n}]}).directive('bsPopover',['$window','$sce','$popover',function(t,e,n){var a=t.requestAnimationFrame||t.setTimeout;return{restrict:'EAC',scope:!0,link:function(t,o,r){var i={scope:t};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','container','delay','trigger','html','animation','customClass','autoClose','id','prefixClass','prefixEvent'],function(t){angular.isDefined(r[t])&&(i[t]=r[t])});var l=/^(false|0|)$/i;angular.forEach(['html','container','autoClose'],function(t){angular.isDefined(r[t])&&l.test(r[t])&&(i[t]=!1)});var c=o.attr('data-target');angular.isDefined(c)&&(l.test(c)?i.target=!1:i.target=c),angular.forEach(['title','content'],function(n){r[n]&&r.$observe(n,function(o,r){t[n]=e.trustAsHtml(o),angular.isDefined(r)&&a(function(){s&&s.$applyPlacement()})})}),r.bsPopover&&t.$watch(r.bsPopover,function(e,n){angular.isObject(e)?angular.extend(t,e):t.content=e,angular.isDefined(n)&&a(function(){s&&s.$applyPlacement()})},!0),r.bsShow&&t.$watch(r.bsShow,function(t,e){s&&angular.isDefined(t)&&(angular.isString(t)&&(t=!!t.match(/true|,?(popover),?/i)),t===!0?s.show():s.hide())}),r.viewport&&t.$watch(r.viewport,function(t){s&&angular.isDefined(t)&&s.setViewport(t)});var s=n(o,i);t.$on('$destroy',function(){s&&s.destroy(),i=null,s=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.popover',['mgcrea.ngStrap.tooltip']).provider('$popover',function(){var e=this.defaults={animation:'am-fade',customClass:'',container:!1,target:!1,placement:'right',templateUrl:'popover/popover.tpl.html',contentTemplate:!1,trigger:'click',keyboard:!0,html:!1,title:'',content:'',delay:0,autoClose:!1};this.$get=['$tooltip',function(t){function n(n,a){var o=angular.extend({},e,a),r=t(n,o);return o.content&&(r.$scope.content=o.content),r}return n}]}).directive('bsPopover',['$window','$sce','$popover',function(e,t,n){var a=e.requestAnimationFrame||e.setTimeout;return{restrict:'EAC',scope:!0,link:function(e,o,r){var i,l={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','contentTemplate','placement','container','delay','trigger','html','animation','customClass','autoClose','id','prefixClass','prefixEvent','bsEnabled'],function(e){angular.isDefined(r[e])&&(l[e]=r[e])});var c=/^(false|0|)$/i;angular.forEach(['html','container','autoClose'],function(e){angular.isDefined(r[e])&&c.test(r[e])&&(l[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var n='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(r[n])&&(l[t]=e.$eval(r[n]))});var s=o.attr('data-target');angular.isDefined(s)&&(c.test(s)?l.target=!1:l.target=s),angular.forEach(['title','content'],function(n){r[n]&&r.$observe(n,function(o,r){e[n]=t.trustAsHtml(o),angular.isDefined(r)&&a(function(){i&&i.$applyPlacement()})})}),r.bsPopover&&e.$watch(r.bsPopover,function(t,n){angular.isObject(t)?angular.extend(e,t):e.content=t,angular.isDefined(n)&&a(function(){i&&i.$applyPlacement()})},!0),r.bsShow&&e.$watch(r.bsShow,function(e,t){i&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(popover),?/i)),e===!0?i.show():i.hide())}),r.bsEnabled&&e.$watch(r.bsEnabled,function(e){i&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|1|,?(popover),?/i)),e===!1?i.setEnabled(!1):i.setEnabled(!0))}),r.viewport&&e.$watch(r.viewport,function(e){i&&angular.isDefined(e)&&i.setViewport(e)}),i=n(o,l),e.$on('$destroy',function(){i&&i.destroy(),l=null,i=null})}}}]); //# sourceMappingURL=popover.min.js.map diff --git a/dist/modules/popover.min.js.map b/dist/modules/popover.min.js.map index 12c2e90f9..ab4a23158 100644 --- a/dist/modules/popover.min.js.map +++ b/dist/modules/popover.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/popover.js"],"names":["angular","templateUrl","provider","contentTemplate","defaults","trigger","keyboard","html","title","content","delay","autoClose","this","$get","options","$popover","PopoverFactory","element","config","$tooltip","restrict","directive","requestAnimationFrame","$window","link","scope","setTimeout","falseValueRegExp","attr","dataTarget","key","isDefined","forEach","$observe","oldValue","popover","test","target","bsPopover","$watch","isObject","newValue","extend","$applyPlacement","show","hide","setViewport","viewport","$on","destroy"],"mappings":"AAOA,YAEAA,SAMMC,OAAAA,0BAAa,2BAAAC,SAAA,WAAA,WALjB,GAMIC,GAAAA,KAAiBC,UACjBC,UAAS,UACTC,YAAU,GACVC,WAAM,EACNC,QAAO,EACPC,UAAS,QACTC,YAAO,2BACPC,iBAAW,EALbN,QAAS,QAQTO,UAAKC,EAEHN,MAAA,EAPFC,MAUQM,GATRL,QAWQM,GAVRL,MAaOI,EAZPH,WAaMI,EAXRH,MAcMC,MAAOE,WAAAA,SAAAA,GAbX,QAASC,GAAeC,EAASC,GAiB/B,GAAAJ,GAAOE,QAAAA,UAAAA,EAAAA,GAfHD,EAAWI,EAASF,EAASH,EA0BjCM,OALHC,GAAUZ,UAELa,EAAAA,OAAAA,QAAwBC,EAAQD,SAGlCF,EAEAI,MAAMR,OApBTK,UAuBoBI,aAAOA,UAAAA,OAAAA,WAAAA,SAAAA,EAAAA,EAAAA,GAtB5B,GAAIH,GAAwBC,EAAQD,uBAAyBC,EAAQG,UACrE,QACEN,SAsBSpB,MArBTyB,OAAO,EACPD,KAwBQG,SAAAA,EAAmBV,EAAAW,GACvB5B,GAAAA,IACEyB,MAAGzB,EAKLA,SAAI6B,SAAaZ,WAAa,cAAA,aAAA,eAAA,kBAAA,YAAA,YAAA,QAAA,UAAA,OAAA,YAAA,cAAA,YAAA,KAAA,cAAA,eAAA,SAAAa,GAC3B9B,QAAQ+B,UAAUF,EAAAA,MAAaf,EAAAgB,GAAAF,EAAAE,KAzBpC,IAAIH,GAAmB,eAiCrB3B,SAAQgC,SAAS,OAAA,YAAS,aAAqBF,SAAAA,GAC7CF,QAAKE,UAAaG,EAASH,KAAKH,EAAmBO,KAAAA,EAAAA,MAAAA,EAAAA,IAAAA,IA9BvD,IAgCMlC,GAAQ+B,EAAUG,KAAAA,cA/BpBlC,SAgCImC,UAAWA,KA/BbR,EAAiBS,KAAKP,GAAaf,EAAQuB,QAAS,EAAYvB,EAAQuB,OAASR,GAEvF7B,QAAQgC,SAAU,QAAS,WAAa,SAASF,GAmC/CF,EAAKU,IAAAA,EAAab,SAAMc,EAAOX,SAAKU,EAAWJ,GAC7CT,EAAGzB,GAAQwC,EAAAA,YAAoBC,GAC7BzC,QAAQ0C,UAAOjB,IAAOgB,EAAAA,WAjCtBN,GAkCKA,EAAAQ,wBA9BXf,EAkCMO,WAAWA,EAAQQ,OAAAA,EAAAA,UAAAA,SAAAA,EAAAA,GAjCnB3C,QAAQwC,SAASC,GAmClBzC,QAAA0C,OAAAjB,EAAAgB,GAIDhB,EAAIU,QAAYnC,EAEhByC,QAAAA,UAAaP,IAAeU,EAAiBC,WAnC7CV,GAAWA,EAAQQ,sBAwCnB,GArCJf,EAsCIO,QAAQW,EAAAA,OAAYL,EAAAA,OAAAA,SAAAA,EAAAA,GArCjBN,GAAYnC,QAAQ+B,UAAUU,KAyC/BN,QAAUpB,SAASE,KAASH,IAAAA,EAAAA,MAAAA,wBAGhCW,KAAU,EAAYU,EAAAS,OAAAT,EAAAU,UAxCxBjB,EA0CId,UAAUW,EAAAc,OAAAX,EAAAmB,SAAA,SAAAN,GACVN,GAAUnC,QAAA+B,UAAAU,IAzCZN,EAAQW,YAAYL,IAEtB,IAAIN,GAAUpB,EAASE,EAASH,EAChCW,GAAMuB,IAAI,WAAY,WAChBb,GAASA,EAAQc,UACrBnC,EAAU,KACVqB,EAAU","file":"modules/popover.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.popover', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$popover', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n // uncommenting the next two lines will break backwards compatability\n // prefixClass: 'popover',\n // prefixEvent: 'popover',\n container: false,\n target: false,\n placement: 'right',\n templateUrl: 'popover/popover.tpl.html',\n contentTemplate: false,\n trigger: 'click',\n keyboard: true,\n html: false,\n title: '',\n content: '',\n delay: 0,\n autoClose: false\n };\n\n this.$get = function($tooltip) {\n\n function PopoverFactory(element, config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n var $popover = $tooltip(element, options);\n\n // Support scope as string options [/*title, */content]\n if(options.content) {\n $popover.$scope.content = options.content;\n }\n\n return $popover;\n\n }\n\n return PopoverFactory;\n\n };\n\n })\n\n .directive('bsPopover', function($window, $sce, $popover) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoClose'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if(angular.isDefined(dataTarget)) {\n if(falseValueRegExp.test(dataTarget))\n options.target = false;\n else\n options.target = dataTarget;\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function(key) {\n attr[key] && attr.$observe(key, function(newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n angular.isDefined(oldValue) && requestAnimationFrame(function() {\n popover && popover.$applyPlacement();\n });\n });\n });\n\n // Support scope as an object\n attr.bsPopover && scope.$watch(attr.bsPopover, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n angular.isDefined(oldValue) && requestAnimationFrame(function() {\n popover && popover.$applyPlacement();\n });\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if(!popover || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i);\n newValue === true ? popover.show() : popover.hide();\n });\n\n // Viewport support\n attr.viewport && scope.$watch(attr.viewport, function (newValue) {\n if(!popover || !angular.isDefined(newValue)) return;\n popover.setViewport(newValue);\n });\n\n // Initialize popover\n var popover = $popover(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (popover) popover.destroy();\n options = null;\n popover = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/popover.js"],"names":["angular","templateUrl","provider","contentTemplate","defaults","trigger","keyboard","html","title","content","delay","autoClose","this","$get","options","$popover","PopoverFactory","element","config","$tooltip","restrict","directive","requestAnimationFrame","$window","link","scope","forEach","attr","falseValueRegExp","key","bsKey","isDefined","charAt","toUpperCase","slice","test","dataTarget","target","$eval","$observe","newValue","oldValue","$sce","trustAsHtml","$watch","bsPopover","$applyPlacement","popover","extend","bsShow","isString","match","bsEnabled","viewport","setViewport","$on","destroy"],"mappings":"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","file":"popover.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.popover', ['mgcrea.ngStrap.tooltip'])\n\n .provider('$popover', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n // uncommenting the next two lines will break backwards compatability\n // prefixClass: 'popover',\n // prefixEvent: 'popover',\n container: false,\n target: false,\n placement: 'right',\n templateUrl: 'popover/popover.tpl.html',\n contentTemplate: false,\n trigger: 'click',\n keyboard: true,\n html: false,\n title: '',\n content: '',\n delay: 0,\n autoClose: false\n };\n\n this.$get = function ($tooltip) {\n\n function PopoverFactory (element, config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n var $popover = $tooltip(element, options);\n\n // Support scope as string options [/*title, */content]\n if (options.content) {\n $popover.$scope.content = options.content;\n }\n\n return $popover;\n\n }\n\n return PopoverFactory;\n\n };\n\n })\n\n .directive('bsPopover', function ($window, $sce, $popover) {\n\n var requestAnimationFrame = $window.requestAnimationFrame || $window.setTimeout;\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr) {\n\n var popover;\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'customClass', 'autoClose', 'id', 'prefixClass', 'prefixEvent', 'bsEnabled'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoClose'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if (angular.isDefined(dataTarget)) {\n if (falseValueRegExp.test(dataTarget)) {\n options.target = false;\n } else {\n options.target = dataTarget;\n }\n }\n\n // Support scope as data-attrs\n angular.forEach(['title', 'content'], function (key) {\n if (attr[key]) {\n attr.$observe(key, function (newValue, oldValue) {\n scope[key] = $sce.trustAsHtml(newValue);\n if (angular.isDefined(oldValue)) {\n requestAnimationFrame(function () {\n if (popover) popover.$applyPlacement();\n });\n }\n });\n }\n });\n\n // Support scope as an object\n if (attr.bsPopover) {\n scope.$watch(attr.bsPopover, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.content = newValue;\n }\n if (angular.isDefined(oldValue)) {\n requestAnimationFrame(function () {\n if (popover) popover.$applyPlacement();\n });\n }\n }, true);\n }\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(popover),?/i);\n if (newValue === true) {\n popover.show();\n } else {\n popover.hide();\n }\n });\n }\n\n // Enabled binding support\n if (attr.bsEnabled) {\n scope.$watch(attr.bsEnabled, function (newValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(popover),?/i);\n if (newValue === false) {\n popover.setEnabled(false);\n } else {\n popover.setEnabled(true);\n }\n });\n }\n\n // Viewport support\n if (attr.viewport) {\n scope.$watch(attr.viewport, function (newValue) {\n if (!popover || !angular.isDefined(newValue)) return;\n popover.setViewport(newValue);\n });\n }\n\n // Initialize popover\n popover = $popover(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (popover) popover.destroy();\n options = null;\n popover = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/popover.tpl.js b/dist/modules/popover.tpl.js index a348a4293..6d523d36c 100644 --- a/dist/modules/popover.tpl.js +++ b/dist/modules/popover.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/popover.tpl.min.js b/dist/modules/popover.tpl.min.js index 028618cfa..3fd402768 100644 --- a/dist/modules/popover.tpl.min.js +++ b/dist/modules/popover.tpl.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/raf.js b/dist/modules/raf.js index 6f6d9dd55..6f36e3273 100644 --- a/dist/modules/raf.js +++ b/dist/modules/raf.js @@ -1,27 +1,29 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ 'use strict'; -angular.version.minor < 3 && angular.version.dot < 14 && angular.module('ng').factory('$$rAF', [ '$window', '$timeout', function($window, $timeout) { - var requestAnimationFrame = $window.requestAnimationFrame || $window.webkitRequestAnimationFrame || $window.mozRequestAnimationFrame; - var cancelAnimationFrame = $window.cancelAnimationFrame || $window.webkitCancelAnimationFrame || $window.mozCancelAnimationFrame || $window.webkitCancelRequestAnimationFrame; - var rafSupported = !!requestAnimationFrame; - var raf = rafSupported ? function(fn) { - var id = requestAnimationFrame(fn); - return function() { - cancelAnimationFrame(id); +if (angular.version.minor < 3 && angular.version.dot < 14) { + angular.module('ng').factory('$$rAF', [ '$window', '$timeout', function($window, $timeout) { + var requestAnimationFrame = $window.requestAnimationFrame || $window.webkitRequestAnimationFrame || $window.mozRequestAnimationFrame; + var cancelAnimationFrame = $window.cancelAnimationFrame || $window.webkitCancelAnimationFrame || $window.mozCancelAnimationFrame || $window.webkitCancelRequestAnimationFrame; + var rafSupported = !!requestAnimationFrame; + var raf = rafSupported ? function(fn) { + var id = requestAnimationFrame(fn); + return function() { + cancelAnimationFrame(id); + }; + } : function(fn) { + var timer = $timeout(fn, 16.66, false); + return function() { + $timeout.cancel(timer); + }; }; - } : function(fn) { - var timer = $timeout(fn, 16.66, false); - return function() { - $timeout.cancel(timer); - }; - }; - raf.supported = rafSupported; - return raf; -} ]); \ No newline at end of file + raf.supported = rafSupported; + return raf; + } ]); +} \ No newline at end of file diff --git a/dist/modules/raf.min.js b/dist/modules/raf.min.js index 0b1de934d..dbfacae0a 100644 --- a/dist/modules/raf.min.js +++ b/dist/modules/raf.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/raf.min.js.map b/dist/modules/raf.min.js.map index c4de111e8..2bf4be2dd 100644 --- a/dist/modules/raf.min.js.map +++ b/dist/modules/raf.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/raf.js"],"names":["angular","cancelAnimationFrame","id","requestAnimationFrame","$window","webkitRequestAnimationFrame","mozRequestAnimationFrame","fn","rafSupported","$timeout","raf","timer","cancel","supported"],"mappings":"AAOA,YAEAA,SAWQC,QAAAA,MAAAA,GAAqBC,QAAAA,QAAAA,IAAAA,IAAAA,QAAAA,OAAAA,MAAAA,QAAAA,SAAAA,UAAAA,WAAAA,SAAAA,EAAAA,GAV3B,GAAIC,GAAwBC,EAAQD,uBAAyBC,EAAQC,6BAA+BD,EAAQE,yBAa1GL,EAASM,EAAAA,sBAAAA,EAAAA,4BAAAA,EAAAA,yBAAAA,EAAAA,kCACPC,IAAYC,EACZC,EAAOF,EAAA,SAAAD,GAXT,GAYIE,GAAAA,EAAgBE,EAXpB,OAAO,YACLV,EAAqBC,KAgBzB,SAAOQ,GAbL,GAAIC,GAAQF,EAASF,EAAI,OAAO,EAChC,OAAO,YACLE,EAASG,OAAOD,IAIpB,OADAD,GAAIG,UAAYL,EACTE","file":"modules/raf.min.js","sourcesContent":["'use strict';\n\n(angular.version.minor < 3 && angular.version.dot < 14) && angular.module('ng')\n\n.factory('$$rAF', function($window, $timeout) {\n\n var requestAnimationFrame = $window.requestAnimationFrame ||\n $window.webkitRequestAnimationFrame ||\n $window.mozRequestAnimationFrame;\n\n var cancelAnimationFrame = $window.cancelAnimationFrame ||\n $window.webkitCancelAnimationFrame ||\n $window.mozCancelAnimationFrame ||\n $window.webkitCancelRequestAnimationFrame;\n\n var rafSupported = !!requestAnimationFrame;\n var raf = rafSupported ?\n function(fn) {\n var id = requestAnimationFrame(fn);\n return function() {\n cancelAnimationFrame(id);\n };\n } :\n function(fn) {\n var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666\n return function() {\n $timeout.cancel(timer);\n };\n };\n\n raf.supported = rafSupported;\n\n return raf;\n\n});\n\n// .factory('$$animateReflow', function($$rAF, $document) {\n\n// var bodyEl = $document[0].body;\n\n// return function(fn) {\n// //the returned function acts as the cancellation function\n// return $$rAF(function() {\n// //the line below will force the browser to perform a repaint\n// //so that all the animated elements within the animation frame\n// //will be properly updated and drawn on screen. This is\n// //required to perform multi-class CSS based animations with\n// //Firefox. DO NOT REMOVE THIS LINE.\n// var a = bodyEl.offsetWidth + 1;\n// fn();\n// });\n// };\n\n// });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/raf.js"],"names":["angular","version","minor","dot","cancelAnimationFrame","id","requestAnimationFrame","$window","webkitRequestAnimationFrame","mozRequestAnimationFrame","fn","rafSupported","$timeout","raf","timer","cancel","supported"],"mappings":"AAOA,YAEIA,SAWIC,QAAOC,MAAA,GAAAF,QAAAC,QAAAE,IAAA,IAVbH,QAWQI,OAAAA,MAAAA,QAAqBC,SAAAA,UAAAA,WAAAA,SAAAA,EAAAA,GAV3B,GAAIC,GAAwBC,EAAQD,uBAAyBC,EAAQC,6BAA+BD,EAAQE,yBAa1GL,EAAUM,EAAAA,sBAAAA,EAAAA,4BAAAA,EAAAA,yBAAAA,EAAAA,kCACRC,IAAYC,EACZC,EAAOF,EAAA,SAAAD,GAXT,GAYIE,GAAAA,EAAgBE,EAXpB,OAAO,YACLV,EAAqBC,KAgBzB,SAAOQ,GAbL,GAAIC,GAAQF,EAASF,EAAI,OAAO,EAChC,OAAO,YACLE,EAASG,OAAOD,IAIpB,OADAD,GAAIG,UAAYL,EACTE","file":"raf.min.js","sourcesContent":["'use strict';\n\nif (angular.version.minor < 3 && angular.version.dot < 14) {\n angular.module('ng')\n\n .factory('$$rAF', function ($window, $timeout) {\n\n var requestAnimationFrame = $window.requestAnimationFrame ||\n $window.webkitRequestAnimationFrame ||\n $window.mozRequestAnimationFrame;\n\n var cancelAnimationFrame = $window.cancelAnimationFrame ||\n $window.webkitCancelAnimationFrame ||\n $window.mozCancelAnimationFrame ||\n $window.webkitCancelRequestAnimationFrame;\n\n var rafSupported = !!requestAnimationFrame;\n var raf = rafSupported ?\n function (fn) {\n var id = requestAnimationFrame(fn);\n return function () {\n cancelAnimationFrame(id);\n };\n } :\n function (fn) {\n var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666\n return function () {\n $timeout.cancel(timer);\n };\n };\n\n raf.supported = rafSupported;\n\n return raf;\n\n });\n}\n"]} \ No newline at end of file diff --git a/dist/modules/scrollspy.js b/dist/modules/scrollspy.js index ad47de1a5..b19bb8c90 100644 --- a/dist/modules/scrollspy.js +++ b/dist/modules/scrollspy.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -32,7 +32,8 @@ angular.module('mgcrea.ngStrap.scrollspy', [ 'mgcrea.ngStrap.helpers.debounce', return spies[scrollId]; } var $scrollspy = {}; - var unbindViewContentLoaded, unbindIncludeContentLoaded; + var unbindViewContentLoaded; + var unbindIncludeContentLoaded; var trackedElements = $scrollspy.$trackedElements = []; var sortedElements = []; var activeTarget; @@ -136,7 +137,7 @@ angular.module('mgcrea.ngStrap.scrollspy', [ 'mgcrea.ngStrap.helpers.debounce', break; } } - trackedElements = trackedElements.splice(toDelete, 1); + trackedElements.splice(toDelete, 1); }; $scrollspy.activate = function(i) { trackedElements[i].addClass('active'); diff --git a/dist/modules/scrollspy.min.js b/dist/modules/scrollspy.min.js index 8b0d77f95..5902cfad7 100644 --- a/dist/modules/scrollspy.min.js +++ b/dist/modules/scrollspy.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.scrollspy',['mgcrea.ngStrap.helpers.debounce','mgcrea.ngStrap.helpers.dimensions']).provider('$scrollspy',function(){var e=this.$$spies={},t=this.defaults={debounce:150,throttle:100,offset:100};this.$get=['$window','$document','$rootScope','dimensions','debounce','throttle',function(n,o,r,c,i,s){function a(e,t){return e[0].nodeName&&e[0].nodeName.toLowerCase()===t.toLowerCase()}function l(o){var l=angular.extend({},t,o);l.element||(l.element=p);var d=a(l.element,'body'),h=d?u:l.element,m=d?'window':l.id;if(e[m])return e[m].$$count++,e[m];var g,v,$,k,E,T,y,b,C={},S=C.$trackedElements=[],L=[];return C.init=function(){this.$$count=1,k=i(this.checkPosition,l.debounce),E=s(this.checkPosition,l.throttle),h.on('click',this.checkPositionWithEventLoop),u.on('resize',k),h.on('scroll',E),T=i(this.checkOffsets,l.debounce),g=r.$on('$viewContentLoaded',T),v=r.$on('$includeContentLoaded',T),T(),m&&(e[m]=C)},C.destroy=function(){this.$$count--,this.$$count>0||(h.off('click',this.checkPositionWithEventLoop),u.off('resize',k),h.off('scroll',E),g(),v(),m&&delete e[m])},C.checkPosition=function(){if(L.length){if(b=(d?n.pageYOffset:h.prop('scrollTop'))||0,y=Math.max(n.innerHeight,f.prop('clientHeight')),bL[e+1].offsetTop))return C.$activateElement(L[e])}},C.checkPositionWithEventLoop=function(){setTimeout(C.checkPosition,1)},C.$activateElement=function(e){if($){var t=C.$getTrackedElement($);t&&(t.source.removeClass('active'),a(t.source,'li')&&a(t.source.parent().parent(),'li')&&t.source.parent().parent().removeClass('active'))}$=e.target,e.source.addClass('active'),a(e.source,'li')&&a(e.source.parent().parent(),'li')&&e.source.parent().parent().addClass('active')},C.$getTrackedElement=function(e){return S.filter(function(t){return t.target===e})[0]},C.checkOffsets=function(){angular.forEach(S,function(e){var t=document.querySelector(e.target);e.offsetTop=t?c.offset(t).top:null,l.offset&&null!==e.offsetTop&&(e.offsetTop-=1*l.offset)}),L=S.filter(function(e){return null!==e.offsetTop}).sort(function(e,t){return e.offsetTop-t.offsetTop}),k()},C.trackElement=function(e,t){S.push({target:e,source:t})},C.untrackElement=function(e,t){for(var n,o=S.length;o--;)if(S[o].target===e&&S[o].source===t){n=o;break}S=S.splice(n,1)},C.activate=function(e){S[e].addClass('active')},C.init(),C}var u=angular.element(n),f=angular.element(o.prop('documentElement')),p=angular.element(n.document.body);return l}]}).directive('bsScrollspy',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,o){return{restrict:'EAC',link:function(e,t,n){var r={scope:e};angular.forEach(['offset','target'],function(e){angular.isDefined(n[e])&&(r[e]=n[e])});var c=o(r);c.trackElement(r.target,t),e.$on('$destroy',function(){c&&(c.untrackElement(r.target,t),c.destroy()),r=null,c=null})}}}]).directive('bsScrollspyList',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,o){return{restrict:'A',compile:function(e,t){var n=e[0].querySelectorAll('li > a[href]');angular.forEach(n,function(e){var t=angular.element(e);t.parent().attr('bs-scrollspy','').attr('data-target',t.attr('href'))})}}}]); +'use strict';angular.module('mgcrea.ngStrap.scrollspy',['mgcrea.ngStrap.helpers.debounce','mgcrea.ngStrap.helpers.dimensions']).provider('$scrollspy',function(){var e=this.$$spies={},t=this.defaults={debounce:150,throttle:100,offset:100};this.$get=['$window','$document','$rootScope','dimensions','debounce','throttle',function(n,o,r,c,i,s){function a(e,t){return e[0].nodeName&&e[0].nodeName.toLowerCase()===t.toLowerCase()}function l(o){var l=angular.extend({},t,o);l.element||(l.element=p);var d=a(l.element,'body'),h=d?u:l.element,m=d?'window':l.id;if(e[m])return e[m].$$count++,e[m];var g,v,$,k,E,T,y,b,C={},S=C.$trackedElements=[],L=[];return C.init=function(){this.$$count=1,k=i(this.checkPosition,l.debounce),E=s(this.checkPosition,l.throttle),h.on('click',this.checkPositionWithEventLoop),u.on('resize',k),h.on('scroll',E),T=i(this.checkOffsets,l.debounce),g=r.$on('$viewContentLoaded',T),v=r.$on('$includeContentLoaded',T),T(),m&&(e[m]=C)},C.destroy=function(){this.$$count--,this.$$count>0||(h.off('click',this.checkPositionWithEventLoop),u.off('resize',k),h.off('scroll',E),g(),v(),m&&delete e[m])},C.checkPosition=function(){if(L.length){if(b=(d?n.pageYOffset:h.prop('scrollTop'))||0,y=Math.max(n.innerHeight,f.prop('clientHeight')),bL[e+1].offsetTop))return C.$activateElement(L[e])}},C.checkPositionWithEventLoop=function(){setTimeout(C.checkPosition,1)},C.$activateElement=function(e){if($){var t=C.$getTrackedElement($);t&&(t.source.removeClass('active'),a(t.source,'li')&&a(t.source.parent().parent(),'li')&&t.source.parent().parent().removeClass('active'))}$=e.target,e.source.addClass('active'),a(e.source,'li')&&a(e.source.parent().parent(),'li')&&e.source.parent().parent().addClass('active')},C.$getTrackedElement=function(e){return S.filter(function(t){return t.target===e})[0]},C.checkOffsets=function(){angular.forEach(S,function(e){var t=document.querySelector(e.target);e.offsetTop=t?c.offset(t).top:null,l.offset&&null!==e.offsetTop&&(e.offsetTop-=1*l.offset)}),L=S.filter(function(e){return null!==e.offsetTop}).sort(function(e,t){return e.offsetTop-t.offsetTop}),k()},C.trackElement=function(e,t){S.push({target:e,source:t})},C.untrackElement=function(e,t){for(var n,o=S.length;o--;)if(S[o].target===e&&S[o].source===t){n=o;break}S.splice(n,1)},C.activate=function(e){S[e].addClass('active')},C.init(),C}var u=angular.element(n),f=angular.element(o.prop('documentElement')),p=angular.element(n.document.body);return l}]}).directive('bsScrollspy',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,o){return{restrict:'EAC',link:function(e,t,n){var r={scope:e};angular.forEach(['offset','target'],function(e){angular.isDefined(n[e])&&(r[e]=n[e])});var c=o(r);c.trackElement(r.target,t),e.$on('$destroy',function(){c&&(c.untrackElement(r.target,t),c.destroy()),r=null,c=null})}}}]).directive('bsScrollspyList',['$rootScope','debounce','dimensions','$scrollspy',function(e,t,n,o){return{restrict:'A',compile:function(e,t){var n=e[0].querySelectorAll('li > a[href]');angular.forEach(n,function(e){var t=angular.element(e);t.parent().attr('bs-scrollspy','').attr('data-target',t.attr('href'))})}}}]); //# sourceMappingURL=scrollspy.min.js.map diff --git a/dist/modules/scrollspy.min.js.map b/dist/modules/scrollspy.min.js.map index 29ed03ac8..54373273b 100644 --- a/dist/modules/scrollspy.min.js.map +++ b/dist/modules/scrollspy.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/scrollspy.js"],"names":["angular","$get","$document","$rootScope","dimensions","spies","windowEl","defaults","debounce","bodyEl","throttle","nodeName","offset","this","ScrollSpyFactory","config","scrollEl","isWindowSpy","scrollId","element","options","id","$$count","$scrollspy","unbindViewContentLoaded","unbindIncludeContentLoaded","trackedElements","$trackedElements","sortedElements","activeTarget","debouncedCheckPosition","viewportHeight","throttledCheckPosition","debouncedCheckOffsets","scrollTop","link","checkPositionWithEventLoop","on","checkOffsets","$on","off","checkPosition","i","isUndefined","prop","docEl","target","$activateElement","length","offsetTop","setTimeout","activeElement","source","$getTrackedElement","addClass","removeClass","parent","filter","targetElement","document","querySelector","trackedElement","el","forEach","b","top","trackElement","push","toDelete","untrackElement","splice","activate","directive","restrict","init","extend","body","scope","key","scrollspy","destroy","compile","childEl","attr","children","querySelectorAll","child"],"mappings":"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","file":"modules/scrollspy.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.scrollspy', ['mgcrea.ngStrap.helpers.debounce', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$scrollspy', function() {\n\n // Pool of registered spies\n var spies = this.$$spies = {};\n\n var defaults = this.defaults = {\n debounce: 150,\n throttle: 100,\n offset: 100\n };\n\n this.$get = function($window, $document, $rootScope, dimensions, debounce, throttle) {\n\n var windowEl = angular.element($window);\n var docEl = angular.element($document.prop('documentElement'));\n var bodyEl = angular.element($window.document.body);\n\n // Helper functions\n\n function nodeName(element, name) {\n return element[0].nodeName && element[0].nodeName.toLowerCase() === name.toLowerCase();\n }\n\n function ScrollSpyFactory(config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n if(!options.element) options.element = bodyEl;\n var isWindowSpy = nodeName(options.element, 'body');\n var scrollEl = isWindowSpy ? windowEl : options.element;\n var scrollId = isWindowSpy ? 'window' : options.id;\n\n // Use existing spy\n if(spies[scrollId]) {\n spies[scrollId].$$count++;\n return spies[scrollId];\n }\n\n var $scrollspy = {};\n\n // Private vars\n var unbindViewContentLoaded, unbindIncludeContentLoaded;\n var trackedElements = $scrollspy.$trackedElements = [];\n var sortedElements = [];\n var activeTarget;\n var debouncedCheckPosition;\n var throttledCheckPosition;\n var debouncedCheckOffsets;\n var viewportHeight;\n var scrollTop;\n\n $scrollspy.init = function() {\n\n // Setup internal ref counter\n this.$$count = 1;\n\n // Bind events\n debouncedCheckPosition = debounce(this.checkPosition, options.debounce);\n throttledCheckPosition = throttle(this.checkPosition, options.throttle);\n scrollEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', debouncedCheckPosition);\n scrollEl.on('scroll', throttledCheckPosition);\n\n debouncedCheckOffsets = debounce(this.checkOffsets, options.debounce);\n unbindViewContentLoaded = $rootScope.$on('$viewContentLoaded', debouncedCheckOffsets);\n unbindIncludeContentLoaded = $rootScope.$on('$includeContentLoaded', debouncedCheckOffsets);\n debouncedCheckOffsets();\n\n // Register spy for reuse\n if(scrollId) {\n spies[scrollId] = $scrollspy;\n }\n\n };\n\n $scrollspy.destroy = function() {\n\n // Check internal ref counter\n this.$$count--;\n if(this.$$count > 0) {\n return;\n }\n\n // Unbind events\n scrollEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', debouncedCheckPosition);\n scrollEl.off('scroll', throttledCheckPosition);\n unbindViewContentLoaded();\n unbindIncludeContentLoaded();\n if (scrollId) {\n delete spies[scrollId];\n }\n };\n\n $scrollspy.checkPosition = function() {\n\n // Not ready yet\n if(!sortedElements.length) return;\n\n // Calculate the scroll position\n scrollTop = (isWindowSpy ? $window.pageYOffset : scrollEl.prop('scrollTop')) || 0;\n\n // Calculate the viewport height for use by the components\n viewportHeight = Math.max($window.innerHeight, docEl.prop('clientHeight'));\n\n // Activate first element if scroll is smaller\n if(scrollTop < sortedElements[0].offsetTop && activeTarget !== sortedElements[0].target) {\n return $scrollspy.$activateElement(sortedElements[0]);\n }\n\n // Activate proper element\n for (var i = sortedElements.length; i--;) {\n if(angular.isUndefined(sortedElements[i].offsetTop) || sortedElements[i].offsetTop === null) continue;\n if(activeTarget === sortedElements[i].target) continue;\n if(scrollTop < sortedElements[i].offsetTop) continue;\n if(sortedElements[i + 1] && scrollTop > sortedElements[i + 1].offsetTop) continue;\n return $scrollspy.$activateElement(sortedElements[i]);\n }\n\n };\n\n $scrollspy.checkPositionWithEventLoop = function() {\n // IE 9 throws an error if we use 'this' instead of '$scrollspy'\n // in this setTimeout call\n setTimeout($scrollspy.checkPosition, 1);\n };\n\n // Protected methods\n\n $scrollspy.$activateElement = function(element) {\n if(activeTarget) {\n var activeElement = $scrollspy.$getTrackedElement(activeTarget);\n if(activeElement) {\n activeElement.source.removeClass('active');\n if(nodeName(activeElement.source, 'li') && nodeName(activeElement.source.parent().parent(), 'li')) {\n activeElement.source.parent().parent().removeClass('active');\n }\n }\n }\n activeTarget = element.target;\n element.source.addClass('active');\n if(nodeName(element.source, 'li') && nodeName(element.source.parent().parent(), 'li')) {\n element.source.parent().parent().addClass('active');\n }\n };\n\n $scrollspy.$getTrackedElement = function(target) {\n return trackedElements.filter(function(obj) {\n return obj.target === target;\n })[0];\n };\n\n // Track offsets behavior\n\n $scrollspy.checkOffsets = function() {\n\n angular.forEach(trackedElements, function(trackedElement) {\n var targetElement = document.querySelector(trackedElement.target);\n trackedElement.offsetTop = targetElement ? dimensions.offset(targetElement).top : null;\n if(options.offset && trackedElement.offsetTop !== null) trackedElement.offsetTop -= options.offset * 1;\n });\n\n sortedElements = trackedElements\n .filter(function(el) {\n return el.offsetTop !== null;\n })\n .sort(function(a, b) {\n return a.offsetTop - b.offsetTop;\n });\n\n debouncedCheckPosition();\n\n };\n\n $scrollspy.trackElement = function(target, source) {\n trackedElements.push({target: target, source: source});\n };\n\n $scrollspy.untrackElement = function(target, source) {\n var toDelete;\n for (var i = trackedElements.length; i--;) {\n if(trackedElements[i].target === target && trackedElements[i].source === source) {\n toDelete = i;\n break;\n }\n }\n trackedElements = trackedElements.splice(toDelete, 1);\n };\n\n $scrollspy.activate = function(i) {\n trackedElements[i].addClass('active');\n };\n\n // Initialize plugin\n\n $scrollspy.init();\n return $scrollspy;\n\n }\n\n return ScrollSpyFactory;\n\n };\n\n })\n\n .directive('bsScrollspy', function($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'EAC',\n link: function postLink(scope, element, attr) {\n\n var options = {scope: scope};\n angular.forEach(['offset', 'target'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n var scrollspy = $scrollspy(options);\n scrollspy.trackElement(options.target, element);\n\n scope.$on('$destroy', function() {\n if (scrollspy) {\n scrollspy.untrackElement(options.target, element);\n scrollspy.destroy();\n }\n options = null;\n scrollspy = null;\n });\n\n }\n };\n\n })\n\n\n .directive('bsScrollspyList', function($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'A',\n compile: function postLink(element, attr) {\n var children = element[0].querySelectorAll('li > a[href]');\n angular.forEach(children, function(child) {\n var childEl = angular.element(child);\n childEl.parent().attr('bs-scrollspy', '').attr('data-target', childEl.attr('href'));\n });\n }\n\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/scrollspy.js"],"names":["angular","$get","$document","$rootScope","dimensions","spies","windowEl","defaults","debounce","bodyEl","throttle","nodeName","offset","this","ScrollSpyFactory","config","scrollEl","isWindowSpy","scrollId","element","options","id","$$count","$scrollspy","unbindViewContentLoaded","unbindIncludeContentLoaded","trackedElements","sortedElements","activeTarget","debouncedCheckOffsets","viewportHeight","debouncedCheckPosition","throttledCheckPosition","scrollTop","link","checkPositionWithEventLoop","on","checkOffsets","$on","off","checkPosition","i","isUndefined","prop","docEl","target","$activateElement","length","offsetTop","setTimeout","activeElement","source","$getTrackedElement","addClass","removeClass","parent","filter","targetElement","document","querySelector","trackedElement","el","forEach","b","top","trackElement","push","toDelete","untrackElement","splice","activate","directive","restrict","init","extend","body","scope","key","scrollspy","destroy","compile","childEl","attr","children","querySelectorAll","child"],"mappings":"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","file":"scrollspy.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.scrollspy', ['mgcrea.ngStrap.helpers.debounce', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$scrollspy', function () {\n\n // Pool of registered spies\n var spies = this.$$spies = {};\n\n var defaults = this.defaults = {\n debounce: 150,\n throttle: 100,\n offset: 100\n };\n\n this.$get = function ($window, $document, $rootScope, dimensions, debounce, throttle) {\n\n var windowEl = angular.element($window);\n var docEl = angular.element($document.prop('documentElement'));\n var bodyEl = angular.element($window.document.body);\n\n // Helper functions\n\n function nodeName (element, name) {\n return element[0].nodeName && element[0].nodeName.toLowerCase() === name.toLowerCase();\n }\n\n function ScrollSpyFactory (config) {\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n if (!options.element) options.element = bodyEl;\n var isWindowSpy = nodeName(options.element, 'body');\n var scrollEl = isWindowSpy ? windowEl : options.element;\n var scrollId = isWindowSpy ? 'window' : options.id;\n\n // Use existing spy\n if (spies[scrollId]) {\n spies[scrollId].$$count++;\n return spies[scrollId];\n }\n\n var $scrollspy = {};\n\n // Private vars\n var unbindViewContentLoaded;\n var unbindIncludeContentLoaded;\n var trackedElements = $scrollspy.$trackedElements = [];\n var sortedElements = [];\n var activeTarget;\n var debouncedCheckPosition;\n var throttledCheckPosition;\n var debouncedCheckOffsets;\n /* eslint-disable no-unused-vars */\n var viewportHeight;\n /* eslint-enable no-unused-vars */\n var scrollTop;\n\n $scrollspy.init = function () {\n\n // Setup internal ref counter\n this.$$count = 1;\n\n // Bind events\n debouncedCheckPosition = debounce(this.checkPosition, options.debounce);\n throttledCheckPosition = throttle(this.checkPosition, options.throttle);\n scrollEl.on('click', this.checkPositionWithEventLoop);\n windowEl.on('resize', debouncedCheckPosition);\n scrollEl.on('scroll', throttledCheckPosition);\n\n debouncedCheckOffsets = debounce(this.checkOffsets, options.debounce);\n unbindViewContentLoaded = $rootScope.$on('$viewContentLoaded', debouncedCheckOffsets);\n unbindIncludeContentLoaded = $rootScope.$on('$includeContentLoaded', debouncedCheckOffsets);\n debouncedCheckOffsets();\n\n // Register spy for reuse\n if (scrollId) {\n spies[scrollId] = $scrollspy;\n }\n\n };\n\n $scrollspy.destroy = function () {\n\n // Check internal ref counter\n this.$$count--;\n if (this.$$count > 0) {\n return;\n }\n\n // Unbind events\n scrollEl.off('click', this.checkPositionWithEventLoop);\n windowEl.off('resize', debouncedCheckPosition);\n scrollEl.off('scroll', throttledCheckPosition);\n unbindViewContentLoaded();\n unbindIncludeContentLoaded();\n if (scrollId) {\n delete spies[scrollId];\n }\n };\n\n $scrollspy.checkPosition = function () {\n\n // Not ready yet\n if (!sortedElements.length) return;\n\n // Calculate the scroll position\n scrollTop = (isWindowSpy ? $window.pageYOffset : scrollEl.prop('scrollTop')) || 0;\n\n // Calculate the viewport height for use by the components\n viewportHeight = Math.max($window.innerHeight, docEl.prop('clientHeight'));\n\n // Activate first element if scroll is smaller\n if (scrollTop < sortedElements[0].offsetTop && activeTarget !== sortedElements[0].target) {\n return $scrollspy.$activateElement(sortedElements[0]);\n }\n\n // Activate proper element\n for (var i = sortedElements.length; i--;) {\n if (angular.isUndefined(sortedElements[i].offsetTop) || sortedElements[i].offsetTop === null) continue;\n if (activeTarget === sortedElements[i].target) continue;\n if (scrollTop < sortedElements[i].offsetTop) continue;\n if (sortedElements[i + 1] && scrollTop > sortedElements[i + 1].offsetTop) continue;\n return $scrollspy.$activateElement(sortedElements[i]);\n }\n\n };\n\n $scrollspy.checkPositionWithEventLoop = function () {\n // IE 9 throws an error if we use 'this' instead of '$scrollspy'\n // in this setTimeout call\n setTimeout($scrollspy.checkPosition, 1);\n };\n\n // Protected methods\n\n $scrollspy.$activateElement = function (element) {\n if (activeTarget) {\n var activeElement = $scrollspy.$getTrackedElement(activeTarget);\n if (activeElement) {\n activeElement.source.removeClass('active');\n if (nodeName(activeElement.source, 'li') && nodeName(activeElement.source.parent().parent(), 'li')) {\n activeElement.source.parent().parent().removeClass('active');\n }\n }\n }\n activeTarget = element.target;\n element.source.addClass('active');\n if (nodeName(element.source, 'li') && nodeName(element.source.parent().parent(), 'li')) {\n element.source.parent().parent().addClass('active');\n }\n };\n\n $scrollspy.$getTrackedElement = function (target) {\n return trackedElements.filter(function (obj) {\n return obj.target === target;\n })[0];\n };\n\n // Track offsets behavior\n\n $scrollspy.checkOffsets = function () {\n\n angular.forEach(trackedElements, function (trackedElement) {\n var targetElement = document.querySelector(trackedElement.target);\n trackedElement.offsetTop = targetElement ? dimensions.offset(targetElement).top : null;\n if (options.offset && trackedElement.offsetTop !== null) trackedElement.offsetTop -= options.offset * 1;\n });\n\n sortedElements = trackedElements\n .filter(function (el) {\n return el.offsetTop !== null;\n })\n .sort(function (a, b) {\n return a.offsetTop - b.offsetTop;\n });\n\n debouncedCheckPosition();\n\n };\n\n $scrollspy.trackElement = function (target, source) {\n trackedElements.push({target: target, source: source});\n };\n\n $scrollspy.untrackElement = function (target, source) {\n var toDelete;\n for (var i = trackedElements.length; i--;) {\n if (trackedElements[i].target === target && trackedElements[i].source === source) {\n toDelete = i;\n break;\n }\n }\n trackedElements.splice(toDelete, 1);\n };\n\n $scrollspy.activate = function (i) {\n trackedElements[i].addClass('active');\n };\n\n // Initialize plugin\n\n $scrollspy.init();\n return $scrollspy;\n\n }\n\n return ScrollSpyFactory;\n\n };\n\n })\n\n .directive('bsScrollspy', function ($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'EAC',\n link: function postLink (scope, element, attr) {\n\n var options = {scope: scope};\n angular.forEach(['offset', 'target'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n var scrollspy = $scrollspy(options);\n scrollspy.trackElement(options.target, element);\n\n scope.$on('$destroy', function () {\n if (scrollspy) {\n scrollspy.untrackElement(options.target, element);\n scrollspy.destroy();\n }\n options = null;\n scrollspy = null;\n });\n\n }\n };\n\n })\n\n\n .directive('bsScrollspyList', function ($rootScope, debounce, dimensions, $scrollspy) {\n\n return {\n restrict: 'A',\n compile: function postLink (element, attr) {\n var children = element[0].querySelectorAll('li > a[href]');\n angular.forEach(children, function (child) {\n var childEl = angular.element(child);\n childEl.parent().attr('bs-scrollspy', '').attr('data-target', childEl.attr('href'));\n });\n }\n\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/select.js b/dist/modules/select.js index b0c441436..d404eea60 100644 --- a/dist/modules/select.js +++ b/dist/modules/select.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -28,11 +28,11 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt noneText: 'None', maxLength: 3, maxLengthHtml: 'selected', - iconCheckmark: 'glyphicon glyphicon-ok' + iconCheckmark: 'glyphicon glyphicon-ok', + toggle: false }; this.$get = [ '$window', '$document', '$rootScope', '$tooltip', '$timeout', function($window, $document, $rootScope, $tooltip, $timeout) { - var bodyEl = angular.element($window.document.body); - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; function SelectFactory(element, controller, config) { var $select = {}; @@ -86,7 +86,11 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt }; $select.activate = function(index) { if (options.multiple) { - $select.$isActive(index) ? scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1) : scope.$activeIndex.push(index); + if ($select.$isActive(index)) { + scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1); + } else { + scope.$activeIndex.push(index); + } if (options.sort) scope.$activeIndex.sort(function(a, b) { return a - b; }); @@ -96,6 +100,9 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt return scope.$activeIndex; }; $select.select = function(index) { + if (angular.isUndefined(index) || index < 0 || index >= scope.$matches.length) { + return; + } var value = scope.$matches[index].value; scope.$apply(function() { $select.activate(index); @@ -107,11 +114,18 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt return scope.$matches[index].value; })); } else { - controller.$setViewValue(value); + if (options.toggle) { + controller.$setViewValue(value === controller.$modelValue ? undefined : value); + } else { + controller.$setViewValue(value); + } $select.hide(); } }); scope.$emit(options.prefixEvent + '.select', value, index, $select); + if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) { + options.onSelect(value, index, $select); + } }; $select.$updateActiveIndex = function() { if (options.multiple) { @@ -139,25 +153,36 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt $select.$isActive = function(index) { if (options.multiple) { return scope.$activeIndex.indexOf(index) !== -1; - } else { - return scope.$activeIndex === index; } + return scope.$activeIndex === index; }; $select.$getIndex = function(value) { - var l = scope.$matches.length, i = l; - if (!l) return; - for (i = l; i--; ) { - if (scope.$matches[i].value === value) break; + var index; + for (index = scope.$matches.length; index--; ) { + if (angular.equals(scope.$matches[index].value, value)) break; } - if (i < 0) return; - return i; + return index; }; $select.$onMouseDown = function(evt) { evt.preventDefault(); evt.stopPropagation(); if (isTouch) { var targetEl = angular.element(evt.target); - targetEl.triggerHandler('click'); + var anchor; + if (evt.target.nodeName !== 'A') { + var anchorCandidate = targetEl.parent(); + while (!anchor && anchorCandidate.length > 0) { + if (anchorCandidate[0].nodeName === 'A') { + anchor = anchorCandidate; + } + anchorCandidate = anchorCandidate.parent(); + } + } + if (anchor) { + angular.element(anchor).triggerHandler('click'); + } else { + targetEl.triggerHandler('click'); + } } }; $select.$onKeyDown = function(evt) { @@ -227,16 +252,28 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt scope: scope, placeholder: defaults.placeholder }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent', 'toggle' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'allNoneButtons', 'sort' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } + }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); var dataMultiple = element.attr('data-multiple'); if (angular.isDefined(dataMultiple)) { - if (falseValueRegExp.test(dataMultiple)) options.multiple = false; else options.multiple = dataMultiple; + if (falseValueRegExp.test(dataMultiple)) { + options.multiple = false; + } else { + options.multiple = dataMultiple; + } } if (element[0].nodeName.toLowerCase() === 'select') { var inputEl = element; @@ -250,22 +287,23 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt element[0].addEventListener('blur', select.$selectScrollFix); } var watchedOptions = parsedOptions.$match[7].replace(/\|.+/, '').trim(); - scope.$watchCollection(watchedOptions, function(newValue, oldValue) { + scope.$watch(watchedOptions, function(newValue, oldValue) { parsedOptions.valuesFn(scope, controller).then(function(values) { select.update(values); controller.$render(); }); - }); + }, true); scope.$watch(attr.ngModel, function(newValue, oldValue) { select.$updateActiveIndex(); controller.$render(); }, true); controller.$render = function() { - var selected, index; + var selected; + var index; if (options.multiple && angular.isArray(controller.$modelValue)) { selected = controller.$modelValue.map(function(value) { index = select.$getIndex(value); - return angular.isDefined(index) ? select.$scope.$matches[index].label : false; + return index !== -1 ? select.$scope.$matches[index].label : false; }).filter(angular.isDefined); if (selected.length > (options.maxLength || defaults.maxLength)) { selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml); @@ -274,9 +312,9 @@ angular.module('mgcrea.ngStrap.select', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.ngSt } } else { index = select.$getIndex(controller.$modelValue); - selected = angular.isDefined(index) ? select.$scope.$matches[index].label : false; + selected = index !== -1 ? select.$scope.$matches[index].label : false; } - element.html((selected ? selected : options.placeholder) + (options.caretHtml ? options.caretHtml : defaults.caretHtml)); + element.html((selected || options.placeholder) + (options.caretHtml || defaults.caretHtml)); }; if (options.multiple) { controller.$isEmpty = function(value) { diff --git a/dist/modules/select.min.js b/dist/modules/select.min.js index ffc747497..9cd03e700 100644 --- a/dist/modules/select.min.js +++ b/dist/modules/select.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.select',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$select',function(){var e=this.defaults={animation:'am-fade',prefixClass:'select',prefixEvent:'$select',placement:'bottom-left',templateUrl:'select/select.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,multiple:!1,allNoneButtons:!1,sort:!0,caretHtml:' ',placeholder:'Choose among the following...',allText:'All',noneText:'None',maxLength:3,maxLengthHtml:'selected',iconCheckmark:'glyphicon glyphicon-ok'};this.$get=['$window','$document','$rootScope','$tooltip','$timeout',function(t,n,a,l,i){function o(a,o,c){var u={},s=angular.extend({},e,c);u=l(a,s);var d=u.$scope;d.$matches=[],s.multiple?d.$activeIndex=[]:d.$activeIndex=-1,d.$isMultiple=s.multiple,d.$showAllNoneButtons=s.allNoneButtons&&s.multiple,d.$iconCheckmark=s.iconCheckmark,d.$allText=s.allText,d.$noneText=s.noneText,d.$activate=function(e){d.$$postDigest(function(){u.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){u.select(e)})},d.$isVisible=function(){return u.$isVisible()},d.$isActive=function(e){return u.$isActive(e)},d.$selectAll=function(){for(var e=0;e=s.minLength:d.$matches.length},u.$isActive=function(e){return s.multiple?-1!==d.$activeIndex.indexOf(e):d.$activeIndex===e},u.$getIndex=function(e){var t=d.$matches.length,n=t;if(t){for(n=t;n--&&d.$matches[n].value!==e;);if(!(0>n))return n}},u.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),r){var t=angular.element(e.target);t.triggerHandler('click')}},u.$onKeyDown=function(e){return/(9|13|38|40)/.test(e.keyCode)?(9!==e.keyCode&&(e.preventDefault(),e.stopPropagation()),s.multiple&&9===e.keyCode?u.hide():s.multiple||13!==e.keyCode&&9!==e.keyCode?void(s.multiple||(38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:38===e.keyCode&&d.$activeIndex<0?d.$activeIndex=d.$matches.length-1:40===e.keyCode&&d.$activeIndex0||e.indexOf('Trident/')>0||e.indexOf('Edge/')>0},u.$selectScrollFix=function(e){'UL'===n[0].activeElement.tagName&&(e.preventDefault(),e.stopImmediatePropagation(),e.target.focus())};var $=u.show;u.show=function(){$(),s.multiple&&u.$element.addClass('select-multiple'),i(function(){u.$element.on(r?'touchstart':'mousedown',u.$onMouseDown),s.keyboard&&a.on('keydown',u.$onKeyDown)},0,!1)};var m=u.hide;return u.hide=function(){!s.multiple&&angular.isUndefined(o.$modelValue)&&(d.$activeIndex=-1),u.$element.off(r?'touchstart':'mousedown',u.$onMouseDown),s.keyboard&&a.off('keydown',u.$onKeyDown),m(!0)},u}var c=(angular.element(t.document.body),/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent)),r='createTouch'in t.document&&c;return o.defaults=e,o}]}).directive('bsSelect',['$window','$parse','$q','$select','$parseOptions',function(e,t,n,a,l){var i=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,o){var c={scope:e,placeholder:i.placeholder};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','placeholder','allNoneButtons','maxLength','maxLengthHtml','allText','noneText','iconCheckmark','autoClose','id','sort','caretHtml','prefixClass','prefixEvent'],function(e){angular.isDefined(n[e])&&(c[e]=n[e])});var r=/^(false|0|)$/i;angular.forEach(['html','container','allNoneButtons','sort'],function(e){angular.isDefined(n[e])&&r.test(n[e])&&(c[e]=!1)});var u=t.attr('data-multiple');if(angular.isDefined(u)&&(r.test(u)?c.multiple=!1:c.multiple=u),'select'===t[0].nodeName.toLowerCase()){var s=t;s.css('display','none'),t=angular.element(''),s.after(t)}var d=l(n.bsOptions),$=a(t,o,c);$.$isIE()&&t[0].addEventListener('blur',$.$selectScrollFix);var m=d.$match[7].replace(/\|.+/,'').trim();e.$watchCollection(m,function(t,n){d.valuesFn(e,o).then(function(e){$.update(e),o.$render()})}),e.$watch(n.ngModel,function(e,t){$.$updateActiveIndex(),o.$render()},!0),o.$render=function(){var e,n;c.multiple&&angular.isArray(o.$modelValue)?(e=o.$modelValue.map(function(e){return n=$.$getIndex(e),angular.isDefined(n)?$.$scope.$matches[n].label:!1}).filter(angular.isDefined),e=e.length>(c.maxLength||i.maxLength)?e.length+' '+(c.maxLengthHtml||i.maxLengthHtml):e.join(', ')):(n=$.$getIndex(o.$modelValue),e=angular.isDefined(n)?$.$scope.$matches[n].label:!1),t.html((e?e:c.placeholder)+(c.caretHtml?c.caretHtml:i.caretHtml))},c.multiple&&(o.$isEmpty=function(e){return!e||0===e.length}),e.$on('$destroy',function(){$&&$.destroy(),c=null,$=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.select',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$select',function(){var e=this.defaults={animation:'am-fade',prefixClass:'select',prefixEvent:'$select',placement:'bottom-left',templateUrl:'select/select.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,multiple:!1,allNoneButtons:!1,sort:!0,caretHtml:' ',placeholder:'Choose among the following...',allText:'All',noneText:'None',maxLength:3,maxLengthHtml:'selected',iconCheckmark:'glyphicon glyphicon-ok',toggle:!1};this.$get=['$window','$document','$rootScope','$tooltip','$timeout',function(t,n,a,l,i){function o(a,o,r){var u={},s=angular.extend({},e,r);u=l(a,s);var d=u.$scope;d.$matches=[],s.multiple?d.$activeIndex=[]:d.$activeIndex=-1,d.$isMultiple=s.multiple,d.$showAllNoneButtons=s.allNoneButtons&&s.multiple,d.$iconCheckmark=s.iconCheckmark,d.$allText=s.allText,d.$noneText=s.noneText,d.$activate=function(e){d.$$postDigest(function(){u.activate(e)})},d.$select=function(e,t){d.$$postDigest(function(){u.select(e)})},d.$isVisible=function(){return u.$isVisible()},d.$isActive=function(e){return u.$isActive(e)},d.$selectAll=function(){for(var e=0;e=d.$matches.length)){var t=d.$matches[e].value;d.$apply(function(){u.activate(e),s.multiple?o.$setViewValue(d.$activeIndex.map(function(e){return angular.isUndefined(d.$matches[e])?null:d.$matches[e].value})):(s.toggle?o.$setViewValue(t===o.$modelValue?void 0:t):o.$setViewValue(t),u.hide())}),d.$emit(s.prefixEvent+'.select',t,e,u),angular.isDefined(s.onSelect)&&angular.isFunction(s.onSelect)&&s.onSelect(t,e,u)}},u.$updateActiveIndex=function(){s.multiple?angular.isArray(o.$modelValue)?d.$activeIndex=o.$modelValue.map(function(e){return u.$getIndex(e)}):d.$activeIndex=[]:angular.isDefined(o.$modelValue)&&d.$matches.length?d.$activeIndex=u.$getIndex(o.$modelValue):d.$activeIndex=-1},u.$isVisible=function(){return s.minLength&&o?d.$matches.length&&o.$viewValue.length>=s.minLength:d.$matches.length},u.$isActive=function(e){return s.multiple?d.$activeIndex.indexOf(e)!==-1:d.$activeIndex===e},u.$getIndex=function(e){var t;for(t=d.$matches.length;t--&&!angular.equals(d.$matches[t].value,e););return t},u.$onMouseDown=function(e){if(e.preventDefault(),e.stopPropagation(),c){var t,n=angular.element(e.target);if('A'!==e.target.nodeName)for(var a=n.parent();!t&&a.length>0;)'A'===a[0].nodeName&&(t=a),a=a.parent();t?angular.element(t).triggerHandler('click'):n.triggerHandler('click')}},u.$onKeyDown=function(e){if(/(9|13|38|40)/.test(e.keyCode))return 9!==e.keyCode&&(e.preventDefault(),e.stopPropagation()),s.multiple&&9===e.keyCode?u.hide():s.multiple||13!==e.keyCode&&9!==e.keyCode?void(s.multiple||(38===e.keyCode&&d.$activeIndex>0?d.$activeIndex--:38===e.keyCode&&d.$activeIndex<0?d.$activeIndex=d.$matches.length-1:40===e.keyCode&&d.$activeIndex0||e.indexOf('Trident/')>0||e.indexOf('Edge/')>0},u.$selectScrollFix=function(e){'UL'===n[0].activeElement.tagName&&(e.preventDefault(),e.stopImmediatePropagation(),e.target.focus())};var $=u.show;u.show=function(){$(),s.multiple&&u.$element.addClass('select-multiple'),i(function(){u.$element.on(c?'touchstart':'mousedown',u.$onMouseDown),s.keyboard&&a.on('keydown',u.$onKeyDown)},0,!1)};var m=u.hide;return u.hide=function(){!s.multiple&&angular.isUndefined(o.$modelValue)&&(d.$activeIndex=-1),u.$element.off(c?'touchstart':'mousedown',u.$onMouseDown),s.keyboard&&a.off('keydown',u.$onKeyDown),m(!0)},u}var r=/(ip[ao]d|iphone|android)/gi.test(t.navigator.userAgent),c='createTouch'in t.document&&r;return o.defaults=e,o}]}).directive('bsSelect',['$window','$parse','$q','$select','$parseOptions',function(e,t,n,a,l){var i=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,o){var r={scope:e,placeholder:i.placeholder};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','placeholder','allNoneButtons','maxLength','maxLengthHtml','allText','noneText','iconCheckmark','autoClose','id','sort','caretHtml','prefixClass','prefixEvent','toggle'],function(e){angular.isDefined(n[e])&&(r[e]=n[e])});var c=/^(false|0|)$/i;angular.forEach(['html','container','allNoneButtons','sort'],function(e){angular.isDefined(n[e])&&c.test(n[e])&&(r[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide','onSelect'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(r[t]=e.$eval(n[a]))});var u=t.attr('data-multiple');if(angular.isDefined(u)&&(c.test(u)?r.multiple=!1:r.multiple=u),'select'===t[0].nodeName.toLowerCase()){var s=t;s.css('display','none'),t=angular.element(''),s.after(t)}var d=l(n.bsOptions),$=a(t,o,r);$.$isIE()&&t[0].addEventListener('blur',$.$selectScrollFix);var m=d.$match[7].replace(/\|.+/,'').trim();e.$watch(m,function(t,n){d.valuesFn(e,o).then(function(e){$.update(e),o.$render()})},!0),e.$watch(n.ngModel,function(e,t){$.$updateActiveIndex(),o.$render()},!0),o.$render=function(){var e,n;r.multiple&&angular.isArray(o.$modelValue)?(e=o.$modelValue.map(function(e){return n=$.$getIndex(e),n!==-1&&$.$scope.$matches[n].label}).filter(angular.isDefined),e=e.length>(r.maxLength||i.maxLength)?e.length+' '+(r.maxLengthHtml||i.maxLengthHtml):e.join(', ')):(n=$.$getIndex(o.$modelValue),e=n!==-1&&$.$scope.$matches[n].label),t.html((e||r.placeholder)+(r.caretHtml||i.caretHtml))},r.multiple&&(o.$isEmpty=function(e){return!e||0===e.length}),e.$on('$destroy',function(){$&&$.destroy(),r=null,$=null})}}}]); //# sourceMappingURL=select.min.js.map diff --git a/dist/modules/select.min.js.map b/dist/modules/select.min.js.map index 15120e435..9daeefbbc 100644 --- a/dist/modules/select.min.js.map +++ b/dist/modules/select.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/select.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","multiple","allNoneButtons","sort","caretHtml","placeholder","allText","noneText","maxLength","maxLengthHtml","iconCheckmark","bodyEl","isNative","isTouch","SelectFactory","$select","$tooltip","element","options","scope","$activeIndex","config","$isMultiple","$showAllNoneButtons","$iconCheckmark","$allText","$activate","activate","index","$$postDigest","select","evt","$isActive","$isVisible","i","$selectNone","$matches","length","matches","$updateActiveIndex","b","a","controller","value","$apply","$setViewValue","hide","prefixEvent","$emit","isArray","$modelValue","minLength","$viewValue","$getIndex","l","indexOf","preventDefault","stopPropagation","targetEl","$onMouseDown","keyCode","$onKeyDown","test","$digest","isUndefined","$selectScrollFix","$isIE","stopImmediatePropagation","target","focus","ua","_show","show","activeElement","tagName","e","$element","addClass","$timeout","_hide","on","off","directive","$scope","$window","document","link","require","restrict","forEach","attr","falseValueRegExp","isDefined","dataMultiple","key","nodeName","toLowerCase","inputEl","after","css","addEventListener","watchedOptions","parsedOptions","valuesFn","update","values","ngModel","$match","newValue","oldValue","$render","selected","join","map","label","$isEmpty","$on","destroy"],"mappings":"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","file":"modules/select.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.select', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$select', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'select',\n prefixEvent: '$select',\n placement: 'bottom-left',\n templateUrl: 'select/select.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n multiple: false,\n allNoneButtons: false,\n sort: true,\n caretHtml: ' ',\n placeholder: 'Choose among the following...',\n allText: 'All',\n noneText: 'None',\n maxLength: 3,\n maxLengthHtml: 'selected',\n iconCheckmark: 'glyphicon glyphicon-ok'\n };\n\n this.$get = function($window, $document, $rootScope, $tooltip, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n\n function SelectFactory(element, controller, config) {\n\n var $select = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $select = $tooltip(element, options);\n var scope = $select.$scope;\n\n scope.$matches = [];\n if (options.multiple) {\n scope.$activeIndex = [];\n }\n else {\n scope.$activeIndex = -1;\n }\n scope.$isMultiple = options.multiple;\n scope.$showAllNoneButtons = options.allNoneButtons && options.multiple;\n scope.$iconCheckmark = options.iconCheckmark;\n scope.$allText = options.allText;\n scope.$noneText = options.noneText;\n\n scope.$activate = function(index) {\n scope.$$postDigest(function() {\n $select.activate(index);\n });\n };\n\n scope.$select = function(index, evt) {\n scope.$$postDigest(function() {\n $select.select(index);\n });\n };\n\n scope.$isVisible = function() {\n return $select.$isVisible();\n };\n\n scope.$isActive = function(index) {\n return $select.$isActive(index);\n };\n\n scope.$selectAll = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (!scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n scope.$selectNone = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n // Public methods\n\n $select.update = function(matches) {\n scope.$matches = matches;\n $select.$updateActiveIndex();\n };\n\n $select.activate = function(index) {\n if(options.multiple) {\n $select.$isActive(index) ? scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1) : scope.$activeIndex.push(index);\n if(options.sort) scope.$activeIndex.sort(function(a, b) { return a - b; }); // use numeric sort instead of default sort\n } else {\n scope.$activeIndex = index;\n }\n return scope.$activeIndex;\n };\n\n $select.select = function(index) {\n var value = scope.$matches[index].value;\n scope.$apply(function() {\n $select.activate(index);\n if(options.multiple) {\n controller.$setViewValue(scope.$activeIndex.map(function(index) {\n if (angular.isUndefined(scope.$matches[index])) {\n return null;\n }\n return scope.$matches[index].value;\n }));\n } else {\n controller.$setViewValue(value);\n // Hide if single select\n $select.hide();\n }\n });\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $select);\n };\n\n // Protected methods\n\n $select.$updateActiveIndex = function() {\n if(options.multiple) {\n if(angular.isArray(controller.$modelValue)) {\n scope.$activeIndex = controller.$modelValue.map(function(value) {\n return $select.$getIndex(value);\n });\n } else {\n scope.$activeIndex = [];\n }\n } else {\n if(angular.isDefined(controller.$modelValue) && scope.$matches.length) {\n scope.$activeIndex = $select.$getIndex(controller.$modelValue);\n } else {\n scope.$activeIndex = -1;\n }\n }\n };\n\n $select.$isVisible = function() {\n if(!options.minLength || !controller) {\n return scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && controller.$viewValue.length >= options.minLength;\n };\n\n $select.$isActive = function(index) {\n if(options.multiple) {\n return scope.$activeIndex.indexOf(index) !== -1;\n } else {\n return scope.$activeIndex === index;\n }\n };\n\n $select.$getIndex = function(value) {\n var l = scope.$matches.length, i = l;\n if(!l) return;\n for(i = l; i--;) {\n if(scope.$matches[i].value === value) break;\n }\n if(i < 0) return;\n return i;\n };\n\n $select.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if(isTouch) {\n var targetEl = angular.element(evt.target);\n targetEl.triggerHandler('click');\n }\n };\n\n $select.$onKeyDown = function(evt) {\n if (!/(9|13|38|40)/.test(evt.keyCode)) return;\n // Let tab propagate\n if (evt.keyCode !== 9) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // release focus on tab\n if (options.multiple && evt.keyCode === 9) {\n return $select.hide();\n }\n\n // Select with enter\n if(!options.multiple && (evt.keyCode === 13 || evt.keyCode === 9)) {\n return $select.select(scope.$activeIndex);\n }\n\n if (!options.multiple) {\n // Navigate with keyboard\n if(evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if(evt.keyCode === 38 && scope.$activeIndex < 0) scope.$activeIndex = scope.$matches.length - 1;\n else if(evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if(angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n }\n };\n\n $select.$isIE = function() {\n var ua = $window.navigator.userAgent;\n return ua.indexOf('MSIE ') > 0 || ua.indexOf('Trident/') > 0 || ua.indexOf('Edge/') > 0;\n };\n\n $select.$selectScrollFix = function(e) {\n if ($document[0].activeElement.tagName === 'UL') {\n e.preventDefault();\n e.stopImmediatePropagation();\n e.target.focus();\n }\n };\n\n // Overrides\n\n var _show = $select.show;\n $select.show = function() {\n _show();\n if(options.multiple) {\n $select.$element.addClass('select-multiple');\n }\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n $select.$element.on(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if(options.keyboard) {\n element.on('keydown', $select.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $select.hide;\n $select.hide = function() {\n if(!options.multiple && angular.isUndefined(controller.$modelValue)) {\n scope.$activeIndex = -1;\n }\n $select.$element.off(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if(options.keyboard) {\n element.off('keydown', $select.$onKeyDown);\n }\n _hide(true);\n };\n\n return $select;\n\n }\n\n SelectFactory.defaults = defaults;\n return SelectFactory;\n\n };\n\n })\n\n .directive('bsSelect', function($window, $parse, $q, $select, $parseOptions) {\n\n var defaults = $select.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope, placeholder: defaults.placeholder};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'allNoneButtons', 'sort'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Only parse data-multiple. Angular sets existence attributes to true (multiple/required/etc), they apply this\n // to data-multiple as well for some reason, so we'll parse this ourselves and disregard multiple\n var dataMultiple = element.attr('data-multiple');\n if(angular.isDefined(dataMultiple)) {\n if(falseValueRegExp.test(dataMultiple))\n options.multiple = false;\n else\n options.multiple = dataMultiple;\n }\n\n // Add support for select markup\n if(element[0].nodeName.toLowerCase() === 'select') {\n var inputEl = element;\n inputEl.css('display', 'none');\n element = angular.element('');\n inputEl.after(element);\n }\n\n // Build proper bsOptions\n var parsedOptions = $parseOptions(attr.bsOptions);\n\n // Initialize select\n var select = $select(element, controller, options);\n\n if (select.$isIE()) {\n element[0].addEventListener('blur', select.$selectScrollFix);\n }\n\n // Watch bsOptions values before filtering for changes\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').trim();\n scope.$watchCollection(watchedOptions, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller)\n .then(function(values) {\n select.update(values);\n controller.$render();\n });\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue);\n select.$updateActiveIndex();\n controller.$render();\n }, true);\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var selected, index;\n if(options.multiple && angular.isArray(controller.$modelValue)) {\n selected = controller.$modelValue.map(function(value) {\n index = select.$getIndex(value);\n return angular.isDefined(index) ? select.$scope.$matches[index].label : false;\n }).filter(angular.isDefined);\n if(selected.length > (options.maxLength || defaults.maxLength)) {\n selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml);\n } else {\n selected = selected.join(', ');\n }\n } else {\n index = select.$getIndex(controller.$modelValue);\n selected = angular.isDefined(index) ? select.$scope.$matches[index].label : false;\n }\n element.html((selected ? selected : options.placeholder) + (options.caretHtml ? options.caretHtml : defaults.caretHtml));\n };\n\n if(options.multiple){\n controller.$isEmpty = function(value){\n return !value || value.length === 0;\n };\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (select) select.destroy();\n options = null;\n select = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/select.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","multiple","allNoneButtons","sort","caretHtml","placeholder","allText","noneText","maxLength","maxLengthHtml","iconCheckmark","toggle","$get","isNative","isTouch","SelectFactory","$select","options","scope","$scope","$activeIndex","config","$tooltip","element","$isMultiple","$showAllNoneButtons","$iconCheckmark","$allText","$activate","activate","index","$$postDigest","select","evt","$isActive","$isVisible","i","$selectNone","$matches","length","matches","$updateActiveIndex","splice","a","indexOf","push","b","$apply","controller","$setViewValue","value","map","$emit","prefixEvent","onSelect","hide","isArray","$modelValue","$getIndex","minLength","$viewValue","equals","$onMouseDown","preventDefault","stopPropagation","anchor","anchorCandidate","targetEl","target","nodeName","parent","triggerHandler","$onKeyDown","test","keyCode","$digest","isUndefined","$selectScrollFix","$isIE","stopImmediatePropagation","focus","ua","_show","show","activeElement","tagName","e","$element","addClass","$timeout","_hide","on","off","directive","$window","document","link","require","restrict","forEach","attr","falseValueRegExp","isDefined","key","$eval","bsKey","dataMultiple","slice","inputEl","css","after","addEventListener","watchedOptions","parsedOptions","valuesFn","update","values","ngModel","$match","newValue","oldValue","$render","selected","filter","join","label","$isEmpty","$on","destroy"],"mappings":"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","file":"select.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.select', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$select', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'select',\n prefixEvent: '$select',\n placement: 'bottom-left',\n templateUrl: 'select/select.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n multiple: false,\n allNoneButtons: false,\n sort: true,\n caretHtml: ' ',\n placeholder: 'Choose among the following...',\n allText: 'All',\n noneText: 'None',\n maxLength: 3,\n maxLengthHtml: 'selected',\n iconCheckmark: 'glyphicon glyphicon-ok',\n toggle: false\n };\n\n this.$get = function ($window, $document, $rootScope, $tooltip, $timeout) {\n\n // var bodyEl = angular.element($window.document.body);\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n\n function SelectFactory (element, controller, config) {\n\n var $select = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $select = $tooltip(element, options);\n var scope = $select.$scope;\n\n scope.$matches = [];\n if (options.multiple) {\n scope.$activeIndex = [];\n } else {\n scope.$activeIndex = -1;\n }\n scope.$isMultiple = options.multiple;\n scope.$showAllNoneButtons = options.allNoneButtons && options.multiple;\n scope.$iconCheckmark = options.iconCheckmark;\n scope.$allText = options.allText;\n scope.$noneText = options.noneText;\n\n scope.$activate = function (index) {\n scope.$$postDigest(function () {\n $select.activate(index);\n });\n };\n\n scope.$select = function (index, evt) {\n scope.$$postDigest(function () {\n $select.select(index);\n });\n };\n\n scope.$isVisible = function () {\n return $select.$isVisible();\n };\n\n scope.$isActive = function (index) {\n return $select.$isActive(index);\n };\n\n scope.$selectAll = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (!scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n scope.$selectNone = function () {\n for (var i = 0; i < scope.$matches.length; i++) {\n if (scope.$isActive(i)) {\n scope.$select(i);\n }\n }\n };\n\n // Public methods\n\n $select.update = function (matches) {\n scope.$matches = matches;\n $select.$updateActiveIndex();\n };\n\n $select.activate = function (index) {\n if (options.multiple) {\n if ($select.$isActive(index)) {\n scope.$activeIndex.splice(scope.$activeIndex.indexOf(index), 1);\n } else {\n scope.$activeIndex.push(index);\n }\n if (options.sort) scope.$activeIndex.sort(function (a, b) { return a - b; }); // use numeric sort instead of default sort\n } else {\n scope.$activeIndex = index;\n }\n return scope.$activeIndex;\n };\n\n $select.select = function (index) {\n if (angular.isUndefined(index) || index < 0 || index >= scope.$matches.length) { return; }\n var value = scope.$matches[index].value;\n scope.$apply(function () {\n $select.activate(index);\n if (options.multiple) {\n controller.$setViewValue(scope.$activeIndex.map(function (index) {\n if (angular.isUndefined(scope.$matches[index])) {\n return null;\n }\n return scope.$matches[index].value;\n }));\n } else {\n if (options.toggle) {\n controller.$setViewValue((value === controller.$modelValue) ? undefined : value);\n } else {\n controller.$setViewValue(value);\n }\n // Hide if single select\n $select.hide();\n }\n });\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $select);\n if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) {\n options.onSelect(value, index, $select);\n }\n };\n\n // Protected methods\n\n $select.$updateActiveIndex = function () {\n if (options.multiple) {\n if (angular.isArray(controller.$modelValue)) {\n scope.$activeIndex = controller.$modelValue.map(function (value) {\n return $select.$getIndex(value);\n });\n } else {\n scope.$activeIndex = [];\n }\n } else {\n if (angular.isDefined(controller.$modelValue) && scope.$matches.length) {\n scope.$activeIndex = $select.$getIndex(controller.$modelValue);\n } else {\n scope.$activeIndex = -1;\n }\n }\n };\n\n $select.$isVisible = function () {\n if (!options.minLength || !controller) {\n return scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && controller.$viewValue.length >= options.minLength;\n };\n\n $select.$isActive = function (index) {\n if (options.multiple) {\n return scope.$activeIndex.indexOf(index) !== -1;\n }\n return scope.$activeIndex === index;\n };\n\n $select.$getIndex = function (value) {\n var index;\n for (index = scope.$matches.length; index--;) {\n if (angular.equals(scope.$matches[index].value, value)) break;\n }\n return index;\n };\n\n $select.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n var anchor;\n\n if (evt.target.nodeName !== 'A') {\n var anchorCandidate = targetEl.parent();\n while (!anchor && anchorCandidate.length > 0) {\n if (anchorCandidate[0].nodeName === 'A') {\n anchor = anchorCandidate;\n }\n anchorCandidate = anchorCandidate.parent();\n }\n }\n\n if (anchor) {\n angular.element(anchor).triggerHandler('click');\n } else {\n targetEl.triggerHandler('click');\n }\n }\n };\n\n $select.$onKeyDown = function (evt) {\n if (!/(9|13|38|40)/.test(evt.keyCode)) return;\n // Let tab propagate\n if (evt.keyCode !== 9) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // release focus on tab\n if (options.multiple && evt.keyCode === 9) {\n return $select.hide();\n }\n\n // Select with enter\n if (!options.multiple && (evt.keyCode === 13 || evt.keyCode === 9)) {\n return $select.select(scope.$activeIndex);\n }\n\n if (!options.multiple) {\n // Navigate with keyboard\n if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if (evt.keyCode === 38 && scope.$activeIndex < 0) scope.$activeIndex = scope.$matches.length - 1;\n else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n }\n };\n\n $select.$isIE = function () {\n var ua = $window.navigator.userAgent;\n return ua.indexOf('MSIE ') > 0 || ua.indexOf('Trident/') > 0 || ua.indexOf('Edge/') > 0;\n };\n\n $select.$selectScrollFix = function (e) {\n if ($document[0].activeElement.tagName === 'UL') {\n e.preventDefault();\n e.stopImmediatePropagation();\n e.target.focus();\n }\n };\n\n // Overrides\n\n var _show = $select.show;\n $select.show = function () {\n _show();\n if (options.multiple) {\n $select.$element.addClass('select-multiple');\n }\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n $select.$element.on(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if (options.keyboard) {\n element.on('keydown', $select.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $select.hide;\n $select.hide = function () {\n if (!options.multiple && angular.isUndefined(controller.$modelValue)) {\n scope.$activeIndex = -1;\n }\n $select.$element.off(isTouch ? 'touchstart' : 'mousedown', $select.$onMouseDown);\n if (options.keyboard) {\n element.off('keydown', $select.$onKeyDown);\n }\n _hide(true);\n };\n\n return $select;\n\n }\n\n SelectFactory.defaults = defaults;\n return SelectFactory;\n\n };\n\n })\n\n .directive('bsSelect', function ($window, $parse, $q, $select, $parseOptions) {\n\n var defaults = $select.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {scope: scope, placeholder: defaults.placeholder};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'placeholder', 'allNoneButtons', 'maxLength', 'maxLengthHtml', 'allText', 'noneText', 'iconCheckmark', 'autoClose', 'id', 'sort', 'caretHtml', 'prefixClass', 'prefixEvent', 'toggle'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'allNoneButtons', 'sort'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show, hide and select events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Only parse data-multiple. Angular sets existence attributes to true (multiple/required/etc), they apply this\n // to data-multiple as well for some reason, so we'll parse this ourselves and disregard multiple\n var dataMultiple = element.attr('data-multiple');\n if (angular.isDefined(dataMultiple)) {\n if (falseValueRegExp.test(dataMultiple)) {\n options.multiple = false;\n } else {\n options.multiple = dataMultiple;\n }\n }\n\n // Add support for select markup\n if (element[0].nodeName.toLowerCase() === 'select') {\n var inputEl = element;\n inputEl.css('display', 'none');\n element = angular.element('');\n inputEl.after(element);\n }\n\n // Build proper bsOptions\n var parsedOptions = $parseOptions(attr.bsOptions);\n\n // Initialize select\n var select = $select(element, controller, options);\n\n if (select.$isIE()) {\n element[0].addEventListener('blur', select.$selectScrollFix);\n }\n\n // Watch bsOptions values before filtering for changes\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').trim();\n scope.$watch(watchedOptions, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller)\n .then(function (values) {\n select.update(values);\n controller.$render();\n });\n }, true);\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue);\n select.$updateActiveIndex();\n controller.$render();\n }, true);\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n var selected;\n var index;\n if (options.multiple && angular.isArray(controller.$modelValue)) {\n selected = controller.$modelValue.map(function (value) {\n index = select.$getIndex(value);\n return index !== -1 ? select.$scope.$matches[index].label : false;\n }).filter(angular.isDefined);\n if (selected.length > (options.maxLength || defaults.maxLength)) {\n selected = selected.length + ' ' + (options.maxLengthHtml || defaults.maxLengthHtml);\n } else {\n selected = selected.join(', ');\n }\n } else {\n index = select.$getIndex(controller.$modelValue);\n selected = index !== -1 ? select.$scope.$matches[index].label : false;\n }\n element.html((selected || options.placeholder) + (options.caretHtml || defaults.caretHtml));\n };\n\n if (options.multiple) {\n controller.$isEmpty = function (value) {\n return !value || value.length === 0;\n };\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (select) select.destroy();\n options = null;\n select = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/select.tpl.js b/dist/modules/select.tpl.js index 96aa1d9bc..6e71beac5 100644 --- a/dist/modules/select.tpl.js +++ b/dist/modules/select.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/select.tpl.min.js b/dist/modules/select.tpl.min.js index 10127a76d..14e911868 100644 --- a/dist/modules/select.tpl.min.js +++ b/dist/modules/select.tpl.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/tab.js b/dist/modules/tab.js index ef3a4270f..b865e77bb 100644 --- a/dist/modules/tab.js +++ b/dist/modules/tab.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -107,7 +107,6 @@ angular.module('mgcrea.ngStrap.tab', []).provider('$tab', function() { require: [ '^?ngModel', '^bsTabs' ], scope: true, link: function postLink(scope, element, attrs, controllers) { - var ngModelCtrl = controllers[0]; var bsTabsCtrl = controllers[1]; element.addClass('tab-pane'); attrs.$observe('title', function(newValue, oldValue) { diff --git a/dist/modules/tab.min.js b/dist/modules/tab.min.js index 10f9b46e0..9a9125de4 100644 --- a/dist/modules/tab.min.js +++ b/dist/modules/tab.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.tab',[]).provider('$tab',function(){var e=this.defaults={animation:'am-fade',template:'tab/tab.tpl.html',navClass:'nav-tabs',activeClass:'active'},n=this.controller=function(n,a,t){var s=this;s.$options=angular.copy(e),angular.forEach(['animation','navClass','activeClass'],function(e){angular.isDefined(t[e])&&(s.$options[e]=t[e])}),n.$navClass=s.$options.navClass,n.$activeClass=s.$options.activeClass,s.$panes=n.$panes=[],s.$activePaneChangeListeners=s.$viewChangeListeners=[],s.$push=function(e){angular.isUndefined(s.$panes.$active)&&n.$setActive(e.name||0),s.$panes.push(e)},s.$remove=function(e){var n,a=s.$panes.indexOf(e),t=s.$panes.$active;n=angular.isString(t)?s.$panes.map(function(e){return e.name}).indexOf(t):s.$panes.$active,s.$panes.splice(a,1),n>a?n--:a===n&&n===s.$panes.length&&n--,n>=0&&n=0&&n= 0 && activeIndex < self.$panes.length) {\n self.$setActive(self.$panes[activeIndex].name || activeIndex);\n } else {\n self.$setActive();\n }\n };\n\n self.$setActive = $scope.$setActive = function(value) {\n self.$panes.$active = value;\n self.$activePaneChangeListeners.forEach(function(fn) {\n fn();\n });\n };\n\n self.$isActive = $scope.$isActive = function($pane, $index) {\n return self.$panes.$active === $pane.name || self.$panes.$active === $index;\n };\n\n };\n\n this.$get = function() {\n var $tab = {};\n $tab.defaults = defaults;\n $tab.controller = controller;\n return $tab;\n };\n\n })\n\n .directive('bsTabs', function($window, $animate, $tab, $parse) {\n\n var defaults = $tab.defaults;\n\n return {\n require: ['?ngModel', 'bsTabs'],\n transclude: true,\n scope: true,\n controller: ['$scope', '$element', '$attrs', $tab.controller],\n templateUrl: function(element, attr) {\n return attr.template || defaults.template;\n },\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // 'ngModel' does interfere with form validation\n // and status, use `bsActivePane` instead to avoid it\n if(ngModelCtrl) {\n\n // Update the modelValue following\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n ngModelCtrl.$setViewValue(bsTabsCtrl.$panes.$active);\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n bsTabsCtrl.$setActive(modelValue);\n return modelValue;\n });\n\n }\n\n if (attrs.bsActivePane) {\n // adapted from angularjs ngModelController bindings\n // https://github.com/angular/angular.js/blob/v1.3.1/src%2Fng%2Fdirective%2Finput.js#L1730\n var parsedBsActivePane = $parse(attrs.bsActivePane);\n\n // Update bsActivePane value with change\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n parsedBsActivePane.assign(scope, bsTabsCtrl.$panes.$active);\n });\n\n // watch bsActivePane for value changes\n scope.$watch(attrs.bsActivePane, function(newValue, oldValue) {\n bsTabsCtrl.$setActive(newValue);\n }, true);\n }\n }\n };\n\n })\n\n .directive('bsPane', function($window, $animate, $sce) {\n\n return {\n require: ['^?ngModel', '^bsTabs'],\n scope: true,\n link: function postLink(scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // Add base class\n element.addClass('tab-pane');\n\n // Observe title attribute for change\n attrs.$observe('title', function(newValue, oldValue) {\n scope.title = $sce.trustAsHtml(newValue);\n });\n\n // Save tab name into scope\n scope.name = attrs.name;\n\n // Add animation class\n if(bsTabsCtrl.$options.animation) {\n element.addClass(bsTabsCtrl.$options.animation);\n }\n\n attrs.$observe('disabled', function(newValue, oldValue) {\n scope.disabled = scope.$eval(newValue);\n });\n\n // Push pane to parent bsTabs controller\n bsTabsCtrl.$push(scope);\n\n // remove pane from tab controller when pane is destroyed\n scope.$on('$destroy', function() {\n bsTabsCtrl.$remove(scope);\n });\n\n function render() {\n var index = bsTabsCtrl.$panes.indexOf(scope);\n $animate[bsTabsCtrl.$isActive(scope, index) ? 'addClass' : 'removeClass'](element, bsTabsCtrl.$options.activeClass);\n }\n\n bsTabsCtrl.$activePaneChangeListeners.push(function() {\n render();\n });\n render();\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/tab.js"],"names":["angular","controller","provider","$scope","$element","defaults","this","self","$options","forEach","navClass","activeClass","$activeClass","$panes","$activePaneChangeListeners","$push","pane","isUndefined","$active","key","$attrs","$navClass","$remove","index","active","activeIndex","indexOf","splice","map","$setActive","length","value","name","fn","$pane","$get","$tab","scope","require","transclude","directive","templateUrl","attr","template","link","postLink","ngModelCtrl","bsTabsCtrl","$setViewValue","$formatters","push","attrs","modelValue","bsActivePane","parsedBsActivePane","assign","$parse","$watch","newValue","oldValue","$observe","animation","controllers","render","$animate","$isActive","element","addClass","disabled"],"mappings":"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","file":"tab.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.tab', [])\n\n .provider('$tab', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n template: 'tab/tab.tpl.html',\n navClass: 'nav-tabs',\n activeClass: 'active'\n };\n\n var controller = this.controller = function ($scope, $element, $attrs) {\n var self = this;\n\n // Attributes options\n self.$options = angular.copy(defaults);\n angular.forEach(['animation', 'navClass', 'activeClass'], function (key) {\n if (angular.isDefined($attrs[key])) self.$options[key] = $attrs[key];\n });\n\n // Publish options on scope\n $scope.$navClass = self.$options.navClass;\n $scope.$activeClass = self.$options.activeClass;\n\n self.$panes = $scope.$panes = [];\n\n // Please use $activePaneChangeListeners if you use `bsActivePane`\n // Because we removed `ngModel` as default, we rename viewChangeListeners to\n // activePaneChangeListeners to make more sense.\n self.$activePaneChangeListeners = self.$viewChangeListeners = [];\n\n self.$push = function (pane) {\n if (angular.isUndefined(self.$panes.$active)) {\n $scope.$setActive(pane.name || 0);\n }\n self.$panes.push(pane);\n };\n\n self.$remove = function (pane) {\n var index = self.$panes.indexOf(pane);\n var active = self.$panes.$active;\n var activeIndex;\n if (angular.isString(active)) {\n activeIndex = self.$panes.map(function (pane) {\n return pane.name;\n }).indexOf(active);\n } else {\n activeIndex = self.$panes.$active;\n }\n\n // remove pane from $panes array\n self.$panes.splice(index, 1);\n\n if (index < activeIndex) {\n // we removed a pane before the active pane, so we need to\n // decrement the active pane index\n activeIndex--;\n } else if (index === activeIndex && activeIndex === self.$panes.length) {\n // we remove the active pane and it was the one at the end,\n // so select the previous one\n activeIndex--;\n }\n if (activeIndex >= 0 && activeIndex < self.$panes.length) {\n self.$setActive(self.$panes[activeIndex].name || activeIndex);\n } else {\n self.$setActive();\n }\n };\n\n self.$setActive = $scope.$setActive = function (value) {\n self.$panes.$active = value;\n self.$activePaneChangeListeners.forEach(function (fn) {\n fn();\n });\n };\n\n self.$isActive = $scope.$isActive = function ($pane, $index) {\n return self.$panes.$active === $pane.name || self.$panes.$active === $index;\n };\n\n };\n\n this.$get = function () {\n var $tab = {};\n $tab.defaults = defaults;\n $tab.controller = controller;\n return $tab;\n };\n\n })\n\n .directive('bsTabs', function ($window, $animate, $tab, $parse) {\n\n var defaults = $tab.defaults;\n\n return {\n require: ['?ngModel', 'bsTabs'],\n transclude: true,\n scope: true,\n controller: ['$scope', '$element', '$attrs', $tab.controller],\n templateUrl: function (element, attr) {\n return attr.template || defaults.template;\n },\n link: function postLink (scope, element, attrs, controllers) {\n\n var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // 'ngModel' does interfere with form validation\n // and status, use `bsActivePane` instead to avoid it\n if (ngModelCtrl) {\n\n // Update the modelValue following\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n ngModelCtrl.$setViewValue(bsTabsCtrl.$panes.$active);\n });\n\n // modelValue -> $formatters -> viewValue\n ngModelCtrl.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n bsTabsCtrl.$setActive(modelValue);\n return modelValue;\n });\n\n }\n\n if (attrs.bsActivePane) {\n // adapted from angularjs ngModelController bindings\n // https://github.com/angular/angular.js/blob/v1.3.1/src%2Fng%2Fdirective%2Finput.js#L1730\n var parsedBsActivePane = $parse(attrs.bsActivePane);\n\n // Update bsActivePane value with change\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n parsedBsActivePane.assign(scope, bsTabsCtrl.$panes.$active);\n });\n\n // watch bsActivePane for value changes\n scope.$watch(attrs.bsActivePane, function (newValue, oldValue) {\n bsTabsCtrl.$setActive(newValue);\n }, true);\n }\n }\n };\n\n })\n\n .directive('bsPane', function ($window, $animate, $sce) {\n\n return {\n require: ['^?ngModel', '^bsTabs'],\n scope: true,\n link: function postLink (scope, element, attrs, controllers) {\n\n // var ngModelCtrl = controllers[0];\n var bsTabsCtrl = controllers[1];\n\n // Add base class\n element.addClass('tab-pane');\n\n // Observe title attribute for change\n attrs.$observe('title', function (newValue, oldValue) {\n scope.title = $sce.trustAsHtml(newValue);\n });\n\n // Save tab name into scope\n scope.name = attrs.name;\n\n // Add animation class\n if (bsTabsCtrl.$options.animation) {\n element.addClass(bsTabsCtrl.$options.animation);\n }\n\n attrs.$observe('disabled', function (newValue, oldValue) {\n scope.disabled = scope.$eval(newValue);\n });\n\n // Push pane to parent bsTabs controller\n bsTabsCtrl.$push(scope);\n\n // remove pane from tab controller when pane is destroyed\n scope.$on('$destroy', function () {\n bsTabsCtrl.$remove(scope);\n });\n\n function render () {\n var index = bsTabsCtrl.$panes.indexOf(scope);\n $animate[bsTabsCtrl.$isActive(scope, index) ? 'addClass' : 'removeClass'](element, bsTabsCtrl.$options.activeClass);\n }\n\n bsTabsCtrl.$activePaneChangeListeners.push(function () {\n render();\n });\n render();\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/tab.tpl.js b/dist/modules/tab.tpl.js index 130eaf4e1..615114cbf 100644 --- a/dist/modules/tab.tpl.js +++ b/dist/modules/tab.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -8,5 +8,5 @@ 'use strict'; angular.module('mgcrea.ngStrap.tab').run([ '$templateCache', function($templateCache) { - $templateCache.put('tab/tab.tpl.html', '
'); + $templateCache.put('tab/tab.tpl.html', '
'); } ]); \ No newline at end of file diff --git a/dist/modules/tab.tpl.min.js b/dist/modules/tab.tpl.min.js index 93dae91a6..b16edc1ee 100644 --- a/dist/modules/tab.tpl.min.js +++ b/dist/modules/tab.tpl.min.js @@ -1,8 +1,8 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.tab').run(['$templateCache',function(a){a.put('tab/tab.tpl.html','
')}]); \ No newline at end of file +'use strict';angular.module('mgcrea.ngStrap.tab').run(['$templateCache',function(a){a.put('tab/tab.tpl.html','
')}]); \ No newline at end of file diff --git a/dist/modules/timepicker.js b/dist/modules/timepicker.js index df09f2012..1c9b0c262 100644 --- a/dist/modules/timepicker.js +++ b/dist/modules/timepicker.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -10,6 +10,7 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip' ]).provider('$timepicker', function() { var defaults = this.defaults = { animation: 'am-fade', + defaultDate: 'auto', prefixClass: 'timepicker', placement: 'bottom-left', templateUrl: 'timepicker/timepicker.tpl.html', @@ -36,7 +37,7 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser arrowBehavior: 'pager' }; this.$get = [ '$window', '$document', '$rootScope', '$sce', '$dateFormatter', '$tooltip', '$timeout', function($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) { - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); var isTouch = 'createTouch' in $window.document && isNative; if (!defaults.lang) { defaults.lang = $dateFormatter.getDefaultLocale(); @@ -65,7 +66,12 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser millisecond: startDate.getMilliseconds() }; var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang); - var hoursFormat = $dateFormatter.hoursFormat(format), timeSeparator = $dateFormatter.timeSeparator(format), minutesFormat = $dateFormatter.minutesFormat(format), secondsFormat = $dateFormatter.secondsFormat(format), showSeconds = $dateFormatter.showSeconds(format), showAM = $dateFormatter.showAM(format); + var hoursFormat = $dateFormatter.hoursFormat(format); + var timeSeparator = $dateFormatter.timeSeparator(format); + var minutesFormat = $dateFormatter.minutesFormat(format); + var secondsFormat = $dateFormatter.secondsFormat(format); + var showSeconds = $dateFormatter.showSeconds(format); + var showAM = $dateFormatter.showAM(format); scope.$iconUp = options.iconUp; scope.$iconDown = options.iconDown; scope.$select = function(date, index) { @@ -92,7 +98,9 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser } }; $timepicker.select = function(date, index, keep) { - if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) controller.$dateValue = new Date(1970, 0, 1); + if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) { + controller.$dateValue = options.defaultDate === 'today' ? new Date() : new Date(1970, 0, 1); + } if (!angular.isDate(date)) date = new Date(date); if (index === 0) controller.$dateValue.setHours(date.getHours()); else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes()); else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds()); controller.$setViewValue(angular.copy(controller.$dateValue)); @@ -113,8 +121,10 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser controller.$render(); }; $timepicker.$build = function() { - var i, midIndex = scope.midIndex = parseInt(options.length / 2, 10); - var hours = [], hour; + var i; + var midIndex = scope.midIndex = parseInt(options.length / 2, 10); + var hours = []; + var hour; for (i = 0; i < options.length; i++) { hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep); hours.push({ @@ -124,7 +134,8 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser disabled: $timepicker.$isDisabled(hour, 0) }); } - var minutes = [], minute; + var minutes = []; + var minute; for (i = 0; i < options.length; i++) { minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep); minutes.push({ @@ -134,7 +145,8 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser disabled: $timepicker.$isDisabled(minute, 1) }); } - var seconds = [], second; + var seconds = []; + var second; for (i = 0; i < options.length; i++) { second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep); seconds.push({ @@ -240,9 +252,12 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser return; } var newDate = new Date($timepicker.$date); - var hours = newDate.getHours(), hoursLength = formatDate(newDate, hoursFormat).length; - var minutes = newDate.getMinutes(), minutesLength = formatDate(newDate, minutesFormat).length; - var seconds = newDate.getSeconds(), secondsLength = formatDate(newDate, secondsFormat).length; + var hours = newDate.getHours(); + var hoursLength = formatDate(newDate, hoursFormat).length; + var minutes = newDate.getMinutes(); + var minutesLength = formatDate(newDate, minutesFormat).length; + var seconds = newDate.getSeconds(); + var secondsLength = formatDate(newDate, secondsFormat).length; var sepLength = 1; var lateralMove = /(37|39)/.test(evt.keyCode); var count = 2 + showSeconds * 1 + showAM * 1; @@ -318,18 +333,18 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser if (!isTouch && element.attr('readonly') || element.attr('disabled')) return; _show(); $timeout(function() { - $timepicker.$element && $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); + if ($timepicker.$element) $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); if (options.keyboard) { - element && element.on('keydown', $timepicker.$onKeyDown); + if (element) element.on('keydown', $timepicker.$onKeyDown); } }, 0, false); }; var _hide = $timepicker.hide; $timepicker.hide = function(blur) { if (!$timepicker.$isShown) return; - $timepicker.$element && $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); + if ($timepicker.$element) $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown); if (options.keyboard) { - element && element.off('keydown', $timepicker.$onKeyDown); + if (element) element.off('keydown', $timepicker.$onKeyDown); } _hide(blur); }; @@ -340,7 +355,7 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser } ]; }).directive('bsTimepicker', [ '$window', '$parse', '$q', '$dateFormatter', '$dateParser', '$timepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) { var defaults = $timepicker.defaults; - var isNative = /(ip(a|o)d|iphone|android)/gi.test($window.navigator.userAgent); + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); return { restrict: 'EAC', require: 'ngModel', @@ -348,17 +363,20 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent', 'defaultDate' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container', 'autoclose', 'useNative', 'roundDisplay' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } }); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!timepicker || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i); - newValue === true ? timepicker.show() : timepicker.hide(); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm'; var timepicker = $timepicker(element, controller, options); @@ -367,16 +385,29 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser var formatDate = function(date, format, timezone) { return $dateFormatter.formatDate(date, format, lang, timezone); }; + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!timepicker || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i); + if (newValue === true) { + timepicker.show(); + } else { + timepicker.hide(); + } + }); + } var dateParser = $dateParser({ format: options.timeFormat, lang: lang }); angular.forEach([ 'minTime', 'maxTime' ], function(key) { - angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) { - timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue); - !isNaN(timepicker.$options[key]) && timepicker.$build(); - validateAgainstMinMaxTime(controller.$dateValue); - }); + if (angular.isDefined(attr[key])) { + attr.$observe(key, function(newValue) { + timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue); + if (!isNaN(timepicker.$options[key])) timepicker.$build(); + validateAgainstMinMaxTime(controller.$dateValue); + }); + } }); scope.$watch(attr.ngModel, function(newValue, oldValue) { timepicker.update(controller.$dateValue); @@ -404,9 +435,8 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser if (!parsedTime || isNaN(parsedTime.getTime())) { controller.$setValidity('date', false); return undefined; - } else { - validateAgainstMinMaxTime(parsedTime); } + validateAgainstMinMaxTime(parsedTime); if (options.timeType === 'string') { date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true); return formatDate(date, options.modelTimeFormat || options.timeFormat); @@ -418,9 +448,8 @@ angular.module('mgcrea.ngStrap.timepicker', [ 'mgcrea.ngStrap.helpers.dateParser return date.getTime() / 1e3; } else if (options.timeType === 'iso') { return date.toISOString(); - } else { - return new Date(date); } + return new Date(date); }); controller.$formatters.push(function(modelValue) { var date; diff --git a/dist/modules/timepicker.min.js b/dist/modules/timepicker.min.js index 59c385256..8b64a3f23 100644 --- a/dist/modules/timepicker.min.js +++ b/dist/modules/timepicker.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.timepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$timepicker',function(){var e=this.defaults={animation:'am-fade',prefixClass:'timepicker',placement:'bottom-left',templateUrl:'timepicker/timepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!0,timeType:'date',timeFormat:'shortTime',timezone:null,modelTimeFormat:null,autoclose:!1,minTime:-(1/0),maxTime:+(1/0),length:5,hourStep:1,minuteStep:5,secondStep:5,roundDisplay:!1,iconUp:'glyphicon glyphicon-chevron-up',iconDown:'glyphicon glyphicon-chevron-down',arrowBehavior:'pager'};this.$get=['$window','$document','$rootScope','$sce','$dateFormatter','$tooltip','$timeout',function(t,n,a,i,o,r,s){function u(t,n,a){function i(e){var t=6e4*p.minuteStep;return new Date(Math.floor(e.getTime()/t)*t)}function u(e,n){var a=e+n;if(t[0].createTextRange){var i=t[0].createTextRange();i.collapse(!0),i.moveStart('character',e),i.moveEnd('character',a),i.select()}else t[0].setSelectionRange?t[0].setSelectionRange(e,a):angular.isUndefined(t[0].selectionStart)&&(t[0].selectionStart=e,t[0].selectionEnd=a)}function c(){t[0].focus()}var m=r(t,angular.extend({},e,a)),g=a.scope,p=m.$options,$=m.$scope,f=p.lang,h=function(e,t,n){return o.formatDate(e,t,f,n)},v=0,w=p.roundDisplay?i(new Date):new Date,S=n.$dateValue||w,y={hour:S.getHours(),meridian:S.getHours()<12,minute:S.getMinutes(),second:S.getSeconds(),millisecond:S.getMilliseconds()},D=o.getDatetimeFormat(p.timeFormat,f),T=o.hoursFormat(D),V=o.timeSeparator(D),b=o.minutesFormat(D),k=o.secondsFormat(D),N=o.showSeconds(D),M=o.showAM(D);$.$iconUp=p.iconUp,$.$iconDown=p.iconDown,$.$select=function(e,t){m.select(e,t)},$.$moveIndex=function(e,t){m.$moveIndex(e,t)},$.$switchMeridian=function(e){m.switchMeridian(e)},m.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())?(m.$date=e,angular.extend(y,{hour:e.getHours(),minute:e.getMinutes(),second:e.getSeconds(),millisecond:e.getMilliseconds()}),m.$build()):m.$isBuilt||m.$build()},m.select=function(e,t,a){(!n.$dateValue||isNaN(n.$dateValue.getTime()))&&(n.$dateValue=new Date(1970,0,1)),angular.isDate(e)||(e=new Date(e)),0===t?n.$dateValue.setHours(e.getHours()):1===t?n.$dateValue.setMinutes(e.getMinutes()):2===t&&n.$dateValue.setSeconds(e.getSeconds()),n.$setViewValue(angular.copy(n.$dateValue)),n.$render(),p.autoclose&&!a&&s(function(){m.hide(!0)})},m.switchMeridian=function(e){if(n.$dateValue&&!isNaN(n.$dateValue.getTime())){var t=(e||n.$dateValue).getHours();n.$dateValue.setHours(12>t?t+12:t-12),n.$setViewValue(angular.copy(n.$dateValue)),n.$render()}},m.$build=function(){var e,t,n=$.midIndex=parseInt(p.length/2,10),a=[];for(e=0;e1*p.maxTime},$.$arrowAction=function(e,t){'picker'===p.arrowBehavior?m.$setTimeByStep(e,t):m.$moveIndex(e,t)},m.$setTimeByStep=function(e,t){var n=new Date(m.$date||S),a=n.getHours(),i=n.getMinutes(),o=n.getSeconds();0===t?n.setHours(a-parseInt(p.hourStep,10)*e):1===t?n.setMinutes(i-parseInt(p.minuteStep,10)*e):2===t&&n.setSeconds(o-parseInt(p.secondStep,10)*e),m.select(n,t,!0)},m.$moveIndex=function(e,t){var n;0===t?(n=new Date(1970,0,1,y.hour+e*p.length,y.minute,y.second),angular.extend(y,{hour:n.getHours()})):1===t?(n=new Date(1970,0,1,y.hour,y.minute+e*p.length*p.minuteStep,y.second),angular.extend(y,{minute:n.getMinutes()})):2===t&&(n=new Date(1970,0,1,y.hour,y.minute,y.second+e*p.length*p.secondStep),angular.extend(y,{second:n.getSeconds()})),m.$build()},m.$onMouseDown=function(e){if('input'!==e.target.nodeName.toLowerCase()&&e.preventDefault(),e.stopPropagation(),d){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},m.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void m.hide(!0);var t=new Date(m.$date),n=t.getHours(),a=h(t,T).length,i=t.getMinutes(),o=h(t,b).length,r=t.getSeconds(),s=h(t,k).length,l=1,d=/(37|39)/.test(e.keyCode),c=2+1*N+1*M;d&&(37===e.keyCode?v=1>v?c-1:v-1:39===e.keyCode&&(v=c-1>v?v+1:0));var $=[0,a],f=0;38===e.keyCode&&(f=-1),40===e.keyCode&&(f=1);var w=2===v&&N,S=2===v&&!N||3===v&&N;0===v?(t.setHours(n+f*parseInt(p.hourStep,10)),a=h(t,T).length,$=[0,a]):1===v?(t.setMinutes(i+f*parseInt(p.minuteStep,10)),o=h(t,b).length,$=[a+l,o]):w?(t.setSeconds(r+f*parseInt(p.secondStep,10)),s=h(t,k).length,$=[a+l+o+l,s]):S&&(d||m.switchMeridian(),$=[a+l+o+l+(s+l)*N,2]),m.select(t,v,!0),u($[0],$[1]),g.$digest()}};var x=m.init;m.init=function(){return l&&p.useNative?(t.prop('type','time'),void t.css('-webkit-appearance','textfield')):(d&&(t.prop('type','text'),t.attr('readonly','true'),t.on('click',c)),void x())};var F=m.destroy;m.destroy=function(){l&&p.useNative&&t.off('click',c),F()};var H=m.show;m.show=function(){!d&&t.attr('readonly')||t.attr('disabled')||(H(),s(function(){m.$element&&m.$element.on(d?'touchstart':'mousedown',m.$onMouseDown),p.keyboard&&t&&t.on('keydown',m.$onKeyDown)},0,!1))};var I=m.hide;return m.hide=function(e){m.$isShown&&(m.$element&&m.$element.off(d?'touchstart':'mousedown',m.$onMouseDown),p.keyboard&&t&&t.off('keydown',m.$onKeyDown),I(e))},m}var l=/(ip(a|o)d|iphone|android)/gi.test(t.navigator.userAgent),d='createTouch'in t.document&&l;return e.lang||(e.lang=o.getDefaultLocale()),u.defaults=e,u}]}).directive('bsTimepicker',['$window','$parse','$q','$dateFormatter','$dateParser','$timepicker',function(e,t,n,a,i,o){var r=o.defaults,s=/(ip(a|o)d|iphone|android)/gi.test(e.navigator.userAgent);return{restrict:'EAC',require:'ngModel',link:function(e,t,n,u){function l(e){if(angular.isDate(e)){var t=isNaN(c.minTime)||new Date(e.getTime()).setFullYear(1970,0,1)>=c.minTime,n=isNaN(c.maxTime)||new Date(e.getTime()).setFullYear(1970,0,1)<=c.maxTime,a=t&&n;u.$setValidity('date',a),u.$setValidity('min',t),u.$setValidity('max',n),a&&(u.$dateValue=e)}}function d(){return!u.$dateValue||isNaN(u.$dateValue.getTime())?'':$(u.$dateValue,c.timeFormat)}var c={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','autoclose','timeType','timeFormat','timezone','modelTimeFormat','useNative','hourStep','minuteStep','secondStep','length','arrowBehavior','iconUp','iconDown','roundDisplay','id','prefixClass','prefixEvent'],function(e){angular.isDefined(n[e])&&(c[e]=n[e])});var m=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','roundDisplay'],function(e){angular.isDefined(n[e])&&m.test(n[e])&&(c[e]=!1)}),n.bsShow&&e.$watch(n.bsShow,function(e,t){g&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(timepicker),?/i)),e===!0?g.show():g.hide())}),s&&(c.useNative||r.useNative)&&(c.timeFormat='HH:mm');var g=o(t,u,c);c=g.$options;var p=c.lang,$=function(e,t,n){return a.formatDate(e,t,p,n)},f=i({format:c.timeFormat,lang:p});angular.forEach(['minTime','maxTime'],function(e){angular.isDefined(n[e])&&n.$observe(e,function(t){g.$options[e]=f.getTimeForAttribute(e,t),!isNaN(g.$options[e])&&g.$build(),l(u.$dateValue)})}),e.$watch(n.ngModel,function(e,t){g.update(u.$dateValue)},!0),u.$parsers.unshift(function(e){var t;if(!e)return u.$setValidity('date',!0),null;var n=angular.isDate(e)?e:f.parse(e,u.$dateValue);return!n||isNaN(n.getTime())?void u.$setValidity('date',!1):(l(n),'string'===c.timeType?(t=f.timezoneOffsetAdjust(n,c.timezone,!0),$(t,c.modelTimeFormat||c.timeFormat)):(t=f.timezoneOffsetAdjust(u.$dateValue,c.timezone,!0),'number'===c.timeType?t.getTime():'unix'===c.timeType?t.getTime()/1e3:'iso'===c.timeType?t.toISOString():new Date(t)))}),u.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===c.timeType?f.parse(e,null,c.modelTimeFormat):'unix'===c.timeType?new Date(1e3*e):new Date(e),u.$dateValue=f.timezoneOffsetAdjust(t,c.timezone),d()}),u.$render=function(){t.val(d())},e.$on('$destroy',function(){g&&g.destroy(),c=null,g=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.timepicker',['mgcrea.ngStrap.helpers.dateParser','mgcrea.ngStrap.helpers.dateFormatter','mgcrea.ngStrap.tooltip']).provider('$timepicker',function(){var e=this.defaults={animation:'am-fade',defaultDate:'auto',prefixClass:'timepicker',placement:'bottom-left',templateUrl:'timepicker/timepicker.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,useNative:!0,timeType:'date',timeFormat:'shortTime',timezone:null,modelTimeFormat:null,autoclose:!1,minTime:-(1/0),maxTime:+(1/0),length:5,hourStep:1,minuteStep:5,secondStep:5,roundDisplay:!1,iconUp:'glyphicon glyphicon-chevron-up',iconDown:'glyphicon glyphicon-chevron-down',arrowBehavior:'pager'};this.$get=['$window','$document','$rootScope','$sce','$dateFormatter','$tooltip','$timeout',function(t,n,a,i,o,r,s){function u(t,n,a){function i(e){var t=6e4*p.minuteStep;return new Date(Math.floor(e.getTime()/t)*t)}function u(e,n){var a=e+n;if(t[0].createTextRange){var i=t[0].createTextRange();i.collapse(!0),i.moveStart('character',e),i.moveEnd('character',a),i.select()}else t[0].setSelectionRange?t[0].setSelectionRange(e,a):angular.isUndefined(t[0].selectionStart)&&(t[0].selectionStart=e,t[0].selectionEnd=a)}function c(){t[0].focus()}var m=r(t,angular.extend({},e,a)),g=a.scope,p=m.$options,$=m.$scope,f=p.lang,h=function(e,t,n){return o.formatDate(e,t,f,n)},v=0,w=p.roundDisplay?i(new Date):new Date,S=n.$dateValue||w,D={hour:S.getHours(),meridian:S.getHours()<12,minute:S.getMinutes(),second:S.getSeconds(),millisecond:S.getMilliseconds()},y=o.getDatetimeFormat(p.timeFormat,f),T=o.hoursFormat(y),V=o.timeSeparator(y),b=o.minutesFormat(y),k=o.secondsFormat(y),N=o.showSeconds(y),M=o.showAM(y);$.$iconUp=p.iconUp,$.$iconDown=p.iconDown,$.$select=function(e,t){m.select(e,t)},$.$moveIndex=function(e,t){m.$moveIndex(e,t)},$.$switchMeridian=function(e){m.switchMeridian(e)},m.update=function(e){angular.isDate(e)&&!isNaN(e.getTime())?(m.$date=e,angular.extend(D,{hour:e.getHours(),minute:e.getMinutes(),second:e.getSeconds(),millisecond:e.getMilliseconds()}),m.$build()):m.$isBuilt||m.$build()},m.select=function(e,t,a){n.$dateValue&&!isNaN(n.$dateValue.getTime())||(n.$dateValue='today'===p.defaultDate?new Date:new Date(1970,0,1)),angular.isDate(e)||(e=new Date(e)),0===t?n.$dateValue.setHours(e.getHours()):1===t?n.$dateValue.setMinutes(e.getMinutes()):2===t&&n.$dateValue.setSeconds(e.getSeconds()),n.$setViewValue(angular.copy(n.$dateValue)),n.$render(),p.autoclose&&!a&&s(function(){m.hide(!0)})},m.switchMeridian=function(e){if(n.$dateValue&&!isNaN(n.$dateValue.getTime())){var t=(e||n.$dateValue).getHours();n.$dateValue.setHours(t<12?t+12:t-12),n.$setViewValue(angular.copy(n.$dateValue)),n.$render()}},m.$build=function(){var e,t,n=$.midIndex=parseInt(p.length/2,10),a=[];for(e=0;e1*p.maxTime},$.$arrowAction=function(e,t){'picker'===p.arrowBehavior?m.$setTimeByStep(e,t):m.$moveIndex(e,t)},m.$setTimeByStep=function(e,t){var n=new Date(m.$date||S),a=n.getHours(),i=n.getMinutes(),o=n.getSeconds();0===t?n.setHours(a-parseInt(p.hourStep,10)*e):1===t?n.setMinutes(i-parseInt(p.minuteStep,10)*e):2===t&&n.setSeconds(o-parseInt(p.secondStep,10)*e),m.select(n,t,!0)},m.$moveIndex=function(e,t){var n;0===t?(n=new Date(1970,0,1,D.hour+e*p.length,D.minute,D.second),angular.extend(D,{hour:n.getHours()})):1===t?(n=new Date(1970,0,1,D.hour,D.minute+e*p.length*p.minuteStep,D.second),angular.extend(D,{minute:n.getMinutes()})):2===t&&(n=new Date(1970,0,1,D.hour,D.minute,D.second+e*p.length*p.secondStep),angular.extend(D,{second:n.getSeconds()})),m.$build()},m.$onMouseDown=function(e){if('input'!==e.target.nodeName.toLowerCase()&&e.preventDefault(),e.stopPropagation(),d){var t=angular.element(e.target);'button'!==t[0].nodeName.toLowerCase()&&(t=t.parent()),t.triggerHandler('click')}},m.$onKeyDown=function(e){if(/(38|37|39|40|13)/.test(e.keyCode)&&!e.shiftKey&&!e.altKey){if(e.preventDefault(),e.stopPropagation(),13===e.keyCode)return void m.hide(!0);var t=new Date(m.$date),n=t.getHours(),a=h(t,T).length,i=t.getMinutes(),o=h(t,b).length,r=t.getSeconds(),s=h(t,k).length,l=1,d=/(37|39)/.test(e.keyCode),c=2+1*N+1*M;d&&(37===e.keyCode?v=v<1?c-1:v-1:39===e.keyCode&&(v=v=c.minTime,n=isNaN(c.maxTime)||new Date(e.getTime()).setFullYear(1970,0,1)<=c.maxTime,a=t&&n;u.$setValidity('date',a),u.$setValidity('min',t),u.$setValidity('max',n),a&&(u.$dateValue=e)}}function d(){return!u.$dateValue||isNaN(u.$dateValue.getTime())?'':$(u.$dateValue,c.timeFormat)}var c={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','autoclose','timeType','timeFormat','timezone','modelTimeFormat','useNative','hourStep','minuteStep','secondStep','length','arrowBehavior','iconUp','iconDown','roundDisplay','id','prefixClass','prefixEvent','defaultDate'],function(e){angular.isDefined(n[e])&&(c[e]=n[e])});var m=/^(false|0|)$/i;angular.forEach(['html','container','autoclose','useNative','roundDisplay'],function(e){angular.isDefined(n[e])&&m.test(n[e])&&(c[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(c[t]=e.$eval(n[a]))}),s&&(c.useNative||r.useNative)&&(c.timeFormat='HH:mm');var g=o(t,u,c);c=g.$options;var p=c.lang,$=function(e,t,n){return a.formatDate(e,t,p,n)};n.bsShow&&e.$watch(n.bsShow,function(e,t){g&&angular.isDefined(e)&&(angular.isString(e)&&(e=!!e.match(/true|,?(timepicker),?/i)),e===!0?g.show():g.hide())});var f=i({format:c.timeFormat,lang:p});angular.forEach(['minTime','maxTime'],function(e){angular.isDefined(n[e])&&n.$observe(e,function(t){g.$options[e]=f.getTimeForAttribute(e,t),isNaN(g.$options[e])||g.$build(),l(u.$dateValue)})}),e.$watch(n.ngModel,function(e,t){g.update(u.$dateValue)},!0),u.$parsers.unshift(function(e){var t;if(!e)return u.$setValidity('date',!0),null;var n=angular.isDate(e)?e:f.parse(e,u.$dateValue);return!n||isNaN(n.getTime())?void u.$setValidity('date',!1):(l(n),'string'===c.timeType?(t=f.timezoneOffsetAdjust(n,c.timezone,!0),$(t,c.modelTimeFormat||c.timeFormat)):(t=f.timezoneOffsetAdjust(u.$dateValue,c.timezone,!0),'number'===c.timeType?t.getTime():'unix'===c.timeType?t.getTime()/1e3:'iso'===c.timeType?t.toISOString():new Date(t)))}),u.$formatters.push(function(e){var t;return t=angular.isUndefined(e)||null===e?NaN:angular.isDate(e)?e:'string'===c.timeType?f.parse(e,null,c.modelTimeFormat):'unix'===c.timeType?new Date(1e3*e):new Date(e),u.$dateValue=f.timezoneOffsetAdjust(t,c.timezone),d()}),u.$render=function(){t.val(d())},e.$on('$destroy',function(){g&&g.destroy(),c=null,g=null})}}}]); //# sourceMappingURL=timepicker.min.js.map diff --git a/dist/modules/timepicker.min.js.map b/dist/modules/timepicker.min.js.map index 890f57660..10515429c 100644 --- a/dist/modules/timepicker.min.js.map +++ b/dist/modules/timepicker.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/timepicker.js"],"names":["angular","container","provider","keyboard","this","defaults","html","delay","useNative","timeType","timeFormat","timezone","modelTimeFormat","autoclose","minTime","maxTime","Infinity","length","hourStep","minuteStep","secondStep","roundDisplay","iconUp","iconDown","arrowBehavior","isNative","$get","$timepicker","$tooltip","element","extend","config","timepickerFactory","formatDate","format","viewDate","hour","startDate","getHours","meridian","coeff","selRange","select","end","start","setSelectionRange","isUndefined","collapse","selectionStart","moveStart","selectionEnd","moveEnd","focusElement","focus","_init","init","parentScope","floorMinutes","time","options","scope","floor","lang","selectedIndex","date","defaultDate","second","getSeconds","millisecond","getMilliseconds","controller","$dateValue","hoursFormat","$dateFormatter","$iconUp","$iconDown","$select","$moveIndex","value","index","$switchMeridian","switchMeridian","isDate","isNaN","getMinutes","getTime","$build","minute","$setViewValue","$timeout","keep","hide","Date","setHours","setMinutes","setSeconds","copy","$render","hours","midIndex","push","label","selected","$isSelected","disabled","i","minutes","seconds","minutesFormat","rows","showSeconds","secondsFormat","$isDisabled","showAM","timeSeparator","$date","isAM","selectedTime","$arrowAction","$setTimeByStep","newDate","parseInt","targetDate","$onMouseDown","evt","target","nodeName","stopPropagation","targetEl","triggerHandler","preventDefault","isTouch","$onKeyDown","test","keyCode","sepLength","lateralMove","count","minutesLength","selectRange","hoursLength","incr","isSeconds","isMeridian","secondsLength","createSelection","$digest","createTextRange","prop","attr","_destroy","destroy","off","_show","show","_hide","blur","$element","directive","$options","require","restrict","$window","navigator","userAgent","link","falseValueRegExp","isMaxValid","isValid","parsedTime","isMinValid","setFullYear","$setValidity","$parsers","unshift","viewValue","getTimeFormattedString","bsShow","newValue","oldValue","timepicker","isDefined","key","forEach","dateParser","$dateParser","$observe","validateAgainstMinMaxTime","$watch","ngModel","getTimeForAttribute","parse","timezoneOffsetAdjust","$formatters","modelValue","NaN","$on"],"mappings":"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","file":"modules/timepicker.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.timepicker', ['mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip'])\n\n .provider('$timepicker', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n //uncommenting the following line will break backwards compatability\n // prefixEvent: 'timepicker',\n prefixClass: 'timepicker',\n placement: 'bottom-left',\n templateUrl: 'timepicker/timepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: true,\n timeType: 'date',\n timeFormat: 'shortTime',\n timezone: null,\n modelTimeFormat: null,\n autoclose: false,\n minTime: -Infinity,\n maxTime: +Infinity,\n length: 5,\n hourStep: 1,\n minuteStep: 5,\n secondStep: 5,\n roundDisplay: false,\n iconUp: 'glyphicon glyphicon-chevron-up',\n iconDown: 'glyphicon glyphicon-chevron-down',\n arrowBehavior: 'pager'\n };\n\n this.$get = function($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) {\n\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) {\n defaults.lang = $dateFormatter.getDefaultLocale();\n }\n\n function timepickerFactory(element, controller, config) {\n\n var $timepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $timepicker.$options;\n var scope = $timepicker.$scope;\n\n var lang = options.lang;\n var formatDate = function(date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n function floorMinutes(time) {\n // coeff used to floor current time to nearest minuteStep interval\n var coeff = 1000 * 60 * options.minuteStep;\n return new Date(Math.floor(time.getTime() / coeff) * coeff);\n }\n\n // View vars\n\n var selectedIndex = 0;\n var defaultDate = options.roundDisplay ? floorMinutes(new Date()) : new Date();\n var startDate = controller.$dateValue || defaultDate;\n var viewDate = {\n hour: startDate.getHours(),\n meridian: startDate.getHours() < 12,\n minute: startDate.getMinutes(),\n second: startDate.getSeconds(),\n millisecond: startDate.getMilliseconds()\n };\n\n var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang);\n\n var hoursFormat = $dateFormatter.hoursFormat(format),\n timeSeparator = $dateFormatter.timeSeparator(format),\n minutesFormat = $dateFormatter.minutesFormat(format),\n secondsFormat = $dateFormatter.secondsFormat(format),\n showSeconds = $dateFormatter.showSeconds(format),\n showAM = $dateFormatter.showAM(format);\n\n scope.$iconUp = options.iconUp;\n scope.$iconDown = options.iconDown;\n\n // Scope methods\n\n scope.$select = function(date, index) {\n $timepicker.select(date, index);\n };\n scope.$moveIndex = function(value, index) {\n $timepicker.$moveIndex(value, index);\n };\n scope.$switchMeridian = function(date) {\n $timepicker.switchMeridian(date);\n };\n\n // Public methods\n\n $timepicker.update = function(date) {\n // console.warn('$timepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $timepicker.$date = date;\n angular.extend(viewDate, {\n hour: date.getHours(),\n minute: date.getMinutes(),\n second: date.getSeconds(),\n millisecond: date.getMilliseconds()\n });\n $timepicker.$build();\n } else if (!$timepicker.$isBuilt) {\n $timepicker.$build();\n }\n };\n\n $timepicker.select = function(date, index, keep) {\n // console.warn('$timepicker.select', date, scope.$mode);\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) controller.$dateValue = new Date(1970, 0, 1);\n if (!angular.isDate(date)) date = new Date(date);\n if (index === 0) controller.$dateValue.setHours(date.getHours());\n else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes());\n else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds());\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function() {\n $timepicker.hide(true);\n });\n }\n };\n\n $timepicker.switchMeridian = function(date) {\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n return;\n }\n var hours = (date || controller.$dateValue).getHours();\n controller.$dateValue.setHours(hours < 12 ? hours + 12 : hours - 12);\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n };\n\n // Protected methods\n\n $timepicker.$build = function() {\n // console.warn('$timepicker.$build() viewDate=%o', viewDate);\n var i, midIndex = scope.midIndex = parseInt(options.length / 2, 10);\n var hours = [],\n hour;\n for (i = 0; i < options.length; i++) {\n hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep);\n hours.push({\n date: hour,\n label: formatDate(hour, hoursFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(hour, 0),\n disabled: $timepicker.$isDisabled(hour, 0)\n });\n }\n var minutes = [],\n minute;\n for (i = 0; i < options.length; i++) {\n minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep);\n minutes.push({\n date: minute,\n label: formatDate(minute, minutesFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(minute, 1),\n disabled: $timepicker.$isDisabled(minute, 1)\n });\n }\n var seconds = [],\n second;\n for (i = 0; i < options.length; i++) {\n second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep);\n seconds.push({\n date: second,\n label: formatDate(second, secondsFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(second, 2),\n disabled: $timepicker.$isDisabled(second, 2)\n });\n }\n\n var rows = [];\n for (i = 0; i < options.length; i++) {\n if (showSeconds) {\n rows.push([hours[i], minutes[i], seconds[i]]);\n } else {\n rows.push([hours[i], minutes[i]]);\n }\n }\n scope.rows = rows;\n scope.showSeconds = showSeconds;\n scope.showAM = showAM;\n scope.isAM = ($timepicker.$date || hours[midIndex].date).getHours() < 12;\n scope.timeSeparator = timeSeparator;\n $timepicker.$isBuilt = true;\n };\n\n $timepicker.$isSelected = function(date, index) {\n if (!$timepicker.$date) return false;\n else if (index === 0) {\n return date.getHours() === $timepicker.$date.getHours();\n } else if (index === 1) {\n return date.getMinutes() === $timepicker.$date.getMinutes();\n } else if (index === 2) {\n return date.getSeconds() === $timepicker.$date.getSeconds();\n }\n };\n\n $timepicker.$isDisabled = function(date, index) {\n var selectedTime;\n if (index === 0) {\n selectedTime = date.getTime() + viewDate.minute * 6e4 + viewDate.second * 1e3;\n } else if (index === 1) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.second * 1e3;\n } else if (index === 2) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.minute * 6e4;\n }\n return selectedTime < options.minTime * 1 || selectedTime > options.maxTime * 1;\n };\n\n scope.$arrowAction = function(value, index) {\n if (options.arrowBehavior === 'picker') {\n $timepicker.$setTimeByStep(value, index);\n } else {\n $timepicker.$moveIndex(value, index);\n }\n };\n\n $timepicker.$setTimeByStep = function(value, index) {\n var newDate = new Date($timepicker.$date || startDate);\n var hours = newDate.getHours();\n var minutes = newDate.getMinutes();\n var seconds = newDate.getSeconds();\n if (index === 0) {\n newDate.setHours(hours - (parseInt(options.hourStep, 10) * value));\n } else if (index === 1) {\n newDate.setMinutes(minutes - (parseInt(options.minuteStep, 10) * value));\n } else if (index === 2) {\n newDate.setSeconds(seconds - (parseInt(options.secondStep, 10) * value));\n }\n $timepicker.select(newDate, index, true);\n };\n\n $timepicker.$moveIndex = function(value, index) {\n var targetDate;\n if (index === 0) {\n targetDate = new Date(1970, 0, 1, viewDate.hour + (value * options.length), viewDate.minute, viewDate.second);\n angular.extend(viewDate, {\n hour: targetDate.getHours()\n });\n } else if (index === 1) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute + (value * options.length * options.minuteStep), viewDate.second);\n angular.extend(viewDate, {\n minute: targetDate.getMinutes()\n });\n } else if (index === 2) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute, viewDate.second + (value * options.length * options.secondStep));\n angular.extend(viewDate, {\n second: targetDate.getSeconds()\n });\n }\n $timepicker.$build();\n };\n\n $timepicker.$onMouseDown = function(evt) {\n // Prevent blur on mousedown on .dropdown-menu\n if (evt.target.nodeName.toLowerCase() !== 'input') evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $timepicker.$onKeyDown = function(evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Close on enter\n if (evt.keyCode === 13) {\n $timepicker.hide(true);\n return;\n }\n\n // Navigate with keyboard\n var newDate = new Date($timepicker.$date);\n var hours = newDate.getHours(),\n hoursLength = formatDate(newDate, hoursFormat).length;\n var minutes = newDate.getMinutes(),\n minutesLength = formatDate(newDate, minutesFormat).length;\n var seconds = newDate.getSeconds(),\n secondsLength = formatDate(newDate, secondsFormat).length;\n var sepLength = 1;\n var lateralMove = /(37|39)/.test(evt.keyCode);\n var count = 2 + showSeconds * 1 + showAM * 1;\n\n // Navigate indexes (left, right)\n if (lateralMove) {\n if (evt.keyCode === 37) selectedIndex = selectedIndex < 1 ? count - 1 : selectedIndex - 1;\n else if (evt.keyCode === 39) selectedIndex = selectedIndex < count - 1 ? selectedIndex + 1 : 0;\n }\n\n // Update values (up, down)\n var selectRange = [0, hoursLength];\n var incr = 0;\n if (evt.keyCode === 38) incr = -1;\n if (evt.keyCode === 40) incr = +1;\n var isSeconds = selectedIndex === 2 && showSeconds;\n var isMeridian = selectedIndex === 2 && !showSeconds || selectedIndex === 3 && showSeconds;\n if (selectedIndex === 0) {\n newDate.setHours(hours + incr * parseInt(options.hourStep, 10));\n // re-calculate hours length because we have changed hours value\n hoursLength = formatDate(newDate, hoursFormat).length;\n selectRange = [0, hoursLength];\n } else if (selectedIndex === 1) {\n newDate.setMinutes(minutes + incr * parseInt(options.minuteStep, 10));\n // re-calculate minutes length because we have changes minutes value\n minutesLength = formatDate(newDate, minutesFormat).length;\n selectRange = [hoursLength + sepLength, minutesLength];\n } else if (isSeconds) {\n newDate.setSeconds(seconds + incr * parseInt(options.secondStep, 10));\n // re-calculate seconds length because we have changes seconds value\n secondsLength = formatDate(newDate, secondsFormat).length;\n selectRange = [hoursLength + sepLength + minutesLength + sepLength, secondsLength];\n } else if (isMeridian) {\n if (!lateralMove) $timepicker.switchMeridian();\n selectRange = [hoursLength + sepLength + minutesLength + sepLength + (secondsLength + sepLength) * showSeconds, 2];\n }\n $timepicker.select(newDate, selectedIndex, true);\n createSelection(selectRange[0], selectRange[1]);\n parentScope.$digest();\n };\n\n // Private\n\n function createSelection(start, length) {\n var end = start + length;\n if (element[0].createTextRange) {\n var selRange = element[0].createTextRange();\n selRange.collapse(true);\n selRange.moveStart('character', start);\n selRange.moveEnd('character', end);\n selRange.select();\n } else if (element[0].setSelectionRange) {\n element[0].setSelectionRange(start, end);\n } else if (angular.isUndefined(element[0].selectionStart)) {\n element[0].selectionStart = start;\n element[0].selectionEnd = end;\n }\n }\n\n function focusElement() {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $timepicker.init;\n $timepicker.init = function() {\n if (isNative && options.useNative) {\n element.prop('type', 'time');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $timepicker.destroy;\n $timepicker.destroy = function() {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $timepicker.show;\n $timepicker.show = function() {\n if((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n $timepicker.$element && $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n element && element.on('keydown', $timepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $timepicker.hide;\n $timepicker.hide = function(blur) {\n if (!$timepicker.$isShown) return;\n $timepicker.$element && $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n element && element.off('keydown', $timepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $timepicker;\n\n }\n\n timepickerFactory.defaults = defaults;\n return timepickerFactory;\n\n };\n\n })\n\n\n .directive('bsTimepicker', function($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) {\n\n var defaults = $timepicker.defaults;\n var isNative = /(ip(a|o)d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'roundDisplay'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if (!timepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i);\n newValue === true ? timepicker.show() : timepicker.hide();\n });\n\n // Initialize timepicker\n if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm';\n var timepicker = $timepicker(element, controller, options);\n options = timepicker.$options;\n\n var lang = options.lang;\n var formatDate = function(date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n // Initialize parser\n var dateParser = $dateParser({\n format: options.timeFormat,\n lang: lang\n });\n\n // Observe attributes for changes\n angular.forEach(['minTime', 'maxTime'], function(key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n angular.isDefined(attr[key]) && attr.$observe(key, function(newValue) {\n timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue);\n !isNaN(timepicker.$options[key]) && timepicker.$build();\n validateAgainstMinMaxTime(controller.$dateValue);\n });\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue, controller.$dateValue);\n timepicker.update(controller.$dateValue);\n }, true);\n\n function validateAgainstMinMaxTime(parsedTime) {\n if (!angular.isDate(parsedTime)) return;\n var isMinValid = isNaN(options.minTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) >= options.minTime;\n var isMaxValid = isNaN(options.maxTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) <= options.maxTime;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (!isValid) {\n return;\n }\n controller.$dateValue = parsedTime;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function(viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n controller.$setValidity('date', true);\n return null;\n }\n var parsedTime = angular.isDate(viewValue) ? viewValue : dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedTime || isNaN(parsedTime.getTime())) {\n controller.$setValidity('date', false);\n // Return undefined, causes ngModelController to\n // invalidate model value\n return undefined;\n } else {\n validateAgainstMinMaxTime(parsedTime);\n }\n\n if (options.timeType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true);\n return formatDate(date, options.modelTimeFormat || options.timeFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.timeType === 'number') {\n return date.getTime();\n } else if (options.timeType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.timeType === 'iso') {\n return date.toISOString();\n } else {\n return new Date(date);\n }\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.timeType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelTimeFormat);\n } else if (options.timeType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if(isNaN(date.getTime())) date = new Date(new Date().setMinutes(0) + 36e5);\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getTimeFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function() {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getTimeFormattedString());\n };\n\n function getTimeFormattedString() {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.timeFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (timepicker) timepicker.destroy();\n options = null;\n timepicker = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/timepicker.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","useNative","timeType","timeFormat","timezone","modelTimeFormat","autoclose","minTime","maxTime","Infinity","length","hourStep","minuteStep","secondStep","roundDisplay","iconUp","iconDown","arrowBehavior","isNative","$get","$timepicker","$tooltip","element","extend","config","timepickerFactory","formatDate","format","viewDate","hour","startDate","getHours","meridian","coeff","selRange","select","end","start","setSelectionRange","isUndefined","collapse","selectionStart","moveStart","selectionEnd","moveEnd","focusElement","focus","_init","init","parentScope","floorMinutes","time","options","scope","floor","lang","selectedIndex","date","defaultDate","second","getSeconds","millisecond","getMilliseconds","controller","$dateValue","hoursFormat","$dateFormatter","timeSeparator","minute","minutesFormat","secondsFormat","showSeconds","$iconUp","index","$moveIndex","value","$switchMeridian","switchMeridian","isDate","isNaN","getMinutes","getTime","$build","$isBuilt","$render","keep","$timeout","Date","setHours","setMinutes","setSeconds","$setViewValue","copy","hide","midIndex","parseInt","hours","push","disabled","label","selected","i","minutes","$date","seconds","rows","$isDisabled","showAM","$isSelected","isAM","selectedTime","$arrowAction","$setTimeByStep","newDate","targetDate","$onMouseDown","evt","target","nodeName","stopPropagation","targetEl","triggerHandler","preventDefault","isTouch","$onKeyDown","test","keyCode","hoursLength","minutesLength","sepLength","lateralMove","count","selectRange","incr","isSeconds","isMeridian","secondsLength","createSelection","$digest","createTextRange","prop","attr","_destroy","destroy","off","_show","show","_hide","$element","blur","directive","$options","require","restrict","$window","navigator","userAgent","link","falseValueRegExp","isMaxValid","isValid","parsedTime","isMinValid","setFullYear","$setValidity","$parsers","unshift","viewValue","getTimeFormattedString","forEach","key","isDefined","$eval","bsKey","timepicker","newValue","oldValue","isString","$watch","bsShow","match","dateParser","$dateParser","validateAgainstMinMaxTime","ngModel","update","timezoneOffsetAdjust","$formatters","modelValue","NaN","parse","$on"],"mappings":"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","file":"timepicker.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.timepicker', ['mgcrea.ngStrap.helpers.dateParser', 'mgcrea.ngStrap.helpers.dateFormatter', 'mgcrea.ngStrap.tooltip'])\n\n .provider('$timepicker', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n defaultDate: 'auto',\n // uncommenting the following line will break backwards compatability\n // prefixEvent: 'timepicker',\n prefixClass: 'timepicker',\n placement: 'bottom-left',\n templateUrl: 'timepicker/timepicker.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n // lang: $locale.id,\n useNative: true,\n timeType: 'date',\n timeFormat: 'shortTime',\n timezone: null,\n modelTimeFormat: null,\n autoclose: false,\n minTime: -Infinity,\n maxTime: +Infinity,\n length: 5,\n hourStep: 1,\n minuteStep: 5,\n secondStep: 5,\n roundDisplay: false,\n iconUp: 'glyphicon glyphicon-chevron-up',\n iconDown: 'glyphicon glyphicon-chevron-down',\n arrowBehavior: 'pager'\n };\n\n this.$get = function ($window, $document, $rootScope, $sce, $dateFormatter, $tooltip, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n if (!defaults.lang) {\n defaults.lang = $dateFormatter.getDefaultLocale();\n }\n\n function timepickerFactory (element, controller, config) {\n\n var $timepicker = $tooltip(element, angular.extend({}, defaults, config));\n var parentScope = config.scope;\n var options = $timepicker.$options;\n var scope = $timepicker.$scope;\n\n var lang = options.lang;\n var formatDate = function (date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n function floorMinutes (time) {\n // coeff used to floor current time to nearest minuteStep interval\n var coeff = 1000 * 60 * options.minuteStep;\n return new Date(Math.floor(time.getTime() / coeff) * coeff);\n }\n\n // View vars\n\n var selectedIndex = 0;\n var defaultDate = options.roundDisplay ? floorMinutes(new Date()) : new Date();\n var startDate = controller.$dateValue || defaultDate;\n var viewDate = {\n hour: startDate.getHours(),\n meridian: startDate.getHours() < 12,\n minute: startDate.getMinutes(),\n second: startDate.getSeconds(),\n millisecond: startDate.getMilliseconds()\n };\n\n var format = $dateFormatter.getDatetimeFormat(options.timeFormat, lang);\n\n var hoursFormat = $dateFormatter.hoursFormat(format);\n var timeSeparator = $dateFormatter.timeSeparator(format);\n var minutesFormat = $dateFormatter.minutesFormat(format);\n var secondsFormat = $dateFormatter.secondsFormat(format);\n var showSeconds = $dateFormatter.showSeconds(format);\n var showAM = $dateFormatter.showAM(format);\n\n scope.$iconUp = options.iconUp;\n scope.$iconDown = options.iconDown;\n\n // Scope methods\n\n scope.$select = function (date, index) {\n $timepicker.select(date, index);\n };\n scope.$moveIndex = function (value, index) {\n $timepicker.$moveIndex(value, index);\n };\n scope.$switchMeridian = function (date) {\n $timepicker.switchMeridian(date);\n };\n\n // Public methods\n\n $timepicker.update = function (date) {\n // console.warn('$timepicker.update() newValue=%o', date);\n if (angular.isDate(date) && !isNaN(date.getTime())) {\n $timepicker.$date = date;\n angular.extend(viewDate, {\n hour: date.getHours(),\n minute: date.getMinutes(),\n second: date.getSeconds(),\n millisecond: date.getMilliseconds()\n });\n $timepicker.$build();\n } else if (!$timepicker.$isBuilt) {\n $timepicker.$build();\n }\n };\n\n $timepicker.select = function (date, index, keep) {\n // console.warn('$timepicker.select', date, scope.$mode);\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n controller.$dateValue = options.defaultDate === 'today' ? new Date() : new Date(1970, 0, 1);\n }\n\n if (!angular.isDate(date)) date = new Date(date);\n if (index === 0) controller.$dateValue.setHours(date.getHours());\n else if (index === 1) controller.$dateValue.setMinutes(date.getMinutes());\n else if (index === 2) controller.$dateValue.setSeconds(date.getSeconds());\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n if (options.autoclose && !keep) {\n $timeout(function () {\n $timepicker.hide(true);\n });\n }\n };\n\n $timepicker.switchMeridian = function (date) {\n if (!controller.$dateValue || isNaN(controller.$dateValue.getTime())) {\n return;\n }\n var hours = (date || controller.$dateValue).getHours();\n controller.$dateValue.setHours(hours < 12 ? hours + 12 : hours - 12);\n controller.$setViewValue(angular.copy(controller.$dateValue));\n controller.$render();\n };\n\n // Protected methods\n\n $timepicker.$build = function () {\n // console.warn('$timepicker.$build() viewDate=%o', viewDate);\n var i;\n var midIndex = scope.midIndex = parseInt(options.length / 2, 10);\n var hours = [];\n var hour;\n for (i = 0; i < options.length; i++) {\n hour = new Date(1970, 0, 1, viewDate.hour - (midIndex - i) * options.hourStep);\n hours.push({\n date: hour,\n label: formatDate(hour, hoursFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(hour, 0),\n disabled: $timepicker.$isDisabled(hour, 0)\n });\n }\n var minutes = [];\n var minute;\n for (i = 0; i < options.length; i++) {\n minute = new Date(1970, 0, 1, 0, viewDate.minute - (midIndex - i) * options.minuteStep);\n minutes.push({\n date: minute,\n label: formatDate(minute, minutesFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(minute, 1),\n disabled: $timepicker.$isDisabled(minute, 1)\n });\n }\n var seconds = [];\n var second;\n for (i = 0; i < options.length; i++) {\n second = new Date(1970, 0, 1, 0, 0, viewDate.second - (midIndex - i) * options.secondStep);\n seconds.push({\n date: second,\n label: formatDate(second, secondsFormat),\n selected: $timepicker.$date && $timepicker.$isSelected(second, 2),\n disabled: $timepicker.$isDisabled(second, 2)\n });\n }\n\n var rows = [];\n for (i = 0; i < options.length; i++) {\n if (showSeconds) {\n rows.push([hours[i], minutes[i], seconds[i]]);\n } else {\n rows.push([hours[i], minutes[i]]);\n }\n }\n scope.rows = rows;\n scope.showSeconds = showSeconds;\n scope.showAM = showAM;\n scope.isAM = ($timepicker.$date || hours[midIndex].date).getHours() < 12;\n scope.timeSeparator = timeSeparator;\n $timepicker.$isBuilt = true;\n };\n\n $timepicker.$isSelected = function (date, index) {\n if (!$timepicker.$date) return false;\n else if (index === 0) {\n return date.getHours() === $timepicker.$date.getHours();\n } else if (index === 1) {\n return date.getMinutes() === $timepicker.$date.getMinutes();\n } else if (index === 2) {\n return date.getSeconds() === $timepicker.$date.getSeconds();\n }\n };\n\n $timepicker.$isDisabled = function (date, index) {\n var selectedTime;\n if (index === 0) {\n selectedTime = date.getTime() + viewDate.minute * 6e4 + viewDate.second * 1e3;\n } else if (index === 1) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.second * 1e3;\n } else if (index === 2) {\n selectedTime = date.getTime() + viewDate.hour * 36e5 + viewDate.minute * 6e4;\n }\n return selectedTime < options.minTime * 1 || selectedTime > options.maxTime * 1;\n };\n\n scope.$arrowAction = function (value, index) {\n if (options.arrowBehavior === 'picker') {\n $timepicker.$setTimeByStep(value, index);\n } else {\n $timepicker.$moveIndex(value, index);\n }\n };\n\n $timepicker.$setTimeByStep = function (value, index) {\n var newDate = new Date($timepicker.$date || startDate);\n var hours = newDate.getHours();\n var minutes = newDate.getMinutes();\n var seconds = newDate.getSeconds();\n if (index === 0) {\n newDate.setHours(hours - (parseInt(options.hourStep, 10) * value));\n } else if (index === 1) {\n newDate.setMinutes(minutes - (parseInt(options.minuteStep, 10) * value));\n } else if (index === 2) {\n newDate.setSeconds(seconds - (parseInt(options.secondStep, 10) * value));\n }\n $timepicker.select(newDate, index, true);\n };\n\n $timepicker.$moveIndex = function (value, index) {\n var targetDate;\n if (index === 0) {\n targetDate = new Date(1970, 0, 1, viewDate.hour + (value * options.length), viewDate.minute, viewDate.second);\n angular.extend(viewDate, {\n hour: targetDate.getHours()\n });\n } else if (index === 1) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute + (value * options.length * options.minuteStep), viewDate.second);\n angular.extend(viewDate, {\n minute: targetDate.getMinutes()\n });\n } else if (index === 2) {\n targetDate = new Date(1970, 0, 1, viewDate.hour, viewDate.minute, viewDate.second + (value * options.length * options.secondStep));\n angular.extend(viewDate, {\n second: targetDate.getSeconds()\n });\n }\n $timepicker.$build();\n };\n\n $timepicker.$onMouseDown = function (evt) {\n // Prevent blur on mousedown on .dropdown-menu\n if (evt.target.nodeName.toLowerCase() !== 'input') evt.preventDefault();\n evt.stopPropagation();\n // Emulate click for mobile devices\n if (isTouch) {\n var targetEl = angular.element(evt.target);\n if (targetEl[0].nodeName.toLowerCase() !== 'button') {\n targetEl = targetEl.parent();\n }\n targetEl.triggerHandler('click');\n }\n };\n\n $timepicker.$onKeyDown = function (evt) {\n if (!/(38|37|39|40|13)/.test(evt.keyCode) || evt.shiftKey || evt.altKey) return;\n evt.preventDefault();\n evt.stopPropagation();\n\n // Close on enter\n if (evt.keyCode === 13) {\n $timepicker.hide(true);\n return;\n }\n\n // Navigate with keyboard\n var newDate = new Date($timepicker.$date);\n var hours = newDate.getHours();\n var hoursLength = formatDate(newDate, hoursFormat).length;\n var minutes = newDate.getMinutes();\n var minutesLength = formatDate(newDate, minutesFormat).length;\n var seconds = newDate.getSeconds();\n var secondsLength = formatDate(newDate, secondsFormat).length;\n var sepLength = 1;\n var lateralMove = /(37|39)/.test(evt.keyCode);\n var count = 2 + showSeconds * 1 + showAM * 1;\n\n // Navigate indexes (left, right)\n if (lateralMove) {\n if (evt.keyCode === 37) selectedIndex = selectedIndex < 1 ? count - 1 : selectedIndex - 1;\n else if (evt.keyCode === 39) selectedIndex = selectedIndex < count - 1 ? selectedIndex + 1 : 0;\n }\n\n // Update values (up, down)\n var selectRange = [0, hoursLength];\n var incr = 0;\n if (evt.keyCode === 38) incr = -1;\n if (evt.keyCode === 40) incr = +1;\n var isSeconds = selectedIndex === 2 && showSeconds;\n var isMeridian = selectedIndex === 2 && !showSeconds || selectedIndex === 3 && showSeconds;\n if (selectedIndex === 0) {\n newDate.setHours(hours + incr * parseInt(options.hourStep, 10));\n // re-calculate hours length because we have changed hours value\n hoursLength = formatDate(newDate, hoursFormat).length;\n selectRange = [0, hoursLength];\n } else if (selectedIndex === 1) {\n newDate.setMinutes(minutes + incr * parseInt(options.minuteStep, 10));\n // re-calculate minutes length because we have changes minutes value\n minutesLength = formatDate(newDate, minutesFormat).length;\n selectRange = [hoursLength + sepLength, minutesLength];\n } else if (isSeconds) {\n newDate.setSeconds(seconds + incr * parseInt(options.secondStep, 10));\n // re-calculate seconds length because we have changes seconds value\n secondsLength = formatDate(newDate, secondsFormat).length;\n selectRange = [hoursLength + sepLength + minutesLength + sepLength, secondsLength];\n } else if (isMeridian) {\n if (!lateralMove) $timepicker.switchMeridian();\n selectRange = [hoursLength + sepLength + minutesLength + sepLength + (secondsLength + sepLength) * showSeconds, 2];\n }\n $timepicker.select(newDate, selectedIndex, true);\n createSelection(selectRange[0], selectRange[1]);\n parentScope.$digest();\n };\n\n // Private\n\n function createSelection (start, length) {\n var end = start + length;\n if (element[0].createTextRange) {\n var selRange = element[0].createTextRange();\n selRange.collapse(true);\n selRange.moveStart('character', start);\n selRange.moveEnd('character', end);\n selRange.select();\n } else if (element[0].setSelectionRange) {\n element[0].setSelectionRange(start, end);\n } else if (angular.isUndefined(element[0].selectionStart)) {\n element[0].selectionStart = start;\n element[0].selectionEnd = end;\n }\n }\n\n function focusElement () {\n element[0].focus();\n }\n\n // Overrides\n\n var _init = $timepicker.init;\n $timepicker.init = function () {\n if (isNative && options.useNative) {\n element.prop('type', 'time');\n element.css('-webkit-appearance', 'textfield');\n return;\n } else if (isTouch) {\n element.prop('type', 'text');\n element.attr('readonly', 'true');\n element.on('click', focusElement);\n }\n _init();\n };\n\n var _destroy = $timepicker.destroy;\n $timepicker.destroy = function () {\n if (isNative && options.useNative) {\n element.off('click', focusElement);\n }\n _destroy();\n };\n\n var _show = $timepicker.show;\n $timepicker.show = function () {\n if ((!isTouch && element.attr('readonly')) || element.attr('disabled')) return;\n _show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n if ($timepicker.$element) $timepicker.$element.on(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n if (element) element.on('keydown', $timepicker.$onKeyDown);\n }\n }, 0, false);\n };\n\n var _hide = $timepicker.hide;\n $timepicker.hide = function (blur) {\n if (!$timepicker.$isShown) return;\n if ($timepicker.$element) $timepicker.$element.off(isTouch ? 'touchstart' : 'mousedown', $timepicker.$onMouseDown);\n if (options.keyboard) {\n if (element) element.off('keydown', $timepicker.$onKeyDown);\n }\n _hide(blur);\n };\n\n return $timepicker;\n\n }\n\n timepickerFactory.defaults = defaults;\n return timepickerFactory;\n\n };\n\n })\n\n\n .directive('bsTimepicker', function ($window, $parse, $q, $dateFormatter, $dateParser, $timepicker) {\n\n var defaults = $timepicker.defaults;\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'autoclose', 'timeType', 'timeFormat', 'timezone', 'modelTimeFormat', 'useNative', 'hourStep', 'minuteStep', 'secondStep', 'length', 'arrowBehavior', 'iconUp', 'iconDown', 'roundDisplay', 'id', 'prefixClass', 'prefixEvent', 'defaultDate'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'autoclose', 'useNative', 'roundDisplay'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Initialize timepicker\n if (isNative && (options.useNative || defaults.useNative)) options.timeFormat = 'HH:mm';\n var timepicker = $timepicker(element, controller, options);\n options = timepicker.$options;\n\n var lang = options.lang;\n var formatDate = function (date, format, timezone) {\n return $dateFormatter.formatDate(date, format, lang, timezone);\n };\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!timepicker || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(timepicker),?/i);\n if (newValue === true) {\n timepicker.show();\n } else {\n timepicker.hide();\n }\n });\n }\n\n // Initialize parser\n var dateParser = $dateParser({\n format: options.timeFormat,\n lang: lang\n });\n\n // Observe attributes for changes\n angular.forEach(['minTime', 'maxTime'], function (key) {\n // console.warn('attr.$observe(%s)', key, attr[key]);\n if (angular.isDefined(attr[key])) {\n attr.$observe(key, function (newValue) {\n timepicker.$options[key] = dateParser.getTimeForAttribute(key, newValue);\n if (!isNaN(timepicker.$options[key])) timepicker.$build();\n validateAgainstMinMaxTime(controller.$dateValue);\n });\n }\n });\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.ngModel, newValue, oldValue, controller.$dateValue);\n timepicker.update(controller.$dateValue);\n }, true);\n\n function validateAgainstMinMaxTime (parsedTime) {\n if (!angular.isDate(parsedTime)) return;\n var isMinValid = isNaN(options.minTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) >= options.minTime;\n var isMaxValid = isNaN(options.maxTime) || new Date(parsedTime.getTime()).setFullYear(1970, 0, 1) <= options.maxTime;\n var isValid = isMinValid && isMaxValid;\n controller.$setValidity('date', isValid);\n controller.$setValidity('min', isMinValid);\n controller.$setValidity('max', isMaxValid);\n // Only update the model when we have a valid date\n if (!isValid) {\n return;\n }\n controller.$dateValue = parsedTime;\n }\n\n // viewValue -> $parsers -> modelValue\n controller.$parsers.unshift(function (viewValue) {\n // console.warn('$parser(\"%s\"): viewValue=%o', element.attr('ng-model'), viewValue);\n var date;\n // Null values should correctly reset the model value & validity\n if (!viewValue) {\n // BREAKING CHANGE:\n // return null (not undefined) when input value is empty, so angularjs 1.3\n // ngModelController can go ahead and run validators, like ngRequired\n controller.$setValidity('date', true);\n return null;\n }\n var parsedTime = angular.isDate(viewValue) ? viewValue : dateParser.parse(viewValue, controller.$dateValue);\n if (!parsedTime || isNaN(parsedTime.getTime())) {\n controller.$setValidity('date', false);\n // Return undefined, causes ngModelController to\n // invalidate model value\n return undefined;\n }\n validateAgainstMinMaxTime(parsedTime);\n\n if (options.timeType === 'string') {\n date = dateParser.timezoneOffsetAdjust(parsedTime, options.timezone, true);\n return formatDate(date, options.modelTimeFormat || options.timeFormat);\n }\n date = dateParser.timezoneOffsetAdjust(controller.$dateValue, options.timezone, true);\n if (options.timeType === 'number') {\n return date.getTime();\n } else if (options.timeType === 'unix') {\n return date.getTime() / 1000;\n } else if (options.timeType === 'iso') {\n return date.toISOString();\n }\n return new Date(date);\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var date;\n if (angular.isUndefined(modelValue) || modelValue === null) {\n date = NaN;\n } else if (angular.isDate(modelValue)) {\n date = modelValue;\n } else if (options.timeType === 'string') {\n date = dateParser.parse(modelValue, null, options.modelTimeFormat);\n } else if (options.timeType === 'unix') {\n date = new Date(modelValue * 1000);\n } else {\n date = new Date(modelValue);\n }\n // Setup default value?\n // if(isNaN(date.getTime())) date = new Date(new Date().setMinutes(0) + 36e5);\n controller.$dateValue = dateParser.timezoneOffsetAdjust(date, options.timezone);\n return getTimeFormattedString();\n });\n\n // viewValue -> element\n controller.$render = function () {\n // console.warn('$render(\"%s\"): viewValue=%o', element.attr('ng-model'), controller.$viewValue);\n element.val(getTimeFormattedString());\n };\n\n function getTimeFormattedString () {\n return !controller.$dateValue || isNaN(controller.$dateValue.getTime()) ? '' : formatDate(controller.$dateValue, options.timeFormat);\n }\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (timepicker) timepicker.destroy();\n options = null;\n timepicker = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/timepicker.tpl.js b/dist/modules/timepicker.tpl.js index bde6f14b9..e8d6845ea 100644 --- a/dist/modules/timepicker.tpl.js +++ b/dist/modules/timepicker.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -8,5 +8,5 @@ 'use strict'; angular.module('mgcrea.ngStrap.timepicker').run([ '$templateCache', function($templateCache) { - $templateCache.put('timepicker/timepicker.tpl.html', ''); + $templateCache.put('timepicker/timepicker.tpl.html', ''); } ]); \ No newline at end of file diff --git a/dist/modules/timepicker.tpl.min.js b/dist/modules/timepicker.tpl.min.js index 3b0bc1ac9..ba7f68114 100644 --- a/dist/modules/timepicker.tpl.min.js +++ b/dist/modules/timepicker.tpl.min.js @@ -1,8 +1,8 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.timepicker').run(['$templateCache',function(t){t.put('timepicker/timepicker.tpl.html','')}]); \ No newline at end of file +'use strict';angular.module('mgcrea.ngStrap.timepicker').run(['$templateCache',function(t){t.put('timepicker/timepicker.tpl.html','')}]); \ No newline at end of file diff --git a/dist/modules/tooltip.js b/dist/modules/tooltip.js index e59b92d69..fa4908609 100644 --- a/dist/modules/tooltip.js +++ b/dist/modules/tooltip.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -18,7 +18,7 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra placement: 'top', templateUrl: 'tooltip/tooltip.tpl.html', template: '', - contentTemplate: false, + titleTemplate: false, trigger: 'hover focus', keyboard: false, html: false, @@ -28,15 +28,16 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra delay: 0, autoClose: false, bsEnabled: true, + mouseDownPreventDefault: true, + mouseDownStopPropagation: true, viewport: { selector: 'body', padding: 0 } }; this.$get = [ '$window', '$rootScope', '$bsCompiler', '$q', '$templateCache', '$http', '$animate', '$sce', 'dimensions', '$$rAF', '$timeout', function($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) { - var trim = String.prototype.trim; - var isTouch = 'createTouch' in $window.document; - var htmlReplaceRegExp = /ng-bind="/gi; + var isNative = /(ip[ao]d|iphone|android)/gi.test($window.navigator.userAgent); + var isTouch = 'createTouch' in $window.document && isNative; var $body = angular.element($window.document); function TooltipFactory(element, config) { var $tooltip = {}; @@ -76,8 +77,12 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra }); }; $tooltip.$isShown = scope.$isShown = false; - var timeout, hoverState; - var compileData, tipElement, tipContainer, tipScope; + var timeout; + var hoverState; + var compileData; + var tipElement; + var tipContainer; + var tipScope; promise.then(function(data) { compileData = data; $tooltip.init(); @@ -102,7 +107,11 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra } if (options.show) { scope.$$postDigest(function() { - options.trigger === 'focus' ? element[0].focus() : $tooltip.show(); + if (options.trigger === 'focus') { + element[0].focus(); + } else { + $tooltip.show(); + } }); } }; @@ -124,7 +133,11 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra $tooltip.show = function() { if (!options.bsEnabled || $tooltip.$isShown) return; scope.$emit(options.prefixEvent + '.show.before', $tooltip); - var parent, after; + if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) { + options.onBeforeShow($tooltip); + } + var parent; + var after; if (options.container) { parent = tipContainer; if (tipContainer[0].lastChild) { @@ -149,7 +162,11 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra if (options.animation) tipElement.addClass(options.animation); if (options.type) tipElement.addClass(options.prefixClass + '-' + options.type); if (options.customClass) tipElement.addClass(options.customClass); - after ? after.after(tipElement) : parent.prepend(tipElement); + if (after) { + after.after(tipElement); + } else { + parent.prepend(tipElement); + } $tooltip.$isShown = scope.$isShown = true; safeDigest(scope); $tooltip.$applyPlacement(); @@ -176,6 +193,9 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra }; function enterAnimateCallback() { scope.$emit(options.prefixEvent + '.show', $tooltip); + if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) { + options.onShow($tooltip); + } } $tooltip.leave = function() { clearTimeout(timeout); @@ -194,12 +214,17 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra $tooltip.hide = function(blur) { if (!$tooltip.$isShown) return; scope.$emit(options.prefixEvent + '.hide.before', $tooltip); + if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) { + options.onBeforeHide($tooltip); + } _blur = blur; _tipToHide = tipElement; - if (angular.version.minor <= 2) { - $animate.leave(tipElement, leaveAnimateCallback); - } else { - $animate.leave(tipElement).then(leaveAnimateCallback); + if (tipElement !== null) { + if (angular.version.minor <= 2) { + $animate.leave(tipElement, leaveAnimateCallback); + } else { + $animate.leave(tipElement).then(leaveAnimateCallback); + } } $tooltip.$isShown = scope.$isShown = false; safeDigest(scope); @@ -212,6 +237,9 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra }; function leaveAnimateCallback() { scope.$emit(options.prefixEvent + '.hide', $tooltip); + if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) { + options.onHide($tooltip); + } if (tipElement === _tipToHide) { if (_blur && options.trigger === 'focus') { return element[0].blur(); @@ -219,8 +247,15 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra destroyTipElement(); } } - $tooltip.toggle = function() { - $tooltip.$isShown ? $tooltip.leave() : $tooltip.enter(); + $tooltip.toggle = function(evt) { + if (evt) { + evt.preventDefault(); + } + if ($tooltip.$isShown) { + $tooltip.leave(); + } else { + $tooltip.enter(); + } }; $tooltip.focus = function() { tipElement[0].focus(); @@ -233,25 +268,29 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra }; $tooltip.$applyPlacement = function() { if (!tipElement) return; - var placement = options.placement, autoToken = /\s?auto?\s?/i, autoPlace = autoToken.test(placement); + var placement = options.placement; + var autoToken = /\s?auto?\s?/i; + var autoPlace = autoToken.test(placement); if (autoPlace) { placement = placement.replace(autoToken, '') || defaults.placement; } tipElement.addClass(options.placement); - var elementPosition = getPosition(), tipWidth = tipElement.prop('offsetWidth'), tipHeight = tipElement.prop('offsetHeight'); + var elementPosition = getPosition(); + var tipWidth = tipElement.prop('offsetWidth'); + var tipHeight = tipElement.prop('offsetHeight'); $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport); if (autoPlace) { var originalPlacement = placement; var viewportPosition = getPosition($tooltip.$viewport); - if (originalPlacement.indexOf('bottom') >= 0 && elementPosition.bottom + tipHeight > viewportPosition.bottom) { + if (/bottom/.test(originalPlacement) && elementPosition.bottom + tipHeight > viewportPosition.bottom) { placement = originalPlacement.replace('bottom', 'top'); - } else if (originalPlacement.indexOf('top') >= 0 && elementPosition.top - tipHeight < viewportPosition.top) { + } else if (/top/.test(originalPlacement) && elementPosition.top - tipHeight < viewportPosition.top) { placement = originalPlacement.replace('top', 'bottom'); } - if ((originalPlacement === 'right' || originalPlacement === 'bottom-left' || originalPlacement === 'top-left') && elementPosition.right + tipWidth > viewportPosition.width) { - placement = originalPlacement === 'right' ? 'left' : placement.replace('left', 'right'); - } else if ((originalPlacement === 'left' || originalPlacement === 'bottom-right' || originalPlacement === 'top-right') && elementPosition.left - tipWidth < viewportPosition.left) { - placement = originalPlacement === 'left' ? 'right' : placement.replace('right', 'left'); + if (/left/.test(originalPlacement) && elementPosition.left - tipWidth < viewportPosition.left) { + placement = placement.replace('left', 'right'); + } else if (/right/.test(originalPlacement) && elementPosition.right + tipWidth > viewportPosition.width) { + placement = placement.replace('right', 'left'); } tipElement.removeClass(originalPlacement).addClass(placement); } @@ -271,19 +310,29 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra } }; $tooltip.$onFocusElementMouseDown = function(evt) { - evt.preventDefault(); - evt.stopPropagation(); - $tooltip.$isShown ? element[0].blur() : element[0].focus(); + if (options.mouseDownPreventDefault) { + evt.preventDefault(); + } + if (options.mouseDownStopPropagation) { + evt.stopPropagation(); + } + if ($tooltip.$isShown) { + element[0].blur(); + } else { + element[0].focus(); + } }; function bindTriggerEvents() { var triggers = options.trigger.split(' '); angular.forEach(triggers, function(trigger) { - if (trigger === 'click') { - element.on('click', $tooltip.toggle); + if (trigger === 'click' || trigger === 'contextmenu') { + element.on(trigger, $tooltip.toggle); } else if (trigger !== 'manual') { element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter); element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave); - nodeName === 'button' && trigger !== 'hover' && element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + if (nodeName === 'button' && trigger !== 'hover') { + element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + } } }); } @@ -291,12 +340,14 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra var triggers = options.trigger.split(' '); for (var i = triggers.length; i--; ) { var trigger = triggers[i]; - if (trigger === 'click') { - element.off('click', $tooltip.toggle); + if (trigger === 'click' || trigger === 'contextmenu') { + element.off(trigger, $tooltip.toggle); } else if (trigger !== 'manual') { element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter); element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave); - nodeName === 'button' && trigger !== 'hover' && element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + if (nodeName === 'button' && trigger !== 'hover') { + element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown); + } } } } @@ -334,7 +385,8 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra } function getPosition($element) { $element = $element || (options.target || element); - var el = $element[0], isBody = el.tagName === 'BODY'; + var el = $element[0]; + var isBody = el.tagName === 'BODY'; var elRect = el.getBoundingClientRect(); var rect = {}; for (var p in elRect) { @@ -349,9 +401,11 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra var elOffset = isBody ? { top: 0, left: 0 - } : dimensions.offset(el), scroll = { + } : dimensions.offset(el); + var scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0 - }, outerDims = isBody ? { + }; + var outerDims = isBody ? { width: document.documentElement.clientWidth, height: $window.innerHeight } : null; @@ -400,22 +454,33 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra case 'right': offset.left = position.left + position.width - actualWidth; + break; + + default: + break; } } else if (split[0] === 'left' || split[0] === 'right') { switch (split[1]) { case 'top': - offset.top = position.top - actualHeight; + offset.top = position.top - actualHeight + position.height; break; case 'bottom': - offset.top = position.top + position.height; + offset.top = position.top; + break; + + default: + break; } } return offset; } function applyPlacement(offset, placement) { - var tip = tipElement[0], width = tip.offsetWidth, height = tip.offsetHeight; - var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10), marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10); + var tip = tipElement[0]; + var width = tip.offsetWidth; + var height = tip.offsetHeight; + var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10); + var marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10); if (isNaN(marginTop)) marginTop = 0; if (isNaN(marginLeft)) marginLeft = 0; offset.top = offset.top + marginTop; @@ -429,7 +494,8 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra }); } }, offset), 0); - var actualWidth = tip.offsetWidth, actualHeight = tip.offsetHeight; + var actualWidth = tip.offsetWidth; + var actualHeight = tip.offsetHeight; if (placement === 'top' && actualHeight !== height) { offset.top = offset.top + height - actualHeight; } @@ -442,7 +508,9 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra } dimensions.setOffset(tip, offset); if (/top|right|bottom|left/.test(placement)) { - var isVertical = /top|bottom/.test(placement), arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight, arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'; + var isVertical = /top|bottom/.test(placement); + var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight; + var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'; replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical); } } @@ -504,35 +572,39 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra function findElement(query, element) { return angular.element((element || document).querySelectorAll(query)); } - var fetchPromises = {}; - function fetchTemplate(template) { - if (fetchPromises[template]) return fetchPromises[template]; - return fetchPromises[template] = $http.get(template, { - cache: $templateCache - }).then(function(res) { - return res.data; - }); - } return TooltipFactory; } ]; -}).directive('bsTooltip', [ '$window', '$location', '$sce', '$tooltip', '$$rAF', function($window, $location, $sce, $tooltip, $$rAF) { +}).directive('bsTooltip', [ '$window', '$location', '$sce', '$parse', '$tooltip', '$$rAF', function($window, $location, $sce, $parse, $tooltip, $$rAF) { return { restrict: 'EAC', scope: true, link: function postLink(scope, element, attr, transclusion) { + var tooltip; var options = { scope: scope }; - angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id' ], function(key) { + angular.forEach([ 'template', 'templateUrl', 'controller', 'controllerAs', 'titleTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id' ], function(key) { if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; angular.forEach([ 'html', 'container' ], function(key) { - if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; + if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) { + options[key] = false; + } + }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } }); var dataTarget = element.attr('data-target'); if (angular.isDefined(dataTarget)) { - if (falseValueRegExp.test(dataTarget)) options.target = false; else options.target = dataTarget; + if (falseValueRegExp.test(dataTarget)) { + options.target = false; + } else { + options.target = dataTarget; + } } if (!scope.hasOwnProperty('title')) { scope.title = ''; @@ -541,36 +613,61 @@ angular.module('mgcrea.ngStrap.tooltip', [ 'mgcrea.ngStrap.core', 'mgcrea.ngStra if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) { var oldValue = scope.title; scope.title = $sce.trustAsHtml(newValue); - angular.isDefined(oldValue) && $$rAF(function() { - tooltip && tooltip.$applyPlacement(); - }); + if (angular.isDefined(oldValue)) { + $$rAF(function() { + if (tooltip) tooltip.$applyPlacement(); + }); + } } }); - attr.bsTooltip && scope.$watch(attr.bsTooltip, function(newValue, oldValue) { - if (angular.isObject(newValue)) { - angular.extend(scope, newValue); - } else { - scope.title = newValue; + attr.$observe('disabled', function(newValue) { + if (newValue && tooltip.$isShown) { + tooltip.hide(); } - angular.isDefined(oldValue) && $$rAF(function() { - tooltip && tooltip.$applyPlacement(); - }); - }, true); - attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i); - newValue === true ? tooltip.show() : tooltip.hide(); }); - attr.bsEnabled && scope.$watch(attr.bsEnabled, function(newValue, oldValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i); - newValue === false ? tooltip.setEnabled(false) : tooltip.setEnabled(true); - }); - attr.viewport && scope.$watch(attr.viewport, function(newValue) { - if (!tooltip || !angular.isDefined(newValue)) return; - tooltip.setViewport(newValue); - }); - var tooltip = $tooltip(element, options); + if (attr.bsTooltip) { + scope.$watch(attr.bsTooltip, function(newValue, oldValue) { + if (angular.isObject(newValue)) { + angular.extend(scope, newValue); + } else { + scope.title = newValue; + } + if (angular.isDefined(oldValue)) { + $$rAF(function() { + if (tooltip) tooltip.$applyPlacement(); + }); + } + }, true); + } + if (attr.bsShow) { + scope.$watch(attr.bsShow, function(newValue, oldValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i); + if (newValue === true) { + tooltip.show(); + } else { + tooltip.hide(); + } + }); + } + if (attr.bsEnabled) { + scope.$watch(attr.bsEnabled, function(newValue, oldValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i); + if (newValue === false) { + tooltip.setEnabled(false); + } else { + tooltip.setEnabled(true); + } + }); + } + if (attr.viewport) { + scope.$watch(attr.viewport, function(newValue) { + if (!tooltip || !angular.isDefined(newValue)) return; + tooltip.setViewport(newValue); + }); + } + tooltip = $tooltip(element, options); scope.$on('$destroy', function() { if (tooltip) tooltip.destroy(); options = null; diff --git a/dist/modules/tooltip.min.js b/dist/modules/tooltip.min.js index b0e260297..85f9eac5d 100644 --- a/dist/modules/tooltip.min.js +++ b/dist/modules/tooltip.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.tooltip',['mgcrea.ngStrap.core','mgcrea.ngStrap.helpers.dimensions']).provider('$tooltip',function(){var t=this.defaults={animation:'am-fade',customClass:'',prefixClass:'tooltip',prefixEvent:'tooltip',container:!1,target:!1,placement:'top',templateUrl:'tooltip/tooltip.tpl.html',template:'',contentTemplate:!1,trigger:'hover focus',keyboard:!1,html:!1,show:!1,title:'',type:'',delay:0,autoClose:!1,bsEnabled:!0,viewport:{selector:'body',padding:0}};this.$get=['$window','$rootScope','$bsCompiler','$q','$templateCache','$http','$animate','$sce','dimensions','$$rAF','$timeout',function(e,o,n,i,r,l,a,s,f,c,u){function p(i,r){function l(){U.$emit(A.prefixEvent+'.show',F)}function p(){if(U.$emit(A.prefixEvent+'.hide',F),q===j){if(V&&'focus'===A.trigger)return i[0].blur();P()}}function $(){var t=A.trigger.split(' ');angular.forEach(t,function(t){'click'===t?i.on('click',F.toggle):'manual'!==t&&(i.on('hover'===t?'mouseenter':'focus',F.enter),i.on('hover'===t?'mouseleave':'blur',F.leave),'button'===H&&'hover'!==t&&i.on(g?'touchstart':'mousedown',F.$onFocusElementMouseDown))})}function v(){for(var t=A.trigger.split(' '),e=t.length;e--;){var o=t[e];'click'===o?i.off('click',F.toggle):'manual'!==o&&(i.off('hover'===o?'mouseenter':'focus',F.enter),i.off('hover'===o?'mouseleave':'blur',F.leave),'button'===H&&'hover'!==o&&i.off(g?'touchstart':'mousedown',F.$onFocusElementMouseDown))}}function w(){'focus'!==A.trigger?q.on('keyup',F.$onKeyUp):i.on('keyup',F.$onFocusKeyUp)}function b(){'focus'!==A.trigger?q.off('keyup',F.$onKeyUp):i.off('keyup',F.$onFocusKeyUp)}function y(){u(function(){q.on('click',k),m.on('click',F.hide),L=!0},0,!1)}function E(){L&&(q.off('click',k),m.off('click',F.hide),L=!1)}function k(t){t.stopPropagation()}function C(t){t=t||A.target||i;var o=t[0],n='BODY'===o.tagName,r=o.getBoundingClientRect(),l={};for(var a in r)l[a]=r[a];null===l.width&&(l=angular.extend({},l,{width:r.right-r.left,height:r.bottom-r.top}));var s=n?{top:0,left:0}:f.offset(o),c={scroll:n?document.documentElement.scrollTop||document.body.scrollTop:t.prop('scrollTop')||0},u=n?{width:document.documentElement.clientWidth,height:e.innerHeight}:null;return angular.extend({},l,c,u,s)}function S(t,e,o,n){var i,r=t.split('-');switch(r[0]){case'right':i={top:e.top+e.height/2-n/2,left:e.left+e.width};break;case'bottom':i={top:e.top+e.height,left:e.left+e.width/2-o/2};break;case'left':i={top:e.top+e.height/2-n/2,left:e.left-o};break;default:i={top:e.top-n,left:e.left+e.width/2-o/2}}if(!r[1])return i;if('top'===r[0]||'bottom'===r[0])switch(r[1]){case'left':i.left=e.left;break;case'right':i.left=e.left+e.width-o}else if('left'===r[0]||'right'===r[0])switch(r[1]){case'top':i.top=e.top-n;break;case'bottom':i.top=e.top+e.height}return i}function D(t,e){var o=q[0],n=o.offsetWidth,i=o.offsetHeight,r=parseInt(f.css(o,'margin-top'),10),l=parseInt(f.css(o,'margin-left'),10);isNaN(r)&&(r=0),isNaN(l)&&(l=0),t.top=t.top+r,t.left=t.left+l,f.setOffset(o,angular.extend({using:function(t){q.css({top:Math.round(t.top)+'px',left:Math.round(t.left)+'px',right:''})}},t),0);var a=o.offsetWidth,s=o.offsetHeight;if('top'===e&&s!==i&&(t.top=t.top+i-s),!/top-left|top-right|bottom-left|bottom-right/.test(e)){var c=x(e,t,a,s);if(c.left?t.left+=c.left:t.top+=c.top,f.setOffset(o,t),/top|right|bottom|left/.test(e)){var u=/top|bottom/.test(e),p=u?2*c.left-n+a:2*c.top-i+s,h=u?'offsetWidth':'offsetHeight';T(p,o[h],u)}}}function x(t,e,o,n){var i={top:0,left:0};if(!F.$viewport)return i;var r=A.viewport&&A.viewport.padding||0,l=C(F.$viewport);if(/right|left/.test(t)){var a=e.top-r-l.scroll,s=e.top+r-l.scroll+n;al.top+l.height&&(i.top=l.top+l.height-s)}else{var f=e.left-r,c=e.left+r+o;fl.right&&(i.left=l.left+l.width-c)}return i}function T(t,e,o){var n=d('.tooltip-arrow, .arrow',q[0]);n.css(o?'left':'top',50*(1-t/e)+'%').css(o?'top':'left','')}function P(){clearTimeout(K),F.$isShown&&null!==q&&(A.autoClose&&E(),A.keyboard&&b()),I&&(I.$destroy(),I=null),q&&(q.remove(),q=F.$element=null)}var F={},A=F.$options=angular.extend({},t,r),O=F.$promise=n.compile(A),U=F.$scope=A.scope&&A.scope.$new()||o.$new(),H=i[0].nodeName.toLowerCase();if(A.delay&&angular.isString(A.delay)){var N=A.delay.split(',').map(parseFloat);A.delay=N.length>1?{show:N[0],hide:N[1]}:N[0]}F.$id=A.id||i.attr('id')||'',A.title&&(U.title=s.trustAsHtml(A.title)),U.$setEnabled=function(t){U.$$postDigest(function(){F.setEnabled(t)})},U.$hide=function(){U.$$postDigest(function(){F.hide()})},U.$show=function(){U.$$postDigest(function(){F.show()})},U.$toggle=function(){U.$$postDigest(function(){F.toggle()})},F.$isShown=U.$isShown=!1;var K,M,W,q,B,I;O.then(function(t){W=t,F.init()}),F.init=function(){A.delay&&angular.isNumber(A.delay)&&(A.delay={show:A.delay,hide:A.delay}),'self'===A.container?B=i:angular.isElement(A.container)?B=A.container:A.container&&(B=d(A.container)),$(),A.target&&(A.target=angular.isElement(A.target)?A.target:d(A.target)),A.show&&U.$$postDigest(function(){'focus'===A.trigger?i[0].focus():F.show()})},F.destroy=function(){v(),P(),U.$destroy()},F.enter=function(){return clearTimeout(K),M='in',A.delay&&A.delay.show?void(K=setTimeout(function(){'in'===M&&F.show()},A.delay.show)):F.show()},F.show=function(){if(A.bsEnabled&&!F.$isShown){U.$emit(A.prefixEvent+'.show.before',F);var t,e;A.container?(t=B,e=B[0].lastChild?angular.element(B[0].lastChild):null):(t=null,e=i),q&&P(),I=F.$scope.$new(),q=F.$element=W.link(I,function(t,e){}),q.css({top:'-9999px',left:'-9999px',right:'auto',display:'block',visibility:'hidden'}),A.animation&&q.addClass(A.animation),A.type&&q.addClass(A.prefixClass+'-'+A.type),A.customClass&&q.addClass(A.customClass),e?e.after(q):t.prepend(q),F.$isShown=U.$isShown=!0,h(U),F.$applyPlacement(),angular.version.minor<=2?a.enter(q,t,e,l):a.enter(q,t,e).then(l),h(U),c(function(){q&&q.css({visibility:'visible'}),A.keyboard&&('focus'!==A.trigger&&F.focus(),w())}),A.autoClose&&y()}},F.leave=function(){return clearTimeout(K),M='out',A.delay&&A.delay.hide?void(K=setTimeout(function(){'out'===M&&F.hide()},A.delay.hide)):F.hide()};var V,j;F.hide=function(t){F.$isShown&&(U.$emit(A.prefixEvent+'.hide.before',F),V=t,j=q,angular.version.minor<=2?a.leave(q,p):a.leave(q).then(p),F.$isShown=U.$isShown=!1,h(U),A.keyboard&&null!==q&&b(),A.autoClose&&null!==q&&E())},F.toggle=function(){F.$isShown?F.leave():F.enter()},F.focus=function(){q[0].focus()},F.setEnabled=function(t){A.bsEnabled=t},F.setViewport=function(t){A.viewport=t},F.$applyPlacement=function(){if(q){var e=A.placement,o=/\s?auto?\s?/i,n=o.test(e);n&&(e=e.replace(o,'')||t.placement),q.addClass(A.placement);var i=C(),r=q.prop('offsetWidth'),l=q.prop('offsetHeight');if(F.$viewport=A.viewport&&d(A.viewport.selector||A.viewport),n){var a=e,s=C(F.$viewport);a.indexOf('bottom')>=0&&i.bottom+l>s.bottom?e=a.replace('bottom','top'):a.indexOf('top')>=0&&i.top-ls.width?e='right'===a?'left':e.replace('left','right'):('left'===a||'bottom-right'===a||'top-right'===a)&&i.left-rr.top+r.height&&(i.top=r.top+r.height-s)}else{var u=e.left-a,f=e.left+a+o;ur.right&&(i.left=r.left+r.width-f)}return i}function H(t,e,o){var n=d('.tooltip-arrow, .arrow',W[0]);n.css(o?'left':'top',50*(1-t/e)+'%').css(o?'top':'left','')}function P(){clearTimeout(K),T.$isShown&&null!==W&&(F.autoClose&&S(),F.keyboard&&b()),I&&(I.$destroy(),I=null),W&&(W.remove(),W=T.$element=null)}var T={},F=T.$options=angular.extend({},t,a),A=T.$promise=n.compile(F),B=T.$scope=F.scope&&F.scope.$new()||o.$new(),U=i[0].nodeName.toLowerCase();if(F.delay&&angular.isString(F.delay)){var N=F.delay.split(',').map(parseFloat);F.delay=N.length>1?{show:N[0],hide:N[1]}:N[0]}T.$id=F.id||i.attr('id')||'',F.title&&(B.title=s.trustAsHtml(F.title)),B.$setEnabled=function(t){B.$$postDigest(function(){T.setEnabled(t)})},B.$hide=function(){B.$$postDigest(function(){T.hide()})},B.$show=function(){B.$$postDigest(function(){T.show()})},B.$toggle=function(){B.$$postDigest(function(){T.toggle()})},T.$isShown=B.$isShown=!1;var K,O,M,W,q,I;A.then(function(t){M=t,T.init()}),T.init=function(){F.delay&&angular.isNumber(F.delay)&&(F.delay={show:F.delay,hide:F.delay}),'self'===F.container?q=i:angular.isElement(F.container)?q=F.container:F.container&&(q=d(F.container)),g(),F.target&&(F.target=angular.isElement(F.target)?F.target:d(F.target)),F.show&&B.$$postDigest(function(){'focus'===F.trigger?i[0].focus():T.show()})},T.destroy=function(){w(),P(),B.$destroy()},T.enter=function(){return clearTimeout(K),O='in',F.delay&&F.delay.show?void(K=setTimeout(function(){'in'===O&&T.show()},F.delay.show)):T.show()},T.show=function(){if(F.bsEnabled&&!T.$isShown){B.$emit(F.prefixEvent+'.show.before',T),angular.isDefined(F.onBeforeShow)&&angular.isFunction(F.onBeforeShow)&&F.onBeforeShow(T);var t,e;F.container?(t=q,e=q[0].lastChild?angular.element(q[0].lastChild):null):(t=null,e=i),W&&P(),I=T.$scope.$new(),W=T.$element=M.link(I,function(t,e){}),W.css({top:'-9999px',left:'-9999px',right:'auto',display:'block',visibility:'hidden'}),F.animation&&W.addClass(F.animation),F.type&&W.addClass(F.prefixClass+'-'+F.type),F.customClass&&W.addClass(F.customClass),e?e.after(W):t.prepend(W),T.$isShown=B.$isShown=!0,h(B),T.$applyPlacement(),angular.version.minor<=2?l.enter(W,t,e,r):l.enter(W,t,e).then(r),h(B),f(function(){W&&W.css({visibility:'visible'}),F.keyboard&&('focus'!==F.trigger&&T.focus(),v())}),F.autoClose&&y()}},T.leave=function(){return clearTimeout(K),O='out',F.delay&&F.delay.hide?void(K=setTimeout(function(){'out'===O&&T.hide()},F.delay.hide)):T.hide()};var V,j;T.hide=function(t){T.$isShown&&(B.$emit(F.prefixEvent+'.hide.before',T),angular.isDefined(F.onBeforeHide)&&angular.isFunction(F.onBeforeHide)&&F.onBeforeHide(T),V=t,j=W,null!==W&&(angular.version.minor<=2?l.leave(W,p):l.leave(W).then(p)),T.$isShown=B.$isShown=!1,h(B),F.keyboard&&null!==W&&b(),F.autoClose&&null!==W&&S())},T.toggle=function(t){t&&t.preventDefault(),T.$isShown?T.leave():T.enter()},T.focus=function(){W[0].focus()},T.setEnabled=function(t){F.bsEnabled=t},T.setViewport=function(t){F.viewport=t},T.$applyPlacement=function(){if(W){var e=F.placement,o=/\s?auto?\s?/i,n=o.test(e);n&&(e=e.replace(o,'')||t.placement),W.addClass(F.placement);var i=E(),a=W.prop('offsetWidth'),r=W.prop('offsetHeight');if(T.$viewport=F.viewport&&d(F.viewport.selector||F.viewport),n){var l=e,s=E(T.$viewport);/bottom/.test(l)&&i.bottom+r>s.bottom?e=l.replace('bottom','top'):/top/.test(l)&&i.top-rs.width&&(e=e.replace('right','left')),W.removeClass(l).addClass(e)}var u=C(e,i,a,r);k(u,e)}},T.$onKeyUp=function(t){27===t.which&&T.$isShown&&(T.hide(),t.stopPropagation())},T.$onFocusKeyUp=function(t){27===t.which&&(i[0].blur(),t.stopPropagation())},T.$onFocusElementMouseDown=function(t){F.mouseDownPreventDefault&&t.preventDefault(),F.mouseDownStopPropagation&&t.stopPropagation(),T.$isShown?i[0].blur():i[0].focus()};var L=!1;return T}function h(t){t.$$phase||t.$root&&t.$root.$$phase||t.$digest()}function d(t,e){return angular.element((e||document).querySelectorAll(t))}var g=/(ip[ao]d|iphone|android)/gi.test(e.navigator.userAgent),m='createTouch'in e.document&&g,$=angular.element(e.document);return p}]}).directive('bsTooltip',['$window','$location','$sce','$parse','$tooltip','$$rAF',function(t,e,o,n,i,a){return{restrict:'EAC',scope:!0,link:function(t,e,n,r){var l,s={scope:t};angular.forEach(['template','templateUrl','controller','controllerAs','titleTemplate','placement','container','delay','trigger','html','animation','backdropAnimation','type','customClass','id'],function(t){angular.isDefined(n[t])&&(s[t]=n[t])});var u=/^(false|0|)$/i;angular.forEach(['html','container'],function(t){angular.isDefined(n[t])&&u.test(n[t])&&(s[t]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide'],function(e){var o='bs'+e.charAt(0).toUpperCase()+e.slice(1);angular.isDefined(n[o])&&(s[e]=t.$eval(n[o]))});var f=e.attr('data-target');angular.isDefined(f)&&(u.test(f)?s.target=!1:s.target=f),t.hasOwnProperty('title')||(t.title=''),n.$observe('title',function(e){if(angular.isDefined(e)||!t.hasOwnProperty('title')){var n=t.title;t.title=o.trustAsHtml(e),angular.isDefined(n)&&a(function(){l&&l.$applyPlacement()})}}),n.$observe('disabled',function(t){t&&l.$isShown&&l.hide()}),n.bsTooltip&&t.$watch(n.bsTooltip,function(e,o){angular.isObject(e)?angular.extend(t,e):t.title=e,angular.isDefined(o)&&a(function(){l&&l.$applyPlacement()})},!0),n.bsShow&&t.$watch(n.bsShow,function(t,e){l&&angular.isDefined(t)&&(angular.isString(t)&&(t=!!t.match(/true|,?(tooltip),?/i)),t===!0?l.show():l.hide())}),n.bsEnabled&&t.$watch(n.bsEnabled,function(t,e){l&&angular.isDefined(t)&&(angular.isString(t)&&(t=!!t.match(/true|1|,?(tooltip),?/i)),t===!1?l.setEnabled(!1):l.setEnabled(!0))}),n.viewport&&t.$watch(n.viewport,function(t){l&&angular.isDefined(t)&&l.setViewport(t)}),l=i(e,s),t.$on('$destroy',function(){l&&l.destroy(),s=null,l=null})}}}]); //# sourceMappingURL=tooltip.min.js.map diff --git a/dist/modules/tooltip.min.js.map b/dist/modules/tooltip.min.js.map index 468b5deec..c4917d063 100644 --- a/dist/modules/tooltip.min.js.map +++ b/dist/modules/tooltip.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/tooltip.js"],"names":["angular","target","provider","placement","this","defaults","templateUrl","template","contentTemplate","trigger","keyboard","html","show","title","type","delay","autoClose","bsEnabled","viewport","selector","padding","$get","String","htmlReplaceRegExp","$body","options","$tooltip","extend","config","split","enterAnimateCallback","scope","$emit","prefixEvent","leaveAnimateCallback","_tipToHide","leave","$isShown","element","blur","tipElement","nodeName","forEach","triggers","on","toggle","unbindTriggerEvents","enter","i","length","off","bindKeyboardEvents","$onKeyUp","isTouch","$onFocusElementMouseDown","unbindKeyboardEvents","_autoCloseEventsBinded","bindAutoCloseEvents","$timeout","hide","unbindAutoCloseEvents","stopEventPropagation","event","getPosition","$element","rect","stopPropagation","width","elRect","height","p","top","left","dimensions","offset","el","scroll","isBody","document","documentElement","$window","getCalculatedOffset","position","actualWidth","actualHeight","outerDims","clientWidth","innerHeight","tip","marginTop","parseInt","using","props","css","isNaN","right","marginLeft","setOffset","delta","getViewportAdjustedDelta","isVertical","replaceArrow","arrowDelta","test","arrowOffsetPosition","viewportDimensions","$viewport","topEdgeOffset","bottomEdgeOffset","viewportPadding","leftEdgeOffset","rightEdgeOffset","dimension","isHorizontal","findElement","$arrow","clearTimeout","timeout","tipScope","$destroy","$$phase","$options","$promise","$bsCompiler","compile","$scope","$new","$rootScope","toLowerCase","$id","attr","map","parseFloat","$sce","trustAsHtml","$setEnabled","$$postDigest","id","$hide","setEnabled","isEnabled","compileData","promise","then","init","container","tipContainer","bindTriggerEvents","isElement","destroyTipElement","hoverState","parent","after","lastChild","display","visibility","link","clonedElement","safeDigest","version","minor","addClass","animation","$animate","customClass","$$rAF","$applyPlacement","focus","_blur","replace","elementPosition","autoPlace","autoToken","viewportPosition","originalPlacement","indexOf","tipHeight","removeClass","tipPosition","applyPlacement","tipWidth","evt","preventDefault","fetchTemplate","fetchPromises","cache","$templateCache","res","isString","directive","$location","restrict","falseValueRegExp","transclusion","dataTarget","key","isDefined","hasOwnProperty","oldValue","tooltip","$observe","newValue","bsTooltip","$watch","isObject","bsShow","match","setViewport","$on","destroy"],"mappings":"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","file":"modules/tooltip.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.tooltip', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$tooltip', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n prefixClass: 'tooltip',\n prefixEvent: 'tooltip',\n container: false,\n target: false,\n placement: 'top',\n templateUrl: 'tooltip/tooltip.tpl.html',\n template: '',\n contentTemplate: false,\n trigger: 'hover focus',\n keyboard: false,\n html: false,\n show: false,\n title: '',\n type: '',\n delay: 0,\n autoClose: false,\n bsEnabled: true,\n viewport: {\n selector: 'body',\n padding: 0\n }\n };\n\n this.$get = function($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) {\n\n var trim = String.prototype.trim;\n var isTouch = 'createTouch' in $window.document;\n var htmlReplaceRegExp = /ng-bind=\"/ig;\n var $body = angular.element($window.document);\n\n function TooltipFactory(element, config) {\n\n var $tooltip = {};\n\n // Common vars\n var options = $tooltip.$options = angular.extend({}, defaults, config);\n var promise = $tooltip.$promise = $bsCompiler.compile(options);\n var scope = $tooltip.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n var nodeName = element[0].nodeName.toLowerCase();\n if(options.delay && angular.isString(options.delay)) {\n var split = options.delay.split(',').map(parseFloat);\n options.delay = split.length > 1 ? {show: split[0], hide: split[1]} : split[0];\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $tooltip.$id = options.id || element.attr('id') || '';\n\n // Support scope as string options\n if(options.title) {\n scope.title = $sce.trustAsHtml(options.title);\n }\n\n // Provide scope helpers\n scope.$setEnabled = function(isEnabled) {\n scope.$$postDigest(function() {\n $tooltip.setEnabled(isEnabled);\n });\n };\n scope.$hide = function() {\n scope.$$postDigest(function() {\n $tooltip.hide();\n });\n };\n scope.$show = function() {\n scope.$$postDigest(function() {\n $tooltip.show();\n });\n };\n scope.$toggle = function() {\n scope.$$postDigest(function() {\n $tooltip.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $tooltip.$isShown = scope.$isShown = false;\n\n // Private vars\n var timeout, hoverState;\n\n // Fetch, compile then initialize tooltip\n var compileData, tipElement, tipContainer, tipScope;\n promise.then(function(data) {\n compileData = data;\n $tooltip.init();\n });\n\n $tooltip.init = function() {\n\n // Options: delay\n if (options.delay && angular.isNumber(options.delay)) {\n options.delay = {\n show: options.delay,\n hide: options.delay\n };\n }\n\n // Replace trigger on touch devices ?\n // if(isTouch && options.trigger === defaults.trigger) {\n // options.trigger.replace(/hover/g, 'click');\n // }\n\n // Options : container\n if(options.container === 'self') {\n tipContainer = element;\n } else if(angular.isElement(options.container)) {\n tipContainer = options.container;\n } else if(options.container) {\n tipContainer = findElement(options.container);\n }\n\n // Options: trigger\n bindTriggerEvents();\n\n // Options: target\n if(options.target) {\n options.target = angular.isElement(options.target) ? options.target : findElement(options.target);\n }\n\n // Options: show\n if(options.show) {\n scope.$$postDigest(function() {\n options.trigger === 'focus' ? element[0].focus() : $tooltip.show();\n });\n }\n\n };\n\n $tooltip.destroy = function() {\n\n // Unbind events\n unbindTriggerEvents();\n\n // Remove element\n destroyTipElement();\n\n // Destroy scope\n scope.$destroy();\n\n };\n\n $tooltip.enter = function() {\n\n clearTimeout(timeout);\n hoverState = 'in';\n if (!options.delay || !options.delay.show) {\n return $tooltip.show();\n }\n\n timeout = setTimeout(function() {\n if (hoverState ==='in') $tooltip.show();\n }, options.delay.show);\n\n };\n\n $tooltip.show = function() {\n if (!options.bsEnabled || $tooltip.$isShown) return;\n\n scope.$emit(options.prefixEvent + '.show.before', $tooltip);\n var parent, after;\n if (options.container) {\n parent = tipContainer;\n if (tipContainer[0].lastChild) {\n after = angular.element(tipContainer[0].lastChild);\n } else {\n after = null;\n }\n } else {\n parent = null;\n after = element;\n }\n\n\n // Hide any existing tipElement\n if(tipElement) destroyTipElement();\n // Fetch a cloned element linked from template\n tipScope = $tooltip.$scope.$new();\n tipElement = $tooltip.$element = compileData.link(tipScope, function(clonedElement, scope) {});\n\n // Set the initial positioning. Make the tooltip invisible\n // so IE doesn't try to focus on it off screen.\n tipElement.css({top: '-9999px', left: '-9999px', right: 'auto', display: 'block', visibility: 'hidden'});\n\n // Options: animation\n if(options.animation) tipElement.addClass(options.animation);\n // Options: type\n if(options.type) tipElement.addClass(options.prefixClass + '-' + options.type);\n // Options: custom classes\n if(options.customClass) tipElement.addClass(options.customClass);\n\n // Append the element, without any animations. If we append\n // using $animate.enter, some of the animations cause the placement\n // to be off due to the transforms.\n after ? after.after(tipElement) : parent.prepend(tipElement);\n\n $tooltip.$isShown = scope.$isShown = true;\n safeDigest(scope);\n\n // Now, apply placement\n $tooltip.$applyPlacement();\n\n // Once placed, animate it.\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.enter(tipElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(tipElement, parent, after).then(enterAnimateCallback);\n }\n safeDigest(scope);\n\n $$rAF(function () {\n // Once the tooltip is placed and the animation starts, make the tooltip visible\n if(tipElement) tipElement.css({visibility: 'visible'});\n\n // Bind events\n if(options.keyboard) {\n if(options.trigger !== 'focus') {\n $tooltip.focus();\n }\n bindKeyboardEvents();\n }\n });\n\n if(options.autoClose) {\n bindAutoCloseEvents();\n }\n\n };\n\n function enterAnimateCallback() {\n scope.$emit(options.prefixEvent + '.show', $tooltip);\n }\n\n $tooltip.leave = function() {\n\n clearTimeout(timeout);\n hoverState = 'out';\n if (!options.delay || !options.delay.hide) {\n return $tooltip.hide();\n }\n timeout = setTimeout(function () {\n if (hoverState === 'out') {\n $tooltip.hide();\n }\n }, options.delay.hide);\n\n };\n\n var _blur;\n var _tipToHide;\n $tooltip.hide = function(blur) {\n\n if(!$tooltip.$isShown) return;\n scope.$emit(options.prefixEvent + '.hide.before', $tooltip);\n\n // store blur value for leaveAnimateCallback to use\n _blur = blur;\n\n // store current tipElement reference to use\n // in leaveAnimateCallback\n _tipToHide = tipElement;\n\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if(angular.version.minor <= 2) {\n $animate.leave(tipElement, leaveAnimateCallback);\n } else {\n $animate.leave(tipElement).then(leaveAnimateCallback);\n }\n\n $tooltip.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n if(options.keyboard && tipElement !== null) {\n unbindKeyboardEvents();\n }\n\n if(options.autoClose && tipElement !== null) {\n unbindAutoCloseEvents();\n }\n };\n\n function leaveAnimateCallback() {\n scope.$emit(options.prefixEvent + '.hide', $tooltip);\n\n // check if current tipElement still references\n // the same element when hide was called\n if (tipElement === _tipToHide) {\n // Allow to blur the input when hidden, like when pressing enter key\n if(_blur && options.trigger === 'focus') {\n return element[0].blur();\n }\n\n // clean up child scopes\n destroyTipElement();\n }\n }\n\n $tooltip.toggle = function() {\n $tooltip.$isShown ? $tooltip.leave() : $tooltip.enter();\n };\n\n $tooltip.focus = function() {\n tipElement[0].focus();\n };\n\n $tooltip.setEnabled = function(isEnabled) {\n options.bsEnabled = isEnabled;\n };\n\n $tooltip.setViewport = function(viewport) {\n options.viewport = viewport;\n };\n\n // Protected methods\n\n $tooltip.$applyPlacement = function() {\n if(!tipElement) return;\n\n // Determine if we're doing an auto or normal placement\n var placement = options.placement,\n autoToken = /\\s?auto?\\s?/i,\n autoPlace = autoToken.test(placement);\n\n if (autoPlace) {\n placement = placement.replace(autoToken, '') || defaults.placement;\n }\n\n // Need to add the position class before we get\n // the offsets\n tipElement.addClass(options.placement);\n\n // Get the position of the target element\n // and the height and width of the tooltip so we can center it.\n var elementPosition = getPosition(),\n tipWidth = tipElement.prop('offsetWidth'),\n tipHeight = tipElement.prop('offsetHeight');\n\n // Refresh viewport position\n $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport);\n\n // If we're auto placing, we need to check the positioning\n if (autoPlace) {\n var originalPlacement = placement;\n var viewportPosition = getPosition($tooltip.$viewport);\n\n // Determine if the vertical placement\n if (originalPlacement.indexOf('bottom') >= 0 && elementPosition.bottom + tipHeight > viewportPosition.bottom) {\n placement = originalPlacement.replace('bottom', 'top');\n } else if (originalPlacement.indexOf('top') >= 0 && elementPosition.top - tipHeight < viewportPosition.top) {\n placement = originalPlacement.replace('top', 'bottom');\n }\n\n // Determine the horizontal placement\n // The exotic placements of left and right are opposite of the standard placements. Their arrows are put on the left/right\n // and flow in the opposite direction of their placement.\n if ((originalPlacement === 'right' || originalPlacement === 'bottom-left' || originalPlacement === 'top-left') &&\n elementPosition.right + tipWidth > viewportPosition.width) {\n\n placement = originalPlacement === 'right' ? 'left' : placement.replace('left', 'right');\n } else if ((originalPlacement === 'left' || originalPlacement === 'bottom-right' || originalPlacement === 'top-right') &&\n elementPosition.left - tipWidth < viewportPosition.left) {\n\n placement = originalPlacement === 'left' ? 'right' : placement.replace('right', 'left');\n }\n\n tipElement.removeClass(originalPlacement).addClass(placement);\n }\n\n // Get the tooltip's top and left coordinates to center it with this directive.\n var tipPosition = getCalculatedOffset(placement, elementPosition, tipWidth, tipHeight);\n applyPlacement(tipPosition, placement);\n };\n\n $tooltip.$onKeyUp = function(evt) {\n if (evt.which === 27 && $tooltip.$isShown) {\n $tooltip.hide();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusKeyUp = function(evt) {\n if (evt.which === 27) {\n element[0].blur();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusElementMouseDown = function(evt) {\n evt.preventDefault();\n evt.stopPropagation();\n // Some browsers do not auto-focus buttons (eg. Safari)\n $tooltip.$isShown ? element[0].blur() : element[0].focus();\n };\n\n // bind/unbind events\n function bindTriggerEvents() {\n var triggers = options.trigger.split(' ');\n angular.forEach(triggers, function(trigger) {\n if(trigger === 'click') {\n element.on('click', $tooltip.toggle);\n } else if(trigger !== 'manual') {\n element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n nodeName === 'button' && trigger !== 'hover' && element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n });\n }\n\n function unbindTriggerEvents() {\n var triggers = options.trigger.split(' ');\n for (var i = triggers.length; i--;) {\n var trigger = triggers[i];\n if(trigger === 'click') {\n element.off('click', $tooltip.toggle);\n } else if(trigger !== 'manual') {\n element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n nodeName === 'button' && trigger !== 'hover' && element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n }\n\n function bindKeyboardEvents() {\n if(options.trigger !== 'focus') {\n tipElement.on('keyup', $tooltip.$onKeyUp);\n } else {\n element.on('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n function unbindKeyboardEvents() {\n if(options.trigger !== 'focus') {\n tipElement.off('keyup', $tooltip.$onKeyUp);\n } else {\n element.off('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n var _autoCloseEventsBinded = false;\n function bindAutoCloseEvents() {\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function() {\n // Stop propagation when clicking inside tooltip\n tipElement.on('click', stopEventPropagation);\n\n // Hide when clicking outside tooltip\n $body.on('click', $tooltip.hide);\n\n _autoCloseEventsBinded = true;\n }, 0, false);\n }\n\n function unbindAutoCloseEvents() {\n if (_autoCloseEventsBinded) {\n tipElement.off('click', stopEventPropagation);\n $body.off('click', $tooltip.hide);\n _autoCloseEventsBinded = false;\n }\n }\n\n function stopEventPropagation(event) {\n event.stopPropagation();\n }\n\n // Private methods\n\n function getPosition($element) {\n $element = $element || (options.target || element);\n\n var el = $element[0],\n isBody = el.tagName === 'BODY';\n\n var elRect = el.getBoundingClientRect();\n var rect = {};\n\n // IE8 has issues with angular.extend and using elRect directly.\n // By coping the values of elRect into a new object, we can continue to use extend\n for (var p in elRect) {\n // DO NOT use hasOwnProperty when inspecting the return of getBoundingClientRect.\n rect[p] = elRect[p];\n }\n\n if (rect.width === null) {\n // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093\n rect = angular.extend({}, rect, { width: elRect.right - elRect.left, height: elRect.bottom - elRect.top });\n }\n var elOffset = isBody ? { top: 0, left: 0 } : dimensions.offset(el),\n scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0 },\n outerDims = isBody ? { width: document.documentElement.clientWidth, height: $window.innerHeight } : null;\n\n return angular.extend({}, rect, scroll, outerDims, elOffset);\n }\n\n function getCalculatedOffset(placement, position, actualWidth, actualHeight) {\n var offset;\n var split = placement.split('-');\n\n switch (split[0]) {\n case 'right':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left + position.width\n };\n break;\n case 'bottom':\n offset = {\n top: position.top + position.height,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n case 'left':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left - actualWidth\n };\n break;\n default:\n offset = {\n top: position.top - actualHeight,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n }\n\n if(!split[1]) {\n return offset;\n }\n\n // Add support for corners @todo css\n if(split[0] === 'top' || split[0] === 'bottom') {\n switch (split[1]) {\n case 'left':\n offset.left = position.left;\n break;\n case 'right':\n offset.left = position.left + position.width - actualWidth;\n }\n } else if(split[0] === 'left' || split[0] === 'right') {\n switch (split[1]) {\n case 'top':\n offset.top = position.top - actualHeight;\n break;\n case 'bottom':\n offset.top = position.top + position.height;\n }\n }\n\n return offset;\n }\n\n function applyPlacement(offset, placement) {\n var tip = tipElement[0],\n width = tip.offsetWidth,\n height = tip.offsetHeight;\n\n // manually read margins because getBoundingClientRect includes difference\n var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10),\n marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10);\n\n // we must check for NaN for ie 8/9\n if (isNaN(marginTop)) marginTop = 0;\n if (isNaN(marginLeft)) marginLeft = 0;\n\n offset.top = offset.top + marginTop;\n offset.left = offset.left + marginLeft;\n\n // dimensions setOffset doesn't round pixel values\n // so we use setOffset directly with our own function\n dimensions.setOffset(tip, angular.extend({\n using: function (props) {\n tipElement.css({\n top: Math.round(props.top) + 'px',\n left: Math.round(props.left) + 'px',\n right: ''\n });\n }\n }, offset), 0);\n\n // check to see if placing tip in new offset caused the tip to resize itself\n var actualWidth = tip.offsetWidth,\n actualHeight = tip.offsetHeight;\n\n if (placement === 'top' && actualHeight !== height) {\n offset.top = offset.top + height - actualHeight;\n }\n\n // If it's an exotic placement, exit now instead of\n // applying a delta and changing the arrow\n if (/top-left|top-right|bottom-left|bottom-right/.test(placement)) return;\n\n var delta = getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight);\n\n if (delta.left) {\n offset.left += delta.left;\n } else {\n offset.top += delta.top;\n }\n\n dimensions.setOffset(tip, offset);\n\n if (/top|right|bottom|left/.test(placement)) {\n var isVertical = /top|bottom/.test(placement),\n arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight,\n arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight';\n\n replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical);\n }\n }\n\n // @source https://github.com/twbs/bootstrap/blob/v3.3.5/js/tooltip.js#L380\n function getViewportAdjustedDelta(placement, position, actualWidth, actualHeight) {\n var delta = {top: 0, left: 0};\n if (!$tooltip.$viewport) return delta;\n\n var viewportPadding = options.viewport && options.viewport.padding || 0;\n var viewportDimensions = getPosition($tooltip.$viewport);\n\n if (/right|left/.test(placement)) {\n var topEdgeOffset = position.top - viewportPadding - viewportDimensions.scroll;\n var bottomEdgeOffset = position.top + viewportPadding - viewportDimensions.scroll + actualHeight;\n if (topEdgeOffset < viewportDimensions.top) { // top overflow\n delta.top = viewportDimensions.top - topEdgeOffset;\n } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow\n delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset;\n }\n } else {\n var leftEdgeOffset = position.left - viewportPadding;\n var rightEdgeOffset = position.left + viewportPadding + actualWidth;\n if (leftEdgeOffset < viewportDimensions.left) { // left overflow\n delta.left = viewportDimensions.left - leftEdgeOffset;\n } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow\n delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset;\n }\n }\n\n return delta;\n }\n\n function replaceArrow(delta, dimension, isHorizontal) {\n var $arrow = findElement('.tooltip-arrow, .arrow', tipElement[0]);\n\n $arrow.css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')\n .css(isHorizontal ? 'top' : 'left', '');\n }\n\n function destroyTipElement() {\n // Cancel pending callbacks\n clearTimeout(timeout);\n\n if($tooltip.$isShown && tipElement !== null) {\n if(options.autoClose) {\n unbindAutoCloseEvents();\n }\n\n if(options.keyboard) {\n unbindKeyboardEvents();\n }\n }\n\n if(tipScope) {\n tipScope.$destroy();\n tipScope = null;\n }\n\n if(tipElement) {\n tipElement.remove();\n tipElement = $tooltip.$element = null;\n }\n }\n\n return $tooltip;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n function findElement(query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n var fetchPromises = {};\n function fetchTemplate(template) {\n if(fetchPromises[template]) return fetchPromises[template];\n return (fetchPromises[template] = $http.get(template, {cache: $templateCache}).then(function(res) {\n return res.data;\n }));\n }\n\n return TooltipFactory;\n\n };\n\n })\n\n .directive('bsTooltip', function($window, $location, $sce, $tooltip, $$rAF) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink(scope, element, attr, transclusion) {\n\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'contentTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id'], function(key) {\n if(angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function(key) {\n if(angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key]))\n options[key] = false;\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if(angular.isDefined(dataTarget)) {\n if(falseValueRegExp.test(dataTarget))\n options.target = false;\n else\n options.target = dataTarget;\n }\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')){\n scope.title = '';\n }\n\n // Observe scope attributes for change\n attr.$observe('title', function(newValue) {\n if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) {\n var oldValue = scope.title;\n scope.title = $sce.trustAsHtml(newValue);\n angular.isDefined(oldValue) && $$rAF(function() {\n tooltip && tooltip.$applyPlacement();\n });\n }\n });\n\n // Support scope as an object\n attr.bsTooltip && scope.$watch(attr.bsTooltip, function(newValue, oldValue) {\n if(angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.title = newValue;\n }\n angular.isDefined(oldValue) && $$rAF(function() {\n tooltip && tooltip.$applyPlacement();\n });\n }, true);\n\n // Visibility binding support\n attr.bsShow && scope.$watch(attr.bsShow, function(newValue, oldValue) {\n if(!tooltip || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i);\n newValue === true ? tooltip.show() : tooltip.hide();\n });\n\n // Enabled binding support\n attr.bsEnabled && scope.$watch(attr.bsEnabled, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.bsEnabled, newValue, oldValue);\n if(!tooltip || !angular.isDefined(newValue)) return;\n if(angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i);\n newValue === false ? tooltip.setEnabled(false) : tooltip.setEnabled(true);\n });\n\n // Viewport support\n attr.viewport && scope.$watch(attr.viewport, function (newValue) {\n if(!tooltip || !angular.isDefined(newValue)) return;\n tooltip.setViewport(newValue);\n });\n\n // Initialize popover\n var tooltip = $tooltip(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if(tooltip) tooltip.destroy();\n options = null;\n tooltip = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/tooltip.js"],"names":["angular","target","provider","placement","this","defaults","templateUrl","template","titleTemplate","trigger","keyboard","html","show","title","type","delay","autoClose","bsEnabled","mouseDownPreventDefault","mouseDownStopPropagation","viewport","selector","padding","TooltipFactory","$get","promise","$tooltip","$bsCompiler","compile","options","split","clearTimeout","hoverState","prefixEvent","hide","isFunction","onShow","leaveAnimateCallback","scope","$emit","destroyTipElement","onHide","_tipToHide","evt","preventDefault","element","triggers","nodeName","on","isTouch","toggle","enter","unbindTriggerEvents","length","$onFocusElementMouseDown","off","i","bindKeyboardEvents","tipElement","$onKeyUp","unbindKeyboardEvents","_autoCloseEventsBinded","bindAutoCloseEvents","$timeout","$body","unbindAutoCloseEvents","stopEventPropagation","event","getPosition","$element","elRect","getBoundingClientRect","rect","p","width","height","top","left","extend","dimensions","offset","el","scroll","outerDims","isBody","document","documentElement","scrollTop","body","prop","clientWidth","innerHeight","position","actualWidth","actualHeight","marginLeft","isNaN","setOffset","using","parseInt","props","css","right","marginTop","tip","offsetHeight","getViewportAdjustedDelta","delta","offsetWidth","isVertical","arrowDelta","replaceArrow","arrowOffsetPosition","viewportDimensions","$viewport","topEdgeOffset","bottomEdgeOffset","viewportPadding","test","leftEdgeOffset","rightEdgeOffset","dimension","isHorizontal","findElement","$arrow","timeout","$isShown","tipScope","$destroy","$$phase","$options","config","$promise","$scope","$new","$rootScope","toLowerCase","$id","attr","map","parseFloat","$sce","trustAsHtml","$setEnabled","$$postDigest","id","$hide","setEnabled","isEnabled","compileData","then","data","tipContainer","init","container","bindTriggerEvents","isElement","destroy","focus","isDefined","onBeforeShow","after","parent","lastChild","display","visibility","link","clonedElement","animation","addClass","prefixClass","safeDigest","version","prepend","$$rAF","$applyPlacement","$animate","enterAnimateCallback","_blur","onBeforeHide","blur","leave","autoPlace","autoToken","elementPosition","tipWidth","tipHeight","replace","viewportPosition","originalPlacement","removeClass","tipPosition","getCalculatedOffset","applyPlacement","stopPropagation","forEach","directive","$root","$digest","restrict","query","querySelectorAll","isNative","isString","tooltip","transclusion","falseValueRegExp","key","$eval","bsKey","dataTarget","slice","hasOwnProperty","oldValue","$observe","newValue","bsTooltip","bsShow","$watch","match","setViewport","$on"],"mappings":"AAOA,YAEAA,SAGMC,OAAQ,0BAAA,sBAAA,sCAAAC,SAAA,WAAA,WAFZ,GAGIC,GAAWC,KAAAC,UACXC,UAAAA,UACAC,YAAU,GACVC,YAAAA,UACAC,YAAS,UACTC,WAAU,EACVC,QAAM,EACNC,UAAM,MACNC,YAAO,2BACPC,SAAM,GACNC,eAAO,EACPC,QAAAA,cACAC,UAAAA,EACAC,MAAAA,EACAC,MAAAA,EACAC,MAAAA,GAFFN,KAGIO,GAFJN,MAGIO,EAFJN,WAAW,EACXC,WAAW,EAKXb,yBAAK,EAEHe,0BAAe,EACfC,UACAC,SAAYrB,OAEZsB,QAASC,GAHbnB,MASMoB,MAAIC,UAAUC,aAAoBC,cAAYC,KAAQC,iBAAAA,QAAAA,WAAAA,OAAAA,aAAAA,QAAAA,WAAAA,SAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,GAL1D,QAUUC,GAAgBf,EAAMe,GA8I9B,QA0EIC,KACAC,EAAAA,MAAAA,EAAaC,YAAA,QAAAP,GACb1B,QAAK6B,UAAQd,EAAUc,SAAcK,QAAMC,WAAAN,EAAAO,SACzCP,EAAOH,OAAAA,GAjCb,QA8FQW,KAQNX,GArGAY,EAAMC,MAAMV,EAAQI,YAAc,QAASP,GAiGvCc,QAAAA,UAAAA,EAAAA,SAAAA,QAAAA,WAAAA,EAAAA,SA/FFX,EAAQY,OAAOf,GAmGjBA,IAAkBgB,EAAUC,CAC1B,GAAIA,GAAK,UAAAd,EAAApB,QAAEkC,MAAIC,GAAAA,GAAAA,MACfJ,MAjBJ,QAyHQK,KAxHN,GAyHMC,GAAIC,EAAatC,QAAAqB,MAAYrB,IAxHnCT,SAyHQ6C,QAAQG,EAAGC,SAAUxC,GAxHX,UAAZA,GAAmC,gBAAZA,EACzBoC,EAAQG,GAAGvC,EAASiB,EAASwB,QACR,WAAZzC,IACToC,EAAQG,GAAe,UAAZvC,EAAsB,aAAe,QAASiB,EAASyB,OA2HtEN,EAASO,GAAAA,UAAAA,EAAAA,aAAAA,OAAAA,EAAAA,OACQvB,WAAXiB,GAAiC,UAANhB,GAC1Be,EAAQC,GAAAA,EAASO,aAAc,YAAA3B,EAAA4B,6BArHxC,QA0HQT,KAxHN,IA0HM,GADAA,GAAQU,EAAI9C,QAAYqB,MAAA,KACpBiB,EAAAA,EAAaM,OAAAG,KAAY/C,CAzHjC,GA0HMoC,GAAQU,EAAIN,EAzHF,WAAZxC,GAAmC,gBAAZA,EACzBoC,EAAQU,IAAI9C,EAASiB,EAASwB,QACT,WAAZzC,IACToC,EAAQU,IAAgB,UAAZ9C,EAAsB,aAAe,QAASiB,EAASyB,OA4HvEN,EAASY,IAAAA,UAAAA,EAAAA,aAAAA,OAAAA,EAAAA,OACKhD,WAARoB,GAA6B,UAAApB,GAC/BiD,EAAWV,IAAGC,EAASvB,aAASiC,YAAAA,EAAAA,4BAMpC,QAASC,KACiB,UAApB/B,EAAQpB,QACViD,EAAWH,GAAAA,QAAI7B,EAASA,UAExBmB,EAAQU,GAAAA,QAAI7B,EAASA,eAIzB,QAAImC,KACKC,UAATjC,EAASiC,QAGPC,EAASR,IAAA,QAAA7B,EAAAiC,UAKPK,EAAMhB,IAAG,QAAStB,EAASQ,eAM/B,QAAS+B,KACPF,EAAIF,WACFH,EAAWH,GAAAA,QAAIW,GACfF,EAAMT,GAAAA,QAAI7B,EAASA,MACnBmC,GAAyB,GAnI1B,GAAG,GAuIN,QAASK,KACPC,IApIAT,EAAWH,IAAI,QAASW,GAyI1BF,EAASI,IAAAA,QAAaC,EAAAA,MACpBA,GAAwBxC,GArI5B,QA0IQyC,GAAYC,GAChBJ,EAAIK,kBAxIR,QA+IWC,GAAYA,GA9IrBJ,EAAWA,GAAaxC,EAAQ5B,QAAU4C,CAkJxC,IAAI2B,GAAKE,EAAAA,GAEPF,EAA0BA,SAAnBxE,EAAAA,QAA0B0E,EAAOJ,EAAAA,wBAA4BK,IA/IxE,KAAK,GAAIF,KAAKH,GACZE,EAAKC,GAAKH,EAAOG,EAgJa,QAALG,EAAKF,QAAGG,EAAM7E,QAAA8E,UAAAN,GAAKO,MAAAA,EAAWC,MAAOC,EAAAA,KAC1DC,OAAAA,EAAAA,OAAAA,EAAAA,MACJ,IAAIC,GAAAA,GAAsBT,IAAAA,EAA6CC,KAAAA,GAA+BI,EAAAC,OAAAC,GAEtGC,GAvIAA,OAAQE,EAASC,SAASC,gBAAgBC,WAAaF,SAASG,KAAKD,UAAYlB,EAASoB,KAAK,cAAgB,GA2I3GT,EAAAA,GACJN,MAAI5C,SAAQ3B,gBAAgBuF,YAE5Bf,OAAQ7C,EAAM6D,aACZ,IAzIJ,OA0IMX,SAAAA,UAAAA,EAAAA,EAAAA,EAAAA,GAxIR,QA0IUH,GAAsBe,EAASlB,EAAAA,EAAAA,GAzIvC,GAAIM,GA2IElD,EAAA3B,EAAA2B,MAAA,IAzIN,QAAQA,EAAM,IA0IV,IAAK,QACHkD,GACEJ,IAAKgB,EAAShB,IAAMgB,EAASjB,OAAAA,EAAAA,EAAAA,EAC7BE,KAAMe,EAASf,KAAOe,EAASlB,MAEjC,MACF,KAAK,SACHM,GACEJ,IAAKgB,EAAShB,IAAMgB,EAASjB,OAC7BE,KAAMe,EAASf,KAAOgB,EAAAA,MAAAA,EAAAA,EAAAA,EAExB,MACF,KAAA,OACEb,GACEJ,IAAKgB,EAAShB,IAAMkB,EAAAA,OAAAA,EAAAA,EAAAA,EACpBjB,KAAMe,EAASf,KAAOe,EAExB,MAGJ,SACEZ,GAvIAJ,IAAKgB,EAAShB,IAAMkB,EA2IlBhE,KAAM8D,EAAOf,KAAS/C,EAAM4C,MAAO,EAAAmB,EAAU,GAtInD,IA0IQ/D,EAAA,GAzIN,MAAOkD,EAET,IAyIeH,QAAPG,EAAAA,IAA8BY,WAAhBA,EAASf,GAxI7B,OAyIM/C,EAAA,IAxIL,IAAK,OAyIFkD,EAAAH,KAAAe,EAAAf,IACE,MAtIL,KAwIM,QACLG,EAAQlD,KAAM8D,EAAAf,KAAAe,EAAAlB,MAAAmB,MAKVb,IAAaY,SAANhB,EAAMgB,IAAShB,UAAAA,EAAAA,GAtI5B,OAuIM9C,EAAA,IAtIL,IAAK,MAuIFkD,EAAAJ,IAAAgB,EAAAhB,IAAAkB,EAAAF,EAAAjB,MACE,MApIL,KAAK,SAwINK,EAAOA,IAAAA,EAAAA,IAUP,MAAIe,GAxIR,QA4IQC,GAAMD,EAAaA,GAEvBf,GAAAA,GAAOJ,EAAaA,GACpBI,EAAOH,EAAOG,YAIdD,EAAWkB,EAAAA,aACTC,EAAOC,SAAUC,EAAAA,IAAAA,EAAAA,cAAAA,IACf1C,EAAW2C,SAAAA,EAAAA,IAAAA,EAAAA,eAAAA,GA/IbL,OAgJIpB,KAAgBwB,EAAMxB,GA/I1BoB,MAgJInB,KAAiBuB,EAAMvB,GA/I/BG,EAgJQsB,IAAOtB,EAAAJ,IAAA2B,EA/IfvB,EAAOH,KAAOG,EAAOH,KAAOkB,EAC5BhB,EAAWkB,UAAUO,EAAKxG,QAAQ8E,QAChCoB,MAgJGlB,SAASoB,GAGRP,EAAAA,KACAC,IAAAA,KAAAA,MAAeU,EAAIC,KAAAA,KAEnBtG,KAAAA,KAAAA,MAAciG,EAASN,MAAAA,KACzBd,MAAOJ,OAOTI,GAAY0B,EAEZ,IAAIC,GAAYH,EAAAI,YACd5B,EAAe2B,EAAM9B,YAKvBE,IAJO,QAxJL5E,GAwJK2F,IAAAnB,IACLK,EAAOJ,IAAAA,EAAaA,IAAAA,EAAAA,IAGtBG,8CAA0BC,KAAAA,GAA1BD,CAEA,GAAI4B,GAAAD,EAA6BvG,EAAY6E,EAAAa,EAAAC,EAU/C,IATIa,EAAIE,KACJ7B,EAAI8B,MAAAA,EAAaD,KAGjBE,EAAAA,KAAaD,EAAAA,IAzJjB/B,EAAWkB,UAAUO,EAAKxB,GA8J1B,wBAAS0B,KAA0BvG,GAAWyF,CAC5C,GAAIe,GAAAA,aAAAA,KAAAA,GAAS/B,EAAKiC,EAAA,EAAAF,EAAA9B,KAAAH,EAAAmB,EAAA,EAAAc,EAAA/B,IAAAD,EAAAmB,EAAGjB,EAAMgC,EAAA,cAAA,cA1J3BE,GAAaD,EAAYN,EAAIQ,GAAsBH,KAGvD,QA2JQI,GAAqB7C,EAAY1C,EAASwF,EAAAA,GAE9C,GAAIP,IACF/B,IAAIuC,EACJtC,KAAIuC,EA1JR,KA4JMT,EAAM/B,UAAMqC,MAAAA,EA3JlB,IAAII,GA4JWD,EAAAA,UAAmBH,EAAAA,SAAyBA,SAAAA,EACrDN,EAAYM,EAAmBrC,EAAMqC,UA3J3C,IAAI,aAAaK,KAAKnH,GAAY,CAChC,GAAIgH,GA4JGvB,EAAAhB,IAAAyC,EAAAJ,EAAA/B,OACDqC,EAAiB3B,EAASf,IAAOwC,EAAAA,EAAAA,OAAAA,CACjCG,GAAAA,EAAkCH,IACtCV,EAAIY,IAAAA,EAAiBN,IAAAA,EACbpC,EAAOoC,EAA0BM,IAAAA,EAAAA,SA3JzCZ,EA4JO/B,IAAI4C,EAAkBP,IAAAA,EAA0BtC,OAAAyC,OA1JpD,CACL,GAAIG,GAAiB3B,EAASf,KAAOwC,EA8JrCG,EAAOb,EAAAA,KAAAA,EAAAA,CA5JHY,GAAiBN,EAAmBpC,KA+J1C8B,EAASI,KAAAA,EAAqBU,KAAWC,EAC1BC,EAAYV,EAAAX,QAEzBsB,EAAOvB,KAAIqB,EAAwB7C,KAAOoC,EAAkBQ,MAAAA,GAM5D1F,MAAAA,GAhKJ,QAmKUF,GAAQb,EAAWyG,EAAAC,GAlK3B,GAmKMzD,GAAAA,EAAAA,yBAAAA,EAAAA,GAlKN2D,GAAOvB,IAAIqB,EAAe,OAAS,MAAO,IAAM,EAAIf,EAAQc,GAAa,KAAKpB,IAAIqB,EAAe,MAAQ,OAAQ,IAEnH,QAoKQ9D,KAnKN7B,aAAa8F,GACTnG,EAASoG,UAA2B,OAAfpE,IAsKnBqE,EAAU/G,WACZ+G,IAnKElG,EAAQnB,UAuKRgD,KAnKFqE,IACFA,EAASC,WAwKXD,EAAOrG,MAMTgC,IAEEpB,EAAM2F,SA3KJvE,EAAahC,EAAS2C,SAAW,MAzgBrC,GAUIxC,MAAoCjB,EAAMkB,EAAMoG,SAAAlI,QAAA8E,UAAAzE,EAAA8H,GAAIjG,EAAMJ,EAAMsG,SAAAzG,EAAAC,QAAAC,GAPhES,EAO4EZ,EAAA2G,OAAAxG,EAAAS,OAAAT,EAAAS,MAAAgG,QAAAC,EAAAD,OAN5EvF,EAAWF,EAAQ,GAAGE,SAASyF,aAYjC9G,IAAAA,EAAS+G,OAAM5G,QAAcgB,SAAQ6F,EAAK3H,OAAS,CAGnD,GAAIc,GAAQhB,EAAOE,MAAAe,MAAA,KAAA6G,IAAAC,WACjBtG,GAAMzB,MAAQgI,EAAKC,OAAYjH,GAZ/BjB,KAAMkB,EAAM,GAgBdQ,KAAMyG,EAAAA,IACJzG,EAAM0G,GAbVtH,EAAS+G,IAAM5G,EAAQoH,IAAMpG,EAAQ6F,KAAK,OAAS,GAC/C7G,EAAQhB,QAgBVyB,EAAM4G,MAAQL,EAAAC,YAAAjH,EAAAhB,QAbhByB,EAeMZ,YAASQ,SAAAA,GAdbI,EAAM0G,aAAa,WACjBtH,EAASyH,WAAWC,MAGxB9G,EAeMZ,MAASd,WAdb0B,EAAM0G,aAAa,WACjBtH,EAASQ,UAGbI,EAeMZ,MAASwB,WAdbZ,EAAM0G,aAAa,WACjBtH,EAASd,UAqBX0B,EAAIN,QAAAA,WAGJM,EAAI+G,aAAAA,WACA3F,EAAAA,YAGJjC,EAAQ6H,SAAKhH,EAAUiH,UAAAA,CAnBzB,IAoBIF,GACA3H,EAnBA2H,EAsBF3H,EAGE8H,EACE3H,CAtBNJ,GAuBQb,KAAMiB,SAAQd,GAtBpBsI,EAuBYxH,EAtBZH,EAAS+H,SAEX/H,EA8BQG,KAAQ6H,WACVF,EAAAA,OAAe3G,QAAAA,SAAAA,EAAAA,SA7BjBhB,EA8BOd,OACLyI,KAAAA,EAAe3H,MA7BfK,KA8BKL,EAAIA,QAKX8H,SAAAA,EAAAA,UAGAH,EAAYvJ,EACFA,QAASD,UAAQ4J,EAAU/H,WAjCrC2H,EAAe3H,EAAQ6H,UAqCnB7H,EAAc6H,YAChBpH,EAAM0G,EAAanH,EAAA6H,YAlCvBC,IACI9H,EAoCS5B,SAnCX4B,EAoCMH,OAASd,QAAAA,UAAAA,EAAAA,QAAAA,EAAAA,OAAAA,EAAAA,EAAAA,SAlCbiB,EAAQjB,MACV0B,EAAM0G,aAAa,WACO,UAApBnH,EAAQpB,QAuChBiB,EAASmI,GAAUC,QAMjBtH,EAAAA,UArCJd,EA+CIM,QAAa,WACboB,IA9CFZ,IACAF,EAAM0F,YAERtG,EAgDUM,MAAAA,WAKRN,MApDAK,cAgDahB,GA/CbiB,EAAa,KAmDbN,EAASd,OAAOiB,EAAAd,MAAAH,UAIdiH,EAAI7H,WAAQ+J,WACFC,OAARnI,GAAqBH,EAAAA,QAlDtBG,EAAQd,MAAMH,OA8CViB,EAAQZ,QA5CjBS,EAmDQuI,KAAAA,WACJ,GAAIpI,EAAQ6H,YAAWhI,EAAAoG,SAAvB,CAlDFxF,EAmDI4H,MAASV,EAAAA,YAAAA,eAAAA,GACTxJ,QAAIwJ,UAAgBW,EAAAA,eAAWnK,QAAAmC,WAAAN,EAAAmI,eAlDjCnI,EAmDIoI,aAAgBpH,EAjDtB,IAmDMoH,GAlDFA,CACApI,GAmDK6H,WACLQ,EAASV,EAjDTS,EAkDAA,EAAQpH,GAAAA,UAlDA7C,QAAQ6C,QAAQ2G,EAAa,GAAGW,WAyD/BzI,OAKKkD,EAAK,KAAWC,EAAMhC,GAA0BuH,GAAS5H,IArD3EuF,EAqDoFsC,EAAYhC,OAAAC,OApDhG5E,EAAahC,EAAS2C,SAAWgF,EAAYiB,KAAKvC,EAAU,SAASwC,EAAejI,MAuDlFoB,EAAI7B,KAEJ+C,IAAI/C,UAEJgD,KAAIhD,UAKJyE,MAAI2D,OACFA,QAAMA,QA3DRI,WA4DO,WA1DLxI,EAAQ2I,WAAW9G,EAAW+G,SAAS5I,EAAQ2I,WA8DjD9I,EAASoG,MAAAA,EAAiBA,SAAWjG,EAAA6I,YAAA,IAAA7I,EAAAf,MACrC6J,EAAWrI,aAAAA,EAAAA,SAAAA,EAAAA,aAGXZ,EAKAuI,EAAIjK,MAAQ4K,GAjEZV,EAmEOW,QAAAnH,GAjEThC,EAASoG,SAAWxF,EAAMwF,UAAW,EAoEnC6C,EAAWrI,GAEXwI,EAAMC,kBAEJ/K,QAAI0D,QAAYA,OAAW2C,EApE7B2E,EAoEkCX,MAAY3G,EAAAwG,EAAAD,EAAAgB,GAG5CD,EAAInJ,MAAQnB,EAAUwJ,EAAAD,GAAAX,KAAA2B,GAnE1BN,EAqEQjJ,GApERoJ,EAAM,WAsEArH,GAAAA,EAAAA,KApEF4G,WAAY,YAwEVxI,EAAQb,WACV8C,UAAAA,EAAAA,SArEEpC,EAASoI,QA0EfrG,OAGI5B,EAAQO,WAvEV0B,MASJpC,EA0EUM,MAAAA,WAvER,MAFAD,cA0EeG,GAzEfF,EAAa,MA2ERH,EAAQd,OAAMmB,EAAAA,MAAAA,UAKnB2F,EAAInF,WAAAA,WACY,QAAhBhB,GAEEA,EAAKA,QAELG,EAAI7B,MAAQ+J,OAnFLrI,EAASQ,OAQpB,IAAIgJ,GAgFAA,CA9EJxJ,GAkFIgB,KAAagB,SAAAA,GAETA,EAAAA,WAlFNpB,EAqFQtC,MAAAA,EAAQ4K,YAAoB,eAAAlJ,GAC9BsJ,QAAAA,UAAetH,EAAAA,eAAYrB,QAAAA,WAAAA,EAAAA,eApF/BR,EAqFSsJ,aAAAzJ,GAnFXwJ,EAAQE,EACR1I,EAAagB,EAuFFoE,OAATpG,IACAiJ,QAAWrI,QAAAA,OAAAA,EAGPT,EAAQnB,MAAAA,EAAYgD,GAtFtBsH,EAASK,MAAM3H,GAAY4F,KAAKjH,IAGpCX,EAASoG,SAAWxF,EAAMwF,UAAW,EACrC6C,EAAWrI,GA2FXT,EAASQ,UAAAA,OAAAA,GACPC,IAEET,EAAQY,WAAOf,OAAAA,GAzFjBuC,MAeJvC,EA+FMA,OAASyB,SAAAA,GA9FTR,GACFA,EAAIC,iBAkGJc,EAAWoE,SA/FXpG,EAAS2J,QAmGTxJ,EAAQZ,SA9FZS,EAkGIG,MAAQT,WAjGVsC,EAAW,GAAGoG,SAEhBpI,EAqGSgC,WAAY,SAAA0F,GAGjBvH,EAAI1B,UAAY0B,GArGpBH,EAuGQ4J,YAAYC,SAAUjE,GAE1BzF,EAAIyJ,SAAWlK,GAtGnBM,EAASqJ,gBAAkB,WA4GvBrH,GAAAA,EAAAA,CAIA,GAAI8H,GAAAA,EAAkBpH,UAClBqH,EAAW/H,eACXgI,EAAYhI,EAAAA,KAAW+B,EAG3B/D,KAGAvB,EAAImL,EAAWK,QAAAJ,EAAA,KAAAlL,EAAAF,WAhHjBuD,EAkHQkI,SAAAA,EAAmBxH,UAjH3B,IAmHIoH,GAAkBK,IAChB1L,EAAAA,EAAY0L,KAAAA,eAlHdH,EAmHWhI,EAAWmI,KAAAA,eAjH1B,IADAnK,EAmHMvB,UAAY0L,EAAAA,UAAkBF,EAAe9J,EAAAT,SAAAC,UAAAQ,EAAAT,UAlH/CkK,EAAW,CAqHX,GAAIO,GAAYA,EACd1L,EAAYA,EAAkBuB,EAAQwF,UAnHtC,UAoHSI,KAAAuE,IAAaA,EAAsBL,OAAgBlF,EAAQmF,EAAWG,OAC/EzL,EAAYA,EAAkBwL,QAAA,SAAS,OAnHhC,MAAMrE,KAAKuE,IAAsBL,EAAgB5G,IAAM8G,EAAYE,EAAiBhH,MAsH7FlB,EAAWoI,EAAYD,QAAAA,MAAmBpB,WAIxCsB,OAAAA,KAAcC,IAAoB7L,EAAWqL,KAAAA,EAAiBC,EAAUC,KAC5EO,EAAeF,EAAAA,QAAa5L,OAAAA,SAtHjB,QAAQmH,KAAKuE,IAAsBL,EAAgBlF,MAAQmF,EAAWG,EAAiBlH,QAyHpGhD,EAASiC,EAAWgI,QAAUhJ,QAAAA,SAE1BjB,EAASQ,YAAAA,GAAAA,SAAAA,GAtHb,GAAI6J,GAAcC,EAAoB7L,EAAWqL,EAAiBC,EAAUC,EAC5EO,GAAeF,EAAa5L,KAE9BuB,EAyHQiB,SAAc,SAAIA,GACTyI,KAAXvI,EAAAA,OAAWuI,EAAAA,WACXzI,EAAIuJ,OAxHNvJ,EAAIuJ,oBAGRxK,EA0HQG,cAAQX,SAAAA,GAA+B0B,KAAJD,EAAIC,QAxH3CC,EAAQ,GAAGuI,OAyHXzI,EAAId,oBArHRH,EAuHQA,yBAAmB,SAAAiB,GACrBE,EAAQ3B,yBAtHVyB,EAAIC,iBAEFf,EAAQV,0BACVwB,EAAIuJ,kBA2HApJ,EAAAA,SACJ9C,EAAQmM,GAAAA,OAvHRtJ,EAyHIA,GAAQG,QA3EhB,IAAIa,IAAyB,CAuM7B,OAAOnC,GAET,QAASiJ,GAAWrI,GAoLrB8J,EAAAA,SAAU9J,EAAA+J,OAAA/J,EAAA+J,MAAApE,SAAA3F,EAAAgK,UAGPC,QAAAA,GAAUC,EAAA3J,GACVP,MAAOtC,SAAA6C,SAAAA,GAAAwC,UAAAoH,iBAAAD,IA1sBT,GASIE,GAAYhL,6BAA0BY,KAAST,EAAQS,UAAMgG,WAE7DrF,EAAIF,eAAsBA,GAASyF,UAAAA,EACnCxE,EAAInC,QAAQd,QAASf,EAAQ2M,SA4gBjC,OAqLQC,OAnLTR,UAqLoB9J,aAAOA,UAAAA,YAAAA,OAAAA,SAAAA,WAAAA,QAAAA,SAAAA,EAAAA,EAAAA,EAAAA,EAAAA,EAAAA,GApL5B,OACEiK,SAoLYJ,MAnLZ7J,OAoLM,EAnLNgI,KAAM,SAAkBhI,EAAOO,EAAS6F,EAAMmE,GAuL1C,GAAIC,GACJ9M,GACEsC,MAAItC,EApLRA,SAAQmM,SAAU,WAAY,cAAe,aAAc,eAAgB,gBAAiB,YAAa,YAAa,QAAS,UAAW,OAAQ,YAAa,oBAAqB,OAAQ,cAAe,MAAQ,SAASY,GACtN/M,QAAQ+J,UAAUrB,EAAKqE,MAAOlL,EAAQkL,GAAOrE,EAAKqE,KAExD,IAwLID,GAAmBC,eAvLvB/M,SAwLQA,SAAQ+J,OAAUrB,aAAc,SAAAqE,GAClClL,QAAQkL,UAAOzK,EAAM0K,KAAWC,EAAAA,KAAAA,EAAAA,MAvLlCpL,EAAQkL,IAAO,KA6LjB/M,QAAIA,SAAQ+J,eAAUmD,SAAa,eAAA,UAAA,SAAAH,GACjC,GAAID,GAAAA,KAAAA,EAAiBxF,OAAK4F,GAAAA,cAAaH,EAAAI,MAAA,EACrCtL,SAAQ5B,UAASyI,EAAAuE,MAzLnBpL,EA0LOkL,GAAAzK,EAAA0K,MAAAtE,EAAAuE,MAvLX,IAAIC,GAAarK,EAAQ6F,KAAK,cA8L5B1I,SAAWoN,UAAAA,KACT9K,EAAcgF,KAAA4F,GA5LdrL,EAAQ5B,QAAS,EAiMjB4B,EAAI7B,OAAQ+J,GAGVzH,EAAItC,eAAkBqN,WA9L1B/K,EA+LMwI,MAAM,IA7LdpC,EAAK4E,SAAS,QAAS,SAASC,GAC9B,GAAIvN,QAAQ+J,UAAUwD,KAAcjL,EAAM8K,eAAe,SAAU,CACjE,GAAIC,GAAW/K,EAAMzB,KACrByB,GAAMzB,MAAQgI,EAAKC,YAAYyE,GAiM5BD,QAASvD,UAAYsD,IACpBE,EAAAA,WACMrL,GAAAA,EAAAA,uBA3LdwG,EAkMM4E,SAAItN,WAAiBuN,SAAAA,GACnBvN,GAAQ8E,EAAOxC,UAjMnBsK,EAkMS1K,SAGPwG,EAAI1I,WAjMRsC,EAkMMwI,OAAMpC,EAAA8E,UAAA,SAAAD,EAAAF,GACJrN,QAAI4M,SAASA,GAjMjB5M,QAAQ8E,OAAOxC,EAAOiL,GAoMrBjL,EAAAzB,MAAA0M,EAID7E,QAAK+E,UAAQJ,IACf/K,EAAMoL,WACCd,GAAY5M,EAAQ+J,sBAjM1B,GAEDrB,EAoMIkE,QAnMNtK,EAAMoL,OAAOhF,EAAK+E,OAAQ,SAASF,EAAUF,GACtCT,GAAY5M,QAAQ+J,UAAUwD,KAC/BvN,QAAQ2M,SAASY,KAAWA,IAAaA,EAASI,MAAM,wBAuM1DjF,KAAgB,EAClBpG,EAAMoL,OAGJd,EAAI5M,UAnMN0I,EAuMIkE,WAtMNtK,EAAMoL,OAAOhF,EAAKzH,UAAW,SAASsM,EAAUF,GACzCT,GAAY5M,QAAQ+J,UAAUwD,KAC/BvN,QAAQ2M,SAASY,KAAWA,IAAaA,EAASI,MAAM,0BA0M1DjF,KAAe,EACjBpG,EAAMoL,YAAYtM,GAEhBwL,EAAQgB,YAAYL,MAQxBjL,EAAMuL,UACJvL,EAAIsK,OAASA,EAAAA,SAAQ/C,SAAAA,GACrBhI,GAAU7B,QAAA+J,UAAAwD,IACVX,EAAUgB,YAAAL,KA1MdX,EAAUlL,EAASmB,EAAShB,GAC5BS,EAAMuL,IAAI,WAAY,WAChBjB,GAASA,EAAQ/C,UACrBhI,EAAU,KACV+K,EAAU","file":"tooltip.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.tooltip', ['mgcrea.ngStrap.core', 'mgcrea.ngStrap.helpers.dimensions'])\n\n .provider('$tooltip', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n customClass: '',\n prefixClass: 'tooltip',\n prefixEvent: 'tooltip',\n container: false,\n target: false,\n placement: 'top',\n templateUrl: 'tooltip/tooltip.tpl.html',\n template: '',\n titleTemplate: false,\n trigger: 'hover focus',\n keyboard: false,\n html: false,\n show: false,\n title: '',\n type: '',\n delay: 0,\n autoClose: false,\n bsEnabled: true,\n mouseDownPreventDefault: true,\n mouseDownStopPropagation: true,\n viewport: {\n selector: 'body',\n padding: 0\n }\n };\n\n this.$get = function ($window, $rootScope, $bsCompiler, $q, $templateCache, $http, $animate, $sce, dimensions, $$rAF, $timeout) {\n\n var isNative = /(ip[ao]d|iphone|android)/ig.test($window.navigator.userAgent);\n var isTouch = ('createTouch' in $window.document) && isNative;\n var $body = angular.element($window.document);\n\n function TooltipFactory (element, config) {\n\n var $tooltip = {};\n\n // Common vars\n var options = $tooltip.$options = angular.extend({}, defaults, config);\n var promise = $tooltip.$promise = $bsCompiler.compile(options);\n var scope = $tooltip.$scope = options.scope && options.scope.$new() || $rootScope.$new();\n\n var nodeName = element[0].nodeName.toLowerCase();\n if (options.delay && angular.isString(options.delay)) {\n var split = options.delay.split(',').map(parseFloat);\n options.delay = split.length > 1 ? {show: split[0], hide: split[1]} : split[0];\n }\n\n // Store $id to identify the triggering element in events\n // give priority to options.id, otherwise, try to use\n // element id if defined\n $tooltip.$id = options.id || element.attr('id') || '';\n\n // Support scope as string options\n if (options.title) {\n scope.title = $sce.trustAsHtml(options.title);\n }\n\n // Provide scope helpers\n scope.$setEnabled = function (isEnabled) {\n scope.$$postDigest(function () {\n $tooltip.setEnabled(isEnabled);\n });\n };\n scope.$hide = function () {\n scope.$$postDigest(function () {\n $tooltip.hide();\n });\n };\n scope.$show = function () {\n scope.$$postDigest(function () {\n $tooltip.show();\n });\n };\n scope.$toggle = function () {\n scope.$$postDigest(function () {\n $tooltip.toggle();\n });\n };\n // Publish isShown as a protected var on scope\n $tooltip.$isShown = scope.$isShown = false;\n\n // Private vars\n var timeout;\n var hoverState;\n\n // Fetch, compile then initialize tooltip\n var compileData;\n var tipElement;\n var tipContainer;\n var tipScope;\n promise.then(function (data) {\n compileData = data;\n $tooltip.init();\n });\n\n $tooltip.init = function () {\n\n // Options: delay\n if (options.delay && angular.isNumber(options.delay)) {\n options.delay = {\n show: options.delay,\n hide: options.delay\n };\n }\n\n // Replace trigger on touch devices ?\n // if(isTouch && options.trigger === defaults.trigger) {\n // options.trigger.replace(/hover/g, 'click');\n // }\n\n // Options : container\n if (options.container === 'self') {\n tipContainer = element;\n } else if (angular.isElement(options.container)) {\n tipContainer = options.container;\n } else if (options.container) {\n tipContainer = findElement(options.container);\n }\n\n // Options: trigger\n bindTriggerEvents();\n\n // Options: target\n if (options.target) {\n options.target = angular.isElement(options.target) ? options.target : findElement(options.target);\n }\n\n // Options: show\n if (options.show) {\n scope.$$postDigest(function () {\n if (options.trigger === 'focus') {\n element[0].focus();\n } else {\n $tooltip.show();\n }\n });\n }\n\n };\n\n $tooltip.destroy = function () {\n\n // Unbind events\n unbindTriggerEvents();\n\n // Remove element\n destroyTipElement();\n\n // Destroy scope\n scope.$destroy();\n\n };\n\n $tooltip.enter = function () {\n\n clearTimeout(timeout);\n hoverState = 'in';\n if (!options.delay || !options.delay.show) {\n return $tooltip.show();\n }\n\n timeout = setTimeout(function () {\n if (hoverState === 'in') $tooltip.show();\n }, options.delay.show);\n\n };\n\n $tooltip.show = function () {\n if (!options.bsEnabled || $tooltip.$isShown) return;\n\n scope.$emit(options.prefixEvent + '.show.before', $tooltip);\n if (angular.isDefined(options.onBeforeShow) && angular.isFunction(options.onBeforeShow)) {\n options.onBeforeShow($tooltip);\n }\n var parent;\n var after;\n if (options.container) {\n parent = tipContainer;\n if (tipContainer[0].lastChild) {\n after = angular.element(tipContainer[0].lastChild);\n } else {\n after = null;\n }\n } else {\n parent = null;\n after = element;\n }\n\n\n // Hide any existing tipElement\n if (tipElement) destroyTipElement();\n // Fetch a cloned element linked from template\n tipScope = $tooltip.$scope.$new();\n tipElement = $tooltip.$element = compileData.link(tipScope, function (clonedElement, scope) {});\n\n // Set the initial positioning. Make the tooltip invisible\n // so IE doesn't try to focus on it off screen.\n tipElement.css({top: '-9999px', left: '-9999px', right: 'auto', display: 'block', visibility: 'hidden'});\n\n // Options: animation\n if (options.animation) tipElement.addClass(options.animation);\n // Options: type\n if (options.type) tipElement.addClass(options.prefixClass + '-' + options.type);\n // Options: custom classes\n if (options.customClass) tipElement.addClass(options.customClass);\n\n // Append the element, without any animations. If we append\n // using $animate.enter, some of the animations cause the placement\n // to be off due to the transforms.\n if (after) {\n after.after(tipElement);\n } else {\n parent.prepend(tipElement);\n }\n\n $tooltip.$isShown = scope.$isShown = true;\n safeDigest(scope);\n\n // Now, apply placement\n $tooltip.$applyPlacement();\n\n // Once placed, animate it.\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.enter(tipElement, parent, after, enterAnimateCallback);\n } else {\n $animate.enter(tipElement, parent, after).then(enterAnimateCallback);\n }\n safeDigest(scope);\n\n $$rAF(function () {\n // Once the tooltip is placed and the animation starts, make the tooltip visible\n if (tipElement) tipElement.css({visibility: 'visible'});\n\n // Bind events\n if (options.keyboard) {\n if (options.trigger !== 'focus') {\n $tooltip.focus();\n }\n bindKeyboardEvents();\n }\n });\n\n if (options.autoClose) {\n bindAutoCloseEvents();\n }\n\n };\n\n function enterAnimateCallback () {\n scope.$emit(options.prefixEvent + '.show', $tooltip);\n if (angular.isDefined(options.onShow) && angular.isFunction(options.onShow)) {\n options.onShow($tooltip);\n }\n }\n\n $tooltip.leave = function () {\n\n clearTimeout(timeout);\n hoverState = 'out';\n if (!options.delay || !options.delay.hide) {\n return $tooltip.hide();\n }\n timeout = setTimeout(function () {\n if (hoverState === 'out') {\n $tooltip.hide();\n }\n }, options.delay.hide);\n\n };\n\n var _blur;\n var _tipToHide;\n $tooltip.hide = function (blur) {\n\n if (!$tooltip.$isShown) return;\n scope.$emit(options.prefixEvent + '.hide.before', $tooltip);\n if (angular.isDefined(options.onBeforeHide) && angular.isFunction(options.onBeforeHide)) {\n options.onBeforeHide($tooltip);\n }\n\n // store blur value for leaveAnimateCallback to use\n _blur = blur;\n\n // store current tipElement reference to use\n // in leaveAnimateCallback\n _tipToHide = tipElement;\n\n if (tipElement !== null) {\n // Support v1.2+ $animate\n // https://github.com/angular/angular.js/issues/11713\n if (angular.version.minor <= 2) {\n $animate.leave(tipElement, leaveAnimateCallback);\n } else {\n $animate.leave(tipElement).then(leaveAnimateCallback);\n }\n }\n\n $tooltip.$isShown = scope.$isShown = false;\n safeDigest(scope);\n\n // Unbind events\n if (options.keyboard && tipElement !== null) {\n unbindKeyboardEvents();\n }\n\n if (options.autoClose && tipElement !== null) {\n unbindAutoCloseEvents();\n }\n };\n\n function leaveAnimateCallback () {\n scope.$emit(options.prefixEvent + '.hide', $tooltip);\n if (angular.isDefined(options.onHide) && angular.isFunction(options.onHide)) {\n options.onHide($tooltip);\n }\n\n // check if current tipElement still references\n // the same element when hide was called\n if (tipElement === _tipToHide) {\n // Allow to blur the input when hidden, like when pressing enter key\n if (_blur && options.trigger === 'focus') {\n return element[0].blur();\n }\n\n // clean up child scopes\n destroyTipElement();\n }\n }\n\n $tooltip.toggle = function (evt) {\n if (evt) { evt.preventDefault(); }\n if ($tooltip.$isShown) {\n $tooltip.leave();\n } else {\n $tooltip.enter();\n }\n };\n\n $tooltip.focus = function () {\n tipElement[0].focus();\n };\n\n $tooltip.setEnabled = function (isEnabled) {\n options.bsEnabled = isEnabled;\n };\n\n $tooltip.setViewport = function (viewport) {\n options.viewport = viewport;\n };\n\n // Protected methods\n\n $tooltip.$applyPlacement = function () {\n if (!tipElement) return;\n\n // Determine if we're doing an auto or normal placement\n var placement = options.placement;\n var autoToken = /\\s?auto?\\s?/i;\n var autoPlace = autoToken.test(placement);\n\n if (autoPlace) {\n placement = placement.replace(autoToken, '') || defaults.placement;\n }\n\n // Need to add the position class before we get\n // the offsets\n tipElement.addClass(options.placement);\n\n // Get the position of the target element\n // and the height and width of the tooltip so we can center it.\n var elementPosition = getPosition();\n var tipWidth = tipElement.prop('offsetWidth');\n var tipHeight = tipElement.prop('offsetHeight');\n\n // Refresh viewport position\n $tooltip.$viewport = options.viewport && findElement(options.viewport.selector || options.viewport);\n\n // If we're auto placing, we need to check the positioning\n if (autoPlace) {\n var originalPlacement = placement;\n var viewportPosition = getPosition($tooltip.$viewport);\n\n if (/bottom/.test(originalPlacement) && elementPosition.bottom + tipHeight > viewportPosition.bottom) {\n placement = originalPlacement.replace('bottom', 'top');\n } else if (/top/.test(originalPlacement) && elementPosition.top - tipHeight < viewportPosition.top) {\n placement = originalPlacement.replace('top', 'bottom');\n }\n\n if (/left/.test(originalPlacement) && elementPosition.left - tipWidth < viewportPosition.left) {\n placement = placement.replace('left', 'right');\n } else if (/right/.test(originalPlacement) && elementPosition.right + tipWidth > viewportPosition.width) {\n placement = placement.replace('right', 'left');\n }\n\n tipElement.removeClass(originalPlacement).addClass(placement);\n }\n\n // Get the tooltip's top and left coordinates to center it with this directive.\n var tipPosition = getCalculatedOffset(placement, elementPosition, tipWidth, tipHeight);\n applyPlacement(tipPosition, placement);\n };\n\n $tooltip.$onKeyUp = function (evt) {\n if (evt.which === 27 && $tooltip.$isShown) {\n $tooltip.hide();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusKeyUp = function (evt) {\n if (evt.which === 27) {\n element[0].blur();\n evt.stopPropagation();\n }\n };\n\n $tooltip.$onFocusElementMouseDown = function (evt) {\n if (options.mouseDownPreventDefault) { evt.preventDefault(); }\n if (options.mouseDownStopPropagation) { evt.stopPropagation(); }\n // Some browsers do not auto-focus buttons (eg. Safari)\n if ($tooltip.$isShown) {\n element[0].blur();\n } else {\n element[0].focus();\n }\n };\n\n // bind/unbind events\n function bindTriggerEvents () {\n var triggers = options.trigger.split(' ');\n angular.forEach(triggers, function (trigger) {\n if (trigger === 'click' || trigger === 'contextmenu') {\n element.on(trigger, $tooltip.toggle);\n } else if (trigger !== 'manual') {\n element.on(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.on(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n if (nodeName === 'button' && trigger !== 'hover') {\n element.on(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n });\n }\n\n function unbindTriggerEvents () {\n var triggers = options.trigger.split(' ');\n for (var i = triggers.length; i--;) {\n var trigger = triggers[i];\n if (trigger === 'click' || trigger === 'contextmenu') {\n element.off(trigger, $tooltip.toggle);\n } else if (trigger !== 'manual') {\n element.off(trigger === 'hover' ? 'mouseenter' : 'focus', $tooltip.enter);\n element.off(trigger === 'hover' ? 'mouseleave' : 'blur', $tooltip.leave);\n if (nodeName === 'button' && trigger !== 'hover') {\n element.off(isTouch ? 'touchstart' : 'mousedown', $tooltip.$onFocusElementMouseDown);\n }\n }\n }\n }\n\n function bindKeyboardEvents () {\n if (options.trigger !== 'focus') {\n tipElement.on('keyup', $tooltip.$onKeyUp);\n } else {\n element.on('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n function unbindKeyboardEvents () {\n if (options.trigger !== 'focus') {\n tipElement.off('keyup', $tooltip.$onKeyUp);\n } else {\n element.off('keyup', $tooltip.$onFocusKeyUp);\n }\n }\n\n var _autoCloseEventsBinded = false;\n function bindAutoCloseEvents () {\n // use timeout to hookup the events to prevent\n // event bubbling from being processed imediately.\n $timeout(function () {\n // Stop propagation when clicking inside tooltip\n tipElement.on('click', stopEventPropagation);\n\n // Hide when clicking outside tooltip\n $body.on('click', $tooltip.hide);\n\n _autoCloseEventsBinded = true;\n }, 0, false);\n }\n\n function unbindAutoCloseEvents () {\n if (_autoCloseEventsBinded) {\n tipElement.off('click', stopEventPropagation);\n $body.off('click', $tooltip.hide);\n _autoCloseEventsBinded = false;\n }\n }\n\n function stopEventPropagation (event) {\n event.stopPropagation();\n }\n\n // Private methods\n\n function getPosition ($element) {\n $element = $element || (options.target || element);\n\n var el = $element[0];\n var isBody = el.tagName === 'BODY';\n\n var elRect = el.getBoundingClientRect();\n var rect = {};\n\n // IE8 has issues with angular.extend and using elRect directly.\n // By coping the values of elRect into a new object, we can continue to use extend\n /* eslint-disable guard-for-in */\n for (var p in elRect) { // eslint-disable-line\n // DO NOT use hasOwnProperty when inspecting the return of getBoundingClientRect.\n rect[p] = elRect[p];\n }\n /* eslint-enable guard-for-in */\n\n if (rect.width === null) {\n // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093\n rect = angular.extend({}, rect, {width: elRect.right - elRect.left, height: elRect.bottom - elRect.top});\n }\n var elOffset = isBody ? {top: 0, left: 0} : dimensions.offset(el);\n var scroll = {scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.prop('scrollTop') || 0};\n var outerDims = isBody ? {width: document.documentElement.clientWidth, height: $window.innerHeight} : null;\n\n return angular.extend({}, rect, scroll, outerDims, elOffset);\n }\n\n function getCalculatedOffset (placement, position, actualWidth, actualHeight) {\n var offset;\n var split = placement.split('-');\n\n switch (split[0]) {\n case 'right':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left + position.width\n };\n break;\n case 'bottom':\n offset = {\n top: position.top + position.height,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n case 'left':\n offset = {\n top: position.top + position.height / 2 - actualHeight / 2,\n left: position.left - actualWidth\n };\n break;\n default:\n offset = {\n top: position.top - actualHeight,\n left: position.left + position.width / 2 - actualWidth / 2\n };\n break;\n }\n\n if (!split[1]) {\n return offset;\n }\n\n // Add support for corners @todo css\n if (split[0] === 'top' || split[0] === 'bottom') {\n switch (split[1]) {\n case 'left':\n offset.left = position.left;\n break;\n case 'right':\n offset.left = position.left + position.width - actualWidth;\n break;\n default:\n break;\n }\n } else if (split[0] === 'left' || split[0] === 'right') {\n switch (split[1]) {\n case 'top':\n offset.top = position.top - actualHeight + position.height;\n break;\n case 'bottom':\n offset.top = position.top;\n break;\n default:\n break;\n }\n }\n\n return offset;\n }\n\n function applyPlacement (offset, placement) {\n var tip = tipElement[0];\n var width = tip.offsetWidth;\n var height = tip.offsetHeight;\n\n // manually read margins because getBoundingClientRect includes difference\n var marginTop = parseInt(dimensions.css(tip, 'margin-top'), 10);\n var marginLeft = parseInt(dimensions.css(tip, 'margin-left'), 10);\n\n // we must check for NaN for ie 8/9\n if (isNaN(marginTop)) marginTop = 0;\n if (isNaN(marginLeft)) marginLeft = 0;\n\n offset.top = offset.top + marginTop;\n offset.left = offset.left + marginLeft;\n\n // dimensions setOffset doesn't round pixel values\n // so we use setOffset directly with our own function\n dimensions.setOffset(tip, angular.extend({\n using: function (props) {\n tipElement.css({\n top: Math.round(props.top) + 'px',\n left: Math.round(props.left) + 'px',\n right: ''\n });\n }\n }, offset), 0);\n\n // check to see if placing tip in new offset caused the tip to resize itself\n var actualWidth = tip.offsetWidth;\n var actualHeight = tip.offsetHeight;\n\n if (placement === 'top' && actualHeight !== height) {\n offset.top = offset.top + height - actualHeight;\n }\n\n // If it's an exotic placement, exit now instead of\n // applying a delta and changing the arrow\n if (/top-left|top-right|bottom-left|bottom-right/.test(placement)) return;\n\n var delta = getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight);\n\n if (delta.left) {\n offset.left += delta.left;\n } else {\n offset.top += delta.top;\n }\n\n dimensions.setOffset(tip, offset);\n\n if (/top|right|bottom|left/.test(placement)) {\n var isVertical = /top|bottom/.test(placement);\n var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight;\n var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight';\n\n replaceArrow(arrowDelta, tip[arrowOffsetPosition], isVertical);\n }\n }\n\n // @source https://github.com/twbs/bootstrap/blob/v3.3.5/js/tooltip.js#L380\n function getViewportAdjustedDelta (placement, position, actualWidth, actualHeight) {\n var delta = {top: 0, left: 0};\n if (!$tooltip.$viewport) return delta;\n\n var viewportPadding = options.viewport && options.viewport.padding || 0;\n var viewportDimensions = getPosition($tooltip.$viewport);\n\n if (/right|left/.test(placement)) {\n var topEdgeOffset = position.top - viewportPadding - viewportDimensions.scroll;\n var bottomEdgeOffset = position.top + viewportPadding - viewportDimensions.scroll + actualHeight;\n if (topEdgeOffset < viewportDimensions.top) { // top overflow\n delta.top = viewportDimensions.top - topEdgeOffset;\n } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow\n delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset;\n }\n } else {\n var leftEdgeOffset = position.left - viewportPadding;\n var rightEdgeOffset = position.left + viewportPadding + actualWidth;\n if (leftEdgeOffset < viewportDimensions.left) { // left overflow\n delta.left = viewportDimensions.left - leftEdgeOffset;\n } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow\n delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset;\n }\n }\n\n return delta;\n }\n\n function replaceArrow (delta, dimension, isHorizontal) {\n var $arrow = findElement('.tooltip-arrow, .arrow', tipElement[0]);\n\n $arrow.css(isHorizontal ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')\n .css(isHorizontal ? 'top' : 'left', '');\n }\n\n function destroyTipElement () {\n // Cancel pending callbacks\n clearTimeout(timeout);\n\n if ($tooltip.$isShown && tipElement !== null) {\n if (options.autoClose) {\n unbindAutoCloseEvents();\n }\n\n if (options.keyboard) {\n unbindKeyboardEvents();\n }\n }\n\n if (tipScope) {\n tipScope.$destroy();\n tipScope = null;\n }\n\n if (tipElement) {\n tipElement.remove();\n tipElement = $tooltip.$element = null;\n }\n }\n\n return $tooltip;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n function findElement (query, element) {\n return angular.element((element || document).querySelectorAll(query));\n }\n\n return TooltipFactory;\n\n };\n\n })\n\n .directive('bsTooltip', function ($window, $location, $sce, $parse, $tooltip, $$rAF) {\n\n return {\n restrict: 'EAC',\n scope: true,\n link: function postLink (scope, element, attr, transclusion) {\n\n var tooltip;\n // Directive options\n var options = {scope: scope};\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'titleTemplate', 'placement', 'container', 'delay', 'trigger', 'html', 'animation', 'backdropAnimation', 'type', 'customClass', 'id'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) {\n options[key] = false;\n }\n });\n\n // bind functions from the attrs to the show and hide events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // should not parse target attribute (anchor tag), only data-target #1454\n var dataTarget = element.attr('data-target');\n if (angular.isDefined(dataTarget)) {\n if (falseValueRegExp.test(dataTarget)) {\n options.target = false;\n } else {\n options.target = dataTarget;\n }\n }\n\n // overwrite inherited title value when no value specified\n // fix for angular 1.3.1 531a8de72c439d8ddd064874bf364c00cedabb11\n if (!scope.hasOwnProperty('title')) {\n scope.title = '';\n }\n\n // Observe scope attributes for change\n attr.$observe('title', function (newValue) {\n if (angular.isDefined(newValue) || !scope.hasOwnProperty('title')) {\n var oldValue = scope.title;\n scope.title = $sce.trustAsHtml(newValue);\n if (angular.isDefined(oldValue)) {\n $$rAF(function () {\n if (tooltip) tooltip.$applyPlacement();\n });\n }\n }\n });\n\n attr.$observe('disabled', function (newValue) {\n if (newValue && tooltip.$isShown) {\n tooltip.hide();\n }\n });\n\n // Support scope as an object\n if (attr.bsTooltip) {\n scope.$watch(attr.bsTooltip, function (newValue, oldValue) {\n if (angular.isObject(newValue)) {\n angular.extend(scope, newValue);\n } else {\n scope.title = newValue;\n }\n if (angular.isDefined(oldValue)) {\n $$rAF(function () {\n if (tooltip) tooltip.$applyPlacement();\n });\n }\n }, true);\n }\n\n // Visibility binding support\n if (attr.bsShow) {\n scope.$watch(attr.bsShow, function (newValue, oldValue) {\n if (!tooltip || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|,?(tooltip),?/i);\n if (newValue === true) {\n tooltip.show();\n } else {\n tooltip.hide();\n }\n });\n }\n\n // Enabled binding support\n if (attr.bsEnabled) {\n scope.$watch(attr.bsEnabled, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', attr.bsEnabled, newValue, oldValue);\n if (!tooltip || !angular.isDefined(newValue)) return;\n if (angular.isString(newValue)) newValue = !!newValue.match(/true|1|,?(tooltip),?/i);\n if (newValue === false) {\n tooltip.setEnabled(false);\n } else {\n tooltip.setEnabled(true);\n }\n });\n }\n\n // Viewport support\n if (attr.viewport) {\n scope.$watch(attr.viewport, function (newValue) {\n if (!tooltip || !angular.isDefined(newValue)) return;\n tooltip.setViewport(newValue);\n });\n }\n\n // Initialize popover\n tooltip = $tooltip(element, options);\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (tooltip) tooltip.destroy();\n options = null;\n tooltip = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/tooltip.tpl.js b/dist/modules/tooltip.tpl.js index 7c15481f0..e308f559d 100644 --- a/dist/modules/tooltip.tpl.js +++ b/dist/modules/tooltip.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/tooltip.tpl.min.js b/dist/modules/tooltip.tpl.min.js index 44e3c454b..2b6a2b7fa 100644 --- a/dist/modules/tooltip.tpl.min.js +++ b/dist/modules/tooltip.tpl.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/typeahead.js b/dist/modules/typeahead.js index 9cd98582b..8b7598348 100644 --- a/dist/modules/typeahead.js +++ b/dist/modules/typeahead.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT @@ -27,7 +27,6 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n trimValue: true }; this.$get = [ '$window', '$rootScope', '$tooltip', '$$rAF', '$timeout', function($window, $rootScope, $tooltip, $$rAF, $timeout) { - var bodyEl = angular.element($window.document.body); function TypeaheadFactory(element, controller, config) { var $typeahead = {}; var options = angular.extend({}, defaults, config); @@ -71,6 +70,9 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n scope.$resetMatches(); if (parentScope) parentScope.$digest(); scope.$emit(options.prefixEvent + '.select', value, index, $typeahead); + if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) { + options.onSelect(value, index, $typeahead); + } }; $typeahead.$isVisible = function() { if (!options.minLength || !controller) { @@ -79,18 +81,30 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength; }; $typeahead.$getIndex = function(value) { - var l = scope.$matches.length, i = l; - if (!l) return; - for (i = l; i--; ) { - if (scope.$matches[i].value === value) break; + var index; + for (index = scope.$matches.length; index--; ) { + if (angular.equals(scope.$matches[index].value, value)) break; } - if (i < 0) return; - return i; + return index; }; $typeahead.$onMouseDown = function(evt) { evt.preventDefault(); evt.stopPropagation(); }; + $typeahead.$$updateScrollTop = function(container, index) { + if (index > -1 && index < container.children.length) { + var active = container.children[index]; + var clientTop = active.offsetTop; + var clientBottom = active.offsetTop + active.clientHeight; + var highWatermark = container.scrollTop; + var lowWatermark = container.scrollTop + container.clientHeight; + if (clientBottom >= highWatermark && clientTop < highWatermark) { + container.scrollTop = Math.max(0, container.scrollTop - container.clientHeight); + } else if (clientBottom > lowWatermark) { + container.scrollTop = clientTop; + } + } + }; $typeahead.$onKeyDown = function(evt) { if (!/(38|40|13)/.test(evt.keyCode)) return; if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) { @@ -99,26 +113,37 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n } if (evt.keyCode === 13 && scope.$matches.length) { $typeahead.select(scope.$activeIndex); - } else if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--; else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++; else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0; + } else if (evt.keyCode === 38 && scope.$activeIndex > 0) { + scope.$activeIndex--; + } else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) { + scope.$activeIndex++; + } else if (angular.isUndefined(scope.$activeIndex)) { + scope.$activeIndex = 0; + } + $typeahead.$$updateScrollTop($typeahead.$element[0], scope.$activeIndex); scope.$digest(); }; var show = $typeahead.show; $typeahead.show = function() { show(); $timeout(function() { - $typeahead.$element && $typeahead.$element.on('mousedown', $typeahead.$onMouseDown); - if (options.keyboard) { - element && element.on('keydown', $typeahead.$onKeyDown); + if ($typeahead.$element) { + $typeahead.$element.on('mousedown', $typeahead.$onMouseDown); + if (options.keyboard) { + if (element) element.on('keydown', $typeahead.$onKeyDown); + } } }, 0, false); }; var hide = $typeahead.hide; $typeahead.hide = function() { - $typeahead.$element && $typeahead.$element.off('mousedown', $typeahead.$onMouseDown); + if ($typeahead.$element) $typeahead.$element.off('mousedown', $typeahead.$onMouseDown); if (options.keyboard) { - element && element.off('keydown', $typeahead.$onKeyDown); + if (element) element.off('keydown', $typeahead.$onKeyDown); + } + if (!options.autoSelect) { + $typeahead.activate(-1); } - if (!options.autoSelect) $typeahead.activate(-1); hide(); }; return $typeahead; @@ -135,9 +160,8 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n return array.then(function(results) { return $filter('filter')(results, expression, comparator); }); - } else { - return $filter('filter')(array, expression, comparator); } + return $filter('filter')(array, expression, comparator); }; } ]).directive('bsTypeahead', [ '$window', '$parse', '$q', '$typeahead', '$parseOptions', function($window, $parse, $q, $typeahead, $parseOptions) { var defaults = $typeahead.defaults; @@ -145,6 +169,7 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n restrict: 'EAC', require: 'ngModel', link: function postLink(scope, element, attr, controller) { + element.off('change'); var options = { scope: scope }; @@ -152,16 +177,24 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n if (angular.isDefined(attr[key])) options[key] = attr[key]; }); var falseValueRegExp = /^(false|0|)$/i; - angular.forEach([ 'html', 'container', 'trimValue' ], function(key) { + angular.forEach([ 'html', 'container', 'trimValue', 'filter' ], function(key) { if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false; }); + angular.forEach([ 'onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect' ], function(key) { + var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1); + if (angular.isDefined(attr[bsKey])) { + options[key] = scope.$eval(attr[bsKey]); + } + }); if (!element.attr('autocomplete')) element.attr('autocomplete', 'off'); - var filter = options.filter || defaults.filter; + var filter = angular.isDefined(options.filter) ? options.filter : defaults.filter; var limit = options.limit || defaults.limit; var comparator = options.comparator || defaults.comparator; var bsOptions = attr.bsOptions; - if (filter) bsOptions += ' | ' + filter + ':$viewValue'; - if (comparator) bsOptions += ':' + comparator; + if (filter) { + bsOptions += ' | ' + filter + ':$viewValue'; + if (comparator) bsOptions += ':' + comparator; + } if (limit) bsOptions += ' | limitTo:' + limit; var parsedOptions = $parseOptions(bsOptions); var typeahead = $typeahead(element, controller, options); @@ -191,7 +224,7 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n if (displayValue) { return displayValue; } - if (modelValue && typeof modelValue !== 'object') { + if (angular.isDefined(modelValue) && typeof modelValue !== 'object') { return modelValue; } return ''; @@ -201,10 +234,13 @@ angular.module('mgcrea.ngStrap.typeahead', [ 'mgcrea.ngStrap.tooltip', 'mgcrea.n return element.val(''); } var index = typeahead.$getIndex(controller.$modelValue); - var selected = angular.isDefined(index) ? typeahead.$scope.$matches[index].label : controller.$viewValue; + var selected = index !== -1 ? typeahead.$scope.$matches[index].label : controller.$viewValue; selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected; var value = selected ? selected.toString().replace(/<(?:.|\n)*?>/gm, '') : ''; + var ss = element[0].selectionStart; + var sd = element[0].selectionEnd; element.val(options.trimValue === false ? value : value.trim()); + element[0].setSelectionRange(ss, sd); }; scope.$on('$destroy', function() { if (typeahead) typeahead.destroy(); diff --git a/dist/modules/typeahead.min.js b/dist/modules/typeahead.min.js index d7bb899c9..bbb9bcadd 100644 --- a/dist/modules/typeahead.min.js +++ b/dist/modules/typeahead.min.js @@ -1,9 +1,9 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT */ -'use strict';angular.module('mgcrea.ngStrap.typeahead',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$typeahead',function(){var e=this.defaults={animation:'am-fade',prefixClass:'typeahead',prefixEvent:'$typeahead',placement:'bottom-left',templateUrl:'typeahead/typeahead.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,minLength:1,filter:'bsAsyncFilter',limit:6,autoSelect:!1,comparator:'',trimValue:!0};this.$get=['$window','$rootScope','$tooltip','$$rAF','$timeout',function(t,n,a,i,o){function r(t,n,r){var c={},u=angular.extend({},e,r);c=a(t,u);var s=r.scope,$=c.$scope;$.$resetMatches=function(){$.$matches=[],$.$activeIndex=u.autoSelect?0:-1},$.$resetMatches(),$.$activate=function(e){$.$$postDigest(function(){c.activate(e)})},$.$select=function(e,t){$.$$postDigest(function(){c.select(e)})},$.$isVisible=function(){return c.$isVisible()},c.update=function(e){$.$matches=e,$.$activeIndex>=e.length&&($.$activeIndex=u.autoSelect?0:-1),l($),i(c.$applyPlacement)},c.activate=function(e){$.$activeIndex=e},c.select=function(e){if(-1!==e){var t=$.$matches[e].value;n.$setViewValue(t),n.$render(),$.$resetMatches(),s&&s.$digest(),$.$emit(u.prefixEvent+'.select',t,e,c)}},c.$isVisible=function(){return u.minLength&&n?$.$matches.length&&angular.isString(n.$viewValue)&&n.$viewValue.length>=u.minLength:!!$.$matches.length},c.$getIndex=function(e){var t=$.$matches.length,n=t;if(t){for(n=t;n--&&$.$matches[n].value!==e;);if(!(0>n))return n}},c.$onMouseDown=function(e){e.preventDefault(),e.stopPropagation()},c.$onKeyDown=function(e){/(38|40|13)/.test(e.keyCode)&&(!c.$isVisible()||13===e.keyCode&&-1===$.$activeIndex||(e.preventDefault(),e.stopPropagation()),13===e.keyCode&&$.$matches.length?c.select($.$activeIndex):38===e.keyCode&&$.$activeIndex>0?$.$activeIndex--:40===e.keyCode&&$.$activeIndex<$.$matches.length-1?$.$activeIndex++:angular.isUndefined($.$activeIndex)&&($.$activeIndex=0),$.$digest())};var d=c.show;c.show=function(){d(),o(function(){c.$element&&c.$element.on('mousedown',c.$onMouseDown),u.keyboard&&t&&t.on('keydown',c.$onKeyDown)},0,!1)};var f=c.hide;return c.hide=function(){c.$element&&c.$element.off('mousedown',c.$onMouseDown),u.keyboard&&t&&t.off('keydown',c.$onKeyDown),u.autoSelect||c.activate(-1),f()},c}function l(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}angular.element(t.document.body);return r.defaults=e,r}]}).filter('bsAsyncFilter',['$filter',function(e){return function(t,n,a){return t&&angular.isFunction(t.then)?t.then(function(t){return e('filter')(t,n,a)}):e('filter')(t,n,a)}}]).directive('bsTypeahead',['$window','$parse','$q','$typeahead','$parseOptions',function(e,t,n,a,i){var o=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,r){var l={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','filter','limit','minLength','watchOptions','selectMode','autoSelect','comparator','id','prefixEvent','prefixClass'],function(e){angular.isDefined(n[e])&&(l[e]=n[e])});var c=/^(false|0|)$/i;angular.forEach(['html','container','trimValue'],function(e){angular.isDefined(n[e])&&c.test(n[e])&&(l[e]=!1)}),t.attr('autocomplete')||t.attr('autocomplete','off');var u=l.filter||o.filter,s=l.limit||o.limit,$=l.comparator||o.comparator,d=n.bsOptions;u&&(d+=' | '+u+':$viewValue'),$&&(d+=':'+$),s&&(d+=' | limitTo:'+s);var f=i(d),p=a(t,r,l);if(l.watchOptions){var m=f.$match[7].replace(/\|.+/,'').replace(/\(.*\)/g,'').trim();e.$watchCollection(m,function(t,n){f.valuesFn(e,r).then(function(e){p.update(e),r.$render()})})}e.$watch(n.ngModel,function(t,n){e.$modelValue=t,f.valuesFn(e,r).then(function(e){return l.selectMode&&!e.length&&t.length>0?void r.$setViewValue(r.$viewValue.substring(0,r.$viewValue.length-1)):(e.length>s&&(e=e.slice(0,s)),p.update(e),void r.$render())})}),r.$formatters.push(function(e){var t=f.displayValue(e);return t?t:e&&'object'!=typeof e?e:''}),r.$render=function(){if(r.$isEmpty(r.$viewValue))return t.val('');var e=p.$getIndex(r.$modelValue),n=angular.isDefined(e)?p.$scope.$matches[e].label:r.$viewValue;n=angular.isObject(n)?f.displayValue(n):n;var a=n?n.toString().replace(/<(?:.|\n)*?>/gm,''):'';t.val(l.trimValue===!1?a:a.trim())},e.$on('$destroy',function(){p&&p.destroy(),l=null,p=null})}}}]); +'use strict';angular.module('mgcrea.ngStrap.typeahead',['mgcrea.ngStrap.tooltip','mgcrea.ngStrap.helpers.parseOptions']).provider('$typeahead',function(){var e=this.defaults={animation:'am-fade',prefixClass:'typeahead',prefixEvent:'$typeahead',placement:'bottom-left',templateUrl:'typeahead/typeahead.tpl.html',trigger:'focus',container:!1,keyboard:!0,html:!1,delay:0,minLength:1,filter:'bsAsyncFilter',limit:6,autoSelect:!1,comparator:'',trimValue:!0};this.$get=['$window','$rootScope','$tooltip','$$rAF','$timeout',function(t,n,a,i,o){function r(t,n,r){var c={},s=angular.extend({},e,r);c=a(t,s);var u=r.scope,$=c.$scope;$.$resetMatches=function(){$.$matches=[],$.$activeIndex=s.autoSelect?0:-1},$.$resetMatches(),$.$activate=function(e){$.$$postDigest(function(){c.activate(e)})},$.$select=function(e,t){$.$$postDigest(function(){c.select(e)})},$.$isVisible=function(){return c.$isVisible()},c.update=function(e){$.$matches=e,$.$activeIndex>=e.length&&($.$activeIndex=s.autoSelect?0:-1),l($),i(c.$applyPlacement)},c.activate=function(e){$.$activeIndex=e},c.select=function(e){if(e!==-1){var t=$.$matches[e].value;n.$setViewValue(t),n.$render(),$.$resetMatches(),u&&u.$digest(),$.$emit(s.prefixEvent+'.select',t,e,c),angular.isDefined(s.onSelect)&&angular.isFunction(s.onSelect)&&s.onSelect(t,e,c)}},c.$isVisible=function(){return s.minLength&&n?$.$matches.length&&angular.isString(n.$viewValue)&&n.$viewValue.length>=s.minLength:!!$.$matches.length},c.$getIndex=function(e){var t;for(t=$.$matches.length;t--&&!angular.equals($.$matches[t].value,e););return t},c.$onMouseDown=function(e){e.preventDefault(),e.stopPropagation()},c.$$updateScrollTop=function(e,t){if(t>-1&&t=o&&ar&&(e.scrollTop=a)}},c.$onKeyDown=function(e){/(38|40|13)/.test(e.keyCode)&&(!c.$isVisible()||13===e.keyCode&&$.$activeIndex===-1||(e.preventDefault(),e.stopPropagation()),13===e.keyCode&&$.$matches.length?c.select($.$activeIndex):38===e.keyCode&&$.$activeIndex>0?$.$activeIndex--:40===e.keyCode&&$.$activeIndex<$.$matches.length-1?$.$activeIndex++:angular.isUndefined($.$activeIndex)&&($.$activeIndex=0),c.$$updateScrollTop(c.$element[0],$.$activeIndex),$.$digest())};var f=c.show;c.show=function(){f(),o(function(){c.$element&&(c.$element.on('mousedown',c.$onMouseDown),s.keyboard&&t&&t.on('keydown',c.$onKeyDown))},0,!1)};var d=c.hide;return c.hide=function(){c.$element&&c.$element.off('mousedown',c.$onMouseDown),s.keyboard&&t&&t.off('keydown',c.$onKeyDown),s.autoSelect||c.activate(-1),d()},c}function l(e){e.$$phase||e.$root&&e.$root.$$phase||e.$digest()}return r.defaults=e,r}]}).filter('bsAsyncFilter',['$filter',function(e){return function(t,n,a){return t&&angular.isFunction(t.then)?t.then(function(t){return e('filter')(t,n,a)}):e('filter')(t,n,a)}}]).directive('bsTypeahead',['$window','$parse','$q','$typeahead','$parseOptions',function(e,t,n,a,i){var o=a.defaults;return{restrict:'EAC',require:'ngModel',link:function(e,t,n,r){t.off('change');var l={scope:e};angular.forEach(['template','templateUrl','controller','controllerAs','placement','container','delay','trigger','keyboard','html','animation','filter','limit','minLength','watchOptions','selectMode','autoSelect','comparator','id','prefixEvent','prefixClass'],function(e){angular.isDefined(n[e])&&(l[e]=n[e])});var c=/^(false|0|)$/i;angular.forEach(['html','container','trimValue','filter'],function(e){angular.isDefined(n[e])&&c.test(n[e])&&(l[e]=!1)}),angular.forEach(['onBeforeShow','onShow','onBeforeHide','onHide','onSelect'],function(t){var a='bs'+t.charAt(0).toUpperCase()+t.slice(1);angular.isDefined(n[a])&&(l[t]=e.$eval(n[a]))}),t.attr('autocomplete')||t.attr('autocomplete','off');var s=angular.isDefined(l.filter)?l.filter:o.filter,u=l.limit||o.limit,$=l.comparator||o.comparator,f=n.bsOptions;s&&(f+=' | '+s+':$viewValue',$&&(f+=':'+$)),u&&(f+=' | limitTo:'+u);var d=i(f),p=a(t,r,l);if(l.watchOptions){var h=d.$match[7].replace(/\|.+/,'').replace(/\(.*\)/g,'').trim();e.$watchCollection(h,function(t,n){d.valuesFn(e,r).then(function(e){p.update(e),r.$render()})})}e.$watch(n.ngModel,function(t,n){e.$modelValue=t,d.valuesFn(e,r).then(function(e){return l.selectMode&&!e.length&&t.length>0?void r.$setViewValue(r.$viewValue.substring(0,r.$viewValue.length-1)):(e.length>u&&(e=e.slice(0,u)),p.update(e),void r.$render())})}),r.$formatters.push(function(e){var t=d.displayValue(e);return t?t:angular.isDefined(e)&&'object'!=typeof e?e:''}),r.$render=function(){if(r.$isEmpty(r.$viewValue))return t.val('');var e=p.$getIndex(r.$modelValue),n=e!==-1?p.$scope.$matches[e].label:r.$viewValue;n=angular.isObject(n)?d.displayValue(n):n;var a=n?n.toString().replace(/<(?:.|\n)*?>/gm,''):'',i=t[0].selectionStart,o=t[0].selectionEnd;t.val(l.trimValue===!1?a:a.trim()),t[0].setSelectionRange(i,o)},e.$on('$destroy',function(){p&&p.destroy(),l=null,p=null})}}}]); //# sourceMappingURL=typeahead.min.js.map diff --git a/dist/modules/typeahead.min.js.map b/dist/modules/typeahead.min.js.map index ee0f1f1d0..8d8d13a6e 100644 --- a/dist/modules/typeahead.min.js.map +++ b/dist/modules/typeahead.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["modules/typeahead.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","minLength","filter","limit","autoSelect","comparator","trimValue","$get","bodyEl","$typeahead","options","scope","$scope","$matches","extend","config","$resetMatches","$$postDigest","activate","$activeIndex","select","index","$select","evt","matches","$isVisible","safeDigest","update","$$rAF","value","controller","$render","$emit","prefixEvent","parentScope","length","l","isString","$viewValue","i","preventDefault","stopPropagation","keyCode","$digest","show","$timeout","$element","$onMouseDown","hide","on","$onKeyDown","element","TypeaheadFactory","array","isFunction","then","$$phase","$window","document","body","$filter","expression","directive","results","restrict","require","link","falseValueRegExp","attr","key","bsOptions","test","parsedOptions","$parseOptions","typeahead","watchOptions","watchedOptions","$watchCollection","values","$match","replace","trim","$watch","ngModel","newValue","oldValue","$modelValue","valuesFn","selectMode","$setViewValue","substring","displayValue","slice","$formatters","push","modelValue","selected","val","destroy","isDefined","label","isObject","toString","$on"],"mappings":"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","file":"modules/typeahead.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.typeahead', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$typeahead', function() {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'typeahead',\n prefixEvent: '$typeahead',\n placement: 'bottom-left',\n templateUrl: 'typeahead/typeahead.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n minLength: 1,\n filter: 'bsAsyncFilter',\n limit: 6,\n autoSelect: false,\n comparator: '',\n trimValue: true\n };\n\n this.$get = function($window, $rootScope, $tooltip, $$rAF, $timeout) {\n\n var bodyEl = angular.element($window.document.body);\n\n function TypeaheadFactory(element, controller, config) {\n\n var $typeahead = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $typeahead = $tooltip(element, options);\n var parentScope = config.scope;\n var scope = $typeahead.$scope;\n\n scope.$resetMatches = function() {\n scope.$matches = [];\n scope.$activeIndex = options.autoSelect ? 0 : -1; // If set to 0, the first match will be highlighted\n };\n scope.$resetMatches();\n\n scope.$activate = function(index) {\n scope.$$postDigest(function() {\n $typeahead.activate(index);\n });\n };\n\n scope.$select = function(index, evt) {\n scope.$$postDigest(function() {\n $typeahead.select(index);\n });\n };\n\n scope.$isVisible = function() {\n return $typeahead.$isVisible();\n };\n\n // Public methods\n\n $typeahead.update = function(matches) {\n scope.$matches = matches;\n if (scope.$activeIndex >= matches.length) {\n scope.$activeIndex = options.autoSelect ? 0 : -1;\n }\n\n // wrap in a $timeout so the results are updated\n // before repositioning\n safeDigest(scope);\n $$rAF($typeahead.$applyPlacement);\n };\n\n $typeahead.activate = function(index) {\n scope.$activeIndex = index;\n };\n\n $typeahead.select = function(index) {\n if (index === -1) return;\n var value = scope.$matches[index].value;\n // console.log('$setViewValue', value);\n controller.$setViewValue(value);\n controller.$render();\n scope.$resetMatches();\n if (parentScope) parentScope.$digest();\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $typeahead);\n };\n\n // Protected methods\n\n $typeahead.$isVisible = function() {\n if (!options.minLength || !controller) {\n return !!scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength;\n };\n\n $typeahead.$getIndex = function(value) {\n var l = scope.$matches.length,\n i = l;\n if (!l) return;\n for (i = l; i--;) {\n if (scope.$matches[i].value === value) break;\n }\n if (i < 0) return;\n return i;\n };\n\n $typeahead.$onMouseDown = function(evt) {\n // Prevent blur on mousedown\n evt.preventDefault();\n evt.stopPropagation();\n };\n\n $typeahead.$onKeyDown = function(evt) {\n if (!/(38|40|13)/.test(evt.keyCode)) return;\n\n // Let ngSubmit pass if the typeahead tip is hidden or no option is selected\n if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // Select with enter\n if (evt.keyCode === 13 && scope.$matches.length) {\n $typeahead.select(scope.$activeIndex);\n }\n\n // Navigate with keyboard\n else if (evt.keyCode === 38 && scope.$activeIndex > 0) scope.$activeIndex--;\n else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) scope.$activeIndex++;\n else if (angular.isUndefined(scope.$activeIndex)) scope.$activeIndex = 0;\n scope.$digest();\n };\n\n // Overrides\n\n var show = $typeahead.show;\n $typeahead.show = function() {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed immediately.\n $timeout(function() {\n $typeahead.$element && $typeahead.$element.on('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n element && element.on('keydown', $typeahead.$onKeyDown);\n }\n }, 0, false);\n };\n\n var hide = $typeahead.hide;\n $typeahead.hide = function() {\n $typeahead.$element && $typeahead.$element.off('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n element && element.off('keydown', $typeahead.$onKeyDown);\n }\n if (!options.autoSelect)\n $typeahead.activate(-1);\n hide();\n };\n\n return $typeahead;\n\n }\n\n // Helper functions\n\n function safeDigest(scope) {\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n }\n\n TypeaheadFactory.defaults = defaults;\n return TypeaheadFactory;\n\n };\n\n })\n\n .filter('bsAsyncFilter', function($filter) {\n return function(array, expression, comparator) {\n if (array && angular.isFunction(array.then)) {\n return array.then(function(results) {\n return $filter('filter')(results, expression, comparator);\n });\n } else {\n return $filter('filter')(array, expression, comparator);\n }\n };\n })\n\n .directive('bsTypeahead', function($window, $parse, $q, $typeahead, $parseOptions) {\n\n var defaults = $typeahead.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink(scope, element, attr, controller) {\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'filter', 'limit', 'minLength', 'watchOptions', 'selectMode', 'autoSelect', 'comparator', 'id', 'prefixEvent', 'prefixClass'], function(key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'trimValue'], function(key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // Disable browser autocompletion\n if (!element.attr('autocomplete')) element.attr('autocomplete', 'off');\n\n // Build proper bsOptions\n var filter = options.filter || defaults.filter;\n var limit = options.limit || defaults.limit;\n var comparator = options.comparator || defaults.comparator;\n\n var bsOptions = attr.bsOptions;\n if (filter) bsOptions += ' | ' + filter + ':$viewValue';\n if (comparator) bsOptions += ':' + comparator;\n if (limit) bsOptions += ' | limitTo:' + limit;\n var parsedOptions = $parseOptions(bsOptions);\n\n // Initialize typeahead\n var typeahead = $typeahead(element, controller, options);\n\n // Watch options on demand\n if (options.watchOptions) {\n // Watch bsOptions values before filtering for changes, drop function calls\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').replace(/\\(.*\\)/g, '').trim();\n scope.$watchCollection(watchedOptions, function(newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller).then(function(values) {\n typeahead.update(values);\n controller.$render();\n });\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function(newValue, oldValue) {\n // console.warn('$watch', element.attr('ng-model'), newValue);\n scope.$modelValue = newValue; // Publish modelValue on scope for custom templates\n parsedOptions.valuesFn(scope, controller)\n .then(function(values) {\n // Prevent input with no future prospect if selectMode is truthy\n // @TODO test selectMode\n if (options.selectMode && !values.length && newValue.length > 0) {\n controller.$setViewValue(controller.$viewValue.substring(0, controller.$viewValue.length - 1));\n return;\n }\n if (values.length > limit) values = values.slice(0, limit);\n typeahead.update(values);\n // Queue a new rendering that will leverage collection loading\n controller.$render();\n });\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function(modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var displayValue = parsedOptions.displayValue(modelValue);\n\n // If we can determine the displayValue, use that\n if (displayValue) {\n return displayValue;\n }\n\n // If there's no display value, attempt to use the modelValue.\n // If the model is an object not much we can do\n if (modelValue && typeof modelValue !== 'object') {\n return modelValue;\n }\n return '';\n });\n\n // Model rendering in view\n controller.$render = function() {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n if (controller.$isEmpty(controller.$viewValue)) {\n return element.val('');\n }\n var index = typeahead.$getIndex(controller.$modelValue);\n var selected = angular.isDefined(index) ? typeahead.$scope.$matches[index].label : controller.$viewValue;\n selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected;\n var value = selected ? selected.toString().replace(/<(?:.|\\n)*?>/gm, '') : '';\n element.val(options.trimValue === false ? value : value.trim());\n };\n\n // Garbage collection\n scope.$on('$destroy', function() {\n if (typeahead) typeahead.destroy();\n options = null;\n typeahead = null;\n });\n\n }\n };\n\n });\n"],"sourceRoot":"/source/"} \ No newline at end of file +{"version":3,"sources":["modules/typeahead.js"],"names":["angular","trigger","provider","container","this","defaults","keyboard","html","delay","minLength","filter","limit","autoSelect","comparator","trimValue","$get","$typeahead","options","parentScope","scope","$resetMatches","$window","$rootScope","$tooltip","$$rAF","$timeout","$matches","extend","config","$$postDigest","$scope","activate","$activeIndex","select","index","$select","evt","matches","$isVisible","safeDigest","update","value","controller","$render","$emit","prefixEvent","onSelect","$setViewValue","length","isDefined","isFunction","equals","$onMouseDown","preventDefault","stopPropagation","clientTop","active","offsetTop","clientBottom","highWatermark","scrollTop","Math","children","lowWatermark","clientHeight","keyCode","$$updateScrollTop","$digest","show","isUndefined","$element","element","hide","$onKeyDown","$$phase","TypeaheadFactory","array","then","$filter","expression","directive","results","postLink","attr","restrict","forEach","require","link","falseValueRegExp","key","bsKey","charAt","toUpperCase","slice","test","bsOptions","parsedOptions","$parseOptions","typeahead","watchedOptions","$watchCollection","values","watchOptions","$match","replace","trim","$watch","ngModel","newValue","oldValue","$modelValue","valuesFn","selectMode","$viewValue","substring","displayValue","$formatters","push","modelValue","selected","ss","selectionStart","val","selectionEnd","setSelectionRange","sd","label","isObject","destroy","$on"],"mappings":"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","file":"typeahead.min.js","sourcesContent":["'use strict';\n\nangular.module('mgcrea.ngStrap.typeahead', ['mgcrea.ngStrap.tooltip', 'mgcrea.ngStrap.helpers.parseOptions'])\n\n .provider('$typeahead', function () {\n\n var defaults = this.defaults = {\n animation: 'am-fade',\n prefixClass: 'typeahead',\n prefixEvent: '$typeahead',\n placement: 'bottom-left',\n templateUrl: 'typeahead/typeahead.tpl.html',\n trigger: 'focus',\n container: false,\n keyboard: true,\n html: false,\n delay: 0,\n minLength: 1,\n filter: 'bsAsyncFilter',\n limit: 6,\n autoSelect: false,\n comparator: '',\n trimValue: true\n };\n\n this.$get = function ($window, $rootScope, $tooltip, $$rAF, $timeout) {\n\n function TypeaheadFactory (element, controller, config) {\n\n var $typeahead = {};\n\n // Common vars\n var options = angular.extend({}, defaults, config);\n\n $typeahead = $tooltip(element, options);\n var parentScope = config.scope;\n var scope = $typeahead.$scope;\n\n scope.$resetMatches = function () {\n scope.$matches = [];\n scope.$activeIndex = options.autoSelect ? 0 : -1; // If set to 0, the first match will be highlighted\n };\n scope.$resetMatches();\n\n scope.$activate = function (index) {\n scope.$$postDigest(function () {\n $typeahead.activate(index);\n });\n };\n\n scope.$select = function (index, evt) {\n scope.$$postDigest(function () {\n $typeahead.select(index);\n });\n };\n\n scope.$isVisible = function () {\n return $typeahead.$isVisible();\n };\n\n // Public methods\n\n $typeahead.update = function (matches) {\n scope.$matches = matches;\n if (scope.$activeIndex >= matches.length) {\n scope.$activeIndex = options.autoSelect ? 0 : -1;\n }\n\n // wrap in a $timeout so the results are updated\n // before repositioning\n safeDigest(scope);\n $$rAF($typeahead.$applyPlacement);\n };\n\n $typeahead.activate = function (index) {\n scope.$activeIndex = index;\n };\n\n $typeahead.select = function (index) {\n if (index === -1) return;\n var value = scope.$matches[index].value;\n // console.log('$setViewValue', value);\n controller.$setViewValue(value);\n controller.$render();\n scope.$resetMatches();\n if (parentScope) parentScope.$digest();\n // Emit event\n scope.$emit(options.prefixEvent + '.select', value, index, $typeahead);\n if (angular.isDefined(options.onSelect) && angular.isFunction(options.onSelect)) {\n options.onSelect(value, index, $typeahead);\n }\n };\n\n // Protected methods\n\n $typeahead.$isVisible = function () {\n if (!options.minLength || !controller) {\n return !!scope.$matches.length;\n }\n // minLength support\n return scope.$matches.length && angular.isString(controller.$viewValue) && controller.$viewValue.length >= options.minLength;\n };\n\n $typeahead.$getIndex = function (value) {\n var index;\n for (index = scope.$matches.length; index--;) {\n if (angular.equals(scope.$matches[index].value, value)) break;\n }\n return index;\n };\n\n $typeahead.$onMouseDown = function (evt) {\n // Prevent blur on mousedown\n evt.preventDefault();\n evt.stopPropagation();\n };\n\n $typeahead.$$updateScrollTop = function (container, index) {\n if (index > -1 && index < container.children.length) {\n var active = container.children[index];\n var clientTop = active.offsetTop;\n var clientBottom = active.offsetTop + active.clientHeight;\n var highWatermark = container.scrollTop;\n var lowWatermark = container.scrollTop + container.clientHeight;\n\n // active entry overlaps top border\n if (clientBottom >= highWatermark && clientTop < highWatermark) {\n container.scrollTop = Math.max(0, container.scrollTop - container.clientHeight);\n } else if (clientBottom > lowWatermark) {\n // top of active element is invisible because it's below the bottom of the visible container window\n container.scrollTop = clientTop;\n }\n }\n };\n\n $typeahead.$onKeyDown = function (evt) {\n if (!/(38|40|13)/.test(evt.keyCode)) return;\n\n // Let ngSubmit pass if the typeahead tip is hidden or no option is selected\n if ($typeahead.$isVisible() && !(evt.keyCode === 13 && scope.$activeIndex === -1)) {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n // Select with enter\n if (evt.keyCode === 13 && scope.$matches.length) {\n $typeahead.select(scope.$activeIndex);\n // Navigate with keyboard\n } else if (evt.keyCode === 38 && scope.$activeIndex > 0) {\n scope.$activeIndex--;\n } else if (evt.keyCode === 40 && scope.$activeIndex < scope.$matches.length - 1) {\n scope.$activeIndex++;\n } else if (angular.isUndefined(scope.$activeIndex)) {\n scope.$activeIndex = 0;\n }\n\n // update scrollTop property on $typeahead when scope.$activeIndex is not in visible area\n $typeahead.$$updateScrollTop($typeahead.$element[0], scope.$activeIndex);\n scope.$digest();\n };\n\n // Overrides\n\n var show = $typeahead.show;\n $typeahead.show = function () {\n show();\n // use timeout to hookup the events to prevent\n // event bubbling from being processed immediately.\n $timeout(function () {\n if ($typeahead.$element) {\n $typeahead.$element.on('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n if (element) element.on('keydown', $typeahead.$onKeyDown);\n }\n }\n }, 0, false);\n };\n\n var hide = $typeahead.hide;\n $typeahead.hide = function () {\n if ($typeahead.$element) $typeahead.$element.off('mousedown', $typeahead.$onMouseDown);\n if (options.keyboard) {\n if (element) element.off('keydown', $typeahead.$onKeyDown);\n }\n if (!options.autoSelect) {\n $typeahead.activate(-1);\n }\n hide();\n };\n\n return $typeahead;\n\n }\n\n // Helper functions\n\n function safeDigest (scope) {\n /* eslint-disable no-unused-expressions */\n scope.$$phase || (scope.$root && scope.$root.$$phase) || scope.$digest();\n /* eslint-enable no-unused-expressions */\n }\n\n TypeaheadFactory.defaults = defaults;\n return TypeaheadFactory;\n\n };\n\n })\n\n .filter('bsAsyncFilter', function ($filter) {\n return function (array, expression, comparator) {\n if (array && angular.isFunction(array.then)) {\n return array.then(function (results) {\n return $filter('filter')(results, expression, comparator);\n });\n }\n return $filter('filter')(array, expression, comparator);\n };\n })\n\n .directive('bsTypeahead', function ($window, $parse, $q, $typeahead, $parseOptions) {\n\n var defaults = $typeahead.defaults;\n\n return {\n restrict: 'EAC',\n require: 'ngModel',\n link: function postLink (scope, element, attr, controller) {\n\n // Fixes firefox bug when using objects in model with typeahead\n // Yes this breaks any other directive using a 'change' event on this input,\n // but if it is using the 'change' event why is it used with typeahead?\n element.off('change');\n\n // Directive options\n var options = {\n scope: scope\n };\n angular.forEach(['template', 'templateUrl', 'controller', 'controllerAs', 'placement', 'container', 'delay', 'trigger', 'keyboard', 'html', 'animation', 'filter', 'limit', 'minLength', 'watchOptions', 'selectMode', 'autoSelect', 'comparator', 'id', 'prefixEvent', 'prefixClass'], function (key) {\n if (angular.isDefined(attr[key])) options[key] = attr[key];\n });\n\n // use string regex match boolean attr falsy values, leave truthy values be\n var falseValueRegExp = /^(false|0|)$/i;\n angular.forEach(['html', 'container', 'trimValue', 'filter'], function (key) {\n if (angular.isDefined(attr[key]) && falseValueRegExp.test(attr[key])) options[key] = false;\n });\n\n // bind functions from the attrs to the show, hide and select events\n angular.forEach(['onBeforeShow', 'onShow', 'onBeforeHide', 'onHide', 'onSelect'], function (key) {\n var bsKey = 'bs' + key.charAt(0).toUpperCase() + key.slice(1);\n if (angular.isDefined(attr[bsKey])) {\n options[key] = scope.$eval(attr[bsKey]);\n }\n });\n\n // Disable browser autocompletion\n if (!element.attr('autocomplete')) element.attr('autocomplete', 'off');\n\n // Build proper bsOptions\n var filter = angular.isDefined(options.filter) ? options.filter : defaults.filter;\n var limit = options.limit || defaults.limit;\n var comparator = options.comparator || defaults.comparator;\n\n var bsOptions = attr.bsOptions;\n if (filter) {\n bsOptions += ' | ' + filter + ':$viewValue';\n if (comparator) bsOptions += ':' + comparator;\n }\n if (limit) bsOptions += ' | limitTo:' + limit;\n var parsedOptions = $parseOptions(bsOptions);\n\n // Initialize typeahead\n var typeahead = $typeahead(element, controller, options);\n\n // Watch options on demand\n if (options.watchOptions) {\n // Watch bsOptions values before filtering for changes, drop function calls\n var watchedOptions = parsedOptions.$match[7].replace(/\\|.+/, '').replace(/\\(.*\\)/g, '').trim();\n scope.$watchCollection(watchedOptions, function (newValue, oldValue) {\n // console.warn('scope.$watch(%s)', watchedOptions, newValue, oldValue);\n parsedOptions.valuesFn(scope, controller).then(function (values) {\n typeahead.update(values);\n controller.$render();\n });\n });\n }\n\n // Watch model for changes\n scope.$watch(attr.ngModel, function (newValue, oldValue) {\n // console.warn('$watch', element.attr('ng-model'), newValue);\n scope.$modelValue = newValue; // Publish modelValue on scope for custom templates\n parsedOptions.valuesFn(scope, controller)\n .then(function (values) {\n // Prevent input with no future prospect if selectMode is truthy\n // @TODO test selectMode\n if (options.selectMode && !values.length && newValue.length > 0) {\n controller.$setViewValue(controller.$viewValue.substring(0, controller.$viewValue.length - 1));\n return;\n }\n if (values.length > limit) values = values.slice(0, limit);\n typeahead.update(values);\n // Queue a new rendering that will leverage collection loading\n controller.$render();\n });\n });\n\n // modelValue -> $formatters -> viewValue\n controller.$formatters.push(function (modelValue) {\n // console.warn('$formatter(\"%s\"): modelValue=%o (%o)', element.attr('ng-model'), modelValue, typeof modelValue);\n var displayValue = parsedOptions.displayValue(modelValue);\n\n // If we can determine the displayValue, use that\n if (displayValue) {\n return displayValue;\n }\n\n // If there's no display value, attempt to use the modelValue.\n // If the model is an object not much we can do\n if (angular.isDefined(modelValue) && typeof modelValue !== 'object') {\n return modelValue;\n }\n return '';\n });\n\n // Model rendering in view\n controller.$render = function () {\n // console.warn('$render', element.attr('ng-model'), 'controller.$modelValue', typeof controller.$modelValue, controller.$modelValue, 'controller.$viewValue', typeof controller.$viewValue, controller.$viewValue);\n if (controller.$isEmpty(controller.$viewValue)) {\n return element.val('');\n }\n var index = typeahead.$getIndex(controller.$modelValue);\n var selected = index !== -1 ? typeahead.$scope.$matches[index].label : controller.$viewValue;\n selected = angular.isObject(selected) ? parsedOptions.displayValue(selected) : selected;\n var value = selected ? selected.toString().replace(/<(?:.|\\n)*?>/gm, '') : '';\n var ss = element[0].selectionStart;\n var sd = element[0].selectionEnd;\n element.val(options.trimValue === false ? value : value.trim());\n element[0].setSelectionRange(ss, sd);\n };\n\n // Garbage collection\n scope.$on('$destroy', function () {\n if (typeahead) typeahead.destroy();\n options = null;\n typeahead = null;\n });\n\n }\n };\n\n });\n"]} \ No newline at end of file diff --git a/dist/modules/typeahead.tpl.js b/dist/modules/typeahead.tpl.js index 28861f3f5..3ad638ad5 100644 --- a/dist/modules/typeahead.tpl.js +++ b/dist/modules/typeahead.tpl.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/dist/modules/typeahead.tpl.min.js b/dist/modules/typeahead.tpl.min.js index 59d3e09c8..225b67747 100644 --- a/dist/modules/typeahead.tpl.min.js +++ b/dist/modules/typeahead.tpl.min.js @@ -1,6 +1,6 @@ /** * angular-strap - * @version v2.3.5 - 2015-10-29 + * @version v2.3.12 - 2017-01-26 * @link http://mgcrea.github.io/angular-strap * @author Olivier Louvignes (https://github.com/mgcrea) * @license MIT License, http://www.opensource.org/licenses/MIT diff --git a/docs/bower.json b/docs/bower.json index 764020ac2..b813f307c 100644 --- a/docs/bower.json +++ b/docs/bower.json @@ -1,16 +1,16 @@ { "name": "angular-strap-docs", "dependencies": { - "angular": "~1.4", - "angular-animate": "~1.4", - "angular-i18n": "~1.4", - "angular-motion": "^0.4.3", - "angular-route": "~1.4", - "bootstrap": "~3.3.5", + "angular": "~1.5", + "angular-animate": "~1.5", + "angular-i18n": "~1.5", + "angular-motion": "^0.4.4", + "angular-route": "~1.5", + "bootstrap": "~3.3.6", "bootstrap-additions": "^0.3.1", "fastclick": "~1.0.6", - "font-awesome": "~4.4.0", - "highlightjs": "~8.8.0" + "font-awesome": "~4.5.0", + "highlightjs": "~9.2.0" }, "devDependencies": { } diff --git a/docs/dev.html b/docs/dev.html index 891a38b5b..36553bc3b 100644 --- a/docs/dev.html +++ b/docs/dev.html @@ -46,7 +46,7 @@
- +
@@ -58,7 +58,7 @@ -
+ diff --git a/docs/fonts/glyphicons-halflings-regular.eot b/docs/fonts/glyphicons-halflings-regular.eot new file mode 100644 index 000000000..b93a4953f Binary files /dev/null and b/docs/fonts/glyphicons-halflings-regular.eot differ diff --git a/docs/fonts/glyphicons-halflings-regular.svg b/docs/fonts/glyphicons-halflings-regular.svg new file mode 100644 index 000000000..94fb5490a --- /dev/null +++ b/docs/fonts/glyphicons-halflings-regular.svg @@ -0,0 +1,288 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/docs/fonts/glyphicons-halflings-regular.ttf b/docs/fonts/glyphicons-halflings-regular.ttf new file mode 100644 index 000000000..1413fc609 Binary files /dev/null and b/docs/fonts/glyphicons-halflings-regular.ttf differ diff --git a/docs/fonts/glyphicons-halflings-regular.woff b/docs/fonts/glyphicons-halflings-regular.woff new file mode 100644 index 000000000..9e612858f Binary files /dev/null and b/docs/fonts/glyphicons-halflings-regular.woff differ diff --git a/docs/fonts/glyphicons-halflings-regular.woff2 b/docs/fonts/glyphicons-halflings-regular.woff2 new file mode 100644 index 000000000..64539b54c Binary files /dev/null and b/docs/fonts/glyphicons-halflings-regular.woff2 differ diff --git a/docs/scripts/app.js b/docs/scripts/app.js index 497fb7236..4c38d961f 100644 --- a/docs/scripts/app.js +++ b/docs/scripts/app.js @@ -1,10 +1,8 @@ 'use strict'; -// bower install angular#~1.3 angular-route#~1.3 angular-animate#~1.3 -// bower install angular#~1.4 angular-route#~1.4 angular-animate#~1.4 angular.module('mgcrea.ngStrapDocs', ['mgcrea.ngStrap', 'mgcrea.ngPlunkr', 'ngRoute', 'ngAnimate']) -.constant('version', 'v2.3.5') +.constant('version', 'v2.3.12') .constant('ngVersion', angular.version.full) .config(function($plunkrProvider, version) { diff --git a/docs/scripts/services/angular-plunkr.js b/docs/scripts/services/angular-plunkr.js index 2bbfcc8a9..dd5ca4891 100644 --- a/docs/scripts/services/angular-plunkr.js +++ b/docs/scripts/services/angular-plunkr.js @@ -4,7 +4,7 @@ angular.module('mgcrea.ngPlunkr', ['mgcrea.ngStrap.helpers.debounce']) .run(function($templateCache, version) { - var ngVersion = '1.4.5'; + var ngVersion = '1.5.5'; var templateHtml = '' + '\n' + @@ -16,8 +16,8 @@ angular.module('mgcrea.ngPlunkr', ['mgcrea.ngStrap.helpers.debounce']) ' \n' + // styles - ' \n' + - ' \n' + + ' \n' + + ' \n' + ' \n' + ' \n' + ' \n' + diff --git a/docs/styles/main.less b/docs/styles/main.less index faaeecb97..c56737a47 100644 --- a/docs/styles/main.less +++ b/docs/styles/main.less @@ -115,7 +115,7 @@ body { .datepicker.dropdown-menu { width: 250px; - height: 270px; + height: 290px; button { outline: none; border: 0px; @@ -319,3 +319,8 @@ input[type="checkbox"] { max-height: 0px; } } + +.typeahead.dropdown-menu { + max-height: 112px; + overflow-y: auto; +} diff --git a/docs/tooltip-viewport.html b/docs/tooltip-viewport.html index 4731182b1..084979249 100644 --- a/docs/tooltip-viewport.html +++ b/docs/tooltip-viewport.html @@ -55,6 +55,9 @@ min-height: 100px; text-align: left; } + .btn { + font-size: 8px; + } .container-viewport { position: absolute; top: 100px; @@ -84,6 +87,12 @@ + + + + + + @@ -99,21 +108,68 @@
- - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + - diff --git a/docs/views/partials/footer.html b/docs/views/partials/footer.html index e38c4fd61..01c1bb3b7 100644 --- a/docs/views/partials/footer.html +++ b/docs/views/partials/footer.html @@ -6,7 +6,7 @@


-

Using Twitter Bootstrap and the Bootstrap's docs styles designed and built by @mdo and @fat.

+

Using Bootstrap and the Bootstrap's docs styles designed and built by @mdo and @fat.

Code licensed under The MIT License, documentation under CC BY 3.0.