@quadrel-enterprise-ui/framework 18.33.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/esm2022/lib/quick-edit/quick-edit.component.mjs +6 -3
- package/fesm2022/quadrel-enterprise-ui-framework.mjs +20 -11
- 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/lib/quick-edit/quick-edit.component.d.ts +1 -0
- package/lib/quick-edit/quick-edit.component.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"]}
|
|
@@ -203,6 +203,9 @@ export class QdQuickEditComponent {
|
|
|
203
203
|
get visibleColumns() {
|
|
204
204
|
return this.config.columns.filter(column => !column.isHidden);
|
|
205
205
|
}
|
|
206
|
+
trackByColumnName(_, column) {
|
|
207
|
+
return column.name;
|
|
208
|
+
}
|
|
206
209
|
get isFormControl() {
|
|
207
210
|
return this.controlContainer != null;
|
|
208
211
|
}
|
|
@@ -353,11 +356,11 @@ export class QdQuickEditComponent {
|
|
|
353
356
|
this.controlContainer.control.valueChanges.pipe(startWith(this.controlContainer.control.value.length), map(() => this.controlContainer.control.value.length), distinctUntilChanged(), tap(() => this.changeDetectorRef.detectChanges())).subscribe();
|
|
354
357
|
}
|
|
355
358
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdQuickEditComponent, deps: [{ token: i1.FormBuilder }, { token: i2.QdPageFooterService, optional: true }, { token: i3.QdPageStoreService, optional: true }, { token: i4.QdSectionToolbarActionService, optional: true }, { token: i5.QdContainerActionService, optional: true }, { token: i6.QdEventBrokerService, optional: true }, { token: i0.ChangeDetectorRef }, { token: i1.ControlContainer, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
356
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdQuickEditComponent, selector: "qd-quick-edit", inputs: { config: "config", data: "data", testId: ["data-test-id", "testId"] }, outputs: { formGroupChange: "formGroupChange", addNewClicked: "addNewClicked" }, viewQueries: [{ propertyName: "customForDirective", first: true, predicate: QdCustomForDirective, descendants: true }, { propertyName: "focusables", predicate: QdFocusableDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"], dependencies: [{ kind: "component", type: i7.QdButtonComponent, selector: "button[qdButton], a[qdButton], button[qd-button]", inputs: ["disabled", "color", "icon", "data-test-id", "additionalInfo"] }, { kind: "directive", type: i8.QdButtonGhostDirective, selector: "button[qdButtonGhost], a[qdButtonGhost]" }, { kind: "directive", type: i9.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i10.QdDropdownComponent, selector: "qd-dropdown", inputs: ["value", "id", "formControlName", "config", "data-test-id", "qdPopoverMaxHeight", "dense"], outputs: ["valueChange", "enterClick", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "component", type: i11.QdInputComponent, selector: "qd-input", inputs: ["formControlName", "value", "config", "isError", "data-test-id"], outputs: ["valueChange", "enterClick", "clickClear", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "component", type: i12.QdIconComponent, selector: "qd-icon", inputs: ["icon"] }, { kind: "directive", type: i13.QdPopoverOnClickDirective, selector: "[qdPopoverOnClick]", inputs: ["qdPopoverOnClick", "positionStrategy", "qdPopoverCloseStrategy", "qdPopoverDisabled", "qdPopoverStopPropagation", "qdPopoverBackgroundColor", "qdPopoverMaxHeight", "qdPopoverMinWidth", "qdPopoverMaxWidth", "qdPopoverAutoSize", "qdPopoverEnableKeyControl"], outputs: ["opened", "closed"], exportAs: ["qdPopoverOnClick"] }, { kind: "directive", type: i14.QdFocusableDirective, selector: "[qdFocusable]" }, { kind: "directive", type: i15.QdTooltipOnClickDirective, selector: "[qdTooltipOnClick]", inputs: ["qdTooltipContent"] }, { kind: "directive", type: i16.QdCustomForDirective, selector: "[qdCustomFor]", inputs: ["qdCustomForOf", "qdCustomForToggler"] }, { kind: "pipe", type: i9.AsyncPipe, name: "async" }, { kind: "pipe", type: i17.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
359
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdQuickEditComponent, selector: "qd-quick-edit", inputs: { config: "config", data: "data", testId: ["data-test-id", "testId"] }, outputs: { formGroupChange: "formGroupChange", addNewClicked: "addNewClicked" }, viewQueries: [{ propertyName: "customForDirective", first: true, predicate: QdCustomForDirective, descendants: true }, { propertyName: "focusables", predicate: QdFocusableDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns; trackBy: trackByColumnName\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"], dependencies: [{ kind: "component", type: i7.QdButtonComponent, selector: "button[qdButton], a[qdButton], button[qd-button]", inputs: ["disabled", "color", "icon", "data-test-id", "additionalInfo"] }, { kind: "directive", type: i8.QdButtonGhostDirective, selector: "button[qdButtonGhost], a[qdButtonGhost]" }, { kind: "directive", type: i9.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i10.QdDropdownComponent, selector: "qd-dropdown", inputs: ["value", "id", "formControlName", "config", "data-test-id", "qdPopoverMaxHeight", "dense"], outputs: ["valueChange", "enterClick", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "component", type: i11.QdInputComponent, selector: "qd-input", inputs: ["formControlName", "value", "config", "isError", "data-test-id"], outputs: ["valueChange", "enterClick", "clickClear", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "component", type: i12.QdIconComponent, selector: "qd-icon", inputs: ["icon"] }, { kind: "directive", type: i13.QdPopoverOnClickDirective, selector: "[qdPopoverOnClick]", inputs: ["qdPopoverOnClick", "positionStrategy", "qdPopoverCloseStrategy", "qdPopoverDisabled", "qdPopoverStopPropagation", "qdPopoverBackgroundColor", "qdPopoverMaxHeight", "qdPopoverMinWidth", "qdPopoverMaxWidth", "qdPopoverAutoSize", "qdPopoverEnableKeyControl"], outputs: ["opened", "closed"], exportAs: ["qdPopoverOnClick"] }, { kind: "directive", type: i14.QdFocusableDirective, selector: "[qdFocusable]" }, { kind: "directive", type: i15.QdTooltipOnClickDirective, selector: "[qdTooltipOnClick]", inputs: ["qdTooltipContent"] }, { kind: "directive", type: i16.QdCustomForDirective, selector: "[qdCustomFor]", inputs: ["qdCustomForOf", "qdCustomForToggler"] }, { kind: "pipe", type: i9.AsyncPipe, name: "async" }, { kind: "pipe", type: i17.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
357
360
|
}
|
|
358
361
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdQuickEditComponent, decorators: [{
|
|
359
362
|
type: Component,
|
|
360
|
-
args: [{ selector: 'qd-quick-edit', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"] }]
|
|
363
|
+
args: [{ selector: 'qd-quick-edit', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns; trackBy: trackByColumnName\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"] }]
|
|
361
364
|
}], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.QdPageFooterService, decorators: [{
|
|
362
365
|
type: Optional
|
|
363
366
|
}] }, { type: i3.QdPageStoreService, decorators: [{
|
|
@@ -391,4 +394,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
|
|
|
391
394
|
type: ViewChild,
|
|
392
395
|
args: [QdCustomForDirective]
|
|
393
396
|
}] } });
|
|
394
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quick-edit.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/quick-edit/quick-edit.component.ts","../../../../../libs/qd-ui/src/lib/quick-edit/quick-edit.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  Self,\n  SimpleChanges,\n  Type,\n  ViewChild,\n  ViewChildren\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ControlContainer, FormArray, FormBuilder, FormGroup } from '@angular/forms';\nimport { combineLatest, filter, Observable, of, Subject, takeUntil } from 'rxjs';\nimport { distinctUntilChanged, map, startWith, tap } from 'rxjs/operators';\n\nimport { QdContainerActionService } from '../container/services/container-action.service';\nimport { COMPONENT_MAP } from '../core/data-facets/data-facets.model';\nimport { QdFocusableDirective } from '../core/focusable/focusable.directive';\nimport { QdEventBrokerService } from '../core/services/event-broker.service';\nimport { QdTooltip } from '../forms/model/forms.interface';\nimport { QdFormControl } from '../forms/qd-form.module';\nimport { QdPageFooterService } from '../page/shared/services/page-footer.service';\nimport { QdPageStoreService } from '../page/shared/services/page-store.service';\nimport { QdSectionToolbarActionService } from '../section/services/section-toolbar-action.service';\nimport { QdQuickEditColumn, QdQuickEditConfig, QdQuickEditSecondaryAction } from './model/quick-edit-config';\nimport { QdQuickEditData, QdQuickEditDataValue, QdQuickEditRow } from './model/quick-edit-data';\nimport { QdCustomForDirective } from './qd-ng-for.directive';\n\n/**\n * The **QdQuickEditComponent** is a dynamic tool that enables users to swiftly edit data without the need for traditional forms.\n * It is specifically designed for inline editing, providing a more seamless and efficient user experience.\n * The component currently supports editing columns with data types such as string, number, and enum.\n * In addition to its editing capabilities, the QdQuickEditComponent also offers features for adding new rows and deleting existing ones.\n *\n * QuickEdit supports two modes:\n * - **Reactive Forms Mode** via `formArrayName` – recommended for full control and validation.\n * - **Template Data Mode** via `data` input – for use cases without external forms.\n *\n * ### Usage\n *\n * #### Configuration\n *\n * ```ts\n * type ColumnDefinition = 'rubrik' | 'data' | 'status';\n * const config: QdQuickEditConfig<ColumnDefinition> = {\n *   columns: [\n *     {\n *       name: 'rubrik',\n *       type: 'text',\n *       i18n: 'translation.key.rubrik',\n *       isEditable: () => false\n *     },\n *     {\n *       name: 'data',\n *       type: 'integer',\n *       i18n: 'translation.key.data',\n *       isEditable: (row, column) => !!row['rubrik'],\n *       validators: [Validators.required] // only relevant in template data mode\n *     }\n *   ],\n *   canAdd: true,\n *   secondaryActions: [...]\n * }\n * ```\n *\n * #### FormArray (recommended)\n *\n * To use QuickEdit with Angular Reactive Forms, define a FormGroup containing a FormArray. The controls must match the defined columns.\n *\n * ```ts\n * form = new FormGroup({\n *   data: new FormArray([\n *     new FormGroup({\n *       rubrik: new QdFormControl('Text'),\n *       data: new QdFormControl(123)\n *     })\n *   ])\n * })\n * ```\n *\n * Then bind the `formArrayName` directive in the template:\n *\n * ```html\n * <form [formGroup]=\"form\">\n *   <qd-quick-edit [config]=\"config\" formArrayName=\"data\"></qd-quick-edit>\n * </form>\n * ```\n *\n * You can add new rows by reacting to `(addNewClicked)`:\n *\n * ```ts\n * addControl(): void {\n *   this.form.get('data')?.insert(0, new FormGroup({ ... }));\n * }\n * ```\n *\n * #### Template Data (optional)\n *\n * Alternatively, you can provide a static `data` array. This is useful for read-only scenarios without reactive forms.\n *\n * ```ts\n * const data: QdQuickEditData<MyColumns> = [\n *   { rubrik: 'Foo', data: 42 },\n *   { rubrik: 'Bar', data: 77 }\n * ];\n * ```\n *\n * ```html\n * <qd-quick-edit [config]=\"config\" [data]=\"data\"></qd-quick-edit>\n * ```\n *\n * #### Validation\n *\n * - In **FormArray mode**, validators must be set directly on your FormControls. Validators in `config.columns` are ignored.\n * - In **template data mode**, validators defined in `config.columns` are applied.\n *\n * **Note: Do not combine both approaches. When using `formArrayName`, only the external FormGroup is used for validation and value access.**\n */\n@Component({\n  selector: 'qd-quick-edit',\n  templateUrl: 'quick-edit.component.html',\n  styleUrls: ['quick-edit.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class QdQuickEditComponent<T extends string> implements OnInit, OnChanges, OnDestroy {\n  /**\n   * Configuration of the QuickEdit. The generic type specifies the column definition. <br />\n   *\n   * **Usage:**\n   *\n   * @example\n   * type ColumnDefinition = 'article' | 'mass' | 'documents';\n   * const config: QdQuickEditConfig<ColumnDefinition> = {\n   *   columns: [\n   *     {\n   *       name: 'rubrik',\n   *       type: 'text',\n   *       i18n: 'translation.key.rubrik',\n   *       isEditable: () => false\n   *     },\n   *     {\n   *       name: 'data',\n   *       type: 'integer',\n   *       i18n: 'translation.key.data',\n   *       isEditable: (row, column) => row['active'] == 1,\n   *       validators: [Validators.required]\n   *     },\n   *     {\n   *       name: 'size',\n   *       type: 'boolean',\n   *       i18n: 'translation.key.size',\n   *       validators: [customValidatorFunction()]\n   *     }\n   *   ]\n   * }\n   */\n  @Input({ required: true }) config!: QdQuickEditConfig<T>;\n\n  /**\n   * Data provider for QuickEdit.\n   */\n  @Input() data: QdQuickEditData<T> | undefined;\n\n  get templateData(): QdQuickEditData<T> {\n    return this.controlContainer?.control?.value ?? this.data;\n  }\n\n  get templateData$(): Observable<QdQuickEditData<T>> {\n    return (\n      this.controlContainer?.control?.valueChanges.pipe(startWith(this.controlContainer.control.value)) ?? of(this.data)\n    );\n  }\n\n  /**\n   * A static test ID for integration tests can be set. <br />\n   * The value for the HTML attribute [data-test-id].\n   */\n  @Input('data-test-id')\n  testId?: string;\n\n  /**\n   * The form group instance of QdQuickEdit.\n   * Use this form group to bind it to step-control of QdStepper or other custom logics.\n   */\n  @Output() formGroupChange = new EventEmitter<any>();\n\n  @Output() addNewClicked = new EventEmitter<void>();\n\n  @ViewChildren(QdFocusableDirective)\n  focusables: QueryList<QdFocusableDirective> | undefined;\n\n  @ViewChild(QdCustomForDirective)\n  customForDirective!: QdCustomForDirective;\n\n  quickEditFormGroup!: FormGroup;\n  viewonly$: Observable<boolean> = of(false);\n  actions$!: Observable<QdQuickEditSecondaryAction<T>[]>;\n\n  private _destroyed$ = new Subject<void>();\n\n  get control(): FormGroup {\n    return (this.controlContainer?.control ?? this.quickEditFormGroup.get('rows')) as FormGroup;\n  }\n\n  get showStandaloneCreate(): boolean {\n    return !this.containerActionService && !this.sectionActionService && !!this.config.canAdd;\n  }\n\n  get canAdd(): boolean {\n    return this.config.canAdd ?? this.sectionActionService != null;\n  }\n\n  get rows(): FormArray {\n    return this.quickEditFormGroup.get('rows') as FormArray;\n  }\n\n  get visibleColumns(): QdQuickEditColumn<T>[] {\n    return this.config.columns.filter(column => !column.isHidden);\n  }\n\n  get isFormControl(): boolean {\n    return this.controlContainer != null;\n  }\n\n  togglerDrawing = false;\n\n  constructor(\n    private fb: FormBuilder,\n    @Optional() readonly footerService: QdPageFooterService,\n    @Optional() readonly pageService: QdPageStoreService<any>,\n    @Optional() private readonly sectionActionService: QdSectionToolbarActionService,\n    @Optional() private readonly containerActionService: QdContainerActionService,\n    @Optional() private readonly eventBrokerService: QdEventBrokerService,\n    private changeDetectorRef: ChangeDetectorRef,\n    @Optional() @Self() private controlContainer: ControlContainer\n  ) {\n    if (!this.isFormControl) {\n      this.quickEditFormGroup = this.fb.group({\n        rows: this.fb.array([])\n      });\n\n      this.quickEditFormGroup.valueChanges.pipe(takeUntilDestroyed()).subscribe(() => {\n        this.formGroupChange.emit(this.quickEditFormGroup);\n      });\n    }\n  }\n\n  ngOnInit(): void {\n    this.validateSetup();\n    this.initOpModeSubscription();\n    this.initCreating();\n    this.initActionsStream();\n    this.initManualChangeDetectionForControlReset();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['data'] || changes['config']) this.initRows();\n  }\n\n  ngOnDestroy(): void {\n    this._destroyed$.next();\n    this._destroyed$.complete();\n  }\n\n  isActionDisabled(action: QdQuickEditSecondaryAction<T>, row: QdQuickEditRow<T>): boolean {\n    if (typeof action.isDisabled === 'function') return action.isDisabled(row);\n    return !!action.isDisabled;\n  }\n\n  handleSecondaryAction(action: QdQuickEditSecondaryAction<T>, rowIndex: number): void {\n    if (!action) return;\n\n    const selectedRowData =\n      (this.controlContainer?.control as FormArray)?.controls[rowIndex].value ?? this.rows.value[rowIndex];\n    action.handler(selectedRowData);\n  }\n\n  getComponentType(column: QdQuickEditColumn<T>): Type<unknown> | null {\n    return COMPONENT_MAP[column.type] || null;\n  }\n\n  getComponentInputs(column: QdQuickEditColumn<T>, row: QdQuickEditRow<T>): any {\n    const testId = this.testId && this.testId + '-cell-' + column.name;\n\n    if (column.type === 'enum')\n      return {\n        data: column.options?.find(option => option.value === row[column.name])?.i18n,\n        testId\n      };\n\n    return { data: row[column.name], testId };\n  }\n\n  createRow(): void {\n    this.redrawOnNextChange();\n\n    if (this.controlContainer) {\n      this.addNewClicked.next();\n    } else {\n      const formGroup = Object.assign(\n        {},\n        ...this.config.columns.map(column => ({\n          [column.name]: new QdFormControl<QdQuickEditDataValue>(undefined, column.validators)\n        }))\n      );\n\n      this.rows.insert(0, new FormGroup(formGroup));\n\n      if (!this.data) this.data = [];\n\n      this.data.unshift(\n        Object.assign({}, ...this.config.columns.map(column => ({ [column.name]: null }))) as QdQuickEditRow<T>\n      );\n    }\n\n    this.changeDetectorRef.detectChanges();\n\n    window.setTimeout(() => this.focusFirstControl());\n  }\n\n  removeRow(index: number): void {\n    this.redrawOnNextChange();\n\n    if (this.controlContainer) {\n      (this.controlContainer.control as FormArray).removeAt(index);\n      this.controlContainer.control?.setValue(this.controlContainer.value);\n    } else {\n      this.rows.removeAt(index);\n      this.data?.splice(index, 1);\n    }\n  }\n\n  hasTooltip(tooltip: QdTooltip): boolean {\n    return !!tooltip && !tooltip.hidden;\n  }\n\n  private initRows(): void {\n    if (this.controlContainer) return;\n\n    while (this.rows.length) {\n      this.rows.removeAt(0);\n    }\n\n    this.data?.forEach(row => this.createFormGroupRow(row));\n  }\n\n  private initCreating(): void {\n    if (this.containerActionService)\n      this.containerActionService.action$.pipe(takeUntil(this._destroyed$)).subscribe(() => this.createRow());\n\n    if (this.sectionActionService)\n      this.sectionActionService.action$\n        .pipe(\n          takeUntil(this._destroyed$),\n          filter(type => type === 'addNew')\n        )\n        .subscribe(() => this.createRow());\n  }\n\n  private createFormGroupRow(row: QdQuickEditRow<T>): void {\n    const group: any = {};\n\n    this.config.columns.forEach(column => {\n      group[column.name] = new QdFormControl<QdQuickEditDataValue>(row[column.name], column.validators);\n    });\n\n    const form = new FormGroup(group);\n    this.rows.push(form);\n  }\n\n  private focusFirstControl(): void {\n    if (!this.focusables) return;\n\n    this.focusables.first?.focus();\n  }\n\n  private initOpModeSubscription(): void {\n    this.viewonly$ =\n      typeof this.config.viewonly === 'boolean'\n        ? of(this.config.viewonly)\n        : this.eventBrokerService?.consume<{ isViewonly: boolean }>('operation_mode_changed').pipe(\n            map(e => e.payload.isViewonly),\n            startWith(false)\n          ) ?? of(false);\n  }\n\n  private validateSetup(): void {\n    if (this.data != null && this.controlContainer != null)\n      console.warn('QdQuickEdit | Data is being ignored when used as FormArray');\n\n    if (!this.data && !this.controlContainer)\n      console.warn(\n        'QdQuickEdit | Either use QuickEdit as FormControl with formArrayName binding or provide data via input.'\n      );\n\n    if (this.controlContainer && this.config.columns.some(column => column.validators != null))\n      console.warn(\n        'QdQuickEdit | Please provide validators in the specific FormControls directly for default validators to be added.'\n      );\n  }\n\n  private redrawOnNextChange(): void {\n    this.togglerDrawing = !this.togglerDrawing;\n  }\n\n  private initActionsStream(): void {\n    this.actions$ = combineLatest([of(this.config.secondaryActions ?? []), this.viewonly$]).pipe(\n      map(([actions, isViewonly]) => this.filterActionsByMode(actions, isViewonly))\n    );\n  }\n\n  private filterActionsByMode(\n    actions: QdQuickEditSecondaryAction<T>[],\n    isViewonly: boolean\n  ): QdQuickEditSecondaryAction<T>[] {\n    const currentMode = isViewonly ? 'view' : 'edit';\n\n    return actions.filter(action => !action.operationMode || action.operationMode === currentMode);\n  }\n\n  private initManualChangeDetectionForControlReset(): void {\n    if (!this.isFormControl) return;\n\n    this.controlContainer!.control!.valueChanges.pipe(\n      startWith(this.controlContainer!.control!.value.length),\n      map(() => this.controlContainer!.control!.value.length),\n      distinctUntilChanged(),\n      tap(() => this.changeDetectorRef.detectChanges())\n    ).subscribe();\n  }\n}\n","<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n  <button\n    qdButton\n    qdButtonGhost\n    icon=\"plus\"\n    (click)=\"createRow()\"\n    [data-test-id]=\"testId + '-button-add-new'\"\n    class=\"create-button-standalone\"\n  >\n    {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n  </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n  <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n    <div class=\"table-row\">\n      <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n        {{ header?.i18n | translate }}\n        <qd-icon\n          *ngIf=\"hasTooltip(header?.tooltip)\"\n          icon=\"circleInfo\"\n          class=\"additional-info\"\n          qdTooltipOnClick\n          [qdTooltipContent]=\"header?.tooltip?.content\"\n        ></qd-icon>\n      </div>\n      <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n        <button class=\"menu-button\">\n          <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n        </button>\n      </div>\n    </div>\n  </div>\n  <div class=\"table-body\">\n    <div\n      class=\"table-row\"\n      *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n      [formGroupName]=\"rowIndex\"\n    >\n      <ng-container *ngFor=\"let column of visibleColumns\">\n        <div class=\"table-cell\">\n          <qd-dropdown\n            [config]=\"{\n              filter: column.filter,\n              options: column.options,\n              placeholder: column.placeholder,\n              placeholderPrefix: column.placeholderPrefix,\n              viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n            }\"\n            [data-test-id]=\"column.name + rowIndex\"\n            *ngIf=\"column.type === 'enum'; else otherTypes\"\n            [dense]=\"true\"\n            [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n            qdFocusable\n            class=\"dropdown\"\n          >\n          </qd-dropdown>\n\n          <ng-template #otherTypes>\n            <qd-input\n              [data-test-id]=\"column.name + rowIndex\"\n              [formControlName]=\"column.name\"\n              *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n              [config]=\"{\n                inputType: column.type === 'integer' ? 'number' : 'text',\n                viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n              }\"\n              qdFocusable\n            ></qd-input>\n          </ng-template>\n        </div>\n      </ng-container>\n      <td\n        *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n        class=\"table-cell actions\"\n        [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n      >\n        <button\n          type=\"button\"\n          [qdPopoverOnClick]=\"menu\"\n          [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n          [qdPopoverStopPropagation]=\"true\"\n          class=\"menu-button\"\n          data-test=\"secondary-actions-toggler\"\n        >\n          <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n        </button>\n\n        <ng-template #menu>\n          <button\n            *ngFor=\"let secondaryAction of actions$ | async\"\n            class=\"secondary-actions\"\n            [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n            [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n            (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n          >\n            {{ secondaryAction.label.i18n | translate }}\n          </button>\n          <button\n            *ngIf=\"canAdd && (viewonly$ | async) === false\"\n            class=\"secondary-actions\"\n            (click)=\"removeRow(rowIndex)\"\n          >\n            {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n          </button>\n        </ng-template>\n      </td>\n    </div>\n  </div>\n  <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n    <p>{{ config.emptyStateView.i18n | translate }}</p>\n  </div>\n</div>\n"]}
|
|
397
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quick-edit.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/quick-edit/quick-edit.component.ts","../../../../../libs/qd-ui/src/lib/quick-edit/quick-edit.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  Self,\n  SimpleChanges,\n  Type,\n  ViewChild,\n  ViewChildren\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ControlContainer, FormArray, FormBuilder, FormGroup } from '@angular/forms';\nimport { combineLatest, filter, Observable, of, Subject, takeUntil } from 'rxjs';\nimport { distinctUntilChanged, map, startWith, tap } from 'rxjs/operators';\n\nimport { QdContainerActionService } from '../container/services/container-action.service';\nimport { COMPONENT_MAP } from '../core/data-facets/data-facets.model';\nimport { QdFocusableDirective } from '../core/focusable/focusable.directive';\nimport { QdEventBrokerService } from '../core/services/event-broker.service';\nimport { QdTooltip } from '../forms/model/forms.interface';\nimport { QdFormControl } from '../forms/qd-form.module';\nimport { QdPageFooterService } from '../page/shared/services/page-footer.service';\nimport { QdPageStoreService } from '../page/shared/services/page-store.service';\nimport { QdSectionToolbarActionService } from '../section/services/section-toolbar-action.service';\nimport { QdQuickEditColumn, QdQuickEditConfig, QdQuickEditSecondaryAction } from './model/quick-edit-config';\nimport { QdQuickEditData, QdQuickEditDataValue, QdQuickEditRow } from './model/quick-edit-data';\nimport { QdCustomForDirective } from './qd-ng-for.directive';\n\n/**\n * The **QdQuickEditComponent** is a dynamic tool that enables users to swiftly edit data without the need for traditional forms.\n * It is specifically designed for inline editing, providing a more seamless and efficient user experience.\n * The component currently supports editing columns with data types such as string, number, and enum.\n * In addition to its editing capabilities, the QdQuickEditComponent also offers features for adding new rows and deleting existing ones.\n *\n * QuickEdit supports two modes:\n * - **Reactive Forms Mode** via `formArrayName` – recommended for full control and validation.\n * - **Template Data Mode** via `data` input – for use cases without external forms.\n *\n * ### Usage\n *\n * #### Configuration\n *\n * ```ts\n * type ColumnDefinition = 'rubrik' | 'data' | 'status';\n * const config: QdQuickEditConfig<ColumnDefinition> = {\n *   columns: [\n *     {\n *       name: 'rubrik',\n *       type: 'text',\n *       i18n: 'translation.key.rubrik',\n *       isEditable: () => false\n *     },\n *     {\n *       name: 'data',\n *       type: 'integer',\n *       i18n: 'translation.key.data',\n *       isEditable: (row, column) => !!row['rubrik'],\n *       validators: [Validators.required] // only relevant in template data mode\n *     }\n *   ],\n *   canAdd: true,\n *   secondaryActions: [...]\n * }\n * ```\n *\n * #### FormArray (recommended)\n *\n * To use QuickEdit with Angular Reactive Forms, define a FormGroup containing a FormArray. The controls must match the defined columns.\n *\n * ```ts\n * form = new FormGroup({\n *   data: new FormArray([\n *     new FormGroup({\n *       rubrik: new QdFormControl('Text'),\n *       data: new QdFormControl(123)\n *     })\n *   ])\n * })\n * ```\n *\n * Then bind the `formArrayName` directive in the template:\n *\n * ```html\n * <form [formGroup]=\"form\">\n *   <qd-quick-edit [config]=\"config\" formArrayName=\"data\"></qd-quick-edit>\n * </form>\n * ```\n *\n * You can add new rows by reacting to `(addNewClicked)`:\n *\n * ```ts\n * addControl(): void {\n *   this.form.get('data')?.insert(0, new FormGroup({ ... }));\n * }\n * ```\n *\n * #### Template Data (optional)\n *\n * Alternatively, you can provide a static `data` array. This is useful for read-only scenarios without reactive forms.\n *\n * ```ts\n * const data: QdQuickEditData<MyColumns> = [\n *   { rubrik: 'Foo', data: 42 },\n *   { rubrik: 'Bar', data: 77 }\n * ];\n * ```\n *\n * ```html\n * <qd-quick-edit [config]=\"config\" [data]=\"data\"></qd-quick-edit>\n * ```\n *\n * #### Validation\n *\n * - In **FormArray mode**, validators must be set directly on your FormControls. Validators in `config.columns` are ignored.\n * - In **template data mode**, validators defined in `config.columns` are applied.\n *\n * **Note: Do not combine both approaches. When using `formArrayName`, only the external FormGroup is used for validation and value access.**\n */\n@Component({\n  selector: 'qd-quick-edit',\n  templateUrl: 'quick-edit.component.html',\n  styleUrls: ['quick-edit.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class QdQuickEditComponent<T extends string> implements OnInit, OnChanges, OnDestroy {\n  /**\n   * Configuration of the QuickEdit. The generic type specifies the column definition. <br />\n   *\n   * **Usage:**\n   *\n   * @example\n   * type ColumnDefinition = 'article' | 'mass' | 'documents';\n   * const config: QdQuickEditConfig<ColumnDefinition> = {\n   *   columns: [\n   *     {\n   *       name: 'rubrik',\n   *       type: 'text',\n   *       i18n: 'translation.key.rubrik',\n   *       isEditable: () => false\n   *     },\n   *     {\n   *       name: 'data',\n   *       type: 'integer',\n   *       i18n: 'translation.key.data',\n   *       isEditable: (row, column) => row['active'] == 1,\n   *       validators: [Validators.required]\n   *     },\n   *     {\n   *       name: 'size',\n   *       type: 'boolean',\n   *       i18n: 'translation.key.size',\n   *       validators: [customValidatorFunction()]\n   *     }\n   *   ]\n   * }\n   */\n  @Input({ required: true }) config!: QdQuickEditConfig<T>;\n\n  /**\n   * Data provider for QuickEdit.\n   */\n  @Input() data: QdQuickEditData<T> | undefined;\n\n  get templateData(): QdQuickEditData<T> {\n    return this.controlContainer?.control?.value ?? this.data;\n  }\n\n  get templateData$(): Observable<QdQuickEditData<T>> {\n    return (\n      this.controlContainer?.control?.valueChanges.pipe(startWith(this.controlContainer.control.value)) ?? of(this.data)\n    );\n  }\n\n  /**\n   * A static test ID for integration tests can be set. <br />\n   * The value for the HTML attribute [data-test-id].\n   */\n  @Input('data-test-id')\n  testId?: string;\n\n  /**\n   * The form group instance of QdQuickEdit.\n   * Use this form group to bind it to step-control of QdStepper or other custom logics.\n   */\n  @Output() formGroupChange = new EventEmitter<any>();\n\n  @Output() addNewClicked = new EventEmitter<void>();\n\n  @ViewChildren(QdFocusableDirective)\n  focusables: QueryList<QdFocusableDirective> | undefined;\n\n  @ViewChild(QdCustomForDirective)\n  customForDirective!: QdCustomForDirective;\n\n  quickEditFormGroup!: FormGroup;\n  viewonly$: Observable<boolean> = of(false);\n  actions$!: Observable<QdQuickEditSecondaryAction<T>[]>;\n\n  private _destroyed$ = new Subject<void>();\n\n  get control(): FormGroup {\n    return (this.controlContainer?.control ?? this.quickEditFormGroup.get('rows')) as FormGroup;\n  }\n\n  get showStandaloneCreate(): boolean {\n    return !this.containerActionService && !this.sectionActionService && !!this.config.canAdd;\n  }\n\n  get canAdd(): boolean {\n    return this.config.canAdd ?? this.sectionActionService != null;\n  }\n\n  get rows(): FormArray {\n    return this.quickEditFormGroup.get('rows') as FormArray;\n  }\n\n  get visibleColumns(): QdQuickEditColumn<T>[] {\n    return this.config.columns.filter(column => !column.isHidden);\n  }\n\n  trackByColumnName(_: number, column: QdQuickEditColumn<T>): string {\n    return column.name;\n  }\n\n  get isFormControl(): boolean {\n    return this.controlContainer != null;\n  }\n\n  togglerDrawing = false;\n\n  constructor(\n    private fb: FormBuilder,\n    @Optional() readonly footerService: QdPageFooterService,\n    @Optional() readonly pageService: QdPageStoreService<any>,\n    @Optional() private readonly sectionActionService: QdSectionToolbarActionService,\n    @Optional() private readonly containerActionService: QdContainerActionService,\n    @Optional() private readonly eventBrokerService: QdEventBrokerService,\n    private changeDetectorRef: ChangeDetectorRef,\n    @Optional() @Self() private controlContainer: ControlContainer\n  ) {\n    if (!this.isFormControl) {\n      this.quickEditFormGroup = this.fb.group({\n        rows: this.fb.array([])\n      });\n\n      this.quickEditFormGroup.valueChanges.pipe(takeUntilDestroyed()).subscribe(() => {\n        this.formGroupChange.emit(this.quickEditFormGroup);\n      });\n    }\n  }\n\n  ngOnInit(): void {\n    this.validateSetup();\n    this.initOpModeSubscription();\n    this.initCreating();\n    this.initActionsStream();\n    this.initManualChangeDetectionForControlReset();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['data'] || changes['config']) this.initRows();\n  }\n\n  ngOnDestroy(): void {\n    this._destroyed$.next();\n    this._destroyed$.complete();\n  }\n\n  isActionDisabled(action: QdQuickEditSecondaryAction<T>, row: QdQuickEditRow<T>): boolean {\n    if (typeof action.isDisabled === 'function') return action.isDisabled(row);\n    return !!action.isDisabled;\n  }\n\n  handleSecondaryAction(action: QdQuickEditSecondaryAction<T>, rowIndex: number): void {\n    if (!action) return;\n\n    const selectedRowData =\n      (this.controlContainer?.control as FormArray)?.controls[rowIndex].value ?? this.rows.value[rowIndex];\n    action.handler(selectedRowData);\n  }\n\n  getComponentType(column: QdQuickEditColumn<T>): Type<unknown> | null {\n    return COMPONENT_MAP[column.type] || null;\n  }\n\n  getComponentInputs(column: QdQuickEditColumn<T>, row: QdQuickEditRow<T>): any {\n    const testId = this.testId && this.testId + '-cell-' + column.name;\n\n    if (column.type === 'enum')\n      return {\n        data: column.options?.find(option => option.value === row[column.name])?.i18n,\n        testId\n      };\n\n    return { data: row[column.name], testId };\n  }\n\n  createRow(): void {\n    this.redrawOnNextChange();\n\n    if (this.controlContainer) {\n      this.addNewClicked.next();\n    } else {\n      const formGroup = Object.assign(\n        {},\n        ...this.config.columns.map(column => ({\n          [column.name]: new QdFormControl<QdQuickEditDataValue>(undefined, column.validators)\n        }))\n      );\n\n      this.rows.insert(0, new FormGroup(formGroup));\n\n      if (!this.data) this.data = [];\n\n      this.data.unshift(\n        Object.assign({}, ...this.config.columns.map(column => ({ [column.name]: null }))) as QdQuickEditRow<T>\n      );\n    }\n\n    this.changeDetectorRef.detectChanges();\n\n    window.setTimeout(() => this.focusFirstControl());\n  }\n\n  removeRow(index: number): void {\n    this.redrawOnNextChange();\n\n    if (this.controlContainer) {\n      (this.controlContainer.control as FormArray).removeAt(index);\n      this.controlContainer.control?.setValue(this.controlContainer.value);\n    } else {\n      this.rows.removeAt(index);\n      this.data?.splice(index, 1);\n    }\n  }\n\n  hasTooltip(tooltip: QdTooltip): boolean {\n    return !!tooltip && !tooltip.hidden;\n  }\n\n  private initRows(): void {\n    if (this.controlContainer) return;\n\n    while (this.rows.length) {\n      this.rows.removeAt(0);\n    }\n\n    this.data?.forEach(row => this.createFormGroupRow(row));\n  }\n\n  private initCreating(): void {\n    if (this.containerActionService)\n      this.containerActionService.action$.pipe(takeUntil(this._destroyed$)).subscribe(() => this.createRow());\n\n    if (this.sectionActionService)\n      this.sectionActionService.action$\n        .pipe(\n          takeUntil(this._destroyed$),\n          filter(type => type === 'addNew')\n        )\n        .subscribe(() => this.createRow());\n  }\n\n  private createFormGroupRow(row: QdQuickEditRow<T>): void {\n    const group: any = {};\n\n    this.config.columns.forEach(column => {\n      group[column.name] = new QdFormControl<QdQuickEditDataValue>(row[column.name], column.validators);\n    });\n\n    const form = new FormGroup(group);\n    this.rows.push(form);\n  }\n\n  private focusFirstControl(): void {\n    if (!this.focusables) return;\n\n    this.focusables.first?.focus();\n  }\n\n  private initOpModeSubscription(): void {\n    this.viewonly$ =\n      typeof this.config.viewonly === 'boolean'\n        ? of(this.config.viewonly)\n        : this.eventBrokerService?.consume<{ isViewonly: boolean }>('operation_mode_changed').pipe(\n            map(e => e.payload.isViewonly),\n            startWith(false)\n          ) ?? of(false);\n  }\n\n  private validateSetup(): void {\n    if (this.data != null && this.controlContainer != null)\n      console.warn('QdQuickEdit | Data is being ignored when used as FormArray');\n\n    if (!this.data && !this.controlContainer)\n      console.warn(\n        'QdQuickEdit | Either use QuickEdit as FormControl with formArrayName binding or provide data via input.'\n      );\n\n    if (this.controlContainer && this.config.columns.some(column => column.validators != null))\n      console.warn(\n        'QdQuickEdit | Please provide validators in the specific FormControls directly for default validators to be added.'\n      );\n  }\n\n  private redrawOnNextChange(): void {\n    this.togglerDrawing = !this.togglerDrawing;\n  }\n\n  private initActionsStream(): void {\n    this.actions$ = combineLatest([of(this.config.secondaryActions ?? []), this.viewonly$]).pipe(\n      map(([actions, isViewonly]) => this.filterActionsByMode(actions, isViewonly))\n    );\n  }\n\n  private filterActionsByMode(\n    actions: QdQuickEditSecondaryAction<T>[],\n    isViewonly: boolean\n  ): QdQuickEditSecondaryAction<T>[] {\n    const currentMode = isViewonly ? 'view' : 'edit';\n\n    return actions.filter(action => !action.operationMode || action.operationMode === currentMode);\n  }\n\n  private initManualChangeDetectionForControlReset(): void {\n    if (!this.isFormControl) return;\n\n    this.controlContainer!.control!.valueChanges.pipe(\n      startWith(this.controlContainer!.control!.value.length),\n      map(() => this.controlContainer!.control!.value.length),\n      distinctUntilChanged(),\n      tap(() => this.changeDetectorRef.detectChanges())\n    ).subscribe();\n  }\n}\n","<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n  <button\n    qdButton\n    qdButtonGhost\n    icon=\"plus\"\n    (click)=\"createRow()\"\n    [data-test-id]=\"testId + '-button-add-new'\"\n    class=\"create-button-standalone\"\n  >\n    {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n  </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n  <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n    <div class=\"table-row\">\n      <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n        {{ header?.i18n | translate }}\n        <qd-icon\n          *ngIf=\"hasTooltip(header?.tooltip)\"\n          icon=\"circleInfo\"\n          class=\"additional-info\"\n          qdTooltipOnClick\n          [qdTooltipContent]=\"header?.tooltip?.content\"\n        ></qd-icon>\n      </div>\n      <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n        <button class=\"menu-button\">\n          <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n        </button>\n      </div>\n    </div>\n  </div>\n  <div class=\"table-body\">\n    <div\n      class=\"table-row\"\n      *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n      [formGroupName]=\"rowIndex\"\n    >\n      <ng-container *ngFor=\"let column of visibleColumns; trackBy: trackByColumnName\">\n        <div class=\"table-cell\">\n          <qd-dropdown\n            [config]=\"{\n              filter: column.filter,\n              options: column.options,\n              placeholder: column.placeholder,\n              placeholderPrefix: column.placeholderPrefix,\n              viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n            }\"\n            [data-test-id]=\"column.name + rowIndex\"\n            *ngIf=\"column.type === 'enum'; else otherTypes\"\n            [dense]=\"true\"\n            [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n            qdFocusable\n            class=\"dropdown\"\n          >\n          </qd-dropdown>\n\n          <ng-template #otherTypes>\n            <qd-input\n              [data-test-id]=\"column.name + rowIndex\"\n              [formControlName]=\"column.name\"\n              *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n              [config]=\"{\n                inputType: column.type === 'integer' ? 'number' : 'text',\n                viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n              }\"\n              qdFocusable\n            ></qd-input>\n          </ng-template>\n        </div>\n      </ng-container>\n      <td\n        *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n        class=\"table-cell actions\"\n        [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n      >\n        <button\n          type=\"button\"\n          [qdPopoverOnClick]=\"menu\"\n          [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n          [qdPopoverStopPropagation]=\"true\"\n          class=\"menu-button\"\n          data-test=\"secondary-actions-toggler\"\n        >\n          <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n        </button>\n\n        <ng-template #menu>\n          <button\n            *ngFor=\"let secondaryAction of actions$ | async\"\n            class=\"secondary-actions\"\n            [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n            [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n            (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n          >\n            {{ secondaryAction.label.i18n | translate }}\n          </button>\n          <button\n            *ngIf=\"canAdd && (viewonly$ | async) === false\"\n            class=\"secondary-actions\"\n            (click)=\"removeRow(rowIndex)\"\n          >\n            {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n          </button>\n        </ng-template>\n      </td>\n    </div>\n  </div>\n  <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n    <p>{{ config.emptyStateView.i18n | translate }}</p>\n  </div>\n</div>\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
|
*
|
|
@@ -38195,6 +38201,9 @@ class QdQuickEditComponent {
|
|
|
38195
38201
|
get visibleColumns() {
|
|
38196
38202
|
return this.config.columns.filter(column => !column.isHidden);
|
|
38197
38203
|
}
|
|
38204
|
+
trackByColumnName(_, column) {
|
|
38205
|
+
return column.name;
|
|
38206
|
+
}
|
|
38198
38207
|
get isFormControl() {
|
|
38199
38208
|
return this.controlContainer != null;
|
|
38200
38209
|
}
|
|
@@ -38345,11 +38354,11 @@ class QdQuickEditComponent {
|
|
|
38345
38354
|
this.controlContainer.control.valueChanges.pipe(startWith(this.controlContainer.control.value.length), map(() => this.controlContainer.control.value.length), distinctUntilChanged(), tap(() => this.changeDetectorRef.detectChanges())).subscribe();
|
|
38346
38355
|
}
|
|
38347
38356
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdQuickEditComponent, deps: [{ token: i1$4.FormBuilder }, { token: QdPageFooterService, optional: true }, { token: QdPageStoreService, optional: true }, { token: QdSectionToolbarActionService, optional: true }, { token: QdContainerActionService, optional: true }, { token: QdEventBrokerService, optional: true }, { token: i0.ChangeDetectorRef }, { token: i1$4.ControlContainer, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
38348
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdQuickEditComponent, selector: "qd-quick-edit", inputs: { config: "config", data: "data", testId: ["data-test-id", "testId"] }, outputs: { formGroupChange: "formGroupChange", addNewClicked: "addNewClicked" }, viewQueries: [{ propertyName: "customForDirective", first: true, predicate: QdCustomForDirective, descendants: true }, { propertyName: "focusables", predicate: QdFocusableDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"], dependencies: [{ kind: "component", type: QdButtonComponent, selector: "button[qdButton], a[qdButton], button[qd-button]", inputs: ["disabled", "color", "icon", "data-test-id", "additionalInfo"] }, { kind: "directive", type: QdButtonGhostDirective, selector: "button[qdButtonGhost], a[qdButtonGhost]" }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: QdDropdownComponent, selector: "qd-dropdown", inputs: ["value", "id", "formControlName", "config", "data-test-id", "qdPopoverMaxHeight", "dense"], outputs: ["valueChange", "enterClick", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "component", type: QdInputComponent, selector: "qd-input", inputs: ["formControlName", "value", "config", "isError", "data-test-id"], outputs: ["valueChange", "enterClick", "clickClear", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "directive", type: i1$4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1$4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1$4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$4.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1$4.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "component", type: QdIconComponent, selector: "qd-icon", inputs: ["icon"] }, { kind: "directive", type: QdPopoverOnClickDirective, selector: "[qdPopoverOnClick]", inputs: ["qdPopoverOnClick", "positionStrategy", "qdPopoverCloseStrategy", "qdPopoverDisabled", "qdPopoverStopPropagation", "qdPopoverBackgroundColor", "qdPopoverMaxHeight", "qdPopoverMinWidth", "qdPopoverMaxWidth", "qdPopoverAutoSize", "qdPopoverEnableKeyControl"], outputs: ["opened", "closed"], exportAs: ["qdPopoverOnClick"] }, { kind: "directive", type: QdFocusableDirective, selector: "[qdFocusable]" }, { kind: "directive", type: QdTooltipOnClickDirective, selector: "[qdTooltipOnClick]", inputs: ["qdTooltipContent"] }, { kind: "directive", type: QdCustomForDirective, selector: "[qdCustomFor]", inputs: ["qdCustomForOf", "qdCustomForToggler"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1$3.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
38357
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdQuickEditComponent, selector: "qd-quick-edit", inputs: { config: "config", data: "data", testId: ["data-test-id", "testId"] }, outputs: { formGroupChange: "formGroupChange", addNewClicked: "addNewClicked" }, viewQueries: [{ propertyName: "customForDirective", first: true, predicate: QdCustomForDirective, descendants: true }, { propertyName: "focusables", predicate: QdFocusableDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns; trackBy: trackByColumnName\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"], dependencies: [{ kind: "component", type: QdButtonComponent, selector: "button[qdButton], a[qdButton], button[qd-button]", inputs: ["disabled", "color", "icon", "data-test-id", "additionalInfo"] }, { kind: "directive", type: QdButtonGhostDirective, selector: "button[qdButtonGhost], a[qdButtonGhost]" }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: QdDropdownComponent, selector: "qd-dropdown", inputs: ["value", "id", "formControlName", "config", "data-test-id", "qdPopoverMaxHeight", "dense"], outputs: ["valueChange", "enterClick", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "component", type: QdInputComponent, selector: "qd-input", inputs: ["formControlName", "value", "config", "isError", "data-test-id"], outputs: ["valueChange", "enterClick", "clickClear", "clickHint", "clickReadonly", "clickViewonly"] }, { kind: "directive", type: i1$4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1$4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1$4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$4.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1$4.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "component", type: QdIconComponent, selector: "qd-icon", inputs: ["icon"] }, { kind: "directive", type: QdPopoverOnClickDirective, selector: "[qdPopoverOnClick]", inputs: ["qdPopoverOnClick", "positionStrategy", "qdPopoverCloseStrategy", "qdPopoverDisabled", "qdPopoverStopPropagation", "qdPopoverBackgroundColor", "qdPopoverMaxHeight", "qdPopoverMinWidth", "qdPopoverMaxWidth", "qdPopoverAutoSize", "qdPopoverEnableKeyControl"], outputs: ["opened", "closed"], exportAs: ["qdPopoverOnClick"] }, { kind: "directive", type: QdFocusableDirective, selector: "[qdFocusable]" }, { kind: "directive", type: QdTooltipOnClickDirective, selector: "[qdTooltipOnClick]", inputs: ["qdTooltipContent"] }, { kind: "directive", type: QdCustomForDirective, selector: "[qdCustomFor]", inputs: ["qdCustomForOf", "qdCustomForToggler"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1$3.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
38349
38358
|
}
|
|
38350
38359
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdQuickEditComponent, decorators: [{
|
|
38351
38360
|
type: Component,
|
|
38352
|
-
args: [{ selector: 'qd-quick-edit', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"] }]
|
|
38361
|
+
args: [{ selector: 'qd-quick-edit', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"editable-actions\" *ngIf=\"showStandaloneCreate\">\n <button\n qdButton\n qdButtonGhost\n icon=\"plus\"\n (click)=\"createRow()\"\n [data-test-id]=\"testId + '-button-add-new'\"\n class=\"create-button-standalone\"\n >\n {{ config.standaloneCreateLabel?.i18n ?? \"i18n.qd.quick.edit.createButtonLabel\" | translate }}\n </button>\n</div>\n\n<div class=\"table\" [formGroup]=\"control\">\n <div class=\"table-header\" *ngIf=\"!config.canAdd || templateData.length > 0 || config.emptyStateView\">\n <div class=\"table-row\">\n <div class=\"table-cell\" *ngFor=\"let header of visibleColumns\">\n {{ header?.i18n | translate }}\n <qd-icon\n *ngIf=\"hasTooltip(header?.tooltip)\"\n icon=\"circleInfo\"\n class=\"additional-info\"\n qdTooltipOnClick\n [qdTooltipContent]=\"header?.tooltip?.content\"\n ></qd-icon>\n </div>\n <div class=\"table-cell actions-column\" *ngIf=\"config?.secondaryActions?.length > 0\">\n <button class=\"menu-button\">\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n </div>\n </div>\n </div>\n <div class=\"table-body\">\n <div\n class=\"table-row\"\n *qdCustomFor=\"let row of templateData; let rowIndex = index; toggler: togglerDrawing\"\n [formGroupName]=\"rowIndex\"\n >\n <ng-container *ngFor=\"let column of visibleColumns; trackBy: trackByColumnName\">\n <div class=\"table-cell\">\n <qd-dropdown\n [config]=\"{\n filter: column.filter,\n options: column.options,\n placeholder: column.placeholder,\n placeholderPrefix: column.placeholderPrefix,\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n [data-test-id]=\"column.name + rowIndex\"\n *ngIf=\"column.type === 'enum'; else otherTypes\"\n [dense]=\"true\"\n [formControl]=\"$any(control.controls[rowIndex])?.controls[column.name]\"\n qdFocusable\n class=\"dropdown\"\n >\n </qd-dropdown>\n\n <ng-template #otherTypes>\n <qd-input\n [data-test-id]=\"column.name + rowIndex\"\n [formControlName]=\"column.name\"\n *ngIf=\"column.type !== 'enum' && $any(control.controls[rowIndex])?.controls[column.name]\"\n [config]=\"{\n inputType: column.type === 'integer' ? 'number' : 'text',\n viewonly: (viewonly$ | async) === true || !column.isEditable(row, column.name)\n }\"\n qdFocusable\n ></qd-input>\n </ng-template>\n </div>\n </ng-container>\n <td\n *ngIf=\"config.secondaryActions?.length > 0 || canAdd\"\n class=\"table-cell actions\"\n [attr.data-test-id]=\"testId + '-cell-inline-actions'\"\n >\n <button\n type=\"button\"\n [qdPopoverOnClick]=\"menu\"\n [qdPopoverCloseStrategy]=\"'onEveryClick'\"\n [qdPopoverStopPropagation]=\"true\"\n class=\"menu-button\"\n data-test=\"secondary-actions-toggler\"\n >\n <qd-icon icon=\"overflowMenuHorizontal\"></qd-icon>\n </button>\n\n <ng-template #menu>\n <button\n *ngFor=\"let secondaryAction of actions$ | async\"\n class=\"secondary-actions\"\n [ngClass]=\"{ disabled: isActionDisabled(secondaryAction, row) }\"\n [attr.disabled]=\"isActionDisabled(secondaryAction, row) || null\"\n (click)=\"handleSecondaryAction(secondaryAction, rowIndex)\"\n >\n {{ secondaryAction.label.i18n | translate }}\n </button>\n <button\n *ngIf=\"canAdd && (viewonly$ | async) === false\"\n class=\"secondary-actions\"\n (click)=\"removeRow(rowIndex)\"\n >\n {{ \"i18n.qd.quick.edit.removeButtonLabel\" | translate }}\n </button>\n </ng-template>\n </td>\n </div>\n </div>\n <div class=\"empty-body\" *ngIf=\"config.emptyStateView && !config.emptyStateView.disabled && templateData.length === 0\">\n <p>{{ config.emptyStateView.i18n | translate }}</p>\n </div>\n</div>\n", styles: [".table{display:flex;width:100%;flex-direction:column;margin:1.25rem auto;background-color:#fff;font-size:.875rem}.table ::ng-deep .qd-input-input{margin-bottom:0!important}.table-header .table-row{padding-top:.125rem;padding-bottom:.125rem;background-color:#e5e5e5;font-weight:700}.table-header .table-row .actions-column{flex:0;border-right:none;visibility:hidden}.table-header,.table-body{display:flex;flex-direction:column}.table-row{display:flex;flex-direction:row;padding:.25rem 1rem;border-bottom:.0625rem solid rgb(213,213,213);gap:1rem}.table-row ::ng-deep qd-form-label{display:none!important}.table-row ::ng-deep qd-input{margin-bottom:0!important}.table-cell{display:flex;height:37px;flex:1;align-items:center;text-align:left}.table-cell:has(.qd-input-error),.table-cell:has(.qd-dropdown-error),.table-cell:has(.qd-form-hint){height:auto;align-items:flex-start}.table-cell.actions{flex:0}.table-row:last-child{border-bottom:none}.table-cell:last-child{border-right:none}.editable-actions{display:flex;justify-content:flex-end;margin-right:.625rem;gap:.625rem}.menu-button{display:flex;padding:0 .625rem 0 .375rem;background:unset;color:#454545;font-size:2rem;vertical-align:middle}.menu-button:hover,.menu-button:focus{color:#000;outline:0!important}.secondary-actions{display:block;width:100%;min-height:2rem;padding:0 1rem;background:#fff0;font-size:.75rem;text-align:left}.secondary-actions:hover{background-color:#f2f7fa}.secondary-actions.disabled{color:#b4b4b4;cursor:not-allowed}.secondary-actions.disabled:hover{background-color:#fff0}.dropdown{min-width:160px}.empty-body{padding:1.5rem;background:#fff}.empty-body p{margin:0 0 .5rem}.additional-info{align-self:center;margin-left:.25rem;color:#069;cursor:pointer;font-size:1rem;font-weight:400;transform:translateY(.0625rem)}.additional-info:hover,.additional-info:focus,.additional-info:active{color:#14516f}\n"] }]
|
|
38353
38362
|
}], ctorParameters: () => [{ type: i1$4.FormBuilder }, { type: QdPageFooterService, decorators: [{
|
|
38354
38363
|
type: Optional
|
|
38355
38364
|
}] }, { type: QdPageStoreService, decorators: [{
|