Angularjs 输入的数字 $filter 指令要求用户按两次小数

Angularjs number $filter directive on input requires a user to press decimal twice

我正在尝试将输入限制为仅数字和小数除外,将分数大小固定为最大值 2,并使用逗号屏蔽 viewValue,并将 modelValue 保留为数字。下面的代码几乎可以完美运行,除了为了添加小数点,用户必须按 . 键两次。知道为什么第一次按下会去掉小数吗?

var app = angular.module('myApp', []);
app.controller('pageCtrl', function($scope){
    $scope.myModel = 0;
});

app.directive('formatter', ['$filter', function ($filter) {
    return {
        require: '?ngModel',
        link: function (scope, elem, attrs, ctrl) {
            if (!ctrl || !attrs.formatter) return;
            ctrl.$formatters.unshift(function (a) {
                return $filter(attrs.formatter)(ctrl.$modelValue, 2)
            });

            ctrl.$parsers.unshift(function (viewValue) {
                var plainNumber = viewValue.replace(/[^0-9.]/g, '');
                var fractionSize = 0;
                if (plainNumber.indexOf('.') >= 0) {
                    var parts = plainNumber.split('.');
                    fractionSize = parts[1].length > 2 ? 2 : parts[1].length;
                }
                elem.val($filter(attrs.formatter)(plainNumber, fractionSize));

                return plainNumber;
            });
        }
    };
}]);
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.2.23/angular.min.js"></script>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<body ng-app="myApp">
   <div ng-controller="pageCtrl">
      <input formatter="number" ng-model="myModel">
      {{ myModel }}
   </div>
</body>

这是因为 AngularJS 中的 number 过滤器在提供的 fractionSize 参数为零时去除了小数点。

你可以自己确认一下:

<!-- 2 -->
{{ 2. | number:0 }}

所以要绕过它,只需有条件地应用 number 过滤器:

elem.val(fractionSize ? $filter(attrs.formatter)(plainNumber, fractionSize) : plainNumber);