/*!
|
|
* Angular Material Design
|
|
* https://github.com/angular/material
|
|
* @license MIT
|
|
* v1.1.3
|
|
*/
|
|
(function( window, angular, undefined ){
|
|
"use strict";
|
|
|
|
/**
|
|
* @ngdoc module
|
|
* @name material.components.switch
|
|
*/
|
|
|
|
MdSwitch['$inject'] = ["mdCheckboxDirective", "$mdUtil", "$mdConstant", "$parse", "$$rAF", "$mdGesture", "$timeout"];
|
|
angular.module('material.components.switch', [
|
|
'material.core',
|
|
'material.components.checkbox'
|
|
])
|
|
.directive('mdSwitch', MdSwitch);
|
|
|
|
/**
|
|
* @ngdoc directive
|
|
* @module material.components.switch
|
|
* @name mdSwitch
|
|
* @restrict E
|
|
*
|
|
* The switch directive is used very much like the normal [angular checkbox](https://docs.angularjs.org/api/ng/input/input%5Bcheckbox%5D).
|
|
*
|
|
* As per the [material design spec](http://www.google.com/design/spec/style/color.html#color-ui-color-application)
|
|
* the switch is in the accent color by default. The primary color palette may be used with
|
|
* the `md-primary` class.
|
|
*
|
|
* @param {string} ng-model Assignable angular expression to data-bind to.
|
|
* @param {string=} name Property name of the form under which the control is published.
|
|
* @param {expression=} ng-true-value The value to which the expression should be set when selected.
|
|
* @param {expression=} ng-false-value The value to which the expression should be set when not selected.
|
|
* @param {string=} ng-change Angular expression to be executed when input changes due to user interaction with the input element.
|
|
* @param {expression=} ng-disabled En/Disable based on the expression.
|
|
* @param {boolean=} md-no-ink Use of attribute indicates use of ripple ink effects.
|
|
* @param {string=} aria-label Publish the button label used by screen-readers for accessibility. Defaults to the switch's text.
|
|
* @param {boolean=} md-invert When set to true, the switch will be inverted.
|
|
*
|
|
* @usage
|
|
* <hljs lang="html">
|
|
* <md-switch ng-model="isActive" aria-label="Finished?">
|
|
* Finished ?
|
|
* </md-switch>
|
|
*
|
|
* <md-switch md-no-ink ng-model="hasInk" aria-label="No Ink Effects">
|
|
* No Ink Effects
|
|
* </md-switch>
|
|
*
|
|
* <md-switch ng-disabled="true" ng-model="isDisabled" aria-label="Disabled">
|
|
* Disabled
|
|
* </md-switch>
|
|
*
|
|
* </hljs>
|
|
*/
|
|
function MdSwitch(mdCheckboxDirective, $mdUtil, $mdConstant, $parse, $$rAF, $mdGesture, $timeout) {
|
|
var checkboxDirective = mdCheckboxDirective[0];
|
|
|
|
return {
|
|
restrict: 'E',
|
|
priority: $mdConstant.BEFORE_NG_ARIA,
|
|
transclude: true,
|
|
template:
|
|
'<div class="md-container">' +
|
|
'<div class="md-bar"></div>' +
|
|
'<div class="md-thumb-container">' +
|
|
'<div class="md-thumb" md-ink-ripple md-ink-ripple-checkbox></div>' +
|
|
'</div>'+
|
|
'</div>' +
|
|
'<div ng-transclude class="md-label"></div>',
|
|
require: ['^?mdInputContainer', '?ngModel', '?^form'],
|
|
compile: mdSwitchCompile
|
|
};
|
|
|
|
function mdSwitchCompile(element, attr) {
|
|
var checkboxLink = checkboxDirective.compile(element, attr).post;
|
|
// No transition on initial load.
|
|
element.addClass('md-dragging');
|
|
|
|
return function (scope, element, attr, ctrls) {
|
|
var containerCtrl = ctrls[0];
|
|
var ngModel = ctrls[1] || $mdUtil.fakeNgModel();
|
|
var formCtrl = ctrls[2];
|
|
|
|
var disabledGetter = null;
|
|
if (attr.disabled != null) {
|
|
disabledGetter = function() { return true; };
|
|
} else if (attr.ngDisabled) {
|
|
disabledGetter = $parse(attr.ngDisabled);
|
|
}
|
|
|
|
var thumbContainer = angular.element(element[0].querySelector('.md-thumb-container'));
|
|
var switchContainer = angular.element(element[0].querySelector('.md-container'));
|
|
var labelContainer = angular.element(element[0].querySelector('.md-label'));
|
|
|
|
// no transition on initial load
|
|
$$rAF(function() {
|
|
element.removeClass('md-dragging');
|
|
});
|
|
|
|
checkboxLink(scope, element, attr, ctrls);
|
|
|
|
if (disabledGetter) {
|
|
scope.$watch(disabledGetter, function(isDisabled) {
|
|
element.attr('tabindex', isDisabled ? -1 : 0);
|
|
});
|
|
}
|
|
|
|
attr.$observe('mdInvert', function(newValue) {
|
|
var isInverted = $mdUtil.parseAttributeBoolean(newValue);
|
|
|
|
isInverted ? element.prepend(labelContainer) : element.prepend(switchContainer);
|
|
|
|
// Toggle a CSS class to update the margin.
|
|
element.toggleClass('md-inverted', isInverted);
|
|
});
|
|
|
|
// These events are triggered by setup drag
|
|
$mdGesture.register(switchContainer, 'drag');
|
|
switchContainer
|
|
.on('$md.dragstart', onDragStart)
|
|
.on('$md.drag', onDrag)
|
|
.on('$md.dragend', onDragEnd);
|
|
|
|
var drag;
|
|
function onDragStart(ev) {
|
|
// Don't go if the switch is disabled.
|
|
if (disabledGetter && disabledGetter(scope)) return;
|
|
ev.stopPropagation();
|
|
|
|
element.addClass('md-dragging');
|
|
drag = {width: thumbContainer.prop('offsetWidth')};
|
|
}
|
|
|
|
function onDrag(ev) {
|
|
if (!drag) return;
|
|
ev.stopPropagation();
|
|
ev.srcEvent && ev.srcEvent.preventDefault();
|
|
|
|
var percent = ev.pointer.distanceX / drag.width;
|
|
|
|
//if checked, start from right. else, start from left
|
|
var translate = ngModel.$viewValue ? 1 + percent : percent;
|
|
// Make sure the switch stays inside its bounds, 0-1%
|
|
translate = Math.max(0, Math.min(1, translate));
|
|
|
|
thumbContainer.css($mdConstant.CSS.TRANSFORM, 'translate3d(' + (100*translate) + '%,0,0)');
|
|
drag.translate = translate;
|
|
}
|
|
|
|
function onDragEnd(ev) {
|
|
if (!drag) return;
|
|
ev.stopPropagation();
|
|
|
|
element.removeClass('md-dragging');
|
|
thumbContainer.css($mdConstant.CSS.TRANSFORM, '');
|
|
|
|
// We changed if there is no distance (this is a click a click),
|
|
// or if the drag distance is >50% of the total.
|
|
var isChanged = ngModel.$viewValue ? drag.translate < 0.5 : drag.translate > 0.5;
|
|
if (isChanged) {
|
|
applyModelValue(!ngModel.$viewValue);
|
|
}
|
|
drag = null;
|
|
|
|
// Wait for incoming mouse click
|
|
scope.skipToggle = true;
|
|
$timeout(function() {
|
|
scope.skipToggle = false;
|
|
}, 1);
|
|
}
|
|
|
|
function applyModelValue(newValue) {
|
|
scope.$apply(function() {
|
|
ngModel.$setViewValue(newValue);
|
|
ngModel.$render();
|
|
});
|
|
}
|
|
|
|
};
|
|
}
|
|
|
|
|
|
}
|
|
|
|
})(window, window.angular);
|