gpt4 book ai didi

javascript - 从指令内部 $watch 对 ngModel 的外部更改的正确方法

转载 作者:行者123 更新时间:2023-11-29 21:44:36 25 4
gpt4 key购买 nike

我正在尝试制定一个从两个输入源读取的指令,然后做一些事情将其变成一个。为此,我正在监听两个输入的变化,并将新的组合值分配给指令的 ngModel。

问题是我还需要知道何时在指令外修改 ngModel 以执行反向过程并正确设置我的两个指令输入源的值。有正确的方法吗?

我做了一个片段来更好地说明问题,这不是我的实际指令。

angular.module('myapp', [])

.controller('AppController', function($scope, $timeout) {
$scope.data = {
value: ''
};

$timeout(function() {
$scope.data.value = 'hellooooo';
}, 5000);
})

.directive('sampleDirective', function() {
return {
require: 'ngModel',
restrict: 'E',
template: '<input ng-model="data.input1" ng-change="changed()"><input ng-model="data.input2" ng-change="changed()">',
scope: {
ngModel: '='
},
controller: function($scope) {
$scope.data = {};
$scope.changed = function() {
$scope.ngModel = $scope.data.input1 + ' + ' + $scope.data.input2;
}

// The watch is running even when the ngModel is modified inside the changed function above
// I want it to only run when the model is changed from outside the directive, like
// I'm doing in the AppController.
$scope.$watch('ngModel', function(value) {
if (value) {
// Just simulating some processing
var length = value.length;

$scope.data.input1 = value.slice(0, length/2);
$scope.data.input2 = value.slice(length/2, length-1);
}
});
}
};
});
<div ng-app="myapp">

<div ng-controller="AppController">
<sample-directive ng-model="data.value"></sample-directive>

<br>
<br>
<div>{{ data.value }}</div>
</div>
</div>
<script src="https://cdnjs.cloudflare.com/ajax/libs/angular.js/1.4.3/angular.js"></script>

最佳答案

您拥有的是自定义输入控件。它是一个具有 2 个文本框的输入控件,但“模型”实际上是两者的组合。

要实现“支持”ngModel 的自定义输入控件,即与其他需要:ngModel 的指令集成,您的指令还需要需要:ngModel 并使用 ngModelController Hook 进行集成。

所以,不要只做 scope: { ngModel: "="}。这样,您就可以双向绑定(bind)到附加到 ng-model 属性的模型,但是当您使用 scope.ngModel = 设置此值时,事情不会像您期望的那样工作“某物”

Angular 提供了一个 example of a custom input control在其 ngModelController 的文档中.简而言之,您需要协调设置 $viewValue - 当底层 View 发生变化时,以及设置 View - 当模型发生变化时。

.directive("sampleDirective", function(){
return {
require: "ngModel",
scope: true,
template: '<input ng-model="d.input1" ng-change="viewChanged()">\
<input ng-model="d.input2" ng-change="viewChanged()">',

link: function(scope, element, attrs, ngModel){
var d = scope.d = {};

ngModel.$render = render;

scope.viewChanged = read;


// defines how to render based on model changes
function render(){
var modelValue = ngModel.$modelValue || "";
var length = modelValue.length;

d.input1 = modelValue.slice(0, length/2);
d.input2 = length > 1 ? modelValue.slice(length/2, length) : "";
};


// defines how to set the model based on DOM changes
function read(){
var newViewValue = d.input1 + d.input2;
ngModel.$setViewValue(newViewValue);
}
}
};
});

然后,您的控件将与支持 ngModel 的任何其他指令一起工作,例如 requiredng-patternng-更改,并可以参与表单验证:

<div ng-form="form1">
<sample-directive ng-model="foo" maxlength="8" ng-change="doSomething()">
</sample-directive>
</div>

Demo

关于javascript - 从指令内部 $watch 对 ngModel 的外部更改的正确方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31668232/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com