@cqa-lib/cqa-ui 1.1.525 → 1.1.526
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/esm2020/lib/assets/images/image-assets.constants.mjs +3 -1
- package/esm2020/lib/compare-runs/compare-runs.component.mjs +1 -1
- package/esm2020/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.mjs +1 -1
- package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +1 -1
- package/esm2020/lib/iterations-loop/iterations-loop.component.mjs +1 -1
- package/esm2020/lib/segment-control/segment-control.component.mjs +6 -3
- package/esm2020/lib/simulator/simulator.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-document-generation-template-step/step-builder-document-generation-template-step.component.mjs +1 -1
- package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +148 -4
- package/esm2020/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.mjs +181 -0
- package/esm2020/lib/templates/modular-table-template/dialogs/move-to-folder-dialog.component.mjs +264 -0
- package/esm2020/lib/templates/modular-table-template/dialogs/new-folder-dialog.component.mjs +352 -0
- package/esm2020/lib/templates/modular-table-template/directives/folder-drag.directive.mjs +45 -0
- package/esm2020/lib/templates/modular-table-template/directives/folder-drop.directive.mjs +95 -0
- package/esm2020/lib/templates/modular-table-template/directives/row-drag.directive.mjs +44 -0
- package/esm2020/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.mjs +479 -0
- package/esm2020/lib/templates/modular-table-template/modular-table-template.component.mjs +1475 -0
- package/esm2020/lib/templates/modular-table-template/modular-table-template.models.mjs +79 -0
- package/esm2020/lib/templates/table-template.component.mjs +88 -12
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +1 -1
- package/esm2020/lib/ui-kit.module.mjs +41 -1
- package/esm2020/public-api.mjs +10 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +3408 -178
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +3388 -176
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/assets/images/image-assets.constants.d.ts +1 -0
- package/lib/segment-control/segment-control.component.d.ts +2 -1
- package/lib/table/dynamic-table/dynamic-table.component.d.ts +43 -1
- package/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.d.ts +34 -0
- package/lib/templates/modular-table-template/dialogs/move-to-folder-dialog.component.d.ts +57 -0
- package/lib/templates/modular-table-template/dialogs/new-folder-dialog.component.d.ts +79 -0
- package/lib/templates/modular-table-template/directives/folder-drag.directive.d.ts +10 -0
- package/lib/templates/modular-table-template/directives/folder-drop.directive.d.ts +22 -0
- package/lib/templates/modular-table-template/directives/row-drag.directive.d.ts +10 -0
- package/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.d.ts +149 -0
- package/lib/templates/modular-table-template/modular-table-template.component.d.ts +453 -0
- package/lib/templates/modular-table-template/modular-table-template.models.d.ts +150 -0
- package/lib/templates/table-template.component.d.ts +40 -2
- package/lib/ui-kit.module.d.ts +153 -145
- package/package.json +1 -1
- package/public-api.d.ts +9 -0
- package/src/lib/assets/images/EmptyFolderState.png +0 -0
- package/src/lib/assets/images/image-assets.constants.ts +3 -0
- package/styles.css +1 -1
|
@@ -5,7 +5,18 @@ import * as i1 from "@angular/platform-browser";
|
|
|
5
5
|
import * as i2 from "../../full-table-loader/full-table-loader.component";
|
|
6
6
|
import * as i3 from "../../table-data-loader/table-data-loader.component";
|
|
7
7
|
import * as i4 from "@angular/common";
|
|
8
|
-
import * as i5 from "
|
|
8
|
+
import * as i5 from "ngx-drag-drop";
|
|
9
|
+
import * as i6 from "./dynamic-cell-container.directive";
|
|
10
|
+
const DYNAMIC_TABLE_REORDER_STYLES = `
|
|
11
|
+
.reorder-handle-cell { width: 44px; }
|
|
12
|
+
.reorder-handle-icon { display: inline-flex; align-items: center; justify-content: center; color: #6B7280; }
|
|
13
|
+
.cqa-row-draggable { cursor: grab; }
|
|
14
|
+
.cqa-row-draggable:active { cursor: grabbing; }
|
|
15
|
+
.cqa-row-draggable.dndDragging { opacity: 0.5; }
|
|
16
|
+
.cqa-row-draggable.dndDraggingSource { opacity: 0.3; }
|
|
17
|
+
tr.cqa-row-drop-above > td { box-shadow: inset 0 3px 0 0 #3F43EE; }
|
|
18
|
+
tr.cqa-row-drop-below > td { box-shadow: inset 0 -3px 0 0 #3F43EE; }
|
|
19
|
+
`;
|
|
9
20
|
export class DynamicTableComponent {
|
|
10
21
|
constructor(sanitizer, cdr) {
|
|
11
22
|
this.sanitizer = sanitizer;
|
|
@@ -18,6 +29,26 @@ export class DynamicTableComponent {
|
|
|
18
29
|
this.enableLocalSort = true;
|
|
19
30
|
// Emit sort changes so parent can perform server-side sort if desired
|
|
20
31
|
this.sortChange = new EventEmitter();
|
|
32
|
+
/**
|
|
33
|
+
* When true, renders a drag-handle column as the leftmost cell and enables
|
|
34
|
+
* row drag-and-drop reordering via ngx-drag-drop. Off by default so existing
|
|
35
|
+
* consumers are unaffected.
|
|
36
|
+
*/
|
|
37
|
+
this.enableRowReorder = false;
|
|
38
|
+
/** Tooltip for the drag handle icon. */
|
|
39
|
+
this.reorderHandleTooltip = 'Drag to reorder';
|
|
40
|
+
/**
|
|
41
|
+
* Emitted after a successful row drop. `orderedData` is a freshly allocated
|
|
42
|
+
* array reflecting the new order — parents should assign it back to `[data]`
|
|
43
|
+
* rather than relying on in-place mutation.
|
|
44
|
+
*/
|
|
45
|
+
this.rowReorder = new EventEmitter();
|
|
46
|
+
/** Index of the row the cursor is currently hovering during a reorder drag. */
|
|
47
|
+
this.dropHoverIndex = null;
|
|
48
|
+
/** Whether the cursor is in the top or bottom half of the hovered row. */
|
|
49
|
+
this.dropPosition = null;
|
|
50
|
+
/** The row currently being dragged — used to suppress the indicator on the source row. */
|
|
51
|
+
this.draggedRow = null;
|
|
21
52
|
// Map to store component references for cleanup: rowIndex_colId -> ComponentRef
|
|
22
53
|
this.componentRefs = new Map();
|
|
23
54
|
// Caches for expensive getters
|
|
@@ -247,6 +278,113 @@ export class DynamicTableComponent {
|
|
|
247
278
|
trackByIndex(index) {
|
|
248
279
|
return index;
|
|
249
280
|
}
|
|
281
|
+
onRowDragStart(row) {
|
|
282
|
+
if (!this.enableRowReorder)
|
|
283
|
+
return;
|
|
284
|
+
this.draggedRow = row;
|
|
285
|
+
}
|
|
286
|
+
onRowDragEnd() {
|
|
287
|
+
this.draggedRow = null;
|
|
288
|
+
this.clearDropIndicator();
|
|
289
|
+
}
|
|
290
|
+
/**
|
|
291
|
+
* Fires continuously while a drag is over a row. Computes whether the cursor
|
|
292
|
+
* is in the top or bottom half of the row and updates the indicator state.
|
|
293
|
+
* Suppresses the indicator on the dragged row itself.
|
|
294
|
+
*/
|
|
295
|
+
onRowDragOver(event, rowIndex) {
|
|
296
|
+
if (!this.enableRowReorder)
|
|
297
|
+
return;
|
|
298
|
+
const drag = event;
|
|
299
|
+
drag.preventDefault();
|
|
300
|
+
const row = this.computedData?.[rowIndex];
|
|
301
|
+
if (this.draggedRow && row === this.draggedRow) {
|
|
302
|
+
if (this.dropHoverIndex !== null || this.dropPosition !== null) {
|
|
303
|
+
this.dropHoverIndex = null;
|
|
304
|
+
this.dropPosition = null;
|
|
305
|
+
this.cdr.markForCheck();
|
|
306
|
+
}
|
|
307
|
+
return;
|
|
308
|
+
}
|
|
309
|
+
const el = drag.currentTarget;
|
|
310
|
+
if (!el)
|
|
311
|
+
return;
|
|
312
|
+
const rect = el.getBoundingClientRect();
|
|
313
|
+
const midY = rect.top + rect.height / 2;
|
|
314
|
+
const pos = drag.clientY < midY ? 'above' : 'below';
|
|
315
|
+
if (this.dropHoverIndex !== rowIndex || this.dropPosition !== pos) {
|
|
316
|
+
this.dropHoverIndex = rowIndex;
|
|
317
|
+
this.dropPosition = pos;
|
|
318
|
+
this.cdr.markForCheck();
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
/** Clears the indicator when the cursor leaves the tbody entirely. */
|
|
322
|
+
onTbodyDragLeave(event) {
|
|
323
|
+
if (!this.enableRowReorder)
|
|
324
|
+
return;
|
|
325
|
+
const drag = event;
|
|
326
|
+
const tbody = drag.currentTarget;
|
|
327
|
+
const related = drag.relatedTarget;
|
|
328
|
+
if (!tbody || !related || !tbody.contains(related)) {
|
|
329
|
+
this.clearDropIndicator();
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
clearDropIndicator() {
|
|
333
|
+
if (this.dropHoverIndex !== null || this.dropPosition !== null) {
|
|
334
|
+
this.dropHoverIndex = null;
|
|
335
|
+
this.dropPosition = null;
|
|
336
|
+
this.cdr.markForCheck();
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
/**
|
|
340
|
+
* Handles a row drop from ngx-drag-drop. Prefers the tracked dragover position
|
|
341
|
+
* (which the user can see live via the insertion line) over `event.index` when
|
|
342
|
+
* computing the final insert slot. Emits a fresh array — does not mutate `data`.
|
|
343
|
+
*/
|
|
344
|
+
onRowDndDrop(event) {
|
|
345
|
+
if (!this.enableRowReorder)
|
|
346
|
+
return;
|
|
347
|
+
const dragged = event?.data;
|
|
348
|
+
if (dragged == null) {
|
|
349
|
+
this.clearDropIndicator();
|
|
350
|
+
return;
|
|
351
|
+
}
|
|
352
|
+
const source = this.computedData || [];
|
|
353
|
+
let previousIndex = source.indexOf(dragged);
|
|
354
|
+
if (previousIndex < 0 && dragged && dragged.id != null) {
|
|
355
|
+
previousIndex = source.findIndex(r => r && r.id === dragged.id);
|
|
356
|
+
}
|
|
357
|
+
if (previousIndex < 0) {
|
|
358
|
+
this.clearDropIndicator();
|
|
359
|
+
return;
|
|
360
|
+
}
|
|
361
|
+
// Prefer the indicator position (what the user just saw) over event.index.
|
|
362
|
+
let insertIndex;
|
|
363
|
+
if (this.dropHoverIndex != null && this.dropPosition) {
|
|
364
|
+
insertIndex = this.dropPosition === 'above' ? this.dropHoverIndex : this.dropHoverIndex + 1;
|
|
365
|
+
}
|
|
366
|
+
else if (typeof event.index === 'number') {
|
|
367
|
+
insertIndex = event.index;
|
|
368
|
+
}
|
|
369
|
+
else {
|
|
370
|
+
insertIndex = source.length;
|
|
371
|
+
}
|
|
372
|
+
const next = [...source];
|
|
373
|
+
next.splice(previousIndex, 1);
|
|
374
|
+
// Removing the source above the target shifts later items up by one.
|
|
375
|
+
if (previousIndex < insertIndex)
|
|
376
|
+
insertIndex -= 1;
|
|
377
|
+
if (insertIndex < 0)
|
|
378
|
+
insertIndex = 0;
|
|
379
|
+
if (insertIndex > next.length)
|
|
380
|
+
insertIndex = next.length;
|
|
381
|
+
this.clearDropIndicator();
|
|
382
|
+
this.draggedRow = null;
|
|
383
|
+
if (insertIndex === previousIndex)
|
|
384
|
+
return;
|
|
385
|
+
next.splice(insertIndex, 0, dragged);
|
|
386
|
+
this.rowReorder.emit({ previousIndex, currentIndex: insertIndex, orderedData: next });
|
|
387
|
+
}
|
|
250
388
|
copyCellJsonPath(path, event, source = 'dbQueryResponse') {
|
|
251
389
|
if (!path)
|
|
252
390
|
return;
|
|
@@ -459,10 +597,10 @@ export class DynamicTableComponent {
|
|
|
459
597
|
}
|
|
460
598
|
}
|
|
461
599
|
DynamicTableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicTableComponent, deps: [{ token: i1.DomSanitizer }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
462
|
-
DynamicTableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DynamicTableComponent, selector: "app-dynamic-table", inputs: { data: "data", columns: "columns", emptyState: "emptyState", gridTemplateColumns: "gridTemplateColumns", screenWidth: "screenWidth", enableSelectAll: "enableSelectAll", rowSelectable: "rowSelectable", enableLocalSort: "enableLocalSort", isTableLoading: "isTableLoading", isTableDataLoading: "isTableDataLoading", cellJsonPathGetter: "cellJsonPathGetter", onJsonPathCopiedHandler: "onJsonPathCopiedHandler" }, outputs: { sortChange: "sortChange" }, host: { classAttribute: "cqa-ui-root" }, queries: [{ propertyName: "emptyTableTpl", first: true, predicate: ["emptyTableTpl"], descendants: true, read: TemplateRef }, { propertyName: "cellTemplates", predicate: DynamicCellTemplateDirective }, { propertyName: "headerTemplates", predicate: DynamicHeaderTemplateDirective }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-ui-root\">\n <div class=\"cqa-relative\" [class.cqa-max-h-[100px]]=\"showTableLoading\">\n <cqa-full-table-loader *ngIf=\"showTableLoading\"></cqa-full-table-loader>\n <cqa-table-data-loader *ngIf=\"showTableDataLoading\"></cqa-table-data-loader>\n <table class=\"table-inner cqa-w-full cqa-border cqa-border-solid cqa-border-gray-200\"\n [class.is-loading]=\"true\"\n [style.min-width.px]=\"computedTableMinWidth || null\"\n *ngIf=\"!showTableLoading\">\n <colgroup>\n <ng-container *ngFor=\"let width of computedColumnWidths; trackBy: trackByIndex\">\n <col [style.width]=\"width\" />\n </ng-container>\n </colgroup>\n\n <thead *ngIf=\"data?.length\">\n <tr class=\"table-header cqa-items-center\">\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <th\n class=\"header-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left'\n ]\">\n <!-- Built-in select-all for checkbox column when enabled and no custom header template -->\n <ng-container\n *ngIf=\"col.fieldId === 'checkbox' && enableSelectAll && !getHeaderTemplate(col.fieldId); else headerTplOrDefault\">\n <div class=\"custom-checkbox\">\n <input type=\"checkbox\" id=\"checkbox\" aria-label=\"Select all rows\" [checked]=\"allSelected\"\n [indeterminate]=\"someSelected\" (change)=\"onSelectAllChange($event)\" class=\"hidden-checkbox\" />\n <label for=\"checkbox\" class=\"custom-checkbox-label\"></label>\n </div>\n </ng-container>\n <ng-template #headerTplOrDefault>\n <ng-container *ngIf=\"getHeaderTemplate(col.fieldId) as headerTpl; else defaultHeader\">\n <ng-container *ngTemplateOutlet=\"headerTpl\"></ng-container>\n </ng-container>\n <ng-template #defaultHeader>\n <ng-container *ngIf=\"col.sortable; else plainHeader\">\n <button type=\"button\" class=\"header-text cqa-inline-flex cqa-items-center cqa-gap-1 cqa-px-0\"\n (click)=\"toggleSort(col)\" [attr.aria-label]=\"'Sort by ' + (col.fieldName || col.fieldId)\">\n <span class=\"cqa-text-[12.3px] cqa-leading-[17.5px] cqa-text-[#0A0A0A] cqa-font-medium\">{{\n col.fieldName }}</span>\n <span *ngIf=\"isSortedAsc(col.fieldId)\">\u25B2</span>\n <span *ngIf=\"isSortedDesc(col.fieldId)\">\u25BC</span>\n </button>\n </ng-container>\n <ng-template #plainHeader>\n <span class=\"header-text\">{{ col.fieldName }}</span>\n </ng-template>\n </ng-template>\n </ng-template>\n </th>\n </ng-container>\n </tr>\n </thead>\n\n <tbody class=\"table-body cqa-w-full\">\n <tr class=\"table-row cqa-bg-surface-default hover:cqa-bg-surface-hover\" *ngFor=\"let row of computedData; let rowIndex = index; trackBy: trackByIndex\"\n [class.selected]=\"row?.isSelected\">\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <td class=\"cell cqa-px-[10.5px] cqa-py-[11px]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left',\n cellJsonPathGetter && col.fieldId !== 'checkbox' && col.fieldId !== 'actions' ? 'cqa-cursor-pointer' : ''\n ]\"\n (dblclick)=\"cellJsonPathGetter && col?.fieldId !== 'checkbox' && col?.fieldId !== 'actions' && copyCellJsonPath(cellJsonPathGetter(rowIndex, col?.fieldId, row), $event, 'dbQueryResponse')\">\n <!-- Built-in checkbox cell when no custom template is provided -->\n <ng-container *ngIf=\"col.fieldId === 'checkbox' && !getCellTemplate(col.fieldId); else regularCell\">\n <div class=\"custom-checkbox\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">\n <input\n type=\"checkbox\"\n class=\"hidden-checkbox\"\n [attr.id]=\"'row-checkbox-' + rowIndex\"\n aria-label=\"Select row\"\n [checked]=\"row?.isSelected\"\n [disabled]=\"!isRowSelectable(row)\"\n (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\n [style.opacity]=\"!isRowSelectable(row) ? '0.45' : null\"\n [style.cursor]=\"!isRowSelectable(row) ? 'not-allowed' : null\"\n [style.pointer-events]=\"!isRowSelectable(row) ? 'none' : null\"\n [attr.for]=\"'row-checkbox-' + rowIndex\">\n </label>\n </div>\n </ng-container>\n <ng-template #regularCell>\n <ng-container *ngIf=\"getCellTemplate(col.fieldId) as cellTpl; else defaultCell\">\n <ng-container\n *ngTemplateOutlet=\"cellTpl; context: {$implicit: row, row: row, value: getCellValue(row, col.fieldValue)}\"></ng-container>\n </ng-container>\n <ng-template #defaultCell>\n <ng-container *ngIf=\"getRenderedValue(row, col) as renderedValue\">\n <ng-container *ngIf=\"isComponent(renderedValue); else nonComponentCell\">\n <ng-container \n cqaCellContainer \n #cellContainer=\"cqaCellContainer\"\n [componentConfig]=\"renderedValue\"\n [row]=\"row\"\n [column]=\"col\"\n [rowIndex]=\"rowIndex\"\n [colId]=\"col.fieldId\">\n </ng-container>\n </ng-container>\n <ng-template #nonComponentCell>\n <ng-container *ngIf=\"isHtmlString(renderedValue); else textCell\">\n <div class=\"cqa-text-xs cqa-leading-[17px]\" [innerHTML]=\"getSanitizedHtml(renderedValue)\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\"></div>\n </ng-container>\n <ng-template #textCell>\n <span class=\"cqa-text-xs cqa-text-[#374151]\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">{{ renderedValue }}</span>\n </ng-template>\n </ng-template>\n </ng-container>\n </ng-template>\n </ng-template>\n </td>\n </ng-container>\n </tr>\n </tbody>\n </table>\n </div>\n</div>", components: [{ type: i2.FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: ["label"] }, { type: i3.TableDataLoaderComponent, selector: "cqa-table-data-loader", inputs: ["label", "size"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i5.DynamicCellContainerDirective, selector: "[cqaCellContainer]", inputs: ["componentConfig", "row", "column", "rowIndex", "colId"], exportAs: ["cqaCellContainer"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
600
|
+
DynamicTableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DynamicTableComponent, selector: "app-dynamic-table", inputs: { data: "data", columns: "columns", emptyState: "emptyState", gridTemplateColumns: "gridTemplateColumns", screenWidth: "screenWidth", enableSelectAll: "enableSelectAll", rowSelectable: "rowSelectable", enableLocalSort: "enableLocalSort", isTableLoading: "isTableLoading", isTableDataLoading: "isTableDataLoading", cellJsonPathGetter: "cellJsonPathGetter", onJsonPathCopiedHandler: "onJsonPathCopiedHandler", enableRowReorder: "enableRowReorder", reorderHandleTooltip: "reorderHandleTooltip" }, outputs: { sortChange: "sortChange", rowReorder: "rowReorder" }, host: { classAttribute: "cqa-ui-root" }, queries: [{ propertyName: "emptyTableTpl", first: true, predicate: ["emptyTableTpl"], descendants: true, read: TemplateRef }, { propertyName: "cellTemplates", predicate: DynamicCellTemplateDirective }, { propertyName: "headerTemplates", predicate: DynamicHeaderTemplateDirective }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-ui-root\">\n <div class=\"cqa-relative\" [class.cqa-max-h-[100px]]=\"showTableLoading\">\n <cqa-full-table-loader *ngIf=\"showTableLoading\"></cqa-full-table-loader>\n <cqa-table-data-loader *ngIf=\"showTableDataLoading\"></cqa-table-data-loader>\n <table class=\"table-inner cqa-w-full cqa-border cqa-border-solid cqa-border-gray-200\"\n [class.is-loading]=\"true\"\n [style.min-width.px]=\"computedTableMinWidth || null\"\n *ngIf=\"!showTableLoading\">\n <colgroup>\n <col *ngIf=\"enableRowReorder\" style=\"width: 44px\" />\n <ng-container *ngFor=\"let width of computedColumnWidths; trackBy: trackByIndex\">\n <col [style.width]=\"width\" />\n </ng-container>\n </colgroup>\n\n <thead *ngIf=\"data?.length\">\n <tr class=\"table-header cqa-items-center\">\n <th\n *ngIf=\"enableRowReorder\"\n class=\"header-cell reorder-handle-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7] cqa-text-center\"\n aria-label=\"Reorder\">\n <span class=\"cqa-sr-only\">Reorder</span>\n </th>\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <th\n class=\"header-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left'\n ]\">\n <!-- Built-in select-all for checkbox column when enabled and no custom header template -->\n <ng-container\n *ngIf=\"col.fieldId === 'checkbox' && enableSelectAll && !getHeaderTemplate(col.fieldId); else headerTplOrDefault\">\n <div class=\"custom-checkbox\">\n <input type=\"checkbox\" id=\"checkbox\" aria-label=\"Select all rows\" [checked]=\"allSelected\"\n [indeterminate]=\"someSelected\" (change)=\"onSelectAllChange($event)\" class=\"hidden-checkbox\" />\n <label for=\"checkbox\" class=\"custom-checkbox-label\"></label>\n </div>\n </ng-container>\n <ng-template #headerTplOrDefault>\n <ng-container *ngIf=\"getHeaderTemplate(col.fieldId) as headerTpl; else defaultHeader\">\n <ng-container *ngTemplateOutlet=\"headerTpl\"></ng-container>\n </ng-container>\n <ng-template #defaultHeader>\n <ng-container *ngIf=\"col.sortable; else plainHeader\">\n <button type=\"button\" class=\"header-text cqa-inline-flex cqa-items-center cqa-gap-1 cqa-px-0\"\n (click)=\"toggleSort(col)\" [attr.aria-label]=\"'Sort by ' + (col.fieldName || col.fieldId)\">\n <span class=\"cqa-text-[12.3px] cqa-leading-[17.5px] cqa-text-[#0A0A0A] cqa-font-medium\">{{\n col.fieldName }}</span>\n <span *ngIf=\"isSortedAsc(col.fieldId)\">\u25B2</span>\n <span *ngIf=\"isSortedDesc(col.fieldId)\">\u25BC</span>\n </button>\n </ng-container>\n <ng-template #plainHeader>\n <span class=\"header-text\">{{ col.fieldName }}</span>\n </ng-template>\n </ng-template>\n </ng-template>\n </th>\n </ng-container>\n </tr>\n </thead>\n\n <tbody class=\"table-body cqa-w-full\"\n [dndDropzone]=\"['cqa-row']\"\n [dndDisableIf]=\"!enableRowReorder\"\n dndEffectAllowed=\"move\"\n (dragleave)=\"onTbodyDragLeave($event)\"\n (dndDrop)=\"onRowDndDrop($event)\">\n <tr class=\"table-row cqa-bg-surface-default hover:cqa-bg-surface-hover\"\n *ngFor=\"let row of computedData; let rowIndex = index; trackBy: trackByIndex\"\n [class.selected]=\"row?.isSelected\"\n [class.cqa-row-draggable]=\"enableRowReorder\"\n [class.cqa-row-drop-above]=\"dropHoverIndex === rowIndex && dropPosition === 'above'\"\n [class.cqa-row-drop-below]=\"dropHoverIndex === rowIndex && dropPosition === 'below'\"\n [dndDraggable]=\"row\"\n [dndDisableIf]=\"!enableRowReorder\"\n dndEffectAllowed=\"move\"\n dndType=\"cqa-row\"\n (dndStart)=\"onRowDragStart(row)\"\n (dndEnd)=\"onRowDragEnd()\"\n (dragover)=\"onRowDragOver($event, rowIndex)\">\n <td *ngIf=\"enableRowReorder\" class=\"cell reorder-handle-cell cqa-px-[10.5px] cqa-py-[11px] cqa-text-center\" [attr.title]=\"reorderHandleTooltip\">\n <span class=\"reorder-handle-icon\" aria-hidden=\"true\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"3\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"10\" cy=\"3\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"6\" cy=\"8\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"10\" cy=\"8\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"6\" cy=\"13\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"10\" cy=\"13\" r=\"1.25\" fill=\"currentColor\"></circle>\n </svg>\n </span>\n </td>\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <td class=\"cell cqa-px-[10.5px] cqa-py-[11px]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left',\n cellJsonPathGetter && col.fieldId !== 'checkbox' && col.fieldId !== 'actions' ? 'cqa-cursor-pointer' : ''\n ]\"\n (dblclick)=\"cellJsonPathGetter && col?.fieldId !== 'checkbox' && col?.fieldId !== 'actions' && copyCellJsonPath(cellJsonPathGetter(rowIndex, col?.fieldId, row), $event, 'dbQueryResponse')\">\n <!-- Built-in checkbox cell when no custom template is provided -->\n <ng-container *ngIf=\"col.fieldId === 'checkbox' && !getCellTemplate(col.fieldId); else regularCell\">\n <div class=\"custom-checkbox\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">\n <input\n type=\"checkbox\"\n class=\"hidden-checkbox\"\n [attr.id]=\"'row-checkbox-' + rowIndex\"\n aria-label=\"Select row\"\n [checked]=\"row?.isSelected\"\n [disabled]=\"!isRowSelectable(row)\"\n (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\n [style.opacity]=\"!isRowSelectable(row) ? '0.45' : null\"\n [style.cursor]=\"!isRowSelectable(row) ? 'not-allowed' : null\"\n [style.pointer-events]=\"!isRowSelectable(row) ? 'none' : null\"\n [attr.for]=\"'row-checkbox-' + rowIndex\">\n </label>\n </div>\n </ng-container>\n <ng-template #regularCell>\n <ng-container *ngIf=\"getCellTemplate(col.fieldId) as cellTpl; else defaultCell\">\n <ng-container\n *ngTemplateOutlet=\"cellTpl; context: {$implicit: row, row: row, value: getCellValue(row, col.fieldValue)}\"></ng-container>\n </ng-container>\n <ng-template #defaultCell>\n <ng-container *ngIf=\"getRenderedValue(row, col) as renderedValue\">\n <ng-container *ngIf=\"isComponent(renderedValue); else nonComponentCell\">\n <ng-container \n cqaCellContainer \n #cellContainer=\"cqaCellContainer\"\n [componentConfig]=\"renderedValue\"\n [row]=\"row\"\n [column]=\"col\"\n [rowIndex]=\"rowIndex\"\n [colId]=\"col.fieldId\">\n </ng-container>\n </ng-container>\n <ng-template #nonComponentCell>\n <ng-container *ngIf=\"isHtmlString(renderedValue); else textCell\">\n <div class=\"cqa-text-xs cqa-leading-[17px]\" [innerHTML]=\"getSanitizedHtml(renderedValue)\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\"></div>\n </ng-container>\n <ng-template #textCell>\n <span class=\"cqa-text-xs cqa-text-[#374151]\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">{{ renderedValue }}</span>\n </ng-template>\n </ng-template>\n </ng-container>\n </ng-template>\n </ng-template>\n </td>\n </ng-container>\n </tr>\n </tbody>\n </table>\n </div>\n</div>", styles: [".reorder-handle-cell{width:44px}.reorder-handle-icon{display:inline-flex;align-items:center;justify-content:center;color:#6b7280}.cqa-row-draggable{cursor:grab}.cqa-row-draggable:active{cursor:grabbing}.cqa-row-draggable.dndDragging{opacity:.5}.cqa-row-draggable.dndDraggingSource{opacity:.3}tr.cqa-row-drop-above>td{box-shadow:inset 0 3px #3f43ee}tr.cqa-row-drop-below>td{box-shadow:inset 0 -3px #3f43ee}\n"], components: [{ type: i2.FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: ["label"] }, { type: i3.TableDataLoaderComponent, selector: "cqa-table-data-loader", inputs: ["label", "size"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i5.DndDropzoneDirective, selector: "[dndDropzone]", inputs: ["dndDropzone", "dndEffectAllowed", "dndAllowExternal", "dndHorizontal", "dndDragoverClass", "dndDropzoneDisabledClass", "dndDisableIf", "dndDisableDropIf"], outputs: ["dndDragover", "dndDrop"] }, { type: i5.DndDraggableDirective, selector: "[dndDraggable]", inputs: ["dndDraggable", "dndEffectAllowed", "dndType", "dndDraggingClass", "dndDraggingSourceClass", "dndDraggableDisabledClass", "dndDragImageOffsetFunction", "dndDisableIf", "dndDisableDragIf"], outputs: ["dndStart", "dndDrag", "dndEnd", "dndMoved", "dndCopied", "dndLinked", "dndCanceled"] }, { type: i6.DynamicCellContainerDirective, selector: "[cqaCellContainer]", inputs: ["componentConfig", "row", "column", "rowIndex", "colId"], exportAs: ["cqaCellContainer"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
463
601
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicTableComponent, decorators: [{
|
|
464
602
|
type: Component,
|
|
465
|
-
args: [{ selector: "app-dynamic-table", host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cqa-ui-root\">\n <div class=\"cqa-relative\" [class.cqa-max-h-[100px]]=\"showTableLoading\">\n <cqa-full-table-loader *ngIf=\"showTableLoading\"></cqa-full-table-loader>\n <cqa-table-data-loader *ngIf=\"showTableDataLoading\"></cqa-table-data-loader>\n <table class=\"table-inner cqa-w-full cqa-border cqa-border-solid cqa-border-gray-200\"\n [class.is-loading]=\"true\"\n [style.min-width.px]=\"computedTableMinWidth || null\"\n *ngIf=\"!showTableLoading\">\n <colgroup>\n <ng-container *ngFor=\"let width of computedColumnWidths; trackBy: trackByIndex\">\n <col [style.width]=\"width\" />\n </ng-container>\n </colgroup>\n\n <thead *ngIf=\"data?.length\">\n <tr class=\"table-header cqa-items-center\">\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <th\n class=\"header-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left'\n ]\">\n <!-- Built-in select-all for checkbox column when enabled and no custom header template -->\n <ng-container\n *ngIf=\"col.fieldId === 'checkbox' && enableSelectAll && !getHeaderTemplate(col.fieldId); else headerTplOrDefault\">\n <div class=\"custom-checkbox\">\n <input type=\"checkbox\" id=\"checkbox\" aria-label=\"Select all rows\" [checked]=\"allSelected\"\n [indeterminate]=\"someSelected\" (change)=\"onSelectAllChange($event)\" class=\"hidden-checkbox\" />\n <label for=\"checkbox\" class=\"custom-checkbox-label\"></label>\n </div>\n </ng-container>\n <ng-template #headerTplOrDefault>\n <ng-container *ngIf=\"getHeaderTemplate(col.fieldId) as headerTpl; else defaultHeader\">\n <ng-container *ngTemplateOutlet=\"headerTpl\"></ng-container>\n </ng-container>\n <ng-template #defaultHeader>\n <ng-container *ngIf=\"col.sortable; else plainHeader\">\n <button type=\"button\" class=\"header-text cqa-inline-flex cqa-items-center cqa-gap-1 cqa-px-0\"\n (click)=\"toggleSort(col)\" [attr.aria-label]=\"'Sort by ' + (col.fieldName || col.fieldId)\">\n <span class=\"cqa-text-[12.3px] cqa-leading-[17.5px] cqa-text-[#0A0A0A] cqa-font-medium\">{{\n col.fieldName }}</span>\n <span *ngIf=\"isSortedAsc(col.fieldId)\">\u25B2</span>\n <span *ngIf=\"isSortedDesc(col.fieldId)\">\u25BC</span>\n </button>\n </ng-container>\n <ng-template #plainHeader>\n <span class=\"header-text\">{{ col.fieldName }}</span>\n </ng-template>\n </ng-template>\n </ng-template>\n </th>\n </ng-container>\n </tr>\n </thead>\n\n <tbody class=\"table-body cqa-w-full\">\n <tr class=\"table-row cqa-bg-surface-default hover:cqa-bg-surface-hover\" *ngFor=\"let row of computedData; let rowIndex = index; trackBy: trackByIndex\"\n [class.selected]=\"row?.isSelected\">\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <td class=\"cell cqa-px-[10.5px] cqa-py-[11px]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left',\n cellJsonPathGetter && col.fieldId !== 'checkbox' && col.fieldId !== 'actions' ? 'cqa-cursor-pointer' : ''\n ]\"\n (dblclick)=\"cellJsonPathGetter && col?.fieldId !== 'checkbox' && col?.fieldId !== 'actions' && copyCellJsonPath(cellJsonPathGetter(rowIndex, col?.fieldId, row), $event, 'dbQueryResponse')\">\n <!-- Built-in checkbox cell when no custom template is provided -->\n <ng-container *ngIf=\"col.fieldId === 'checkbox' && !getCellTemplate(col.fieldId); else regularCell\">\n <div class=\"custom-checkbox\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">\n <input\n type=\"checkbox\"\n class=\"hidden-checkbox\"\n [attr.id]=\"'row-checkbox-' + rowIndex\"\n aria-label=\"Select row\"\n [checked]=\"row?.isSelected\"\n [disabled]=\"!isRowSelectable(row)\"\n (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\n [style.opacity]=\"!isRowSelectable(row) ? '0.45' : null\"\n [style.cursor]=\"!isRowSelectable(row) ? 'not-allowed' : null\"\n [style.pointer-events]=\"!isRowSelectable(row) ? 'none' : null\"\n [attr.for]=\"'row-checkbox-' + rowIndex\">\n </label>\n </div>\n </ng-container>\n <ng-template #regularCell>\n <ng-container *ngIf=\"getCellTemplate(col.fieldId) as cellTpl; else defaultCell\">\n <ng-container\n *ngTemplateOutlet=\"cellTpl; context: {$implicit: row, row: row, value: getCellValue(row, col.fieldValue)}\"></ng-container>\n </ng-container>\n <ng-template #defaultCell>\n <ng-container *ngIf=\"getRenderedValue(row, col) as renderedValue\">\n <ng-container *ngIf=\"isComponent(renderedValue); else nonComponentCell\">\n <ng-container \n cqaCellContainer \n #cellContainer=\"cqaCellContainer\"\n [componentConfig]=\"renderedValue\"\n [row]=\"row\"\n [column]=\"col\"\n [rowIndex]=\"rowIndex\"\n [colId]=\"col.fieldId\">\n </ng-container>\n </ng-container>\n <ng-template #nonComponentCell>\n <ng-container *ngIf=\"isHtmlString(renderedValue); else textCell\">\n <div class=\"cqa-text-xs cqa-leading-[17px]\" [innerHTML]=\"getSanitizedHtml(renderedValue)\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\"></div>\n </ng-container>\n <ng-template #textCell>\n <span class=\"cqa-text-xs cqa-text-[#374151]\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">{{ renderedValue }}</span>\n </ng-template>\n </ng-template>\n </ng-container>\n </ng-template>\n </ng-template>\n </td>\n </ng-container>\n </tr>\n </tbody>\n </table>\n </div>\n</div>", styles: [] }]
|
|
603
|
+
args: [{ selector: "app-dynamic-table", styles: [DYNAMIC_TABLE_REORDER_STYLES], host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cqa-ui-root\">\n <div class=\"cqa-relative\" [class.cqa-max-h-[100px]]=\"showTableLoading\">\n <cqa-full-table-loader *ngIf=\"showTableLoading\"></cqa-full-table-loader>\n <cqa-table-data-loader *ngIf=\"showTableDataLoading\"></cqa-table-data-loader>\n <table class=\"table-inner cqa-w-full cqa-border cqa-border-solid cqa-border-gray-200\"\n [class.is-loading]=\"true\"\n [style.min-width.px]=\"computedTableMinWidth || null\"\n *ngIf=\"!showTableLoading\">\n <colgroup>\n <col *ngIf=\"enableRowReorder\" style=\"width: 44px\" />\n <ng-container *ngFor=\"let width of computedColumnWidths; trackBy: trackByIndex\">\n <col [style.width]=\"width\" />\n </ng-container>\n </colgroup>\n\n <thead *ngIf=\"data?.length\">\n <tr class=\"table-header cqa-items-center\">\n <th\n *ngIf=\"enableRowReorder\"\n class=\"header-cell reorder-handle-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7] cqa-text-center\"\n aria-label=\"Reorder\">\n <span class=\"cqa-sr-only\">Reorder</span>\n </th>\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <th\n class=\"header-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left'\n ]\">\n <!-- Built-in select-all for checkbox column when enabled and no custom header template -->\n <ng-container\n *ngIf=\"col.fieldId === 'checkbox' && enableSelectAll && !getHeaderTemplate(col.fieldId); else headerTplOrDefault\">\n <div class=\"custom-checkbox\">\n <input type=\"checkbox\" id=\"checkbox\" aria-label=\"Select all rows\" [checked]=\"allSelected\"\n [indeterminate]=\"someSelected\" (change)=\"onSelectAllChange($event)\" class=\"hidden-checkbox\" />\n <label for=\"checkbox\" class=\"custom-checkbox-label\"></label>\n </div>\n </ng-container>\n <ng-template #headerTplOrDefault>\n <ng-container *ngIf=\"getHeaderTemplate(col.fieldId) as headerTpl; else defaultHeader\">\n <ng-container *ngTemplateOutlet=\"headerTpl\"></ng-container>\n </ng-container>\n <ng-template #defaultHeader>\n <ng-container *ngIf=\"col.sortable; else plainHeader\">\n <button type=\"button\" class=\"header-text cqa-inline-flex cqa-items-center cqa-gap-1 cqa-px-0\"\n (click)=\"toggleSort(col)\" [attr.aria-label]=\"'Sort by ' + (col.fieldName || col.fieldId)\">\n <span class=\"cqa-text-[12.3px] cqa-leading-[17.5px] cqa-text-[#0A0A0A] cqa-font-medium\">{{\n col.fieldName }}</span>\n <span *ngIf=\"isSortedAsc(col.fieldId)\">\u25B2</span>\n <span *ngIf=\"isSortedDesc(col.fieldId)\">\u25BC</span>\n </button>\n </ng-container>\n <ng-template #plainHeader>\n <span class=\"header-text\">{{ col.fieldName }}</span>\n </ng-template>\n </ng-template>\n </ng-template>\n </th>\n </ng-container>\n </tr>\n </thead>\n\n <tbody class=\"table-body cqa-w-full\"\n [dndDropzone]=\"['cqa-row']\"\n [dndDisableIf]=\"!enableRowReorder\"\n dndEffectAllowed=\"move\"\n (dragleave)=\"onTbodyDragLeave($event)\"\n (dndDrop)=\"onRowDndDrop($event)\">\n <tr class=\"table-row cqa-bg-surface-default hover:cqa-bg-surface-hover\"\n *ngFor=\"let row of computedData; let rowIndex = index; trackBy: trackByIndex\"\n [class.selected]=\"row?.isSelected\"\n [class.cqa-row-draggable]=\"enableRowReorder\"\n [class.cqa-row-drop-above]=\"dropHoverIndex === rowIndex && dropPosition === 'above'\"\n [class.cqa-row-drop-below]=\"dropHoverIndex === rowIndex && dropPosition === 'below'\"\n [dndDraggable]=\"row\"\n [dndDisableIf]=\"!enableRowReorder\"\n dndEffectAllowed=\"move\"\n dndType=\"cqa-row\"\n (dndStart)=\"onRowDragStart(row)\"\n (dndEnd)=\"onRowDragEnd()\"\n (dragover)=\"onRowDragOver($event, rowIndex)\">\n <td *ngIf=\"enableRowReorder\" class=\"cell reorder-handle-cell cqa-px-[10.5px] cqa-py-[11px] cqa-text-center\" [attr.title]=\"reorderHandleTooltip\">\n <span class=\"reorder-handle-icon\" aria-hidden=\"true\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"3\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"10\" cy=\"3\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"6\" cy=\"8\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"10\" cy=\"8\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"6\" cy=\"13\" r=\"1.25\" fill=\"currentColor\"></circle>\n <circle cx=\"10\" cy=\"13\" r=\"1.25\" fill=\"currentColor\"></circle>\n </svg>\n </span>\n </td>\n <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n <td class=\"cell cqa-px-[10.5px] cqa-py-[11px]\"\n [style.min-width.px]=\"col.minWidth\"\n [ngClass]=\"[\n col.fieldId + '-cell',\n col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left',\n cellJsonPathGetter && col.fieldId !== 'checkbox' && col.fieldId !== 'actions' ? 'cqa-cursor-pointer' : ''\n ]\"\n (dblclick)=\"cellJsonPathGetter && col?.fieldId !== 'checkbox' && col?.fieldId !== 'actions' && copyCellJsonPath(cellJsonPathGetter(rowIndex, col?.fieldId, row), $event, 'dbQueryResponse')\">\n <!-- Built-in checkbox cell when no custom template is provided -->\n <ng-container *ngIf=\"col.fieldId === 'checkbox' && !getCellTemplate(col.fieldId); else regularCell\">\n <div class=\"custom-checkbox\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">\n <input\n type=\"checkbox\"\n class=\"hidden-checkbox\"\n [attr.id]=\"'row-checkbox-' + rowIndex\"\n aria-label=\"Select row\"\n [checked]=\"row?.isSelected\"\n [disabled]=\"!isRowSelectable(row)\"\n (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\n [style.opacity]=\"!isRowSelectable(row) ? '0.45' : null\"\n [style.cursor]=\"!isRowSelectable(row) ? 'not-allowed' : null\"\n [style.pointer-events]=\"!isRowSelectable(row) ? 'none' : null\"\n [attr.for]=\"'row-checkbox-' + rowIndex\">\n </label>\n </div>\n </ng-container>\n <ng-template #regularCell>\n <ng-container *ngIf=\"getCellTemplate(col.fieldId) as cellTpl; else defaultCell\">\n <ng-container\n *ngTemplateOutlet=\"cellTpl; context: {$implicit: row, row: row, value: getCellValue(row, col.fieldValue)}\"></ng-container>\n </ng-container>\n <ng-template #defaultCell>\n <ng-container *ngIf=\"getRenderedValue(row, col) as renderedValue\">\n <ng-container *ngIf=\"isComponent(renderedValue); else nonComponentCell\">\n <ng-container \n cqaCellContainer \n #cellContainer=\"cqaCellContainer\"\n [componentConfig]=\"renderedValue\"\n [row]=\"row\"\n [column]=\"col\"\n [rowIndex]=\"rowIndex\"\n [colId]=\"col.fieldId\">\n </ng-container>\n </ng-container>\n <ng-template #nonComponentCell>\n <ng-container *ngIf=\"isHtmlString(renderedValue); else textCell\">\n <div class=\"cqa-text-xs cqa-leading-[17px]\" [innerHTML]=\"getSanitizedHtml(renderedValue)\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\"></div>\n </ng-container>\n <ng-template #textCell>\n <span class=\"cqa-text-xs cqa-text-[#374151]\" [ngClass]=\"[\n col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n ]\">{{ renderedValue }}</span>\n </ng-template>\n </ng-template>\n </ng-container>\n </ng-template>\n </ng-template>\n </td>\n </ng-container>\n </tr>\n </tbody>\n </table>\n </div>\n</div>" }]
|
|
466
604
|
}], ctorParameters: function () { return [{ type: i1.DomSanitizer }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { data: [{
|
|
467
605
|
type: Input
|
|
468
606
|
}], columns: [{
|
|
@@ -489,6 +627,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
489
627
|
type: Input
|
|
490
628
|
}], onJsonPathCopiedHandler: [{
|
|
491
629
|
type: Input
|
|
630
|
+
}], enableRowReorder: [{
|
|
631
|
+
type: Input
|
|
632
|
+
}], reorderHandleTooltip: [{
|
|
633
|
+
type: Input
|
|
634
|
+
}], rowReorder: [{
|
|
635
|
+
type: Output
|
|
492
636
|
}], cellTemplates: [{
|
|
493
637
|
type: ContentChildren,
|
|
494
638
|
args: [DynamicCellTemplateDirective]
|
|
@@ -499,4 +643,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
499
643
|
type: ContentChild,
|
|
500
644
|
args: ['emptyTableTpl', { read: TemplateRef }]
|
|
501
645
|
}] } });
|
|
502
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-table.component.js","sourceRoot":"","sources":["../../../../../../src/lib/table/dynamic-table/dynamic-table.component.ts","../../../../../../src/lib/table/dynamic-table/dynamic-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ContentChild, ContentChildren, Input, QueryList, TemplateRef, EventEmitter, Output, ViewChild, ViewContainerRef, ComponentRef, Type, OnDestroy, ChangeDetectorRef, OnChanges, SimpleChanges, ChangeDetectionStrategy } from \"@angular/core\";\nimport { DomSanitizer, SafeHtml } from \"@angular/platform-browser\";\nimport { DynamicCellTemplateDirective, DynamicHeaderTemplateDirective } from \"./dynamic-cell.directive\";\nimport { ComponentRenderConfig } from \"./component-render-config.interface\";\n\nexport { ComponentRenderConfig } from \"./component-render-config.interface\";\n\nexport interface DynamicTableColumn {\n  fieldId: string;\n  fieldName?: string;\n  fieldValue?: string;\n  /**\n   * Optional custom renderer invoked when no projected cell template is provided.\n   * Returns the string/number/HTML string/Component class/ComponentRenderConfig to display for the given row.\n   * HTML strings will be sanitized and rendered safely.\n   * Component classes will be dynamically instantiated.\n   */\n  render?: (row: any, column: DynamicTableColumn) => string | number | SafeHtml | Type<any> | ComponentRenderConfig;\n  isDefault?: boolean;\n  sortable?: boolean;\n  isShow?: boolean;\n  // New configuration fields to unify layout logic\n  weight?: number; // relative weight for dynamic share\n  fixedPx?: number; // fixed width in px (e.g. checkbox/actions)\n  /** Minimum width in px — applied to both the header cell and every body cell of this column. */\n  minWidth?: number;\n  responsive?: { xs?: boolean; sm?: boolean; md?: boolean }; // hide at specific breakpoints\n  align?: 'left' | 'center' | 'right';\n}\n\n@Component({\n  selector: \"app-dynamic-table\",\n  templateUrl: \"./dynamic-table.component.html\",\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DynamicTableComponent implements OnDestroy, OnChanges {\n  @Input() data: any[] = [];\n  @Input() columns: DynamicTableColumn[] = [];\n  /**\n   * Optional configuration object used when no custom empty template is provided.\n   */\n  @Input() emptyState?: { title?: string; description?: string; imageUrl?: string; actions?: any[] };\n  // Optional override; if omitted we compute from columns\n  @Input() gridTemplateColumns?: string;\n  // Breakpoint-aware sizing; provide from parent to avoid extra listeners\n  @Input() screenWidth?: number;\n  // Enable built-in select-all behavior for a 'checkbox' column\n  @Input() enableSelectAll: boolean = true;\n  // Optional predicate to control whether a row is selectable.\n  // If not provided, all rows are selectable (backward compatible behavior).\n  @Input() rowSelectable?: (row: any) => boolean;\n  // Enable simple client-side sorting when clicking sortable headers (no custom header)\n  @Input() enableLocalSort: boolean = true;\n  // Backward-compat loading flag (applies to both loaders if specific flags not provided)\n  \n  // New: controls the full table overlay loader\n  @Input() isTableLoading?: boolean;\n  // New: controls the inline data loader above the table\n  @Input() isTableDataLoading?: boolean;\n  // Emit sort changes so parent can perform server-side sort if desired\n  @Output() sortChange = new EventEmitter<{ fieldId: string; fieldValue?: string; direction: 'asc' | 'desc' | null }>();\n\n  // Optional getter for JSON path per cell. When provided, double-clicking a cell copies the path.\n  @Input() cellJsonPathGetter?: (rowIndex: number, colId: string, row: any) => string;\n\n  // Optional handler called when a JSON path is copied from a table cell\n  @Input() onJsonPathCopiedHandler?: (event: { path: string; source: string }) => void;\n\n  @ContentChildren(DynamicCellTemplateDirective) cellTemplates!: QueryList<DynamicCellTemplateDirective>;\n  @ContentChildren(DynamicHeaderTemplateDirective) headerTemplates!: QueryList<DynamicHeaderTemplateDirective>;\n  // Allow consumers to project a custom empty-state template using a template ref name\n  // e.g. <ng-template #emptyTableTpl>Custom empty UI</ng-template>\n  @ContentChild('emptyTableTpl', { read: TemplateRef }) emptyTableTpl?: TemplateRef<any>;\n\n  // Map to store component references for cleanup: rowIndex_colId -> ComponentRef\n  private componentRefs = new Map<string, ComponentRef<any>>();\n\n  // Caches for expensive getters\n  visibleColumns: DynamicTableColumn[] = [];\n  computedData: any[] = [];\n  computedGridTemplate: string = '';\n  computedColumnWidths: string[] = [];\n  /**\n   * Hard floor on the `<table>` width = sum of every visible column's minimum (fixed\n   * widths + dynamic columns' `minWidth`). With `table-layout: fixed; width: 100%;` set\n   * by consumers, `<col>` widths become advisory once the container is narrower than\n   * the column-width sum — the browser silently shrinks them. Pinning `min-width` on\n   * the table itself prevents that compaction; the consumer's `overflow-x: auto`\n   * wrapper then scrolls horizontally instead of squishing columns to zero width.\n   */\n  computedTableMinWidth = 0;\n\n  // Cache for sanitized HTML to avoid redundant innerHTML renders\n  private htmlCache = new Map<string, SafeHtml>();\n\n  constructor(\n    private sanitizer: DomSanitizer,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['columns'] || changes['screenWidth']) {\n      this.updateVisibleColumns();\n      this.updateGridTemplate();\n    }\n    if (changes['data'] || changes['columns'] || changes['enableLocalSort']) {\n      this.updateComputedData();\n    }\n  }\n\n  ngOnDestroy(): void {\n    // Clean up all dynamically created components\n    this.componentRefs.forEach(ref => ref.destroy());\n    this.componentRefs.clear();\n    this.htmlCache.clear();\n  }\n\n  private get isXs(): boolean {\n    const w = this.screenWidth || window.innerWidth;\n    return w <= 480;\n  }\n  private get isSm(): boolean {\n    const w = this.screenWidth || window.innerWidth;\n    return w <= 768;\n  }\n  private get isMd(): boolean {\n    const w = this.screenWidth || window.innerWidth;\n    return w <= 992;\n  }\n\n  private updateVisibleColumns(): void {\n    const responsiveFilter = (c: DynamicTableColumn) => {\n      const r = c?.responsive || {};\n      if (r.xs && this.isXs) return false;\n      if (r.sm && this.isSm) return false;\n      if (r.md && this.isMd) return false;\n      return true;\n    };\n    this.visibleColumns = (this.columns || []).filter(c => c.isShow !== false).filter(responsiveFilter);\n  }\n\n  getHeaderTemplate(colId: string): TemplateRef<any> | null {\n    const tpl = this.headerTemplates?.find(t => t.name === colId);\n    return tpl ? tpl.template : null;\n  }\n\n  getCellTemplate(colId: string): TemplateRef<any> | null {\n    const tpl = this.cellTemplates?.find(t => t.name === colId);\n    return tpl ? tpl.template : null;\n  }\n\n  getRenderedValue(row: any, column: DynamicTableColumn): any {\n    if (column?.render && typeof column.render === 'function') {\n      try {\n        return column.render(row, column);\n      } catch (err) {\n        console.warn('DynamicTable render function threw an error for column', column?.fieldId, err);\n        return '';\n      }\n    }\n    return this.getCellValue(row, column?.fieldValue);\n  }\n\n  /**\n   * Checks if the rendered value is a component class or component config\n   */\n  isComponent(value: any): boolean {\n    if (!value) return false;\n    // Check if it's a component class (has @Component decorator metadata)\n    if (typeof value === 'function' && value.prototype) {\n      return true;\n    }\n    // Check if it's a ComponentRenderConfig\n    if (typeof value === 'object' && value.component && typeof value.component === 'function') {\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Gets the component class from a render value (could be class or config)\n   */\n  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  /**\n   * Gets component inputs from a render value\n   */\n  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  /**\n   * Gets component outputs from a render value\n   */\n  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   * Initializes a component in a ViewContainerRef (called from template)\n   */\n  initComponent(\n    vcr: ViewContainerRef,\n    renderedValue: Type<any> | ComponentRenderConfig,\n    row: any,\n    column: DynamicTableColumn,\n    rowIndex: number,\n    colId: string\n  ): boolean {\n    if (!vcr) return false;\n    \n    const key = `${rowIndex}_${colId}`;\n    \n    // If component already exists for this cell, don't recreate\n    if (this.componentRefs.has(key)) {\n      return true;\n    }\n\n    try {\n      const componentClass = this.getComponentClass(renderedValue);\n      const inputs = this.getComponentInputs(renderedValue);\n      const outputs = this.getComponentOutputs(renderedValue);\n\n      // Create the component\n      const componentRef = vcr.createComponent(componentClass);\n      \n      // Set inputs\n      Object.keys(inputs).forEach(inputKey => {\n        if (componentRef.instance && inputKey in componentRef.instance) {\n          (componentRef.instance as any)[inputKey] = inputs[inputKey];\n        }\n      });\n\n      // Subscribe to outputs\n      Object.keys(outputs).forEach(outputKey => {\n        if (componentRef.instance && outputKey in componentRef.instance) {\n          const output = (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          } else if (output && typeof output.emit === 'function') {\n            // For EventEmitter, wrap the emit to call the handler\n            const originalEmit = output.emit.bind(output);\n            (componentRef.instance as any)[outputKey] = {\n              emit: (event: any) => {\n                originalEmit(event);\n                const handler = outputs[outputKey];\n                if (handler) handler(event);\n              },\n              subscribe: output.subscribe.bind(output)\n            };\n          }\n        }\n      });\n\n      // Mark for change detection\n      componentRef.changeDetectorRef.markForCheck();\n      \n      // Store reference for cleanup\n      this.componentRefs.set(key, componentRef);\n      return true;\n    } catch (err) {\n      console.error('Failed to create component in table cell:', err);\n      return false;\n    }\n  }\n\n  /**\n   * Checks if the rendered value is an HTML string that should be rendered with innerHTML\n   */\n  isHtmlString(value: any): boolean {\n    if (typeof value !== 'string') return false;\n    // Faster check: if it contains an opening tag pattern near the start\n    // or just has any brackets. Wrapping in a regex with [\\s\\S]* is very slow.\n    return value.includes('<') && /<[a-z/][^>]*>/i.test(value);\n  }\n\n  /**\n   * Sanitizes HTML string for safe rendering with caching\n   */\n  getSanitizedHtml(value: string): SafeHtml {\n    let cached = this.htmlCache.get(value);\n    if (!cached) {\n      cached = this.sanitizer.bypassSecurityTrustHtml(value);\n      this.htmlCache.set(value, cached);\n      \n      // Simple cache eviction\n      if (this.htmlCache.size > 1000) {\n        const firstKey = this.htmlCache.keys().next().value;\n        if (firstKey !== undefined) {\n          this.htmlCache.delete(firstKey);\n        }\n      }\n    }\n    return cached;\n  }\n\n  getCellValue(row: any, path?: string): any {\n    if (!row || !path) return \"\";\n    const parts = path.split(\".\");\n    let current = row;\n    for (const part of parts) {\n      if (current == null) return \"\";\n      current = current[part];\n    }\n    return current ?? \"\";\n  }\n\n  trackByIndex(index: number): number {\n    return index;\n  }\n\n  copyCellJsonPath(path: string, event: Event, source: string = 'dbQueryResponse'): void {\n    if (!path) return;\n    event.preventDefault();\n    event.stopPropagation();\n    navigator.clipboard.writeText(path).then(() => {\n      // Call the handler if provided\n      if (this.onJsonPathCopiedHandler) {\n        try {\n          this.onJsonPathCopiedHandler({ path, source });\n        } catch (handlerError) {\n          console.error('Error in onJsonPathCopiedHandler:', handlerError);\n        }\n      }\n    }).catch(err => {\n      console.error('Failed to copy jsonPath:', err);\n      // Still call the handler even on failure (similar to api-step pattern)\n      if (this.onJsonPathCopiedHandler) {\n        try {\n          this.onJsonPathCopiedHandler({ path, source });\n        } catch (handlerError) {\n          console.error('Error in onJsonPathCopiedHandler:', handlerError);\n        }\n      }\n    });\n  }\n\n  // Compute grid-template-columns string from column config\n  private updateGridTemplate(): void {\n    const cols = this.visibleColumns;\n    if (!cols?.length) {\n      this.computedGridTemplate = '';\n      this.computedColumnWidths = [];\n      return;\n    }\n    \n    const fixedPx = cols.reduce((sum, c) => sum + (c.fixedPx || 0), 0);\n    const dynamicCols = cols.filter(c => !c.fixedPx);\n    const totalWeight = dynamicCols.reduce((sum, c) => sum + (c.weight || 1), 0) || 1;\n\n    // Floor on the table itself = every column's minimum width contribution. Fixed\n    // columns contribute `fixedPx`; dynamic columns contribute `minWidth` (or 0 when\n    // unset). The template binds this via `[style.min-width.px]` on the <table> so\n    // narrow containers force horizontal scroll instead of squishing columns to 0.\n    this.computedTableMinWidth = cols.reduce(\n      (sum, c) => sum + (c.fixedPx && c.fixedPx > 0 ? c.fixedPx : (c.minWidth || 0)),\n      0,\n    );\n\n    this.computedColumnWidths = cols.map(c => {\n      if (c.fixedPx && c.fixedPx > 0) {\n        // Fixed-width columns still respect minWidth in the (rare) case the consumer\n        // configured both — `max(fixed, min)` picks whichever is larger.\n        return c.minWidth && c.minWidth > c.fixedPx\n          ? `max(${c.fixedPx}px, ${c.minWidth}px)`\n          : `${c.fixedPx}px`;\n      }\n      const share = (c.weight || 1) / totalWeight;\n      const dynamicWidth = `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;\n      // Dynamic-share columns can collapse to ~0 when the table is narrower than\n      // (sum of fixed widths) + the share they'd compute to. Using CSS `max()` floors\n      // the column at `minWidth` so it grows naturally as the table widens but never\n      // shrinks below the configured pixel minimum. With `table-layout: fixed`, the\n      // <col> width is authoritative — without max() here the cell-level `min-width`\n      // we set on <th>/<td> is ignored and columns visually disappear at narrow widths.\n      return c.minWidth && c.minWidth > 0\n        ? `max(${c.minWidth}px, ${dynamicWidth})`\n        : dynamicWidth;\n    });\n\n    console.log('computedColumnWidths', this.computedColumnWidths);\n    \n    if (this.gridTemplateColumns) {\n      this.computedGridTemplate = this.gridTemplateColumns;\n    } else {\n      this.computedGridTemplate = this.computedColumnWidths.join(' ');\n    }\n  }\n\n  // Selection helpers\n  isRowSelectable(row: any): boolean {\n    if (!row) return false;\n    if (typeof this.rowSelectable !== 'function') return true;\n    try {\n      return !!this.rowSelectable(row);\n    } catch {\n      return true;\n    }\n  }\n\n  get allSelected(): boolean {\n    const rows = (this.data || []).filter((r) => this.isRowSelectable(r));\n    if (!rows.length) return false;\n    return rows.every(r => !!r?.isSelected);\n  }\n\n  get someSelected(): boolean {\n    const rows = (this.data || []).filter((r) => this.isRowSelectable(r));\n    if (!rows.length) return false;\n    const anySelected = rows.some(r => !!r?.isSelected);\n    return anySelected && !this.allSelected;\n  }\n\n  onSelectAllChange(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    this.toggleSelectAll(target.checked);\n  }\n\n  onRowSelectChange(event: Event, row: any): void {\n    if (!this.isRowSelectable(row)) {\n      row.isSelected = false;\n      return;\n    }\n    const target = event.target as HTMLInputElement;\n    row.isSelected = target.checked;\n  }\n\n  toggleSelectAll(checked: boolean): void {\n    const rows = this.data || [];\n    for (const row of rows) {\n      if (row) {\n        row.isSelected = this.isRowSelectable(row) ? checked : false;\n      }\n    }\n  }\n\n  // ===== Sorting support =====\n  private _sortActive?: string;\n  private _sortDirection: 'asc' | 'desc' | null = null;\n\n  private updateComputedData(): void {\n    const source = this.data || [];\n    if (!this.enableLocalSort || !this._sortActive || !this._sortDirection) {\n      this.computedData = source;\n      return;\n    }\n    const col = this.visibleColumns.find(c => c.fieldId === this._sortActive);\n    if (!col || !col.fieldValue) {\n      this.computedData = source;\n      return;\n    }\n    const dir = this._sortDirection === 'asc' ? 1 : -1;\n    const fieldPath = col.fieldValue;\n    const out = [...source];\n    out.sort((a, b) => dir * this.compareValues(this.getCellValue(a, fieldPath), this.getCellValue(b, fieldPath)));\n    this.computedData = out;\n  }\n\n  // Computed loading flags to support backward compatibility\n  get showTableLoading(): boolean {\n    return this.isTableLoading ?? false;\n  }\n\n  get showTableDataLoading(): boolean {\n    return this.isTableDataLoading ?? false;\n  }\n\n  // True when table has no data and is not currently loading — used to show an empty state\n  get isEmpty(): boolean {\n    const anyLoading = this.showTableLoading || this.showTableDataLoading;\n    return !anyLoading && (!(this.data && this.data.length) || this.data.length === 0);\n  }\n\n  isSortedAsc(colId: string): boolean {\n    return this._sortActive === colId && this._sortDirection === 'asc';\n  }\n\n  isSortedDesc(colId: string): boolean {\n    return this._sortActive === colId && this._sortDirection === 'desc';\n  }\n\n  toggleSort(col: DynamicTableColumn): void {\n    if (!col?.sortable) return;\n    const colId = col.fieldId;\n    if (this._sortActive !== colId) {\n      this._sortActive = colId;\n      this._sortDirection = 'asc';\n    } else {\n      // cycle asc -> desc -> null -> asc\n      if (this._sortDirection === 'asc') this._sortDirection = 'desc';\n      else if (this._sortDirection === 'desc') this._sortDirection = null;\n      else this._sortDirection = 'asc';\n    }\n    this.updateComputedData();\n    this.sortChange.emit({ fieldId: this._sortActive, fieldValue: col.fieldValue, direction: this._sortDirection });\n  }\n\n  private compareValues(a: any, b: any): number {\n    if (a == null && b == null) return 0;\n    if (a == null) return 1; // nulls last in asc (handled by dir multiplier)\n    if (b == null) return -1;\n\n    const numA = typeof a === 'number' ? a : Number(a);\n    const numB = typeof b === 'number' ? b : Number(b);\n    const aIsNum = !isNaN(numA) && a !== '' && a !== null && a !== false;\n    const bIsNum = !isNaN(numB) && b !== '' && b !== null && b !== false;\n    if (aIsNum && bIsNum) {\n      if (numA < numB) return -1;\n      if (numA > numB) return 1;\n      return 0;\n    }\n\n    // Date detection: attempt parse when both values are strings and parseable\n    if (typeof a === 'string' && typeof b === 'string') {\n      const tsA = Date.parse(a);\n      const tsB = Date.parse(b);\n      if (!isNaN(tsA) && !isNaN(tsB)) {\n        if (tsA < tsB) return -1;\n        if (tsA > tsB) return 1;\n        return 0;\n      }\n      // localeCompare, case-insensitive\n      return a.localeCompare(b, undefined, { sensitivity: 'base', numeric: true });\n    }\n\n    // Fallback to string comparison\n    const sa = String(a);\n    const sb = String(b);\n    return sa.localeCompare(sb, undefined, { sensitivity: 'base', numeric: true });\n  }\n}\n\n\n","<div class=\"cqa-ui-root\">\n  <div class=\"cqa-relative\" [class.cqa-max-h-[100px]]=\"showTableLoading\">\n    <cqa-full-table-loader *ngIf=\"showTableLoading\"></cqa-full-table-loader>\n    <cqa-table-data-loader *ngIf=\"showTableDataLoading\"></cqa-table-data-loader>\n    <table class=\"table-inner cqa-w-full cqa-border cqa-border-solid cqa-border-gray-200\"\n           [class.is-loading]=\"true\"\n           [style.min-width.px]=\"computedTableMinWidth || null\"\n           *ngIf=\"!showTableLoading\">\n      <colgroup>\n        <ng-container *ngFor=\"let width of computedColumnWidths; trackBy: trackByIndex\">\n          <col [style.width]=\"width\" />\n        </ng-container>\n      </colgroup>\n\n      <thead *ngIf=\"data?.length\">\n        <tr class=\"table-header cqa-items-center\">\n          <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n            <th\n              class=\"header-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n              [style.min-width.px]=\"col.minWidth\"\n              [ngClass]=\"[\n                col.fieldId + '-cell',\n                col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left'\n              ]\">\n              <!-- Built-in select-all for checkbox column when enabled and no custom header template -->\n              <ng-container\n                *ngIf=\"col.fieldId === 'checkbox' && enableSelectAll && !getHeaderTemplate(col.fieldId); else headerTplOrDefault\">\n                <div class=\"custom-checkbox\">\n                  <input type=\"checkbox\" id=\"checkbox\" aria-label=\"Select all rows\" [checked]=\"allSelected\"\n                  [indeterminate]=\"someSelected\" (change)=\"onSelectAllChange($event)\" class=\"hidden-checkbox\" />\n                  <label for=\"checkbox\" class=\"custom-checkbox-label\"></label>\n                </div>\n              </ng-container>\n              <ng-template #headerTplOrDefault>\n                <ng-container *ngIf=\"getHeaderTemplate(col.fieldId) as headerTpl; else defaultHeader\">\n                  <ng-container *ngTemplateOutlet=\"headerTpl\"></ng-container>\n                </ng-container>\n                <ng-template #defaultHeader>\n                  <ng-container *ngIf=\"col.sortable; else plainHeader\">\n                    <button type=\"button\" class=\"header-text cqa-inline-flex cqa-items-center cqa-gap-1 cqa-px-0\"\n                      (click)=\"toggleSort(col)\" [attr.aria-label]=\"'Sort by ' + (col.fieldName || col.fieldId)\">\n                      <span class=\"cqa-text-[12.3px] cqa-leading-[17.5px] cqa-text-[#0A0A0A] cqa-font-medium\">{{\n                        col.fieldName }}</span>\n                      <span *ngIf=\"isSortedAsc(col.fieldId)\">▲</span>\n                      <span *ngIf=\"isSortedDesc(col.fieldId)\">▼</span>\n                    </button>\n                  </ng-container>\n                  <ng-template #plainHeader>\n                    <span class=\"header-text\">{{ col.fieldName }}</span>\n                  </ng-template>\n                </ng-template>\n              </ng-template>\n            </th>\n          </ng-container>\n        </tr>\n      </thead>\n\n      <tbody class=\"table-body cqa-w-full\">\n        <tr class=\"table-row cqa-bg-surface-default hover:cqa-bg-surface-hover\" *ngFor=\"let row of computedData; let rowIndex = index; trackBy: trackByIndex\"\n          [class.selected]=\"row?.isSelected\">\n          <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n            <td class=\"cell cqa-px-[10.5px] cqa-py-[11px]\"\n              [style.min-width.px]=\"col.minWidth\"\n              [ngClass]=\"[\n                col.fieldId + '-cell',\n                col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left',\n                cellJsonPathGetter && col.fieldId !== 'checkbox' && col.fieldId !== 'actions' ? 'cqa-cursor-pointer' : ''\n              ]\"\n              (dblclick)=\"cellJsonPathGetter && col?.fieldId !== 'checkbox' && col?.fieldId !== 'actions' && copyCellJsonPath(cellJsonPathGetter(rowIndex, col?.fieldId, row), $event, 'dbQueryResponse')\">\n              <!-- Built-in checkbox cell when no custom template is provided -->\n              <ng-container *ngIf=\"col.fieldId === 'checkbox' && !getCellTemplate(col.fieldId); else regularCell\">\n                <div class=\"custom-checkbox\" [ngClass]=\"[\n                  col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n                ]\">\n                  <input\n                    type=\"checkbox\"\n                    class=\"hidden-checkbox\"\n                    [attr.id]=\"'row-checkbox-' + rowIndex\"\n                    aria-label=\"Select row\"\n                    [checked]=\"row?.isSelected\"\n                    [disabled]=\"!isRowSelectable(row)\"\n                    (change)=\"onRowSelectChange($event, row)\" />\n                  <label\n                    class=\"custom-checkbox-label\"\n                    [style.opacity]=\"!isRowSelectable(row) ? '0.45' : null\"\n                    [style.cursor]=\"!isRowSelectable(row) ? 'not-allowed' : null\"\n                    [style.pointer-events]=\"!isRowSelectable(row) ? 'none' : null\"\n                    [attr.for]=\"'row-checkbox-' + rowIndex\">\n                  </label>\n                </div>\n              </ng-container>\n              <ng-template #regularCell>\n                <ng-container *ngIf=\"getCellTemplate(col.fieldId) as cellTpl; else defaultCell\">\n                  <ng-container\n                    *ngTemplateOutlet=\"cellTpl; context: {$implicit: row, row: row, value: getCellValue(row, col.fieldValue)}\"></ng-container>\n                </ng-container>\n                <ng-template #defaultCell>\n                  <ng-container *ngIf=\"getRenderedValue(row, col) as renderedValue\">\n                    <ng-container *ngIf=\"isComponent(renderedValue); else nonComponentCell\">\n                      <ng-container \n                        cqaCellContainer \n                        #cellContainer=\"cqaCellContainer\"\n                        [componentConfig]=\"renderedValue\"\n                        [row]=\"row\"\n                        [column]=\"col\"\n                        [rowIndex]=\"rowIndex\"\n                        [colId]=\"col.fieldId\">\n                      </ng-container>\n                    </ng-container>\n                    <ng-template #nonComponentCell>\n                      <ng-container *ngIf=\"isHtmlString(renderedValue); else textCell\">\n                        <div class=\"cqa-text-xs cqa-leading-[17px]\" [innerHTML]=\"getSanitizedHtml(renderedValue)\" [ngClass]=\"[\n                        col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n                      ]\"></div>\n                      </ng-container>\n                      <ng-template #textCell>\n                        <span class=\"cqa-text-xs cqa-text-[#374151]\" [ngClass]=\"[\n                        col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n                      ]\">{{ renderedValue }}</span>\n                      </ng-template>\n                    </ng-template>\n                  </ng-container>\n                </ng-template>\n              </ng-template>\n            </td>\n          </ng-container>\n        </tr>\n      </tbody>\n    </table>\n  </div>\n</div>"]}
|
|
646
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-table.component.js","sourceRoot":"","sources":["../../../../../../src/lib/table/dynamic-table/dynamic-table.component.ts","../../../../../../src/lib/table/dynamic-table/dynamic-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ContentChild, ContentChildren, Input, QueryList, TemplateRef, EventEmitter, Output, ViewChild, ViewContainerRef, ComponentRef, Type, OnDestroy, ChangeDetectorRef, OnChanges, SimpleChanges, ChangeDetectionStrategy } from \"@angular/core\";\nimport { DomSanitizer, SafeHtml } from \"@angular/platform-browser\";\nimport { DndDropEvent } from \"ngx-drag-drop\";\nimport { DynamicCellTemplateDirective, DynamicHeaderTemplateDirective } from \"./dynamic-cell.directive\";\nimport { ComponentRenderConfig } from \"./component-render-config.interface\";\n\nexport { ComponentRenderConfig } from \"./component-render-config.interface\";\n\nconst DYNAMIC_TABLE_REORDER_STYLES = `\n  .reorder-handle-cell { width: 44px; }\n  .reorder-handle-icon { display: inline-flex; align-items: center; justify-content: center; color: #6B7280; }\n  .cqa-row-draggable { cursor: grab; }\n  .cqa-row-draggable:active { cursor: grabbing; }\n  .cqa-row-draggable.dndDragging { opacity: 0.5; }\n  .cqa-row-draggable.dndDraggingSource { opacity: 0.3; }\n  tr.cqa-row-drop-above > td { box-shadow: inset 0 3px 0 0 #3F43EE; }\n  tr.cqa-row-drop-below > td { box-shadow: inset 0 -3px 0 0 #3F43EE; }\n`;\n\nexport interface DynamicTableColumn {\n  fieldId: string;\n  fieldName?: string;\n  fieldValue?: string;\n  /**\n   * Optional custom renderer invoked when no projected cell template is provided.\n   * Returns the string/number/HTML string/Component class/ComponentRenderConfig to display for the given row.\n   * HTML strings will be sanitized and rendered safely.\n   * Component classes will be dynamically instantiated.\n   */\n  render?: (row: any, column: DynamicTableColumn) => string | number | SafeHtml | Type<any> | ComponentRenderConfig;\n  isDefault?: boolean;\n  sortable?: boolean;\n  isShow?: boolean;\n  // New configuration fields to unify layout logic\n  weight?: number; // relative weight for dynamic share\n  fixedPx?: number; // fixed width in px (e.g. checkbox/actions)\n  /** Minimum width in px — applied to both the header cell and every body cell of this column. */\n  minWidth?: number;\n  responsive?: { xs?: boolean; sm?: boolean; md?: boolean }; // hide at specific breakpoints\n  align?: 'left' | 'center' | 'right';\n}\n\n@Component({\n  selector: \"app-dynamic-table\",\n  templateUrl: \"./dynamic-table.component.html\",\n  styles: [DYNAMIC_TABLE_REORDER_STYLES],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DynamicTableComponent implements OnDestroy, OnChanges {\n  @Input() data: any[] = [];\n  @Input() columns: DynamicTableColumn[] = [];\n  /**\n   * Optional configuration object used when no custom empty template is provided.\n   */\n  @Input() emptyState?: { title?: string; description?: string; imageUrl?: string; actions?: any[] };\n  // Optional override; if omitted we compute from columns\n  @Input() gridTemplateColumns?: string;\n  // Breakpoint-aware sizing; provide from parent to avoid extra listeners\n  @Input() screenWidth?: number;\n  // Enable built-in select-all behavior for a 'checkbox' column\n  @Input() enableSelectAll: boolean = true;\n  // Optional predicate to control whether a row is selectable.\n  // If not provided, all rows are selectable (backward compatible behavior).\n  @Input() rowSelectable?: (row: any) => boolean;\n  // Enable simple client-side sorting when clicking sortable headers (no custom header)\n  @Input() enableLocalSort: boolean = true;\n  // Backward-compat loading flag (applies to both loaders if specific flags not provided)\n  \n  // New: controls the full table overlay loader\n  @Input() isTableLoading?: boolean;\n  // New: controls the inline data loader above the table\n  @Input() isTableDataLoading?: boolean;\n  // Emit sort changes so parent can perform server-side sort if desired\n  @Output() sortChange = new EventEmitter<{ fieldId: string; fieldValue?: string; direction: 'asc' | 'desc' | null }>();\n\n  // Optional getter for JSON path per cell. When provided, double-clicking a cell copies the path.\n  @Input() cellJsonPathGetter?: (rowIndex: number, colId: string, row: any) => string;\n\n  // Optional handler called when a JSON path is copied from a table cell\n  @Input() onJsonPathCopiedHandler?: (event: { path: string; source: string }) => void;\n\n  /**\n   * When true, renders a drag-handle column as the leftmost cell and enables\n   * row drag-and-drop reordering via ngx-drag-drop. Off by default so existing\n   * consumers are unaffected.\n   */\n  @Input() enableRowReorder: boolean = false;\n  /** Tooltip for the drag handle icon. */\n  @Input() reorderHandleTooltip: string = 'Drag to reorder';\n  /**\n   * Emitted after a successful row drop. `orderedData` is a freshly allocated\n   * array reflecting the new order — parents should assign it back to `[data]`\n   * rather than relying on in-place mutation.\n   */\n  @Output() rowReorder = new EventEmitter<{ previousIndex: number; currentIndex: number; orderedData: any[] }>();\n\n  /** Index of the row the cursor is currently hovering during a reorder drag. */\n  dropHoverIndex: number | null = null;\n  /** Whether the cursor is in the top or bottom half of the hovered row. */\n  dropPosition: 'above' | 'below' | null = null;\n  /** The row currently being dragged — used to suppress the indicator on the source row. */\n  private draggedRow: any = null;\n\n  @ContentChildren(DynamicCellTemplateDirective) cellTemplates!: QueryList<DynamicCellTemplateDirective>;\n  @ContentChildren(DynamicHeaderTemplateDirective) headerTemplates!: QueryList<DynamicHeaderTemplateDirective>;\n  // Allow consumers to project a custom empty-state template using a template ref name\n  // e.g. <ng-template #emptyTableTpl>Custom empty UI</ng-template>\n  @ContentChild('emptyTableTpl', { read: TemplateRef }) emptyTableTpl?: TemplateRef<any>;\n\n  // Map to store component references for cleanup: rowIndex_colId -> ComponentRef\n  private componentRefs = new Map<string, ComponentRef<any>>();\n\n  // Caches for expensive getters\n  visibleColumns: DynamicTableColumn[] = [];\n  computedData: any[] = [];\n  computedGridTemplate: string = '';\n  computedColumnWidths: string[] = [];\n  /**\n   * Hard floor on the `<table>` width = sum of every visible column's minimum (fixed\n   * widths + dynamic columns' `minWidth`). With `table-layout: fixed; width: 100%;` set\n   * by consumers, `<col>` widths become advisory once the container is narrower than\n   * the column-width sum — the browser silently shrinks them. Pinning `min-width` on\n   * the table itself prevents that compaction; the consumer's `overflow-x: auto`\n   * wrapper then scrolls horizontally instead of squishing columns to zero width.\n   */\n  computedTableMinWidth = 0;\n\n  // Cache for sanitized HTML to avoid redundant innerHTML renders\n  private htmlCache = new Map<string, SafeHtml>();\n\n  constructor(\n    private sanitizer: DomSanitizer,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['columns'] || changes['screenWidth']) {\n      this.updateVisibleColumns();\n      this.updateGridTemplate();\n    }\n    if (changes['data'] || changes['columns'] || changes['enableLocalSort']) {\n      this.updateComputedData();\n    }\n  }\n\n  ngOnDestroy(): void {\n    // Clean up all dynamically created components\n    this.componentRefs.forEach(ref => ref.destroy());\n    this.componentRefs.clear();\n    this.htmlCache.clear();\n  }\n\n  private get isXs(): boolean {\n    const w = this.screenWidth || window.innerWidth;\n    return w <= 480;\n  }\n  private get isSm(): boolean {\n    const w = this.screenWidth || window.innerWidth;\n    return w <= 768;\n  }\n  private get isMd(): boolean {\n    const w = this.screenWidth || window.innerWidth;\n    return w <= 992;\n  }\n\n  private updateVisibleColumns(): void {\n    const responsiveFilter = (c: DynamicTableColumn) => {\n      const r = c?.responsive || {};\n      if (r.xs && this.isXs) return false;\n      if (r.sm && this.isSm) return false;\n      if (r.md && this.isMd) return false;\n      return true;\n    };\n    this.visibleColumns = (this.columns || []).filter(c => c.isShow !== false).filter(responsiveFilter);\n  }\n\n  getHeaderTemplate(colId: string): TemplateRef<any> | null {\n    const tpl = this.headerTemplates?.find(t => t.name === colId);\n    return tpl ? tpl.template : null;\n  }\n\n  getCellTemplate(colId: string): TemplateRef<any> | null {\n    const tpl = this.cellTemplates?.find(t => t.name === colId);\n    return tpl ? tpl.template : null;\n  }\n\n  getRenderedValue(row: any, column: DynamicTableColumn): any {\n    if (column?.render && typeof column.render === 'function') {\n      try {\n        return column.render(row, column);\n      } catch (err) {\n        console.warn('DynamicTable render function threw an error for column', column?.fieldId, err);\n        return '';\n      }\n    }\n    return this.getCellValue(row, column?.fieldValue);\n  }\n\n  /**\n   * Checks if the rendered value is a component class or component config\n   */\n  isComponent(value: any): boolean {\n    if (!value) return false;\n    // Check if it's a component class (has @Component decorator metadata)\n    if (typeof value === 'function' && value.prototype) {\n      return true;\n    }\n    // Check if it's a ComponentRenderConfig\n    if (typeof value === 'object' && value.component && typeof value.component === 'function') {\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Gets the component class from a render value (could be class or config)\n   */\n  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  /**\n   * Gets component inputs from a render value\n   */\n  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  /**\n   * Gets component outputs from a render value\n   */\n  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   * Initializes a component in a ViewContainerRef (called from template)\n   */\n  initComponent(\n    vcr: ViewContainerRef,\n    renderedValue: Type<any> | ComponentRenderConfig,\n    row: any,\n    column: DynamicTableColumn,\n    rowIndex: number,\n    colId: string\n  ): boolean {\n    if (!vcr) return false;\n    \n    const key = `${rowIndex}_${colId}`;\n    \n    // If component already exists for this cell, don't recreate\n    if (this.componentRefs.has(key)) {\n      return true;\n    }\n\n    try {\n      const componentClass = this.getComponentClass(renderedValue);\n      const inputs = this.getComponentInputs(renderedValue);\n      const outputs = this.getComponentOutputs(renderedValue);\n\n      // Create the component\n      const componentRef = vcr.createComponent(componentClass);\n      \n      // Set inputs\n      Object.keys(inputs).forEach(inputKey => {\n        if (componentRef.instance && inputKey in componentRef.instance) {\n          (componentRef.instance as any)[inputKey] = inputs[inputKey];\n        }\n      });\n\n      // Subscribe to outputs\n      Object.keys(outputs).forEach(outputKey => {\n        if (componentRef.instance && outputKey in componentRef.instance) {\n          const output = (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          } else if (output && typeof output.emit === 'function') {\n            // For EventEmitter, wrap the emit to call the handler\n            const originalEmit = output.emit.bind(output);\n            (componentRef.instance as any)[outputKey] = {\n              emit: (event: any) => {\n                originalEmit(event);\n                const handler = outputs[outputKey];\n                if (handler) handler(event);\n              },\n              subscribe: output.subscribe.bind(output)\n            };\n          }\n        }\n      });\n\n      // Mark for change detection\n      componentRef.changeDetectorRef.markForCheck();\n      \n      // Store reference for cleanup\n      this.componentRefs.set(key, componentRef);\n      return true;\n    } catch (err) {\n      console.error('Failed to create component in table cell:', err);\n      return false;\n    }\n  }\n\n  /**\n   * Checks if the rendered value is an HTML string that should be rendered with innerHTML\n   */\n  isHtmlString(value: any): boolean {\n    if (typeof value !== 'string') return false;\n    // Faster check: if it contains an opening tag pattern near the start\n    // or just has any brackets. Wrapping in a regex with [\\s\\S]* is very slow.\n    return value.includes('<') && /<[a-z/][^>]*>/i.test(value);\n  }\n\n  /**\n   * Sanitizes HTML string for safe rendering with caching\n   */\n  getSanitizedHtml(value: string): SafeHtml {\n    let cached = this.htmlCache.get(value);\n    if (!cached) {\n      cached = this.sanitizer.bypassSecurityTrustHtml(value);\n      this.htmlCache.set(value, cached);\n      \n      // Simple cache eviction\n      if (this.htmlCache.size > 1000) {\n        const firstKey = this.htmlCache.keys().next().value;\n        if (firstKey !== undefined) {\n          this.htmlCache.delete(firstKey);\n        }\n      }\n    }\n    return cached;\n  }\n\n  getCellValue(row: any, path?: string): any {\n    if (!row || !path) return \"\";\n    const parts = path.split(\".\");\n    let current = row;\n    for (const part of parts) {\n      if (current == null) return \"\";\n      current = current[part];\n    }\n    return current ?? \"\";\n  }\n\n  trackByIndex(index: number): number {\n    return index;\n  }\n\n  onRowDragStart(row: any): void {\n    if (!this.enableRowReorder) return;\n    this.draggedRow = row;\n  }\n\n  onRowDragEnd(): void {\n    this.draggedRow = null;\n    this.clearDropIndicator();\n  }\n\n  /**\n   * Fires continuously while a drag is over a row. Computes whether the cursor\n   * is in the top or bottom half of the row and updates the indicator state.\n   * Suppresses the indicator on the dragged row itself.\n   */\n  onRowDragOver(event: Event, rowIndex: number): void {\n    if (!this.enableRowReorder) return;\n    const drag = event as DragEvent;\n    drag.preventDefault();\n    const row = this.computedData?.[rowIndex];\n    if (this.draggedRow && row === this.draggedRow) {\n      if (this.dropHoverIndex !== null || this.dropPosition !== null) {\n        this.dropHoverIndex = null;\n        this.dropPosition = null;\n        this.cdr.markForCheck();\n      }\n      return;\n    }\n    const el = drag.currentTarget as HTMLElement | null;\n    if (!el) return;\n    const rect = el.getBoundingClientRect();\n    const midY = rect.top + rect.height / 2;\n    const pos: 'above' | 'below' = drag.clientY < midY ? 'above' : 'below';\n    if (this.dropHoverIndex !== rowIndex || this.dropPosition !== pos) {\n      this.dropHoverIndex = rowIndex;\n      this.dropPosition = pos;\n      this.cdr.markForCheck();\n    }\n  }\n\n  /** Clears the indicator when the cursor leaves the tbody entirely. */\n  onTbodyDragLeave(event: Event): void {\n    if (!this.enableRowReorder) return;\n    const drag = event as DragEvent;\n    const tbody = drag.currentTarget as HTMLElement | null;\n    const related = drag.relatedTarget as Node | null;\n    if (!tbody || !related || !tbody.contains(related)) {\n      this.clearDropIndicator();\n    }\n  }\n\n  private clearDropIndicator(): void {\n    if (this.dropHoverIndex !== null || this.dropPosition !== null) {\n      this.dropHoverIndex = null;\n      this.dropPosition = null;\n      this.cdr.markForCheck();\n    }\n  }\n\n  /**\n   * Handles a row drop from ngx-drag-drop. Prefers the tracked dragover position\n   * (which the user can see live via the insertion line) over `event.index` when\n   * computing the final insert slot. Emits a fresh array — does not mutate `data`.\n   */\n  onRowDndDrop(event: DndDropEvent): void {\n    if (!this.enableRowReorder) return;\n    const dragged = event?.data;\n    if (dragged == null) {\n      this.clearDropIndicator();\n      return;\n    }\n    const source = this.computedData || [];\n    let previousIndex = source.indexOf(dragged);\n    if (previousIndex < 0 && dragged && dragged.id != null) {\n      previousIndex = source.findIndex(r => r && r.id === dragged.id);\n    }\n    if (previousIndex < 0) {\n      this.clearDropIndicator();\n      return;\n    }\n\n    // Prefer the indicator position (what the user just saw) over event.index.\n    let insertIndex: number;\n    if (this.dropHoverIndex != null && this.dropPosition) {\n      insertIndex = this.dropPosition === 'above' ? this.dropHoverIndex : this.dropHoverIndex + 1;\n    } else if (typeof event.index === 'number') {\n      insertIndex = event.index;\n    } else {\n      insertIndex = source.length;\n    }\n\n    const next = [...source];\n    next.splice(previousIndex, 1);\n    // Removing the source above the target shifts later items up by one.\n    if (previousIndex < insertIndex) insertIndex -= 1;\n    if (insertIndex < 0) insertIndex = 0;\n    if (insertIndex > next.length) insertIndex = next.length;\n\n    this.clearDropIndicator();\n    this.draggedRow = null;\n\n    if (insertIndex === previousIndex) return;\n    next.splice(insertIndex, 0, dragged);\n    this.rowReorder.emit({ previousIndex, currentIndex: insertIndex, orderedData: next });\n  }\n\n  copyCellJsonPath(path: string, event: Event, source: string = 'dbQueryResponse'): void {\n    if (!path) return;\n    event.preventDefault();\n    event.stopPropagation();\n    navigator.clipboard.writeText(path).then(() => {\n      // Call the handler if provided\n      if (this.onJsonPathCopiedHandler) {\n        try {\n          this.onJsonPathCopiedHandler({ path, source });\n        } catch (handlerError) {\n          console.error('Error in onJsonPathCopiedHandler:', handlerError);\n        }\n      }\n    }).catch(err => {\n      console.error('Failed to copy jsonPath:', err);\n      // Still call the handler even on failure (similar to api-step pattern)\n      if (this.onJsonPathCopiedHandler) {\n        try {\n          this.onJsonPathCopiedHandler({ path, source });\n        } catch (handlerError) {\n          console.error('Error in onJsonPathCopiedHandler:', handlerError);\n        }\n      }\n    });\n  }\n\n  // Compute grid-template-columns string from column config\n  private updateGridTemplate(): void {\n    const cols = this.visibleColumns;\n    if (!cols?.length) {\n      this.computedGridTemplate = '';\n      this.computedColumnWidths = [];\n      return;\n    }\n    \n    const fixedPx = cols.reduce((sum, c) => sum + (c.fixedPx || 0), 0);\n    const dynamicCols = cols.filter(c => !c.fixedPx);\n    const totalWeight = dynamicCols.reduce((sum, c) => sum + (c.weight || 1), 0) || 1;\n\n    // Floor on the table itself = every column's minimum width contribution. Fixed\n    // columns contribute `fixedPx`; dynamic columns contribute `minWidth` (or 0 when\n    // unset). The template binds this via `[style.min-width.px]` on the <table> so\n    // narrow containers force horizontal scroll instead of squishing columns to 0.\n    this.computedTableMinWidth = cols.reduce(\n      (sum, c) => sum + (c.fixedPx && c.fixedPx > 0 ? c.fixedPx : (c.minWidth || 0)),\n      0,\n    );\n\n    this.computedColumnWidths = cols.map(c => {\n      if (c.fixedPx && c.fixedPx > 0) {\n        // Fixed-width columns still respect minWidth in the (rare) case the consumer\n        // configured both — `max(fixed, min)` picks whichever is larger.\n        return c.minWidth && c.minWidth > c.fixedPx\n          ? `max(${c.fixedPx}px, ${c.minWidth}px)`\n          : `${c.fixedPx}px`;\n      }\n      const share = (c.weight || 1) / totalWeight;\n      const dynamicWidth = `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;\n      // Dynamic-share columns can collapse to ~0 when the table is narrower than\n      // (sum of fixed widths) + the share they'd compute to. Using CSS `max()` floors\n      // the column at `minWidth` so it grows naturally as the table widens but never\n      // shrinks below the configured pixel minimum. With `table-layout: fixed`, the\n      // <col> width is authoritative — without max() here the cell-level `min-width`\n      // we set on <th>/<td> is ignored and columns visually disappear at narrow widths.\n      return c.minWidth && c.minWidth > 0\n        ? `max(${c.minWidth}px, ${dynamicWidth})`\n        : dynamicWidth;\n    });\n\n    console.log('computedColumnWidths', this.computedColumnWidths);\n    \n    if (this.gridTemplateColumns) {\n      this.computedGridTemplate = this.gridTemplateColumns;\n    } else {\n      this.computedGridTemplate = this.computedColumnWidths.join(' ');\n    }\n  }\n\n  // Selection helpers\n  isRowSelectable(row: any): boolean {\n    if (!row) return false;\n    if (typeof this.rowSelectable !== 'function') return true;\n    try {\n      return !!this.rowSelectable(row);\n    } catch {\n      return true;\n    }\n  }\n\n  get allSelected(): boolean {\n    const rows = (this.data || []).filter((r) => this.isRowSelectable(r));\n    if (!rows.length) return false;\n    return rows.every(r => !!r?.isSelected);\n  }\n\n  get someSelected(): boolean {\n    const rows = (this.data || []).filter((r) => this.isRowSelectable(r));\n    if (!rows.length) return false;\n    const anySelected = rows.some(r => !!r?.isSelected);\n    return anySelected && !this.allSelected;\n  }\n\n  onSelectAllChange(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    this.toggleSelectAll(target.checked);\n  }\n\n  onRowSelectChange(event: Event, row: any): void {\n    if (!this.isRowSelectable(row)) {\n      row.isSelected = false;\n      return;\n    }\n    const target = event.target as HTMLInputElement;\n    row.isSelected = target.checked;\n  }\n\n  toggleSelectAll(checked: boolean): void {\n    const rows = this.data || [];\n    for (const row of rows) {\n      if (row) {\n        row.isSelected = this.isRowSelectable(row) ? checked : false;\n      }\n    }\n  }\n\n  // ===== Sorting support =====\n  private _sortActive?: string;\n  private _sortDirection: 'asc' | 'desc' | null = null;\n\n  private updateComputedData(): void {\n    const source = this.data || [];\n    if (!this.enableLocalSort || !this._sortActive || !this._sortDirection) {\n      this.computedData = source;\n      return;\n    }\n    const col = this.visibleColumns.find(c => c.fieldId === this._sortActive);\n    if (!col || !col.fieldValue) {\n      this.computedData = source;\n      return;\n    }\n    const dir = this._sortDirection === 'asc' ? 1 : -1;\n    const fieldPath = col.fieldValue;\n    const out = [...source];\n    out.sort((a, b) => dir * this.compareValues(this.getCellValue(a, fieldPath), this.getCellValue(b, fieldPath)));\n    this.computedData = out;\n  }\n\n  // Computed loading flags to support backward compatibility\n  get showTableLoading(): boolean {\n    return this.isTableLoading ?? false;\n  }\n\n  get showTableDataLoading(): boolean {\n    return this.isTableDataLoading ?? false;\n  }\n\n  // True when table has no data and is not currently loading — used to show an empty state\n  get isEmpty(): boolean {\n    const anyLoading = this.showTableLoading || this.showTableDataLoading;\n    return !anyLoading && (!(this.data && this.data.length) || this.data.length === 0);\n  }\n\n  isSortedAsc(colId: string): boolean {\n    return this._sortActive === colId && this._sortDirection === 'asc';\n  }\n\n  isSortedDesc(colId: string): boolean {\n    return this._sortActive === colId && this._sortDirection === 'desc';\n  }\n\n  toggleSort(col: DynamicTableColumn): void {\n    if (!col?.sortable) return;\n    const colId = col.fieldId;\n    if (this._sortActive !== colId) {\n      this._sortActive = colId;\n      this._sortDirection = 'asc';\n    } else {\n      // cycle asc -> desc -> null -> asc\n      if (this._sortDirection === 'asc') this._sortDirection = 'desc';\n      else if (this._sortDirection === 'desc') this._sortDirection = null;\n      else this._sortDirection = 'asc';\n    }\n    this.updateComputedData();\n    this.sortChange.emit({ fieldId: this._sortActive, fieldValue: col.fieldValue, direction: this._sortDirection });\n  }\n\n  private compareValues(a: any, b: any): number {\n    if (a == null && b == null) return 0;\n    if (a == null) return 1; // nulls last in asc (handled by dir multiplier)\n    if (b == null) return -1;\n\n    const numA = typeof a === 'number' ? a : Number(a);\n    const numB = typeof b === 'number' ? b : Number(b);\n    const aIsNum = !isNaN(numA) && a !== '' && a !== null && a !== false;\n    const bIsNum = !isNaN(numB) && b !== '' && b !== null && b !== false;\n    if (aIsNum && bIsNum) {\n      if (numA < numB) return -1;\n      if (numA > numB) return 1;\n      return 0;\n    }\n\n    // Date detection: attempt parse when both values are strings and parseable\n    if (typeof a === 'string' && typeof b === 'string') {\n      const tsA = Date.parse(a);\n      const tsB = Date.parse(b);\n      if (!isNaN(tsA) && !isNaN(tsB)) {\n        if (tsA < tsB) return -1;\n        if (tsA > tsB) return 1;\n        return 0;\n      }\n      // localeCompare, case-insensitive\n      return a.localeCompare(b, undefined, { sensitivity: 'base', numeric: true });\n    }\n\n    // Fallback to string comparison\n    const sa = String(a);\n    const sb = String(b);\n    return sa.localeCompare(sb, undefined, { sensitivity: 'base', numeric: true });\n  }\n}\n\n\n","<div class=\"cqa-ui-root\">\n  <div class=\"cqa-relative\" [class.cqa-max-h-[100px]]=\"showTableLoading\">\n    <cqa-full-table-loader *ngIf=\"showTableLoading\"></cqa-full-table-loader>\n    <cqa-table-data-loader *ngIf=\"showTableDataLoading\"></cqa-table-data-loader>\n    <table class=\"table-inner cqa-w-full cqa-border cqa-border-solid cqa-border-gray-200\"\n           [class.is-loading]=\"true\"\n           [style.min-width.px]=\"computedTableMinWidth || null\"\n           *ngIf=\"!showTableLoading\">\n      <colgroup>\n        <col *ngIf=\"enableRowReorder\" style=\"width: 44px\" />\n        <ng-container *ngFor=\"let width of computedColumnWidths; trackBy: trackByIndex\">\n          <col [style.width]=\"width\" />\n        </ng-container>\n      </colgroup>\n\n      <thead *ngIf=\"data?.length\">\n        <tr class=\"table-header cqa-items-center\">\n          <th\n            *ngIf=\"enableRowReorder\"\n            class=\"header-cell reorder-handle-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7] cqa-text-center\"\n            aria-label=\"Reorder\">\n            <span class=\"cqa-sr-only\">Reorder</span>\n          </th>\n          <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n            <th\n              class=\"header-cell cqa-py-[13.25px] cqa-px-[10.5px] cqa-text-xs cqa-font-semibold cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n              [style.min-width.px]=\"col.minWidth\"\n              [ngClass]=\"[\n                col.fieldId + '-cell',\n                col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left'\n              ]\">\n              <!-- Built-in select-all for checkbox column when enabled and no custom header template -->\n              <ng-container\n                *ngIf=\"col.fieldId === 'checkbox' && enableSelectAll && !getHeaderTemplate(col.fieldId); else headerTplOrDefault\">\n                <div class=\"custom-checkbox\">\n                  <input type=\"checkbox\" id=\"checkbox\" aria-label=\"Select all rows\" [checked]=\"allSelected\"\n                  [indeterminate]=\"someSelected\" (change)=\"onSelectAllChange($event)\" class=\"hidden-checkbox\" />\n                  <label for=\"checkbox\" class=\"custom-checkbox-label\"></label>\n                </div>\n              </ng-container>\n              <ng-template #headerTplOrDefault>\n                <ng-container *ngIf=\"getHeaderTemplate(col.fieldId) as headerTpl; else defaultHeader\">\n                  <ng-container *ngTemplateOutlet=\"headerTpl\"></ng-container>\n                </ng-container>\n                <ng-template #defaultHeader>\n                  <ng-container *ngIf=\"col.sortable; else plainHeader\">\n                    <button type=\"button\" class=\"header-text cqa-inline-flex cqa-items-center cqa-gap-1 cqa-px-0\"\n                      (click)=\"toggleSort(col)\" [attr.aria-label]=\"'Sort by ' + (col.fieldName || col.fieldId)\">\n                      <span class=\"cqa-text-[12.3px] cqa-leading-[17.5px] cqa-text-[#0A0A0A] cqa-font-medium\">{{\n                        col.fieldName }}</span>\n                      <span *ngIf=\"isSortedAsc(col.fieldId)\">▲</span>\n                      <span *ngIf=\"isSortedDesc(col.fieldId)\">▼</span>\n                    </button>\n                  </ng-container>\n                  <ng-template #plainHeader>\n                    <span class=\"header-text\">{{ col.fieldName }}</span>\n                  </ng-template>\n                </ng-template>\n              </ng-template>\n            </th>\n          </ng-container>\n        </tr>\n      </thead>\n\n      <tbody class=\"table-body cqa-w-full\"\n        [dndDropzone]=\"['cqa-row']\"\n        [dndDisableIf]=\"!enableRowReorder\"\n        dndEffectAllowed=\"move\"\n        (dragleave)=\"onTbodyDragLeave($event)\"\n        (dndDrop)=\"onRowDndDrop($event)\">\n        <tr class=\"table-row cqa-bg-surface-default hover:cqa-bg-surface-hover\"\n          *ngFor=\"let row of computedData; let rowIndex = index; trackBy: trackByIndex\"\n          [class.selected]=\"row?.isSelected\"\n          [class.cqa-row-draggable]=\"enableRowReorder\"\n          [class.cqa-row-drop-above]=\"dropHoverIndex === rowIndex && dropPosition === 'above'\"\n          [class.cqa-row-drop-below]=\"dropHoverIndex === rowIndex && dropPosition === 'below'\"\n          [dndDraggable]=\"row\"\n          [dndDisableIf]=\"!enableRowReorder\"\n          dndEffectAllowed=\"move\"\n          dndType=\"cqa-row\"\n          (dndStart)=\"onRowDragStart(row)\"\n          (dndEnd)=\"onRowDragEnd()\"\n          (dragover)=\"onRowDragOver($event, rowIndex)\">\n          <td *ngIf=\"enableRowReorder\" class=\"cell reorder-handle-cell cqa-px-[10.5px] cqa-py-[11px] cqa-text-center\" [attr.title]=\"reorderHandleTooltip\">\n            <span class=\"reorder-handle-icon\" aria-hidden=\"true\">\n              <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n                <circle cx=\"6\" cy=\"3\" r=\"1.25\" fill=\"currentColor\"></circle>\n                <circle cx=\"10\" cy=\"3\" r=\"1.25\" fill=\"currentColor\"></circle>\n                <circle cx=\"6\" cy=\"8\" r=\"1.25\" fill=\"currentColor\"></circle>\n                <circle cx=\"10\" cy=\"8\" r=\"1.25\" fill=\"currentColor\"></circle>\n                <circle cx=\"6\" cy=\"13\" r=\"1.25\" fill=\"currentColor\"></circle>\n                <circle cx=\"10\" cy=\"13\" r=\"1.25\" fill=\"currentColor\"></circle>\n              </svg>\n            </span>\n          </td>\n          <ng-container *ngFor=\"let col of visibleColumns; trackBy: trackByIndex\">\n            <td class=\"cell cqa-px-[10.5px] cqa-py-[11px]\"\n              [style.min-width.px]=\"col.minWidth\"\n              [ngClass]=\"[\n                col.fieldId + '-cell',\n                col.align === 'center' ? 'cqa-text-center' : col.align === 'right' ? 'cqa-text-right' : 'cqa-text-left',\n                cellJsonPathGetter && col.fieldId !== 'checkbox' && col.fieldId !== 'actions' ? 'cqa-cursor-pointer' : ''\n              ]\"\n              (dblclick)=\"cellJsonPathGetter && col?.fieldId !== 'checkbox' && col?.fieldId !== 'actions' && copyCellJsonPath(cellJsonPathGetter(rowIndex, col?.fieldId, row), $event, 'dbQueryResponse')\">\n              <!-- Built-in checkbox cell when no custom template is provided -->\n              <ng-container *ngIf=\"col.fieldId === 'checkbox' && !getCellTemplate(col.fieldId); else regularCell\">\n                <div class=\"custom-checkbox\" [ngClass]=\"[\n                  col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n                ]\">\n                  <input\n                    type=\"checkbox\"\n                    class=\"hidden-checkbox\"\n                    [attr.id]=\"'row-checkbox-' + rowIndex\"\n                    aria-label=\"Select row\"\n                    [checked]=\"row?.isSelected\"\n                    [disabled]=\"!isRowSelectable(row)\"\n                    (change)=\"onRowSelectChange($event, row)\" />\n                  <label\n                    class=\"custom-checkbox-label\"\n                    [style.opacity]=\"!isRowSelectable(row) ? '0.45' : null\"\n                    [style.cursor]=\"!isRowSelectable(row) ? 'not-allowed' : null\"\n                    [style.pointer-events]=\"!isRowSelectable(row) ? 'none' : null\"\n                    [attr.for]=\"'row-checkbox-' + rowIndex\">\n                  </label>\n                </div>\n              </ng-container>\n              <ng-template #regularCell>\n                <ng-container *ngIf=\"getCellTemplate(col.fieldId) as cellTpl; else defaultCell\">\n                  <ng-container\n                    *ngTemplateOutlet=\"cellTpl; context: {$implicit: row, row: row, value: getCellValue(row, col.fieldValue)}\"></ng-container>\n                </ng-container>\n                <ng-template #defaultCell>\n                  <ng-container *ngIf=\"getRenderedValue(row, col) as renderedValue\">\n                    <ng-container *ngIf=\"isComponent(renderedValue); else nonComponentCell\">\n                      <ng-container \n                        cqaCellContainer \n                        #cellContainer=\"cqaCellContainer\"\n                        [componentConfig]=\"renderedValue\"\n                        [row]=\"row\"\n                        [column]=\"col\"\n                        [rowIndex]=\"rowIndex\"\n                        [colId]=\"col.fieldId\">\n                      </ng-container>\n                    </ng-container>\n                    <ng-template #nonComponentCell>\n                      <ng-container *ngIf=\"isHtmlString(renderedValue); else textCell\">\n                        <div class=\"cqa-text-xs cqa-leading-[17px]\" [innerHTML]=\"getSanitizedHtml(renderedValue)\" [ngClass]=\"[\n                        col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n                      ]\"></div>\n                      </ng-container>\n                      <ng-template #textCell>\n                        <span class=\"cqa-text-xs cqa-text-[#374151]\" [ngClass]=\"[\n                        col.align === 'center' ? 'cqa-flex cqa-justify-center' : col.align === 'right' ? 'cqa-flex cqa-justify-end' : ''\n                      ]\">{{ renderedValue }}</span>\n                      </ng-template>\n                    </ng-template>\n                  </ng-container>\n                </ng-template>\n              </ng-template>\n            </td>\n          </ng-container>\n        </tr>\n      </tbody>\n    </table>\n  </div>\n</div>"]}
|