@quadrel-enterprise-ui/framework 18.34.0 → 18.35.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/page/shared/services/form-group-manager.service.mjs +16 -10
- package/fesm2022/quadrel-enterprise-ui-framework.mjs +15 -9
- package/fesm2022/quadrel-enterprise-ui-framework.mjs.map +1 -1
- package/lib/page/shared/services/form-group-manager.service.d.ts +1 -0
- package/lib/page/shared/services/form-group-manager.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -123,18 +123,24 @@ export class QdFormGroupManagerService {
|
|
|
123
123
|
const snapshot = this._formGroupsSnapshot.get(key);
|
|
124
124
|
if (!snapshot)
|
|
125
125
|
return;
|
|
126
|
-
|
|
127
|
-
const newValue = snapshot[ctrlKey];
|
|
128
|
-
if (ctrl instanceof FormArray && Array.isArray(newValue)) {
|
|
129
|
-
this.resetFormArrayToValues(ctrl, newValue);
|
|
130
|
-
}
|
|
131
|
-
else {
|
|
132
|
-
ctrl.reset(newValue);
|
|
133
|
-
}
|
|
134
|
-
});
|
|
126
|
+
this.restoreFormGroup(fg, snapshot);
|
|
135
127
|
});
|
|
136
128
|
this.cancelPendingAsyncValidation();
|
|
137
129
|
}
|
|
130
|
+
restoreFormGroup(fg, snapshot) {
|
|
131
|
+
Object.entries(fg.controls).forEach(([ctrlKey, ctrl]) => {
|
|
132
|
+
const newValue = snapshot[ctrlKey];
|
|
133
|
+
if (ctrl instanceof FormArray && Array.isArray(newValue)) {
|
|
134
|
+
this.resetFormArrayToValues(ctrl, newValue);
|
|
135
|
+
}
|
|
136
|
+
else if (ctrl instanceof FormGroup && newValue && typeof newValue === 'object') {
|
|
137
|
+
this.restoreFormGroup(ctrl, newValue);
|
|
138
|
+
}
|
|
139
|
+
else {
|
|
140
|
+
ctrl.reset(newValue);
|
|
141
|
+
}
|
|
142
|
+
});
|
|
143
|
+
}
|
|
138
144
|
/**
|
|
139
145
|
* Cancels any in-flight async validators on all registered form groups.
|
|
140
146
|
*
|
|
@@ -222,4 +228,4 @@ export class QdFormGroupManagerService {
|
|
|
222
228
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdFormGroupManagerService, decorators: [{
|
|
223
229
|
type: Injectable
|
|
224
230
|
}] });
|
|
225
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-group-manager.service.js","sourceRoot":"","sources":["../../../../../../../libs/qd-ui/src/lib/page/shared/services/form-group-manager.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { AbstractControl, FormArray, FormControl, FormGroup } from '@angular/forms';\nimport isEqual from 'lodash/isEqual';\nimport { BehaviorSubject, combineLatest, Observable, of, queueScheduler } from 'rxjs';\nimport { map, observeOn, startWith, switchMap } from 'rxjs/operators';\n\n/**\n * The QdFormGroupManagerService provides centralized registration, snapshotting, and value tracking\n * for multiple Angular `FormGroup` instances. It supports efficient state restoration, including dynamic\n * resizing of embedded `FormArray` controls, while avoiding any structural cloning in snapshots.\n *\n * #### Purpose:\n * This service is useful when managing multiple reactive forms that must be:\n * - Observed for changes\n * - Checked for validity\n * - Compared against a previous state (snapshot)\n * - Reset to a previous state, including dynamic form arrays\n *\n * #### Value Snapshot\n * Snapshots store only **raw values** (via `getRawValue()`), not any control instances.\n * This allows for minimal memory usage and easy comparison using deep equality checks.\n *\n * #### Smart FormArray Reset\n * When restoring a `FormArray`, the service ensures that the number and structure of controls\n * matches the snapshot values. Controls are cloned dynamically from an existing prototype.\n * If a `FormArray` was emptied (e.g. via `removeAt()`), a minimal control structure is inferred\n * from the snapshot data.\n *\n * #### Example Usage:\n *\n * **Register a FormGroup**\n * ```ts\n * service.setFormGroup('userForm', formGroup);\n * ```\n *\n * **Take a Snapshot**\n * ```ts\n * service.takeFormGroupsSnapshot();\n * ```\n *\n * **Detect Changes**\n * ```ts\n * service.$hasValuesChanged().subscribe(changed => {\n *   if (changed) {\n *     console.log('Form data has changed');\n *   }\n * });\n * ```\n *\n * **Restore from Snapshot**\n * ```ts\n * service.restoreFormGroupsFromSnapshot();\n * ```\n *\n * **Check if All Forms Are Valid**\n * ```ts\n * service.$areFormGroupsValid().subscribe(valid => {\n *   if (valid) {\n *     console.log('All forms valid!');\n *   }\n * });\n * ```\n *\n * #### Notes:\n * - FormArrays are not replaced during restore. Their length is adjusted and their content reset.\n * - Control cloning is recursive and structure-preserving.\n * - No structural information is stored in the snapshot; only data.\n */\n@Injectable()\nexport class QdFormGroupManagerService {\n  private _formGroups = new Map<string, FormGroup>();\n  private _formGroupsSnapshot = new Map<string, Record<string, unknown>>();\n  private _formGroupsChanged$ = new BehaviorSubject<void>(undefined);\n\n  setFormGroup(key: string, formGroup: FormGroup): void {\n    this._formGroups.set(key, formGroup);\n    this._formGroupsChanged$.next();\n  }\n\n  getFormGroup(key: string): FormGroup | undefined {\n    return this._formGroups.get(key);\n  }\n\n  getAllFormGroups(): Map<string, FormGroup> {\n    return this._formGroups;\n  }\n\n  tryRemoveFormGroup(key: string): void {\n    this._formGroups.delete(key);\n    this._formGroupsChanged$.next();\n  }\n\n  hasFormGroups(): boolean {\n    return this._formGroups.size > 0;\n  }\n\n  getAllValues(): Record<string, unknown> {\n    const allValues: Record<string, unknown> = {};\n\n    this._formGroups.forEach((formGroup, key) => (allValues[key] = formGroup.value));\n\n    return allValues;\n  }\n\n  isFormGroupKeyUnique(key: string): boolean {\n    return !this._formGroups.has(key);\n  }\n\n  $areFormGroupsValid(): Observable<boolean> {\n    return this._formGroupsChanged$.pipe(\n      observeOn(queueScheduler),\n      switchMap(() => {\n        if (!this.hasFormGroups()) return of(false);\n\n        const obs = Array.from(this._formGroups.values()).map(fg =>\n          fg.statusChanges.pipe(\n            startWith(fg.status),\n            map(() => this.areFormGroupsValid(fg))\n          )\n        );\n\n        return combineLatest(obs).pipe(map(valids => valids.every(Boolean)));\n      })\n    );\n  }\n\n  $hasValuesChanged(): Observable<boolean> {\n    return this._formGroupsChanged$.pipe(\n      observeOn(queueScheduler),\n      switchMap(() => {\n        if (!this.hasFormGroups()) return of(false);\n\n        const obs = Array.from(this._formGroups.entries()).map(([key, fg]) =>\n          fg.valueChanges.pipe(\n            startWith(fg.getRawValue()),\n            map(() => {\n              const currentValues = fg.getRawValue();\n              const snapshot = this._formGroupsSnapshot.get(key);\n              return snapshot ? !isEqual(currentValues, snapshot) : true;\n            })\n          )\n        );\n\n        return combineLatest(obs).pipe(map(changes => changes.some(Boolean)));\n      })\n    );\n  }\n\n  takeFormGroupsSnapshot(): void {\n    this._formGroups.forEach((fg, key) => this._formGroupsSnapshot.set(key, fg.getRawValue()));\n  }\n\n  restoreFormGroupsFromSnapshot(): void {\n    this._formGroups.forEach((fg, key) => {\n      const snapshot = this._formGroupsSnapshot.get(key);\n      if (!snapshot) return;\n\n      Object.entries(fg.controls).forEach(([ctrlKey, ctrl]) => {\n        const newValue = (snapshot as any)[ctrlKey];\n        if (ctrl instanceof FormArray && Array.isArray(newValue)) {\n          this.resetFormArrayToValues(ctrl, newValue);\n        } else {\n          ctrl.reset(newValue);\n        }\n      });\n    });\n\n    this.cancelPendingAsyncValidation();\n  }\n\n  /**\n   * Cancels any in-flight async validators on all registered form groups.\n   *\n   * 1. Collect all PENDING controls and their async validators\n   * 2. Clear all async validators, then update validity (sync-only, no cascade)\n   * 3. Re-attach async validators for future use\n   */\n  cancelPendingAsyncValidation(): void {\n    this._formGroups.forEach(fg => {\n      const pendingControls = this.collectPendingControls(fg);\n      if (pendingControls.length === 0) return;\n\n      for (const { control } of pendingControls) {\n        control.clearAsyncValidators();\n      }\n\n      for (const { control } of pendingControls) {\n        control.updateValueAndValidity({ onlySelf: true });\n      }\n\n      for (const { control, asyncValidator } of pendingControls) {\n        control.setAsyncValidators(asyncValidator);\n      }\n    });\n  }\n\n  private collectPendingControls(\n    control: AbstractControl\n  ): { control: AbstractControl; asyncValidator: AbstractControl['asyncValidator'] }[] {\n    const result: { control: AbstractControl; asyncValidator: AbstractControl['asyncValidator'] }[] = [];\n\n    if (control instanceof FormGroup) {\n      Object.values(control.controls).forEach(c => result.push(...this.collectPendingControls(c)));\n    } else if (control instanceof FormArray) {\n      control.controls.forEach(c => result.push(...this.collectPendingControls(c)));\n    }\n\n    if (control.status === 'PENDING') {\n      result.push({ control, asyncValidator: control.asyncValidator });\n    }\n\n    return result;\n  }\n\n  private resetFormArrayToValues(array: FormArray, newValues: unknown[]): void {\n    if (!Array.isArray(newValues)) return;\n\n    const oldLen = array.length;\n    const newLen = newValues.length;\n\n    if (oldLen === 0 && newLen === 0) return;\n    if (oldLen === 0 && newLen > 0) return this.initFormArrayFromValues(array, newValues);\n\n    while (array.length < newLen) array.push(this.deepCloneControl(array.at(0)));\n    while (array.length > newLen) array.removeAt(array.length - 1);\n\n    array.reset(newValues);\n  }\n\n  private initFormArrayFromValues(array: FormArray, values: unknown[]): void {\n    const protoValue = values[0];\n\n    if (!protoValue || typeof protoValue !== 'object') {\n      console.warn('QD-UI | QdFormGroupManager - Cannot restore FormArray: no prototype and value is not object-like.');\n      return;\n    }\n\n    for (let i = 0; i < values.length; i++) {\n      const controls: Record<string, AbstractControl> = {};\n      Object.keys(protoValue).forEach(key => (controls[key] = new FormControl()));\n      array.push(new FormGroup({ ...controls }));\n    }\n\n    array.reset(values);\n  }\n\n  private deepCloneControl(control: AbstractControl): AbstractControl {\n    if (control instanceof FormControl) return new FormControl(undefined, control.validator, control.asyncValidator);\n\n    if (control instanceof FormGroup) {\n      const groupControls: Record<string, AbstractControl> = {};\n\n      Object.entries(control.controls).forEach(([key, ctrl]) => (groupControls[key] = this.deepCloneControl(ctrl)));\n\n      return new FormGroup(groupControls, control.validator, control.asyncValidator);\n    }\n\n    if (control instanceof FormArray) {\n      const arrControls = control.controls.map(c => this.deepCloneControl(c));\n\n      return new FormArray(arrControls, control.validator, control.asyncValidator);\n    }\n\n    throw new Error('QD-UI | QdFormGroupManager - Unsupported control type');\n  }\n\n  private areFormGroupsValid(fg: FormGroup): boolean {\n    return Object.values(fg.controls).every(c => (c.disabled && !c.validator && !c.asyncValidator ? true : c.valid));\n  }\n}\n"]}
|
|
231
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-group-manager.service.js","sourceRoot":"","sources":["../../../../../../../libs/qd-ui/src/lib/page/shared/services/form-group-manager.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { AbstractControl, FormArray, FormControl, FormGroup } from '@angular/forms';\nimport isEqual from 'lodash/isEqual';\nimport { BehaviorSubject, combineLatest, Observable, of, queueScheduler } from 'rxjs';\nimport { map, observeOn, startWith, switchMap } from 'rxjs/operators';\n\n/**\n * The QdFormGroupManagerService provides centralized registration, snapshotting, and value tracking\n * for multiple Angular `FormGroup` instances. It supports efficient state restoration, including dynamic\n * resizing of embedded `FormArray` controls, while avoiding any structural cloning in snapshots.\n *\n * #### Purpose:\n * This service is useful when managing multiple reactive forms that must be:\n * - Observed for changes\n * - Checked for validity\n * - Compared against a previous state (snapshot)\n * - Reset to a previous state, including dynamic form arrays\n *\n * #### Value Snapshot\n * Snapshots store only **raw values** (via `getRawValue()`), not any control instances.\n * This allows for minimal memory usage and easy comparison using deep equality checks.\n *\n * #### Smart FormArray Reset\n * When restoring a `FormArray`, the service ensures that the number and structure of controls\n * matches the snapshot values. Controls are cloned dynamically from an existing prototype.\n * If a `FormArray` was emptied (e.g. via `removeAt()`), a minimal control structure is inferred\n * from the snapshot data.\n *\n * #### Example Usage:\n *\n * **Register a FormGroup**\n * ```ts\n * service.setFormGroup('userForm', formGroup);\n * ```\n *\n * **Take a Snapshot**\n * ```ts\n * service.takeFormGroupsSnapshot();\n * ```\n *\n * **Detect Changes**\n * ```ts\n * service.$hasValuesChanged().subscribe(changed => {\n *   if (changed) {\n *     console.log('Form data has changed');\n *   }\n * });\n * ```\n *\n * **Restore from Snapshot**\n * ```ts\n * service.restoreFormGroupsFromSnapshot();\n * ```\n *\n * **Check if All Forms Are Valid**\n * ```ts\n * service.$areFormGroupsValid().subscribe(valid => {\n *   if (valid) {\n *     console.log('All forms valid!');\n *   }\n * });\n * ```\n *\n * #### Notes:\n * - FormArrays are not replaced during restore. Their length is adjusted and their content reset.\n * - Control cloning is recursive and structure-preserving.\n * - No structural information is stored in the snapshot; only data.\n */\n@Injectable()\nexport class QdFormGroupManagerService {\n  private _formGroups = new Map<string, FormGroup>();\n  private _formGroupsSnapshot = new Map<string, Record<string, unknown>>();\n  private _formGroupsChanged$ = new BehaviorSubject<void>(undefined);\n\n  setFormGroup(key: string, formGroup: FormGroup): void {\n    this._formGroups.set(key, formGroup);\n    this._formGroupsChanged$.next();\n  }\n\n  getFormGroup(key: string): FormGroup | undefined {\n    return this._formGroups.get(key);\n  }\n\n  getAllFormGroups(): Map<string, FormGroup> {\n    return this._formGroups;\n  }\n\n  tryRemoveFormGroup(key: string): void {\n    this._formGroups.delete(key);\n    this._formGroupsChanged$.next();\n  }\n\n  hasFormGroups(): boolean {\n    return this._formGroups.size > 0;\n  }\n\n  getAllValues(): Record<string, unknown> {\n    const allValues: Record<string, unknown> = {};\n\n    this._formGroups.forEach((formGroup, key) => (allValues[key] = formGroup.value));\n\n    return allValues;\n  }\n\n  isFormGroupKeyUnique(key: string): boolean {\n    return !this._formGroups.has(key);\n  }\n\n  $areFormGroupsValid(): Observable<boolean> {\n    return this._formGroupsChanged$.pipe(\n      observeOn(queueScheduler),\n      switchMap(() => {\n        if (!this.hasFormGroups()) return of(false);\n\n        const obs = Array.from(this._formGroups.values()).map(fg =>\n          fg.statusChanges.pipe(\n            startWith(fg.status),\n            map(() => this.areFormGroupsValid(fg))\n          )\n        );\n\n        return combineLatest(obs).pipe(map(valids => valids.every(Boolean)));\n      })\n    );\n  }\n\n  $hasValuesChanged(): Observable<boolean> {\n    return this._formGroupsChanged$.pipe(\n      observeOn(queueScheduler),\n      switchMap(() => {\n        if (!this.hasFormGroups()) return of(false);\n\n        const obs = Array.from(this._formGroups.entries()).map(([key, fg]) =>\n          fg.valueChanges.pipe(\n            startWith(fg.getRawValue()),\n            map(() => {\n              const currentValues = fg.getRawValue();\n              const snapshot = this._formGroupsSnapshot.get(key);\n              return snapshot ? !isEqual(currentValues, snapshot) : true;\n            })\n          )\n        );\n\n        return combineLatest(obs).pipe(map(changes => changes.some(Boolean)));\n      })\n    );\n  }\n\n  takeFormGroupsSnapshot(): void {\n    this._formGroups.forEach((fg, key) => this._formGroupsSnapshot.set(key, fg.getRawValue()));\n  }\n\n  restoreFormGroupsFromSnapshot(): void {\n    this._formGroups.forEach((fg, key) => {\n      const snapshot = this._formGroupsSnapshot.get(key);\n      if (!snapshot) return;\n\n      this.restoreFormGroup(fg, snapshot);\n    });\n\n    this.cancelPendingAsyncValidation();\n  }\n\n  private restoreFormGroup(fg: FormGroup, snapshot: Record<string, unknown>): void {\n    Object.entries(fg.controls).forEach(([ctrlKey, ctrl]) => {\n      const newValue = (snapshot as any)[ctrlKey];\n\n      if (ctrl instanceof FormArray && Array.isArray(newValue)) {\n        this.resetFormArrayToValues(ctrl, newValue);\n      } else if (ctrl instanceof FormGroup && newValue && typeof newValue === 'object') {\n        this.restoreFormGroup(ctrl, newValue as Record<string, unknown>);\n      } else {\n        ctrl.reset(newValue);\n      }\n    });\n  }\n\n  /**\n   * Cancels any in-flight async validators on all registered form groups.\n   *\n   * 1. Collect all PENDING controls and their async validators\n   * 2. Clear all async validators, then update validity (sync-only, no cascade)\n   * 3. Re-attach async validators for future use\n   */\n  cancelPendingAsyncValidation(): void {\n    this._formGroups.forEach(fg => {\n      const pendingControls = this.collectPendingControls(fg);\n      if (pendingControls.length === 0) return;\n\n      for (const { control } of pendingControls) {\n        control.clearAsyncValidators();\n      }\n\n      for (const { control } of pendingControls) {\n        control.updateValueAndValidity({ onlySelf: true });\n      }\n\n      for (const { control, asyncValidator } of pendingControls) {\n        control.setAsyncValidators(asyncValidator);\n      }\n    });\n  }\n\n  private collectPendingControls(\n    control: AbstractControl\n  ): { control: AbstractControl; asyncValidator: AbstractControl['asyncValidator'] }[] {\n    const result: { control: AbstractControl; asyncValidator: AbstractControl['asyncValidator'] }[] = [];\n\n    if (control instanceof FormGroup) {\n      Object.values(control.controls).forEach(c => result.push(...this.collectPendingControls(c)));\n    } else if (control instanceof FormArray) {\n      control.controls.forEach(c => result.push(...this.collectPendingControls(c)));\n    }\n\n    if (control.status === 'PENDING') {\n      result.push({ control, asyncValidator: control.asyncValidator });\n    }\n\n    return result;\n  }\n\n  private resetFormArrayToValues(array: FormArray, newValues: unknown[]): void {\n    if (!Array.isArray(newValues)) return;\n\n    const oldLen = array.length;\n    const newLen = newValues.length;\n\n    if (oldLen === 0 && newLen === 0) return;\n    if (oldLen === 0 && newLen > 0) return this.initFormArrayFromValues(array, newValues);\n\n    while (array.length < newLen) array.push(this.deepCloneControl(array.at(0)));\n    while (array.length > newLen) array.removeAt(array.length - 1);\n\n    array.reset(newValues);\n  }\n\n  private initFormArrayFromValues(array: FormArray, values: unknown[]): void {\n    const protoValue = values[0];\n\n    if (!protoValue || typeof protoValue !== 'object') {\n      console.warn('QD-UI | QdFormGroupManager - Cannot restore FormArray: no prototype and value is not object-like.');\n      return;\n    }\n\n    for (let i = 0; i < values.length; i++) {\n      const controls: Record<string, AbstractControl> = {};\n      Object.keys(protoValue).forEach(key => (controls[key] = new FormControl()));\n      array.push(new FormGroup({ ...controls }));\n    }\n\n    array.reset(values);\n  }\n\n  private deepCloneControl(control: AbstractControl): AbstractControl {\n    if (control instanceof FormControl) return new FormControl(undefined, control.validator, control.asyncValidator);\n\n    if (control instanceof FormGroup) {\n      const groupControls: Record<string, AbstractControl> = {};\n\n      Object.entries(control.controls).forEach(([key, ctrl]) => (groupControls[key] = this.deepCloneControl(ctrl)));\n\n      return new FormGroup(groupControls, control.validator, control.asyncValidator);\n    }\n\n    if (control instanceof FormArray) {\n      const arrControls = control.controls.map(c => this.deepCloneControl(c));\n\n      return new FormArray(arrControls, control.validator, control.asyncValidator);\n    }\n\n    throw new Error('QD-UI | QdFormGroupManager - Unsupported control type');\n  }\n\n  private areFormGroupsValid(fg: FormGroup): boolean {\n    return Object.values(fg.controls).every(c => (c.disabled && !c.validator && !c.asyncValidator ? true : c.valid));\n  }\n}\n"]}
|
|
@@ -33372,18 +33372,24 @@ class QdFormGroupManagerService {
|
|
|
33372
33372
|
const snapshot = this._formGroupsSnapshot.get(key);
|
|
33373
33373
|
if (!snapshot)
|
|
33374
33374
|
return;
|
|
33375
|
-
|
|
33376
|
-
const newValue = snapshot[ctrlKey];
|
|
33377
|
-
if (ctrl instanceof FormArray && Array.isArray(newValue)) {
|
|
33378
|
-
this.resetFormArrayToValues(ctrl, newValue);
|
|
33379
|
-
}
|
|
33380
|
-
else {
|
|
33381
|
-
ctrl.reset(newValue);
|
|
33382
|
-
}
|
|
33383
|
-
});
|
|
33375
|
+
this.restoreFormGroup(fg, snapshot);
|
|
33384
33376
|
});
|
|
33385
33377
|
this.cancelPendingAsyncValidation();
|
|
33386
33378
|
}
|
|
33379
|
+
restoreFormGroup(fg, snapshot) {
|
|
33380
|
+
Object.entries(fg.controls).forEach(([ctrlKey, ctrl]) => {
|
|
33381
|
+
const newValue = snapshot[ctrlKey];
|
|
33382
|
+
if (ctrl instanceof FormArray && Array.isArray(newValue)) {
|
|
33383
|
+
this.resetFormArrayToValues(ctrl, newValue);
|
|
33384
|
+
}
|
|
33385
|
+
else if (ctrl instanceof FormGroup && newValue && typeof newValue === 'object') {
|
|
33386
|
+
this.restoreFormGroup(ctrl, newValue);
|
|
33387
|
+
}
|
|
33388
|
+
else {
|
|
33389
|
+
ctrl.reset(newValue);
|
|
33390
|
+
}
|
|
33391
|
+
});
|
|
33392
|
+
}
|
|
33387
33393
|
/**
|
|
33388
33394
|
* Cancels any in-flight async validators on all registered form groups.
|
|
33389
33395
|
*
|