Angular 指令将 ngmodel 值从 null 更改为未定义中断验证
Angular directive change ngmodel value from null to undefined breaks validation
我创建了一个 plnkr 来描述我的问题:Link to plnkr
问题描述:
我有一个数字字段,从中写入模型的值。首先,我像第一个输入一样实现了这个特性。这个实现的问题是,如果我输入一些东西然后删除,我有以下模型:
{"firstNumber":null,"secondNumber":64}
不幸的是,对于我来说,这种表示是不可接受的,我需要以下结果:
{"secondNumber":64}
为此,我接受了指令并实施了第二个字段。现在我收到了正确的输出,但是当我删除该值时,表单变得无效。
我还添加了第三个和第四个输入来证明该指令也破坏了所需的验证。
所以,问题是:
我怎样才能改进输入字段没有模型
{"firstNumber":null,"secondNumber":64}
但是
{"secondNumber":64}
并且不要破坏 Angular 表单的验证机制。
供参考:
我有以下 HTML:
<body ng-controller="MainCtrl as vm">
<h1>Validating input inside ng-repeat with Angular 1.3</h1>
<form name="vm.myForm" novalidate>
<input type="number" ng-model="vm.fields.firstNumber" name="firstNumber">
<input type="number" ng-model="vm.fields.secondNumber" name="secondNumber" null-to-undefined>
<br>
<input type="number" ng-model="vm.fields.thirdNumber" name="thirdNumber" ng-required = "true">
<input type="number" ng-model="vm.fields.fourthNumber" name="fourthNumber" null-to-undefined ng-required="true">
</form>
</body>
以及以下控制器和指令代码:
var app = angular.module('plunker', []);
app.controller('MainCtrl', function($scope) {
var vm = this;
vm.fields = {};
vm.fields.firstNumber = 12;
vm.fields.secondNumber = 24;
vm.fields.thirdNumber = 64;
vm.fields.fourthNumber = 128;
});
app.directive('nullToUndefined', function($timeout) {
return {
restrict: 'A',
require: 'ngModel',
link: function(scope, elem, attrs, ctrl) {
ctrl.$parsers.push(function(viewValue, modelValue) {
if (viewValue === null) {
$timeout(function() {
//ctrl.$setValidity('number', true);
});
return undefined;
}
return viewValue;
});
}
};
});
p.s。我尽可能地削减代码以重现我的主要项目中的问题。我将不胜感激任何解决方案,但如果有人可以提出解决方案,那将非常酷:
1.表现良好
2. 以 angular 方式表现。
您可能应该专注于其他解决方案,而不是编写指令。
您有带有道具(firstNumber、secondNumber、...)的对象。
您需要用 Angular 验证 属性(是否为空?是正确的数字吗?)。
然后,您需要过滤具有非空值的对象道具,而不改变您的模型:您的输入绑定到模型 属性,如果您销毁 属性,Angular 无法验证并失败。
我们可以做到:
- Select all keys of a.fields
- Iterate all keys, filters
- We got all non-null keys
- Reduce non-null keys to a new object, without mutation of "a"
const a = {
fields: {
a: 1,
b: 2,
c: null,
},
};
// After form validation
const newA = Object
.keys(a.fields)
.filter(k => a.fields[k] !== null)
.reduce(
(accumulator, k) =>
Object.assign(
accumulator,
{
fields: Object.assign(accumulator.fields, { [k]: a.fields[k] })
}
),
{ fields: {} }
);
如果您需要过滤 undefined + null,只需对过滤器进行少量检查即可:
.filter(k => a.fields[k] != null)
编辑:修复代码。
我创建了一个 plnkr 来描述我的问题:Link to plnkr
问题描述: 我有一个数字字段,从中写入模型的值。首先,我像第一个输入一样实现了这个特性。这个实现的问题是,如果我输入一些东西然后删除,我有以下模型:
{"firstNumber":null,"secondNumber":64}
不幸的是,对于我来说,这种表示是不可接受的,我需要以下结果:
{"secondNumber":64}
为此,我接受了指令并实施了第二个字段。现在我收到了正确的输出,但是当我删除该值时,表单变得无效。
我还添加了第三个和第四个输入来证明该指令也破坏了所需的验证。
所以,问题是: 我怎样才能改进输入字段没有模型
{"firstNumber":null,"secondNumber":64}
但是
{"secondNumber":64}
并且不要破坏 Angular 表单的验证机制。
供参考: 我有以下 HTML:
<body ng-controller="MainCtrl as vm">
<h1>Validating input inside ng-repeat with Angular 1.3</h1>
<form name="vm.myForm" novalidate>
<input type="number" ng-model="vm.fields.firstNumber" name="firstNumber">
<input type="number" ng-model="vm.fields.secondNumber" name="secondNumber" null-to-undefined>
<br>
<input type="number" ng-model="vm.fields.thirdNumber" name="thirdNumber" ng-required = "true">
<input type="number" ng-model="vm.fields.fourthNumber" name="fourthNumber" null-to-undefined ng-required="true">
</form>
</body>
以及以下控制器和指令代码:
var app = angular.module('plunker', []);
app.controller('MainCtrl', function($scope) {
var vm = this;
vm.fields = {};
vm.fields.firstNumber = 12;
vm.fields.secondNumber = 24;
vm.fields.thirdNumber = 64;
vm.fields.fourthNumber = 128;
});
app.directive('nullToUndefined', function($timeout) {
return {
restrict: 'A',
require: 'ngModel',
link: function(scope, elem, attrs, ctrl) {
ctrl.$parsers.push(function(viewValue, modelValue) {
if (viewValue === null) {
$timeout(function() {
//ctrl.$setValidity('number', true);
});
return undefined;
}
return viewValue;
});
}
};
});
p.s。我尽可能地削减代码以重现我的主要项目中的问题。我将不胜感激任何解决方案,但如果有人可以提出解决方案,那将非常酷: 1.表现良好 2. 以 angular 方式表现。
您可能应该专注于其他解决方案,而不是编写指令。
您有带有道具(firstNumber、secondNumber、...)的对象。
您需要用 Angular 验证 属性(是否为空?是正确的数字吗?)。
然后,您需要过滤具有非空值的对象道具,而不改变您的模型:您的输入绑定到模型 属性,如果您销毁 属性,Angular 无法验证并失败。
我们可以做到:
- Select all keys of a.fields
- Iterate all keys, filters
- We got all non-null keys
- Reduce non-null keys to a new object, without mutation of "a"
const a = {
fields: {
a: 1,
b: 2,
c: null,
},
};
// After form validation
const newA = Object
.keys(a.fields)
.filter(k => a.fields[k] !== null)
.reduce(
(accumulator, k) =>
Object.assign(
accumulator,
{
fields: Object.assign(accumulator.fields, { [k]: a.fields[k] })
}
),
{ fields: {} }
);
如果您需要过滤 undefined + null,只需对过滤器进行少量检查即可:
.filter(k => a.fields[k] != null)
编辑:修复代码。