@cqa-lib/cqa-ui 1.1.499 → 1.1.500

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.
@@ -5,6 +5,7 @@ export class DynamicCellContainerDirective {
5
5
  this.viewContainerRef = viewContainerRef;
6
6
  this.cdr = cdr;
7
7
  this.initialized = false;
8
+ this.isApplyingInputChanges = false;
8
9
  }
9
10
  ngOnInit() {
10
11
  if (this.componentConfig && !this.initialized) {
@@ -16,16 +17,25 @@ export class DynamicCellContainerDirective {
16
17
  if (this.componentRef &&
17
18
  this.componentConfig &&
18
19
  (changes['componentConfig'] || changes['row'] || changes['column'] || changes['rowIndex'] || changes['colId'])) {
19
- // Refresh component inputs when config or row context changes.
20
- const inputs = this.getComponentInputs(this.componentConfig);
21
- Object.keys(inputs).forEach(inputKey => {
22
- if (this.componentRef?.instance && inputKey in this.componentRef.instance) {
23
- this.componentRef.instance[inputKey] = inputs[inputKey];
24
- }
25
- });
26
- // Use detectChanges to ensure styles are updated immediately
27
- this.componentRef.changeDetectorRef.detectChanges();
28
- this.cdr.detectChanges();
20
+ if (this.isApplyingInputChanges) {
21
+ return;
22
+ }
23
+ this.isApplyingInputChanges = true;
24
+ try {
25
+ // Refresh component inputs when config or row context changes.
26
+ const inputs = this.getComponentInputs(this.componentConfig);
27
+ Object.keys(inputs).forEach(inputKey => {
28
+ if (this.componentRef?.instance && inputKey in this.componentRef.instance) {
29
+ this.componentRef.instance[inputKey] = inputs[inputKey];
30
+ }
31
+ });
32
+ // Avoid synchronous detectChanges inside ngOnChanges to prevent re-entrant CD loops.
33
+ this.componentRef.changeDetectorRef.markForCheck();
34
+ this.cdr.markForCheck();
35
+ }
36
+ finally {
37
+ this.isApplyingInputChanges = false;
38
+ }
29
39
  }
30
40
  else if (changes['componentConfig'] && this.componentConfig && !this.initialized) {
31
41
  // Create component if it wasn't created yet
@@ -68,9 +78,9 @@ export class DynamicCellContainerDirective {
68
78
  }
69
79
  }
70
80
  });
71
- // Mark for change detection - use detectChanges to ensure styles are applied
72
- this.componentRef.changeDetectorRef.detectChanges();
73
- this.cdr.detectChanges();
81
+ // Schedule checks without forcing nested synchronous cycles.
82
+ this.componentRef.changeDetectorRef.markForCheck();
83
+ this.cdr.markForCheck();
74
84
  }
75
85
  catch (err) {
76
86
  console.error('Failed to create component in table cell:', err);
@@ -117,4 +127,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
117
127
  }], colId: [{
118
128
  type: Input
119
129
  }] } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-cell-container.directive.js","sourceRoot":"","sources":["../../../../../../src/lib/table/dynamic-table/dynamic-cell-container.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ViewContainerRef, Input, OnInit, OnDestroy, OnChanges, SimpleChanges, ComponentRef, Type, ChangeDetectorRef } from '@angular/core';\nimport { ComponentRenderConfig } from './component-render-config.interface';\n\n@Directive({\n  selector: '[cqaCellContainer]',\n  exportAs: 'cqaCellContainer'\n})\nexport class DynamicCellContainerDirective implements OnInit, OnChanges, OnDestroy {\n  @Input() componentConfig?: Type<any> | ComponentRenderConfig;\n  @Input() row?: any;\n  @Input() column?: any;\n  @Input() rowIndex?: number;\n  @Input() colId?: string;\n  \n  private componentRef?: ComponentRef<any>;\n  private initialized = false;\n\n  constructor(\n    public viewContainerRef: ViewContainerRef,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    if (this.componentConfig && !this.initialized) {\n      this.createComponent();\n      this.initialized = true;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (\n      this.componentRef &&\n      this.componentConfig &&\n      (changes['componentConfig'] || changes['row'] || changes['column'] || changes['rowIndex'] || changes['colId'])\n    ) {\n      // Refresh component inputs when config or row context changes.\n      const inputs = this.getComponentInputs(this.componentConfig);\n      Object.keys(inputs).forEach(inputKey => {\n        if (this.componentRef?.instance && inputKey in this.componentRef.instance) {\n          (this.componentRef.instance as any)[inputKey] = inputs[inputKey];\n        }\n      });\n      // Use detectChanges to ensure styles are updated immediately\n      this.componentRef.changeDetectorRef.detectChanges();\n      this.cdr.detectChanges();\n    } else if (changes['componentConfig'] && this.componentConfig && !this.initialized) {\n      // Create component if it wasn't created yet\n      this.createComponent();\n      this.initialized = true;\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.componentRef) {\n      this.componentRef.destroy();\n      this.componentRef = undefined;\n    }\n  }\n\n  private createComponent(): void {\n    if (!this.componentConfig || !this.viewContainerRef) return;\n\n    try {\n      const componentClass = this.getComponentClass(this.componentConfig);\n      const inputs = this.getComponentInputs(this.componentConfig);\n      const outputs = this.getComponentOutputs(this.componentConfig);\n\n      // Clear any existing content\n      this.viewContainerRef.clear();\n\n      // Create the component\n      this.componentRef = this.viewContainerRef.createComponent(componentClass);\n      \n      // Set inputs immediately after creation, before any change detection\n      // Use Object.assign to set all inputs at once\n      if (this.componentRef?.instance && Object.keys(inputs).length > 0) {\n        Object.assign(this.componentRef.instance, inputs);\n      }\n\n      // Subscribe to outputs\n      Object.keys(outputs).forEach(outputKey => {\n        if (this.componentRef?.instance && outputKey in this.componentRef.instance) {\n          const output = (this.componentRef.instance as any)[outputKey];\n          if (output && typeof output.subscribe === 'function') {\n            output.subscribe((event: any) => {\n              const handler = outputs[outputKey];\n              if (handler) handler(event);\n            });\n          }\n        }\n      });\n\n      // Mark for change detection - use detectChanges to ensure styles are applied\n      this.componentRef.changeDetectorRef.detectChanges();\n      this.cdr.detectChanges();\n    } catch (err) {\n      console.error('Failed to create component in table cell:', err);\n    }\n  }\n\n  private getComponentClass(value: Type<any> | ComponentRenderConfig): Type<any> {\n    if (typeof value === 'function') {\n      return value;\n    }\n    if (typeof value === 'object' && value.component) {\n      return value.component;\n    }\n    throw new Error('Invalid component value');\n  }\n\n  private getComponentInputs(value: Type<any> | ComponentRenderConfig): { [key: string]: any } {\n    if (typeof value === 'object' && value.inputs) {\n      return value.inputs;\n    }\n    return {};\n  }\n\n  private getComponentOutputs(value: Type<any> | ComponentRenderConfig): { [key: string]: (event: any) => void } {\n    if (typeof value === 'object' && value.outputs) {\n      return value.outputs;\n    }\n    return {};\n  }\n}\n\n"]}
130
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-cell-container.directive.js","sourceRoot":"","sources":["../../../../../../src/lib/table/dynamic-table/dynamic-cell-container.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ViewContainerRef, Input, OnInit, OnDestroy, OnChanges, SimpleChanges, ComponentRef, Type, ChangeDetectorRef } from '@angular/core';\nimport { ComponentRenderConfig } from './component-render-config.interface';\n\n@Directive({\n  selector: '[cqaCellContainer]',\n  exportAs: 'cqaCellContainer'\n})\nexport class DynamicCellContainerDirective implements OnInit, OnChanges, OnDestroy {\n  @Input() componentConfig?: Type<any> | ComponentRenderConfig;\n  @Input() row?: any;\n  @Input() column?: any;\n  @Input() rowIndex?: number;\n  @Input() colId?: string;\n  \n  private componentRef?: ComponentRef<any>;\n  private initialized = false;\n  private isApplyingInputChanges = false;\n\n  constructor(\n    public viewContainerRef: ViewContainerRef,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    if (this.componentConfig && !this.initialized) {\n      this.createComponent();\n      this.initialized = true;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (\n      this.componentRef &&\n      this.componentConfig &&\n      (changes['componentConfig'] || changes['row'] || changes['column'] || changes['rowIndex'] || changes['colId'])\n    ) {\n      if (this.isApplyingInputChanges) {\n        return;\n      }\n      this.isApplyingInputChanges = true;\n      try {\n        // Refresh component inputs when config or row context changes.\n        const inputs = this.getComponentInputs(this.componentConfig);\n        Object.keys(inputs).forEach(inputKey => {\n          if (this.componentRef?.instance && inputKey in this.componentRef.instance) {\n            (this.componentRef.instance as any)[inputKey] = inputs[inputKey];\n          }\n        });\n        // Avoid synchronous detectChanges inside ngOnChanges to prevent re-entrant CD loops.\n        this.componentRef.changeDetectorRef.markForCheck();\n        this.cdr.markForCheck();\n      } finally {\n        this.isApplyingInputChanges = false;\n      }\n    } else if (changes['componentConfig'] && this.componentConfig && !this.initialized) {\n      // Create component if it wasn't created yet\n      this.createComponent();\n      this.initialized = true;\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.componentRef) {\n      this.componentRef.destroy();\n      this.componentRef = undefined;\n    }\n  }\n\n  private createComponent(): void {\n    if (!this.componentConfig || !this.viewContainerRef) return;\n\n    try {\n      const componentClass = this.getComponentClass(this.componentConfig);\n      const inputs = this.getComponentInputs(this.componentConfig);\n      const outputs = this.getComponentOutputs(this.componentConfig);\n\n      // Clear any existing content\n      this.viewContainerRef.clear();\n\n      // Create the component\n      this.componentRef = this.viewContainerRef.createComponent(componentClass);\n      \n      // Set inputs immediately after creation, before any change detection\n      // Use Object.assign to set all inputs at once\n      if (this.componentRef?.instance && Object.keys(inputs).length > 0) {\n        Object.assign(this.componentRef.instance, inputs);\n      }\n\n      // Subscribe to outputs\n      Object.keys(outputs).forEach(outputKey => {\n        if (this.componentRef?.instance && outputKey in this.componentRef.instance) {\n          const output = (this.componentRef.instance as any)[outputKey];\n          if (output && typeof output.subscribe === 'function') {\n            output.subscribe((event: any) => {\n              const handler = outputs[outputKey];\n              if (handler) handler(event);\n            });\n          }\n        }\n      });\n\n      // Schedule checks without forcing nested synchronous cycles.\n      this.componentRef.changeDetectorRef.markForCheck();\n      this.cdr.markForCheck();\n    } catch (err) {\n      console.error('Failed to create component in table cell:', err);\n    }\n  }\n\n  private getComponentClass(value: Type<any> | ComponentRenderConfig): Type<any> {\n    if (typeof value === 'function') {\n      return value;\n    }\n    if (typeof value === 'object' && value.component) {\n      return value.component;\n    }\n    throw new Error('Invalid component value');\n  }\n\n  private getComponentInputs(value: Type<any> | ComponentRenderConfig): { [key: string]: any } {\n    if (typeof value === 'object' && value.inputs) {\n      return value.inputs;\n    }\n    return {};\n  }\n\n  private getComponentOutputs(value: Type<any> | ComponentRenderConfig): { [key: string]: (event: any) => void } {\n    if (typeof value === 'object' && value.outputs) {\n      return value.outputs;\n    }\n    return {};\n  }\n}\n\n"]}
@@ -1067,6 +1067,7 @@ class DynamicCellContainerDirective {
1067
1067
  this.viewContainerRef = viewContainerRef;
1068
1068
  this.cdr = cdr;
1069
1069
  this.initialized = false;
1070
+ this.isApplyingInputChanges = false;
1070
1071
  }
1071
1072
  ngOnInit() {
1072
1073
  if (this.componentConfig && !this.initialized) {
@@ -1078,17 +1079,26 @@ class DynamicCellContainerDirective {
1078
1079
  if (this.componentRef &&
1079
1080
  this.componentConfig &&
1080
1081
  (changes['componentConfig'] || changes['row'] || changes['column'] || changes['rowIndex'] || changes['colId'])) {
1081
- // Refresh component inputs when config or row context changes.
1082
- const inputs = this.getComponentInputs(this.componentConfig);
1083
- Object.keys(inputs).forEach(inputKey => {
1084
- var _a;
1085
- if (((_a = this.componentRef) === null || _a === void 0 ? void 0 : _a.instance) && inputKey in this.componentRef.instance) {
1086
- this.componentRef.instance[inputKey] = inputs[inputKey];
1087
- }
1088
- });
1089
- // Use detectChanges to ensure styles are updated immediately
1090
- this.componentRef.changeDetectorRef.detectChanges();
1091
- this.cdr.detectChanges();
1082
+ if (this.isApplyingInputChanges) {
1083
+ return;
1084
+ }
1085
+ this.isApplyingInputChanges = true;
1086
+ try {
1087
+ // Refresh component inputs when config or row context changes.
1088
+ const inputs = this.getComponentInputs(this.componentConfig);
1089
+ Object.keys(inputs).forEach(inputKey => {
1090
+ var _a;
1091
+ if (((_a = this.componentRef) === null || _a === void 0 ? void 0 : _a.instance) && inputKey in this.componentRef.instance) {
1092
+ this.componentRef.instance[inputKey] = inputs[inputKey];
1093
+ }
1094
+ });
1095
+ // Avoid synchronous detectChanges inside ngOnChanges to prevent re-entrant CD loops.
1096
+ this.componentRef.changeDetectorRef.markForCheck();
1097
+ this.cdr.markForCheck();
1098
+ }
1099
+ finally {
1100
+ this.isApplyingInputChanges = false;
1101
+ }
1092
1102
  }
1093
1103
  else if (changes['componentConfig'] && this.componentConfig && !this.initialized) {
1094
1104
  // Create component if it wasn't created yet
@@ -1133,9 +1143,9 @@ class DynamicCellContainerDirective {
1133
1143
  }
1134
1144
  }
1135
1145
  });
1136
- // Mark for change detection - use detectChanges to ensure styles are applied
1137
- this.componentRef.changeDetectorRef.detectChanges();
1138
- this.cdr.detectChanges();
1146
+ // Schedule checks without forcing nested synchronous cycles.
1147
+ this.componentRef.changeDetectorRef.markForCheck();
1148
+ this.cdr.markForCheck();
1139
1149
  }
1140
1150
  catch (err) {
1141
1151
  console.error('Failed to create component in table cell:', err);