/*!
|
|
* Angular Material Design
|
|
* https://github.com/angular/material
|
|
* @license MIT
|
|
* v1.1.3
|
|
*/
|
|
(function( window, angular, undefined ){
|
|
"use strict";
|
|
|
|
/**
|
|
* @ngdoc module
|
|
* @name material.components.input
|
|
*/
|
|
mdInputContainerDirective['$inject'] = ["$mdTheming", "$parse"];
|
|
inputTextareaDirective['$inject'] = ["$mdUtil", "$window", "$mdAria", "$timeout", "$mdGesture"];
|
|
mdMaxlengthDirective['$inject'] = ["$animate", "$mdUtil"];
|
|
placeholderDirective['$inject'] = ["$compile"];
|
|
ngMessageDirective['$inject'] = ["$mdUtil"];
|
|
mdSelectOnFocusDirective['$inject'] = ["$timeout"];
|
|
mdInputInvalidMessagesAnimation['$inject'] = ["$$AnimateRunner", "$animateCss", "$mdUtil", "$log"];
|
|
ngMessagesAnimation['$inject'] = ["$$AnimateRunner", "$animateCss", "$mdUtil", "$log"];
|
|
ngMessageAnimation['$inject'] = ["$$AnimateRunner", "$animateCss", "$mdUtil", "$log"];
|
|
var inputModule = angular.module('material.components.input', [
|
|
'material.core'
|
|
])
|
|
.directive('mdInputContainer', mdInputContainerDirective)
|
|
.directive('label', labelDirective)
|
|
.directive('input', inputTextareaDirective)
|
|
.directive('textarea', inputTextareaDirective)
|
|
.directive('mdMaxlength', mdMaxlengthDirective)
|
|
.directive('placeholder', placeholderDirective)
|
|
.directive('ngMessages', ngMessagesDirective)
|
|
.directive('ngMessage', ngMessageDirective)
|
|
.directive('ngMessageExp', ngMessageDirective)
|
|
.directive('mdSelectOnFocus', mdSelectOnFocusDirective)
|
|
|
|
.animation('.md-input-invalid', mdInputInvalidMessagesAnimation)
|
|
.animation('.md-input-messages-animation', ngMessagesAnimation)
|
|
.animation('.md-input-message-animation', ngMessageAnimation);
|
|
|
|
// If we are running inside of tests; expose some extra services so that we can test them
|
|
if (window._mdMocksIncluded) {
|
|
inputModule.service('$$mdInput', function() {
|
|
return {
|
|
// special accessor to internals... useful for testing
|
|
messages: {
|
|
show : showInputMessages,
|
|
hide : hideInputMessages,
|
|
getElement : getMessagesElement
|
|
}
|
|
}
|
|
})
|
|
|
|
// Register a service for each animation so that we can easily inject them into unit tests
|
|
.service('mdInputInvalidAnimation', mdInputInvalidMessagesAnimation)
|
|
.service('mdInputMessagesAnimation', ngMessagesAnimation)
|
|
.service('mdInputMessageAnimation', ngMessageAnimation);
|
|
}
|
|
|
|
/**
|
|
* @ngdoc directive
|
|
* @name mdInputContainer
|
|
* @module material.components.input
|
|
*
|
|
* @restrict E
|
|
*
|
|
* @description
|
|
* `<md-input-container>` is the parent of any input or textarea element.
|
|
*
|
|
* Input and textarea elements will not behave properly unless the md-input-container
|
|
* parent is provided.
|
|
*
|
|
* A single `<md-input-container>` should contain only one `<input>` element, otherwise it will throw an error.
|
|
*
|
|
* <b>Exception:</b> Hidden inputs (`<input type="hidden" />`) are ignored and will not throw an error, so
|
|
* you may combine these with other inputs.
|
|
*
|
|
* <b>Note:</b> When using `ngMessages` with your input element, make sure the message and container elements
|
|
* are *block* elements, otherwise animations applied to the messages will not look as intended. Either use a `div` and
|
|
* apply the `ng-message` and `ng-messages` classes respectively, or use the `md-block` class on your element.
|
|
*
|
|
* @param md-is-error {expression=} When the given expression evaluates to true, the input container
|
|
* will go into error state. Defaults to erroring if the input has been touched and is invalid.
|
|
* @param md-no-float {boolean=} When present, `placeholder` attributes on the input will not be converted to floating
|
|
* labels.
|
|
*
|
|
* @usage
|
|
* <hljs lang="html">
|
|
*
|
|
* <md-input-container>
|
|
* <label>Username</label>
|
|
* <input type="text" ng-model="user.name">
|
|
* </md-input-container>
|
|
*
|
|
* <md-input-container>
|
|
* <label>Description</label>
|
|
* <textarea ng-model="user.description"></textarea>
|
|
* </md-input-container>
|
|
*
|
|
* </hljs>
|
|
*
|
|
* <h3>When disabling floating labels</h3>
|
|
* <hljs lang="html">
|
|
*
|
|
* <md-input-container md-no-float>
|
|
* <input type="text" placeholder="Non-Floating Label">
|
|
* </md-input-container>
|
|
*
|
|
* </hljs>
|
|
*/
|
|
function mdInputContainerDirective($mdTheming, $parse) {
|
|
|
|
ContainerCtrl['$inject'] = ["$scope", "$element", "$attrs", "$animate"];
|
|
var INPUT_TAGS = ['INPUT', 'TEXTAREA', 'SELECT', 'MD-SELECT'];
|
|
|
|
var LEFT_SELECTORS = INPUT_TAGS.reduce(function(selectors, isel) {
|
|
return selectors.concat(['md-icon ~ ' + isel, '.md-icon ~ ' + isel]);
|
|
}, []).join(",");
|
|
|
|
var RIGHT_SELECTORS = INPUT_TAGS.reduce(function(selectors, isel) {
|
|
return selectors.concat([isel + ' ~ md-icon', isel + ' ~ .md-icon']);
|
|
}, []).join(",");
|
|
|
|
return {
|
|
restrict: 'E',
|
|
compile: compile,
|
|
controller: ContainerCtrl
|
|
};
|
|
|
|
function compile(tElement) {
|
|
// Check for both a left & right icon
|
|
var leftIcon = tElement[0].querySelector(LEFT_SELECTORS);
|
|
var rightIcon = tElement[0].querySelector(RIGHT_SELECTORS);
|
|
|
|
if (leftIcon) { tElement.addClass('md-icon-left'); }
|
|
if (rightIcon) { tElement.addClass('md-icon-right'); }
|
|
|
|
return function postLink(scope, element) {
|
|
$mdTheming(element);
|
|
};
|
|
}
|
|
|
|
function ContainerCtrl($scope, $element, $attrs, $animate) {
|
|
var self = this;
|
|
|
|
self.isErrorGetter = $attrs.mdIsError && $parse($attrs.mdIsError);
|
|
|
|
self.delegateClick = function() {
|
|
self.input.focus();
|
|
};
|
|
self.element = $element;
|
|
self.setFocused = function(isFocused) {
|
|
$element.toggleClass('md-input-focused', !!isFocused);
|
|
};
|
|
self.setHasValue = function(hasValue) {
|
|
$element.toggleClass('md-input-has-value', !!hasValue);
|
|
};
|
|
self.setHasPlaceholder = function(hasPlaceholder) {
|
|
$element.toggleClass('md-input-has-placeholder', !!hasPlaceholder);
|
|
};
|
|
self.setInvalid = function(isInvalid) {
|
|
if (isInvalid) {
|
|
$animate.addClass($element, 'md-input-invalid');
|
|
} else {
|
|
$animate.removeClass($element, 'md-input-invalid');
|
|
}
|
|
};
|
|
$scope.$watch(function() {
|
|
return self.label && self.input;
|
|
}, function(hasLabelAndInput) {
|
|
if (hasLabelAndInput && !self.label.attr('for')) {
|
|
self.label.attr('for', self.input.attr('id'));
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
function labelDirective() {
|
|
return {
|
|
restrict: 'E',
|
|
require: '^?mdInputContainer',
|
|
link: function(scope, element, attr, containerCtrl) {
|
|
if (!containerCtrl || attr.mdNoFloat || element.hasClass('md-container-ignore')) return;
|
|
|
|
containerCtrl.label = element;
|
|
scope.$on('$destroy', function() {
|
|
containerCtrl.label = null;
|
|
});
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @ngdoc directive
|
|
* @name mdInput
|
|
* @restrict E
|
|
* @module material.components.input
|
|
*
|
|
* @description
|
|
* You can use any `<input>` or `<textarea>` element as a child of an `<md-input-container>`. This
|
|
* allows you to build complex forms for data entry.
|
|
*
|
|
* When the input is required and uses a floating label, then the label will automatically contain
|
|
* an asterisk (`*`).<br/>
|
|
* This behavior can be disabled by using the `md-no-asterisk` attribute.
|
|
*
|
|
* @param {number=} md-maxlength The maximum number of characters allowed in this input. If this is
|
|
* specified, a character counter will be shown underneath the input.<br/><br/>
|
|
* The purpose of **`md-maxlength`** is exactly to show the max length counter text. If you don't
|
|
* want the counter text and only need "plain" validation, you can use the "simple" `ng-maxlength`
|
|
* or maxlength attributes.<br/><br/>
|
|
* **Note:** Only valid for text/string inputs (not numeric).
|
|
*
|
|
* @param {string=} aria-label Aria-label is required when no label is present. A warning message
|
|
* will be logged in the console if not present.
|
|
* @param {string=} placeholder An alternative approach to using aria-label when the label is not
|
|
* PRESENT. The placeholder text is copied to the aria-label attribute.
|
|
* @param md-no-autogrow {boolean=} When present, textareas will not grow automatically.
|
|
* @param md-no-asterisk {boolean=} When present, an asterisk will not be appended to the inputs floating label
|
|
* @param md-no-resize {boolean=} Disables the textarea resize handle.
|
|
* @param {number=} max-rows The maximum amount of rows for a textarea.
|
|
* @param md-detect-hidden {boolean=} When present, textareas will be sized properly when they are
|
|
* revealed after being hidden. This is off by default for performance reasons because it
|
|
* guarantees a reflow every digest cycle.
|
|
*
|
|
* @usage
|
|
* <hljs lang="html">
|
|
* <md-input-container>
|
|
* <label>Color</label>
|
|
* <input type="text" ng-model="color" required md-maxlength="10">
|
|
* </md-input-container>
|
|
* </hljs>
|
|
*
|
|
* <h3>With Errors</h3>
|
|
*
|
|
* `md-input-container` also supports errors using the standard `ng-messages` directives and
|
|
* animates the messages when they become visible using from the `ngEnter`/`ngLeave` events or
|
|
* the `ngShow`/`ngHide` events.
|
|
*
|
|
* By default, the messages will be hidden until the input is in an error state. This is based off
|
|
* of the `md-is-error` expression of the `md-input-container`. This gives the user a chance to
|
|
* fill out the form before the errors become visible.
|
|
*
|
|
* <hljs lang="html">
|
|
* <form name="colorForm">
|
|
* <md-input-container>
|
|
* <label>Favorite Color</label>
|
|
* <input name="favoriteColor" ng-model="favoriteColor" required>
|
|
* <div ng-messages="colorForm.favoriteColor.$error">
|
|
* <div ng-message="required">This is required!</div>
|
|
* </div>
|
|
* </md-input-container>
|
|
* </form>
|
|
* </hljs>
|
|
*
|
|
* We automatically disable this auto-hiding functionality if you provide any of the following
|
|
* visibility directives on the `ng-messages` container:
|
|
*
|
|
* - `ng-if`
|
|
* - `ng-show`/`ng-hide`
|
|
* - `ng-switch-when`/`ng-switch-default`
|
|
*
|
|
* You can also disable this functionality manually by adding the `md-auto-hide="false"` expression
|
|
* to the `ng-messages` container. This may be helpful if you always want to see the error messages
|
|
* or if you are building your own visibilty directive.
|
|
*
|
|
* _<b>Note:</b> The `md-auto-hide` attribute is a static string that is only checked upon
|
|
* initialization of the `ng-messages` directive to see if it equals the string `false`._
|
|
*
|
|
* <hljs lang="html">
|
|
* <form name="userForm">
|
|
* <md-input-container>
|
|
* <label>Last Name</label>
|
|
* <input name="lastName" ng-model="lastName" required md-maxlength="10" minlength="4">
|
|
* <div ng-messages="userForm.lastName.$error" ng-show="userForm.lastName.$dirty">
|
|
* <div ng-message="required">This is required!</div>
|
|
* <div ng-message="md-maxlength">That's too long!</div>
|
|
* <div ng-message="minlength">That's too short!</div>
|
|
* </div>
|
|
* </md-input-container>
|
|
* <md-input-container>
|
|
* <label>Biography</label>
|
|
* <textarea name="bio" ng-model="biography" required md-maxlength="150"></textarea>
|
|
* <div ng-messages="userForm.bio.$error" ng-show="userForm.bio.$dirty">
|
|
* <div ng-message="required">This is required!</div>
|
|
* <div ng-message="md-maxlength">That's too long!</div>
|
|
* </div>
|
|
* </md-input-container>
|
|
* <md-input-container>
|
|
* <input aria-label='title' ng-model='title'>
|
|
* </md-input-container>
|
|
* <md-input-container>
|
|
* <input placeholder='title' ng-model='title'>
|
|
* </md-input-container>
|
|
* </form>
|
|
* </hljs>
|
|
*
|
|
* <h3>Notes</h3>
|
|
*
|
|
* - Requires [ngMessages](https://docs.angularjs.org/api/ngMessages).
|
|
* - Behaves like the [AngularJS input directive](https://docs.angularjs.org/api/ng/directive/input).
|
|
*
|
|
* The `md-input` and `md-input-container` directives use very specific positioning to achieve the
|
|
* error animation effects. Therefore, it is *not* advised to use the Layout system inside of the
|
|
* `<md-input-container>` tags. Instead, use relative or absolute positioning.
|
|
*
|
|
*
|
|
* <h3>Textarea directive</h3>
|
|
* The `textarea` element within a `md-input-container` has the following specific behavior:
|
|
* - By default the `textarea` grows as the user types. This can be disabled via the `md-no-autogrow`
|
|
* attribute.
|
|
* - If a `textarea` has the `rows` attribute, it will treat the `rows` as the minimum height and will
|
|
* continue growing as the user types. For example a textarea with `rows="3"` will be 3 lines of text
|
|
* high initially. If no rows are specified, the directive defaults to 1.
|
|
* - The textarea's height gets set on initialization, as well as while the user is typing. In certain situations
|
|
* (e.g. while animating) the directive might have been initialized, before the element got it's final height. In
|
|
* those cases, you can trigger a resize manually by broadcasting a `md-resize-textarea` event on the scope.
|
|
* - If you wan't a `textarea` to stop growing at a certain point, you can specify the `max-rows` attribute.
|
|
* - The textarea's bottom border acts as a handle which users can drag, in order to resize the element vertically.
|
|
* Once the user has resized a `textarea`, the autogrowing functionality becomes disabled. If you don't want a
|
|
* `textarea` to be resizeable by the user, you can add the `md-no-resize` attribute.
|
|
*/
|
|
|
|
function inputTextareaDirective($mdUtil, $window, $mdAria, $timeout, $mdGesture) {
|
|
return {
|
|
restrict: 'E',
|
|
require: ['^?mdInputContainer', '?ngModel', '?^form'],
|
|
link: postLink
|
|
};
|
|
|
|
function postLink(scope, element, attr, ctrls) {
|
|
|
|
var containerCtrl = ctrls[0];
|
|
var hasNgModel = !!ctrls[1];
|
|
var ngModelCtrl = ctrls[1] || $mdUtil.fakeNgModel();
|
|
var parentForm = ctrls[2];
|
|
var isReadonly = angular.isDefined(attr.readonly);
|
|
var mdNoAsterisk = $mdUtil.parseAttributeBoolean(attr.mdNoAsterisk);
|
|
var tagName = element[0].tagName.toLowerCase();
|
|
|
|
|
|
if (!containerCtrl) return;
|
|
if (attr.type === 'hidden') {
|
|
element.attr('aria-hidden', 'true');
|
|
return;
|
|
} else if (containerCtrl.input) {
|
|
if (containerCtrl.input[0].contains(element[0])) {
|
|
return;
|
|
} else {
|
|
throw new Error("<md-input-container> can only have *one* <input>, <textarea> or <md-select> child element!");
|
|
}
|
|
}
|
|
containerCtrl.input = element;
|
|
|
|
setupAttributeWatchers();
|
|
|
|
// Add an error spacer div after our input to provide space for the char counter and any ng-messages
|
|
var errorsSpacer = angular.element('<div class="md-errors-spacer">');
|
|
element.after(errorsSpacer);
|
|
|
|
if (!containerCtrl.label) {
|
|
$mdAria.expect(element, 'aria-label', attr.placeholder);
|
|
}
|
|
|
|
element.addClass('md-input');
|
|
if (!element.attr('id')) {
|
|
element.attr('id', 'input_' + $mdUtil.nextUid());
|
|
}
|
|
|
|
// This works around a Webkit issue where number inputs, placed in a flexbox, that have
|
|
// a `min` and `max` will collapse to about 1/3 of their proper width. Please check #7349
|
|
// for more info. Also note that we don't override the `step` if the user has specified it,
|
|
// in order to prevent some unexpected behaviour.
|
|
if (tagName === 'input' && attr.type === 'number' && attr.min && attr.max && !attr.step) {
|
|
element.attr('step', 'any');
|
|
} else if (tagName === 'textarea') {
|
|
setupTextarea();
|
|
}
|
|
|
|
// If the input doesn't have an ngModel, it may have a static value. For that case,
|
|
// we have to do one initial check to determine if the container should be in the
|
|
// "has a value" state.
|
|
if (!hasNgModel) {
|
|
inputCheckValue();
|
|
}
|
|
|
|
var isErrorGetter = containerCtrl.isErrorGetter || function() {
|
|
return ngModelCtrl.$invalid && (ngModelCtrl.$touched || (parentForm && parentForm.$submitted));
|
|
};
|
|
|
|
scope.$watch(isErrorGetter, containerCtrl.setInvalid);
|
|
|
|
// When the developer uses the ngValue directive for the input, we have to observe the attribute, because
|
|
// Angular's ngValue directive is just setting the `value` attribute.
|
|
if (attr.ngValue) {
|
|
attr.$observe('value', inputCheckValue);
|
|
}
|
|
|
|
ngModelCtrl.$parsers.push(ngModelPipelineCheckValue);
|
|
ngModelCtrl.$formatters.push(ngModelPipelineCheckValue);
|
|
|
|
element.on('input', inputCheckValue);
|
|
|
|
if (!isReadonly) {
|
|
element
|
|
.on('focus', function(ev) {
|
|
$mdUtil.nextTick(function() {
|
|
containerCtrl.setFocused(true);
|
|
});
|
|
})
|
|
.on('blur', function(ev) {
|
|
$mdUtil.nextTick(function() {
|
|
containerCtrl.setFocused(false);
|
|
inputCheckValue();
|
|
});
|
|
});
|
|
}
|
|
|
|
scope.$on('$destroy', function() {
|
|
containerCtrl.setFocused(false);
|
|
containerCtrl.setHasValue(false);
|
|
containerCtrl.input = null;
|
|
});
|
|
|
|
/** Gets run through ngModel's pipeline and set the `has-value` class on the container. */
|
|
function ngModelPipelineCheckValue(arg) {
|
|
containerCtrl.setHasValue(!ngModelCtrl.$isEmpty(arg));
|
|
return arg;
|
|
}
|
|
|
|
function setupAttributeWatchers() {
|
|
if (containerCtrl.label) {
|
|
attr.$observe('required', function (value) {
|
|
// We don't need to parse the required value, it's always a boolean because of angular's
|
|
// required directive.
|
|
containerCtrl.label.toggleClass('md-required', value && !mdNoAsterisk);
|
|
});
|
|
}
|
|
}
|
|
|
|
function inputCheckValue() {
|
|
// An input's value counts if its length > 0,
|
|
// or if the input's validity state says it has bad input (eg string in a number input)
|
|
containerCtrl.setHasValue(element.val().length > 0 || (element[0].validity || {}).badInput);
|
|
}
|
|
|
|
function setupTextarea() {
|
|
var isAutogrowing = !attr.hasOwnProperty('mdNoAutogrow');
|
|
|
|
attachResizeHandle();
|
|
|
|
if (!isAutogrowing) return;
|
|
|
|
// Can't check if height was or not explicity set,
|
|
// so rows attribute will take precedence if present
|
|
var minRows = attr.hasOwnProperty('rows') ? parseInt(attr.rows) : NaN;
|
|
var maxRows = attr.hasOwnProperty('maxRows') ? parseInt(attr.maxRows) : NaN;
|
|
var scopeResizeListener = scope.$on('md-resize-textarea', growTextarea);
|
|
var lineHeight = null;
|
|
var node = element[0];
|
|
|
|
// This timeout is necessary, because the browser needs a little bit
|
|
// of time to calculate the `clientHeight` and `scrollHeight`.
|
|
$timeout(function() {
|
|
$mdUtil.nextTick(growTextarea);
|
|
}, 10, false);
|
|
|
|
// We could leverage ngModel's $parsers here, however it
|
|
// isn't reliable, because Angular trims the input by default,
|
|
// which means that growTextarea won't fire when newlines and
|
|
// spaces are added.
|
|
element.on('input', growTextarea);
|
|
|
|
// We should still use the $formatters, because they fire when
|
|
// the value was changed from outside the textarea.
|
|
if (hasNgModel) {
|
|
ngModelCtrl.$formatters.push(formattersListener);
|
|
}
|
|
|
|
if (!minRows) {
|
|
element.attr('rows', 1);
|
|
}
|
|
|
|
angular.element($window).on('resize', growTextarea);
|
|
scope.$on('$destroy', disableAutogrow);
|
|
|
|
function growTextarea() {
|
|
// temporarily disables element's flex so its height 'runs free'
|
|
element
|
|
.attr('rows', 1)
|
|
.css('height', 'auto')
|
|
.addClass('md-no-flex');
|
|
|
|
var height = getHeight();
|
|
|
|
if (!lineHeight) {
|
|
// offsetHeight includes padding which can throw off our value
|
|
var originalPadding = element[0].style.padding || '';
|
|
lineHeight = element.css('padding', 0).prop('offsetHeight');
|
|
element[0].style.padding = originalPadding;
|
|
}
|
|
|
|
if (minRows && lineHeight) {
|
|
height = Math.max(height, lineHeight * minRows);
|
|
}
|
|
|
|
if (maxRows && lineHeight) {
|
|
var maxHeight = lineHeight * maxRows;
|
|
|
|
if (maxHeight < height) {
|
|
element.attr('md-no-autogrow', '');
|
|
height = maxHeight;
|
|
} else {
|
|
element.removeAttr('md-no-autogrow');
|
|
}
|
|
}
|
|
|
|
if (lineHeight) {
|
|
element.attr('rows', Math.round(height / lineHeight));
|
|
}
|
|
|
|
element
|
|
.css('height', height + 'px')
|
|
.removeClass('md-no-flex');
|
|
}
|
|
|
|
function getHeight() {
|
|
var offsetHeight = node.offsetHeight;
|
|
var line = node.scrollHeight - offsetHeight;
|
|
return offsetHeight + Math.max(line, 0);
|
|
}
|
|
|
|
function formattersListener(value) {
|
|
$mdUtil.nextTick(growTextarea);
|
|
return value;
|
|
}
|
|
|
|
function disableAutogrow() {
|
|
if (!isAutogrowing) return;
|
|
|
|
isAutogrowing = false;
|
|
angular.element($window).off('resize', growTextarea);
|
|
scopeResizeListener && scopeResizeListener();
|
|
element
|
|
.attr('md-no-autogrow', '')
|
|
.off('input', growTextarea);
|
|
|
|
if (hasNgModel) {
|
|
var listenerIndex = ngModelCtrl.$formatters.indexOf(formattersListener);
|
|
|
|
if (listenerIndex > -1) {
|
|
ngModelCtrl.$formatters.splice(listenerIndex, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
function attachResizeHandle() {
|
|
if (attr.hasOwnProperty('mdNoResize')) return;
|
|
|
|
var handle = angular.element('<div class="md-resize-handle"></div>');
|
|
var isDragging = false;
|
|
var dragStart = null;
|
|
var startHeight = 0;
|
|
var container = containerCtrl.element;
|
|
var dragGestureHandler = $mdGesture.register(handle, 'drag', { horizontal: false });
|
|
|
|
|
|
element.wrap('<div class="md-resize-wrapper">').after(handle);
|
|
handle.on('mousedown', onMouseDown);
|
|
|
|
container
|
|
.on('$md.dragstart', onDragStart)
|
|
.on('$md.drag', onDrag)
|
|
.on('$md.dragend', onDragEnd);
|
|
|
|
scope.$on('$destroy', function() {
|
|
handle
|
|
.off('mousedown', onMouseDown)
|
|
.remove();
|
|
|
|
container
|
|
.off('$md.dragstart', onDragStart)
|
|
.off('$md.drag', onDrag)
|
|
.off('$md.dragend', onDragEnd);
|
|
|
|
dragGestureHandler();
|
|
handle = null;
|
|
container = null;
|
|
dragGestureHandler = null;
|
|
});
|
|
|
|
function onMouseDown(ev) {
|
|
ev.preventDefault();
|
|
isDragging = true;
|
|
dragStart = ev.clientY;
|
|
startHeight = parseFloat(element.css('height')) || element.prop('offsetHeight');
|
|
}
|
|
|
|
function onDragStart(ev) {
|
|
if (!isDragging) return;
|
|
ev.preventDefault();
|
|
disableAutogrow();
|
|
container.addClass('md-input-resized');
|
|
}
|
|
|
|
function onDrag(ev) {
|
|
if (!isDragging) return;
|
|
|
|
element.css('height', (startHeight + ev.pointer.distanceY) + 'px');
|
|
}
|
|
|
|
function onDragEnd(ev) {
|
|
if (!isDragging) return;
|
|
isDragging = false;
|
|
container.removeClass('md-input-resized');
|
|
}
|
|
}
|
|
|
|
// Attach a watcher to detect when the textarea gets shown.
|
|
if (attr.hasOwnProperty('mdDetectHidden')) {
|
|
|
|
var handleHiddenChange = function() {
|
|
var wasHidden = false;
|
|
|
|
return function() {
|
|
var isHidden = node.offsetHeight === 0;
|
|
|
|
if (isHidden === false && wasHidden === true) {
|
|
growTextarea();
|
|
}
|
|
|
|
wasHidden = isHidden;
|
|
};
|
|
}();
|
|
|
|
// Check every digest cycle whether the visibility of the textarea has changed.
|
|
// Queue up to run after the digest cycle is complete.
|
|
scope.$watch(function() {
|
|
$mdUtil.nextTick(handleHiddenChange, false);
|
|
return true;
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function mdMaxlengthDirective($animate, $mdUtil) {
|
|
return {
|
|
restrict: 'A',
|
|
require: ['ngModel', '^mdInputContainer'],
|
|
link: postLink
|
|
};
|
|
|
|
function postLink(scope, element, attr, ctrls) {
|
|
var maxlength;
|
|
var ngModelCtrl = ctrls[0];
|
|
var containerCtrl = ctrls[1];
|
|
var charCountEl, errorsSpacer;
|
|
|
|
// Wait until the next tick to ensure that the input has setup the errors spacer where we will
|
|
// append our counter
|
|
$mdUtil.nextTick(function() {
|
|
errorsSpacer = angular.element(containerCtrl.element[0].querySelector('.md-errors-spacer'));
|
|
charCountEl = angular.element('<div class="md-char-counter">');
|
|
|
|
// Append our character counter inside the errors spacer
|
|
errorsSpacer.append(charCountEl);
|
|
|
|
// Stop model from trimming. This makes it so whitespace
|
|
// over the maxlength still counts as invalid.
|
|
attr.$set('ngTrim', 'false');
|
|
|
|
scope.$watch(attr.mdMaxlength, function(value) {
|
|
maxlength = value;
|
|
if (angular.isNumber(value) && value > 0) {
|
|
if (!charCountEl.parent().length) {
|
|
$animate.enter(charCountEl, errorsSpacer);
|
|
}
|
|
renderCharCount();
|
|
} else {
|
|
$animate.leave(charCountEl);
|
|
}
|
|
});
|
|
|
|
ngModelCtrl.$validators['md-maxlength'] = function(modelValue, viewValue) {
|
|
if (!angular.isNumber(maxlength) || maxlength < 0) {
|
|
return true;
|
|
}
|
|
|
|
// We always update the char count, when the modelValue has changed.
|
|
// Using the $validators for triggering the update works very well.
|
|
renderCharCount();
|
|
|
|
return ( modelValue || element.val() || viewValue || '' ).length <= maxlength;
|
|
};
|
|
});
|
|
|
|
function renderCharCount(value) {
|
|
// If we have not been appended to the body yet; do not render
|
|
if (!charCountEl.parent) {
|
|
return value;
|
|
}
|
|
|
|
// Force the value into a string since it may be a number,
|
|
// which does not have a length property.
|
|
charCountEl.text(String(element.val() || value || '').length + ' / ' + maxlength);
|
|
return value;
|
|
}
|
|
}
|
|
}
|
|
|
|
function placeholderDirective($compile) {
|
|
return {
|
|
restrict: 'A',
|
|
require: '^^?mdInputContainer',
|
|
priority: 200,
|
|
link: {
|
|
// Note that we need to do this in the pre-link, as opposed to the post link, if we want to
|
|
// support data bindings in the placeholder. This is necessary, because we have a case where
|
|
// we transfer the placeholder value to the `<label>` and we remove it from the original `<input>`.
|
|
// If we did this in the post-link, Angular would have set up the observers already and would be
|
|
// re-adding the attribute, even though we removed it from the element.
|
|
pre: preLink
|
|
}
|
|
};
|
|
|
|
function preLink(scope, element, attr, inputContainer) {
|
|
// If there is no input container, just return
|
|
if (!inputContainer) return;
|
|
|
|
var label = inputContainer.element.find('label');
|
|
var noFloat = inputContainer.element.attr('md-no-float');
|
|
|
|
// If we have a label, or they specify the md-no-float attribute, just return
|
|
if ((label && label.length) || noFloat === '' || scope.$eval(noFloat)) {
|
|
// Add a placeholder class so we can target it in the CSS
|
|
inputContainer.setHasPlaceholder(true);
|
|
return;
|
|
}
|
|
|
|
// md-select handles placeholders on it's own
|
|
if (element[0].nodeName != 'MD-SELECT') {
|
|
// Move the placeholder expression to the label
|
|
var newLabel = angular.element('<label ng-click="delegateClick()" tabindex="-1">' + attr.placeholder + '</label>');
|
|
|
|
// Note that we unset it via `attr`, in order to get Angular
|
|
// to remove any observers that it might have set up. Otherwise
|
|
// the attribute will be added on the next digest.
|
|
attr.$set('placeholder', null);
|
|
|
|
// We need to compile the label manually in case it has any bindings.
|
|
// A gotcha here is that we first add the element to the DOM and we compile
|
|
// it later. This is necessary, because if we compile the element beforehand,
|
|
// it won't be able to find the `mdInputContainer` controller.
|
|
inputContainer.element
|
|
.addClass('md-icon-float')
|
|
.prepend(newLabel);
|
|
|
|
$compile(newLabel)(scope);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @ngdoc directive
|
|
* @name mdSelectOnFocus
|
|
* @module material.components.input
|
|
*
|
|
* @restrict A
|
|
*
|
|
* @description
|
|
* The `md-select-on-focus` directive allows you to automatically select the element's input text on focus.
|
|
*
|
|
* <h3>Notes</h3>
|
|
* - The use of `md-select-on-focus` is restricted to `<input>` and `<textarea>` elements.
|
|
*
|
|
* @usage
|
|
* <h3>Using with an Input</h3>
|
|
* <hljs lang="html">
|
|
*
|
|
* <md-input-container>
|
|
* <label>Auto Select</label>
|
|
* <input type="text" md-select-on-focus>
|
|
* </md-input-container>
|
|
* </hljs>
|
|
*
|
|
* <h3>Using with a Textarea</h3>
|
|
* <hljs lang="html">
|
|
*
|
|
* <md-input-container>
|
|
* <label>Auto Select</label>
|
|
* <textarea md-select-on-focus>This text will be selected on focus.</textarea>
|
|
* </md-input-container>
|
|
*
|
|
* </hljs>
|
|
*/
|
|
function mdSelectOnFocusDirective($timeout) {
|
|
|
|
return {
|
|
restrict: 'A',
|
|
link: postLink
|
|
};
|
|
|
|
function postLink(scope, element, attr) {
|
|
if (element[0].nodeName !== 'INPUT' && element[0].nodeName !== "TEXTAREA") return;
|
|
|
|
var preventMouseUp = false;
|
|
|
|
element
|
|
.on('focus', onFocus)
|
|
.on('mouseup', onMouseUp);
|
|
|
|
scope.$on('$destroy', function() {
|
|
element
|
|
.off('focus', onFocus)
|
|
.off('mouseup', onMouseUp);
|
|
});
|
|
|
|
function onFocus() {
|
|
preventMouseUp = true;
|
|
|
|
$timeout(function() {
|
|
// Use HTMLInputElement#select to fix firefox select issues.
|
|
// The debounce is here for Edge's sake, otherwise the selection doesn't work.
|
|
element[0].select();
|
|
|
|
// This should be reset from inside the `focus`, because the event might
|
|
// have originated from something different than a click, e.g. a keyboard event.
|
|
preventMouseUp = false;
|
|
}, 1, false);
|
|
}
|
|
|
|
// Prevents the default action of the first `mouseup` after a focus.
|
|
// This is necessary, because browsers fire a `mouseup` right after the element
|
|
// has been focused. In some browsers (Firefox in particular) this can clear the
|
|
// selection. There are examples of the problem in issue #7487.
|
|
function onMouseUp(event) {
|
|
if (preventMouseUp) {
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
var visibilityDirectives = ['ngIf', 'ngShow', 'ngHide', 'ngSwitchWhen', 'ngSwitchDefault'];
|
|
function ngMessagesDirective() {
|
|
return {
|
|
restrict: 'EA',
|
|
link: postLink,
|
|
|
|
// This is optional because we don't want target *all* ngMessage instances, just those inside of
|
|
// mdInputContainer.
|
|
require: '^^?mdInputContainer'
|
|
};
|
|
|
|
function postLink(scope, element, attrs, inputContainer) {
|
|
// If we are not a child of an input container, don't do anything
|
|
if (!inputContainer) return;
|
|
|
|
// Add our animation class
|
|
element.toggleClass('md-input-messages-animation', true);
|
|
|
|
// Add our md-auto-hide class to automatically hide/show messages when container is invalid
|
|
element.toggleClass('md-auto-hide', true);
|
|
|
|
// If we see some known visibility directives, remove the md-auto-hide class
|
|
if (attrs.mdAutoHide == 'false' || hasVisibiltyDirective(attrs)) {
|
|
element.toggleClass('md-auto-hide', false);
|
|
}
|
|
}
|
|
|
|
function hasVisibiltyDirective(attrs) {
|
|
return visibilityDirectives.some(function(attr) {
|
|
return attrs[attr];
|
|
});
|
|
}
|
|
}
|
|
|
|
function ngMessageDirective($mdUtil) {
|
|
return {
|
|
restrict: 'EA',
|
|
compile: compile,
|
|
priority: 100
|
|
};
|
|
|
|
function compile(tElement) {
|
|
if (!isInsideInputContainer(tElement)) {
|
|
|
|
// When the current element is inside of a document fragment, then we need to check for an input-container
|
|
// in the postLink, because the element will be later added to the DOM and is currently just in a temporary
|
|
// fragment, which causes the input-container check to fail.
|
|
if (isInsideFragment()) {
|
|
return function (scope, element) {
|
|
if (isInsideInputContainer(element)) {
|
|
// Inside of the postLink function, a ngMessage directive will be a comment element, because it's
|
|
// currently hidden. To access the shown element, we need to use the element from the compile function.
|
|
initMessageElement(tElement);
|
|
}
|
|
};
|
|
}
|
|
} else {
|
|
initMessageElement(tElement);
|
|
}
|
|
|
|
function isInsideFragment() {
|
|
var nextNode = tElement[0];
|
|
while (nextNode = nextNode.parentNode) {
|
|
if (nextNode.nodeType === Node.DOCUMENT_FRAGMENT_NODE) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function isInsideInputContainer(element) {
|
|
return !!$mdUtil.getClosest(element, "md-input-container");
|
|
}
|
|
|
|
function initMessageElement(element) {
|
|
// Add our animation class
|
|
element.toggleClass('md-input-message-animation', true);
|
|
}
|
|
}
|
|
}
|
|
|
|
var $$AnimateRunner, $animateCss, $mdUtil, $log;
|
|
|
|
function mdInputInvalidMessagesAnimation($$AnimateRunner, $animateCss, $mdUtil, $log) {
|
|
saveSharedServices($$AnimateRunner, $animateCss, $mdUtil, $log);
|
|
|
|
return {
|
|
addClass: function(element, className, done) {
|
|
showInputMessages(element, done);
|
|
}
|
|
|
|
// NOTE: We do not need the removeClass method, because the message ng-leave animation will fire
|
|
};
|
|
}
|
|
|
|
function ngMessagesAnimation($$AnimateRunner, $animateCss, $mdUtil, $log) {
|
|
saveSharedServices($$AnimateRunner, $animateCss, $mdUtil, $log);
|
|
|
|
return {
|
|
enter: function(element, done) {
|
|
showInputMessages(element, done);
|
|
},
|
|
|
|
leave: function(element, done) {
|
|
hideInputMessages(element, done);
|
|
},
|
|
|
|
addClass: function(element, className, done) {
|
|
if (className == "ng-hide") {
|
|
hideInputMessages(element, done);
|
|
} else {
|
|
done();
|
|
}
|
|
},
|
|
|
|
removeClass: function(element, className, done) {
|
|
if (className == "ng-hide") {
|
|
showInputMessages(element, done);
|
|
} else {
|
|
done();
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
function ngMessageAnimation($$AnimateRunner, $animateCss, $mdUtil, $log) {
|
|
saveSharedServices($$AnimateRunner, $animateCss, $mdUtil, $log);
|
|
|
|
return {
|
|
enter: function(element, done) {
|
|
var animator = showMessage(element);
|
|
|
|
animator.start().done(done);
|
|
},
|
|
|
|
leave: function(element, done) {
|
|
var animator = hideMessage(element);
|
|
|
|
animator.start().done(done);
|
|
}
|
|
};
|
|
}
|
|
|
|
function showInputMessages(element, done) {
|
|
var animators = [], animator;
|
|
var messages = getMessagesElement(element);
|
|
var children = messages.children();
|
|
|
|
if (messages.length == 0 || children.length == 0) {
|
|
$log.warn('mdInput messages show animation called on invalid messages element: ', element);
|
|
done();
|
|
return;
|
|
}
|
|
|
|
angular.forEach(children, function(child) {
|
|
animator = showMessage(angular.element(child));
|
|
|
|
animators.push(animator.start());
|
|
});
|
|
|
|
$$AnimateRunner.all(animators, done);
|
|
}
|
|
|
|
function hideInputMessages(element, done) {
|
|
var animators = [], animator;
|
|
var messages = getMessagesElement(element);
|
|
var children = messages.children();
|
|
|
|
if (messages.length == 0 || children.length == 0) {
|
|
$log.warn('mdInput messages hide animation called on invalid messages element: ', element);
|
|
done();
|
|
return;
|
|
}
|
|
|
|
angular.forEach(children, function(child) {
|
|
animator = hideMessage(angular.element(child));
|
|
|
|
animators.push(animator.start());
|
|
});
|
|
|
|
$$AnimateRunner.all(animators, done);
|
|
}
|
|
|
|
function showMessage(element) {
|
|
var height = parseInt(window.getComputedStyle(element[0]).height);
|
|
var topMargin = parseInt(window.getComputedStyle(element[0]).marginTop);
|
|
|
|
var messages = getMessagesElement(element);
|
|
var container = getInputElement(element);
|
|
|
|
// Check to see if the message is already visible so we can skip
|
|
var alreadyVisible = (topMargin > -height);
|
|
|
|
// If we have the md-auto-hide class, the md-input-invalid animation will fire, so we can skip
|
|
if (alreadyVisible || (messages.hasClass('md-auto-hide') && !container.hasClass('md-input-invalid'))) {
|
|
return $animateCss(element, {});
|
|
}
|
|
|
|
return $animateCss(element, {
|
|
event: 'enter',
|
|
structural: true,
|
|
from: {"opacity": 0, "margin-top": -height + "px"},
|
|
to: {"opacity": 1, "margin-top": "0"},
|
|
duration: 0.3
|
|
});
|
|
}
|
|
|
|
function hideMessage(element) {
|
|
var height = element[0].offsetHeight;
|
|
var styles = window.getComputedStyle(element[0]);
|
|
|
|
// If we are already hidden, just return an empty animation
|
|
if (parseInt(styles.opacity) === 0) {
|
|
return $animateCss(element, {});
|
|
}
|
|
|
|
// Otherwise, animate
|
|
return $animateCss(element, {
|
|
event: 'leave',
|
|
structural: true,
|
|
from: {"opacity": 1, "margin-top": 0},
|
|
to: {"opacity": 0, "margin-top": -height + "px"},
|
|
duration: 0.3
|
|
});
|
|
}
|
|
|
|
function getInputElement(element) {
|
|
var inputContainer = element.controller('mdInputContainer');
|
|
|
|
return inputContainer.element;
|
|
}
|
|
|
|
function getMessagesElement(element) {
|
|
// If we ARE the messages element, just return ourself
|
|
if (element.hasClass('md-input-messages-animation')) {
|
|
return element;
|
|
}
|
|
|
|
// If we are a ng-message element, we need to traverse up the DOM tree
|
|
if (element.hasClass('md-input-message-animation')) {
|
|
return angular.element($mdUtil.getClosest(element, function(node) {
|
|
return node.classList.contains('md-input-messages-animation');
|
|
}));
|
|
}
|
|
|
|
// Otherwise, we can traverse down
|
|
return angular.element(element[0].querySelector('.md-input-messages-animation'));
|
|
}
|
|
|
|
function saveSharedServices(_$$AnimateRunner_, _$animateCss_, _$mdUtil_, _$log_) {
|
|
$$AnimateRunner = _$$AnimateRunner_;
|
|
$animateCss = _$animateCss_;
|
|
$mdUtil = _$mdUtil_;
|
|
$log = _$log_;
|
|
}
|
|
|
|
})(window, window.angular);
|