I have a ControlValueAccesor with one FormControl. I need to create a new observable from the control's valueChanges and use it in the template via AsyncPipe. So in CVA's writeValue I update the form control's value using setValue().
public isOdd$ = new Observable<boolean>();
nestedFc = new FormControl([null]);
writeValue() {
this.nestedFc.setValue(22);
}
ngOnInit() {
this.isOdd$ = this.nestedFc.valueChanges.pipe(
map((value) => value % 2 !== 0)
);
}
<span> <input [formControl]="nestedFc"/> </span>
<span *ngIf="{value: isOdd$ | async} as context"> {{context.value}}</span>
The problem is that valueChanges is triggered when writeValue is first called but the async pipe does not "see" these changes, and so the view does not show the update.
There is a couple GitHub issues around this: ng 11: patchValue, valueChanges & async pipe and https://github.com/angular/angular/issues/40826.
Based on the last one I have created a stackblitz to reproduce the problem. If writeValue uses setTimeout to patch the form control's value, valueChanges is triggered and the async pipe "sees" the change.
Is this correct? Is there really no other way but to use setTimeout?
valueChanges
will only emit the changes, not the initial value. Note, you do not need to initializeisOdd$
with an empty observable, then reassign it, you can simply initialize it to the value from your form control valueChanges StackBlitz