From 7ed5f1f3e417605bce5a8d16e79a016bd96c7ef2 Mon Sep 17 00:00:00 2001 From: Egor Volvachev Date: Fri, 19 Apr 2024 22:37:39 +0300 Subject: [PATCH] fix(primeng/p-checkbox): support formControlName in multiple mode Fixes #15185. --- src/app/components/checkbox/checkbox.ts | 18 ++++++++++++++---- 1 file changed, 14 insertions(+), 4 deletions(-) diff --git a/src/app/components/checkbox/checkbox.ts b/src/app/components/checkbox/checkbox.ts index 3c451482fa1..9bcaefbff45 100755 --- a/src/app/components/checkbox/checkbox.ts +++ b/src/app/components/checkbox/checkbox.ts @@ -8,6 +8,7 @@ import { ElementRef, EventEmitter, forwardRef, + Injector, Input, NgModule, numberAttribute, @@ -17,7 +18,7 @@ import { ViewChild, ViewEncapsulation } from '@angular/core'; -import { ControlValueAccessor, FormControl, NG_VALUE_ACCESSOR } from '@angular/forms'; +import { ControlValueAccessor, FormControl, NG_VALUE_ACCESSOR, NgControl } from '@angular/forms'; import { PrimeTemplate, SharedModule } from 'primeng/api'; import { AutoFocusModule } from 'primeng/autofocus'; import { CheckIcon } from 'primeng/icons/check'; @@ -235,7 +236,7 @@ export class Checkbox implements ControlValueAccessor { focused: boolean = false; - constructor(public cd: ChangeDetectorRef) {} + constructor(public cd: ChangeDetectorRef, private readonly injector: Injector) {} ngAfterContentInit() { this.templates.forEach((item) => { @@ -264,9 +265,18 @@ export class Checkbox implements ControlValueAccessor { updateModel(event) { let newModelValue; + /* + * When `formControlName` or `formControl` is used - `writeValue` is not called after control changes. + * Otherwise it is causing multiple references to the actual value: there is one array reference inside the component and another one in the control value. + * `selfControl` is the source of truth of references, it is made to avoid reference loss. + * */ + const selfControl = this.injector.get(NgControl, null, { optional: true, self: true }); + + const currentModelValue = selfControl && !this.formControl ? selfControl.value : this.model; + if (!this.binary) { - if (this.checked()) newModelValue = this.model.filter((val) => !ObjectUtils.equals(val, this.value)); - else newModelValue = this.model ? [...this.model, this.value] : [this.value]; + if (this.checked()) newModelValue = currentModelValue.filter((val) => !ObjectUtils.equals(val, this.value)); + else newModelValue = currentModelValue ? [...currentModelValue, this.value] : [this.value]; this.onModelChange(newModelValue); this.model = newModelValue;