AngularJS用于监视更改的输入类型=数字的指令

时间:2017-03-20 12:37:50

标签: javascript angularjs input angularjs-directive

上下文:我正在开发我的应用的移动版本。我希望用户能够使用手机/平板电脑的numeric keybord输入数字字段。 AngularJS使得type = "number"的输入字段对于任何非整数数据都是不可达的(这是任意的,因为输入是一个字符串)。

这个想法:我想制定一个指令,以某种方式(使用$watch$parsers或任何其他古代巫术)将输入字符串转换为数字。

问题:当我在输入字段中输入任何内容时,我的指令不会触发$watch的更改。 但是当我从我的cotroller更改范围字段时,$ watch会按计划触发。

控制器:

        // code minimized and omitted for clarity
        .controller('carController', ['$scope', function ($scope) {
            $scope.car = {
                power : null,
                // other fields
            }
        }]);

指令

angular.module('numeric', []).directive('numeric', function () {
    return {
        restrict: 'A',
        require: 'ngModel',

        scope: {
            model: '=ngModel'
        },
        link: function (scope, element, attrs, ngModelCtrl) {
            // parsers does not affect anything
            ngModelCtrl.$parsers.push(function(value) {
                return parseInt(value);
            });

            // watcher does not watch
            scope.$watch('model', function(newVal, old) {
                if (typeof newVal == 'string') {
                    scope.car.power = parseInt(newVal);
                }
            }, true);

        }
    };
});

HTML

    <div ng-controller="carController">
     <input   
            ng-model="car.power"                      
            numeric
            name="power"
            type="number"
            pattern="[0-9]">
       </div>

这是一个演示这种行为的小提琴http://jsfiddle.net/xo94sw7m/1/

问题:我错过了什么以及如何使指令按计划工作?


我尝试过:使用$ formatters- $ parsers,使用不同的方法来观察$(使用scope:false,隔离范围,尝试使用attrs等观察范围变化),到目前为止似乎没有任何工作

1 个答案:

答案 0 :(得分:1)

myApp.directive('number', ['$parse', function($parse) {
    return {
        require: 'ngModel',
        link: function(scope, element, attrs, ngModelController) {

            ngModelController.$parsers.push(function(data) {
                return parseInt(data);
            });
            ngModelController.$formatters.push(function(data) {
                if (data) {
                    var model = $parse(attrs['ngModel']);
                    model.assign(scope, parseInt(data));
                }
                return parseInt(data);

            });
        }
    }
}]);

Here is working fiddle