vin*_*nce 2 recursion clone typescript angular angular-reactive-forms
我正在尝试构建一个能够生成给定副本的函数FormGroup.我开始时:
function copyForm(form: FormGroup): FormGroup {
const copy = new FormGroup({});
for (let key of Object.keys(form.value)) {
const control = form.controls[key];
/* Copy the data from the control into a new control */
const copyControl = new FormControl({[key]: control.value});
copy.addControl(key, copyControl);
}
Run Code Online (Sandbox Code Playgroud)
但如果有一个FormArray或那个,这不起作用FormGroup.如果它是递归的,这个可能会起作用,但我无法很好地处理它.
我也试图解决它
function copyForm(form: FormGroup): FormGroup {
const copy = new FormGroup({});
for (let key of Object.keys(form.value)) {
const control = form.controls[key];
const copyControl = new FormControl({...control.value});
copy.addControl(key, copyControl);
}
return copy;
Run Code Online (Sandbox Code Playgroud)
}
但这不适用于双嵌套FormGroups,任何FormArrays或常规FormControls......
我也尝试过:
function copyForm(form: FormGroup): FormGroup {
const copy = new FormGroup(Object.assign({}, form.value));
return copy;
}
Run Code Online (Sandbox Code Playgroud)
但这给了我错误:
ERROR TypeError: control.setParent is not a function
Run Code Online (Sandbox Code Playgroud)
我很难过.
Joh*_*ohn 13
这是我提出的深层复制功能,它还保留了每个AbstractControl的相关验证器/异步验证器功能和禁用状态.
/**
* Deep clones the given AbstractControl, preserving values, validators, async validators, and disabled status.
* @param control AbstractControl
* @returns AbstractControl
*/
export function cloneAbstractControl<T extends AbstractControl>(control: T): T {
let newControl: T;
if (control instanceof FormGroup) {
const formGroup = new FormGroup({}, control.validator, control.asyncValidator);
const controls = control.controls;
Object.keys(controls).forEach(key => {
formGroup.addControl(key, cloneAbstractControl(controls[key]));
})
newControl = formGroup as any;
}
else if (control instanceof FormArray) {
const formArray = new FormArray([], control.validator, control.asyncValidator);
control.controls.forEach(formControl => formArray.push(cloneAbstractControl(formControl)))
newControl = formArray as any;
}
else if (control instanceof FormControl) {
newControl = new FormControl(control.value, control.validator, control.asyncValidator) as any;
}
else {
throw new Error('Error: unexpected control value');
}
if (control.disabled) newControl.disable({emitEvent: false});
return newControl;
}
Run Code Online (Sandbox Code Playgroud)
如果您有仅包含 FormControls(即不包含 FormGroups 或 FormArrays)的简单 FormGroups,并且您提前知道它们的结构,那么这是一个简单的解决方案:
注意:通过使用initaliseFormGroup初始化原始 FormGroup,您将匹配每个控件的验证器。
initaliseFormGroup(){
return new FormGroup({
x : new FormControl('', Validators.required),
y : new FormControl('', Validators.minLength(10))
});
}
cloneFormGroup(oldForm: FormGroup){
let newForm = this.initaliseFormGroup()
newForm.patchValue(oldForm.value);
return newForm;
}
Run Code Online (Sandbox Code Playgroud)
如果您有更复杂的表单(带有子 FormGroups),或者想要动态克隆它们而不提前知道它们的结构,那么其他答案会更适合。
这就是我要做的:
copyFormControl(control: AbstractControl) {
if (control instanceof FormControl) {
return new FormControl(control.value);
} else if (control instanceof FormGroup) {
const copy = new FormGroup({});
Object.keys(control.getRawValue()).forEach(key => {
copy.addControl(key, copyFormControl(control.controls[key]));
});
return copy;
} else if (control instanceof FormArray) {
const copy = new FormArray([]);
control.controls.forEach(control => {
copy.push(copyFormControl(control));
})
return copy;
}
}
Run Code Online (Sandbox Code Playgroud)
我使用getRawValue()而不是value,因为value其中不会包含已禁用的控件。
我个人使用lodash cloneDeep()函数在这里找到:
https://lodash.com/docs/#cloneDeep
我用这种方式:
const newFormGroup: any = _.cloneDeep(myFormGroup);
而且,如果您想对其进行强类型输入,则可以添加as FormGroup@Andre Elrico作为注释中建议的内容:
const newFormGroup = _.cloneDeep(myFormGroup) as FormGroup;
| 归档时间: |
|
| 查看次数: |
7787 次 |
| 最近记录: |