如何根据用户输入的$ viewValue更新来更新ngModel的$ modelValue

Cod*_*ody 9 angularjs angularjs-directive angular-ngmodel

说我有以下指令:

myApp.directive('myDirective', function() {
    return {
        restrict: 'A',
        require: 'ngModel',
        scope: {
            ngModel: '='
        },
        link: function(scope, elem, attrs, ngModelCtrl) {
            scope.$watch('ngModel', function() {
                ngModelCtrl.$modelValue = 'foo';
            });
        }
    }
}); 
Run Code Online (Sandbox Code Playgroud)

以下html:

<input ng-model="name" my-directive></input>
Run Code Online (Sandbox Code Playgroud)

基本上,每当用户更改输入时,my-directive理想情况下将内部模型值更改为"foo",同时保持视图值不变.

但是当我$scope.name在相应的控制器中打印出来时,它不记录"foo",它会记录用户输入的内容.

看起来这ngModelCtrl.$modelValue不是控制器正在访问的 - 我是不正确地接近这个问题?

(另外观察ngModel范围感觉确实是错误的,但我不确定是否有任何其他方式.任何建议都会非常感激!)

PSL*_*PSL 11

如果您正在寻找视图更改,则不应注册手表.ngModelController $viewChangeListeners是专为此目的而设计的,可避免在ngModel上创建任何额外的监视.您还可以删除ngModel上的双向绑定设置.

我能想到这种方式.

.directive('myDirective', function($parse) {
    return {
        restrict: 'A',
        require: 'ngModel',
        link: function(scope, elem, attrs, ngModelCtrl) {
          /*Register a viewchange listener*/
          ngModelCtrl.$viewChangeListeners.push(function(){ 
              /*Set model value differently based on the viewvalue entered*/
              $parse(attrs.ngModel).assign(scope, ngModelCtrl.$viewValue.split(',')); 
          });
        }
    }
});
Run Code Online (Sandbox Code Playgroud)

演示

虽然反过来考虑它(Credits @Cody),但在使用时更加简洁和恰当$parser.

 ngModelCtrl.$parsers.push(function(val) { return val.split(',') });
Run Code Online (Sandbox Code Playgroud)

  • 为了将来的参考,我检查了`$ parsers`,我想我找到了一个更简洁的解决方案:`ngModelCtrl.$ parsers.push(function(val){return val.split(',')})`. (3认同)