@cqa-lib/cqa-ui 1.1.16 → 1.1.17
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/compare-runs/compare-runs.component.mjs +21 -10
- package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +41 -34
- package/esm2020/lib/templates/table-template.component.mjs +14 -8
- package/fesm2015/cqa-lib-cqa-ui.mjs +73 -49
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +73 -49
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/compare-runs/compare-runs.component.d.ts +3 -0
- package/lib/table/dynamic-table/dynamic-table.component.d.ts +10 -6
- package/lib/templates/table-template.component.d.ts +4 -2
- package/package.json +1 -1
|
@@ -107,12 +107,15 @@ export class TableTemplateComponent {
|
|
|
107
107
|
this.pagedRows = [];
|
|
108
108
|
this.onReload = new EventEmitter();
|
|
109
109
|
this.onAutoRefreshClick = new EventEmitter();
|
|
110
|
+
// Derived columns with visibility applied.
|
|
111
|
+
this.computedColumns = [];
|
|
112
|
+
// Auto-generated visibility columns from columns input
|
|
113
|
+
this.visibilityColumns = [];
|
|
110
114
|
}
|
|
111
|
-
|
|
112
|
-
get computedColumns() {
|
|
115
|
+
updateComputedColumns() {
|
|
113
116
|
const visibility = this._columnVisibility || {};
|
|
114
117
|
const source = this.columns || [];
|
|
115
|
-
|
|
118
|
+
this.computedColumns = source.map(col => {
|
|
116
119
|
if (['checkbox', 'actions'].includes(col.fieldId)) {
|
|
117
120
|
return col;
|
|
118
121
|
}
|
|
@@ -120,10 +123,9 @@ export class TableTemplateComponent {
|
|
|
120
123
|
return { ...col, isShow: show !== false };
|
|
121
124
|
});
|
|
122
125
|
}
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
return this._cachedVisibilityColumns;
|
|
126
|
+
updateVisibilityColumns() {
|
|
127
|
+
this._cachedVisibilityColumns = this.mapVisibilityColumns();
|
|
128
|
+
this.visibilityColumns = this._cachedVisibilityColumns;
|
|
127
129
|
}
|
|
128
130
|
// Internal column visibility state
|
|
129
131
|
get columnVisibility() {
|
|
@@ -131,6 +133,8 @@ export class TableTemplateComponent {
|
|
|
131
133
|
}
|
|
132
134
|
ngOnInit() {
|
|
133
135
|
this.initializeComponent();
|
|
136
|
+
this.updateComputedColumns();
|
|
137
|
+
this.updateVisibilityColumns();
|
|
134
138
|
}
|
|
135
139
|
ngOnChanges(changes) {
|
|
136
140
|
if (changes['data'] || changes['isEmptyState']) {
|
|
@@ -138,6 +142,8 @@ export class TableTemplateComponent {
|
|
|
138
142
|
}
|
|
139
143
|
if (changes['columns']) {
|
|
140
144
|
this.initializeColumnVisibility();
|
|
145
|
+
this.updateComputedColumns();
|
|
146
|
+
this.updateVisibilityColumns();
|
|
141
147
|
}
|
|
142
148
|
}
|
|
143
149
|
initializeComponent() {
|
|
@@ -443,4 +449,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
443
449
|
}], onAutoRefreshClick: [{
|
|
444
450
|
type: Output
|
|
445
451
|
}] } });
|
|
446
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-template.component.js","sourceRoot":"","sources":["../../../../../src/lib/templates/table-template.component.ts","../../../../../src/lib/templates/table-template.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n  OnDestroy,\n} from '@angular/core';\nimport { DynamicFilterItem } from '../filters/dynamic-filter/dynamic-filter.component';\nimport { DynamicTableColumn } from '../table/dynamic-table/dynamic-table.component';\nimport { ColumnVisibilityConfig } from '../column-visibility/column-visibility.component';\nimport { TableAction } from '../table-action-toolbar/table-action-toolbar.component';\nimport { EmptyStateConfig } from '../empty-state/empty-state-config.interface';\nimport { EMPTY_STATE_IMAGES } from '../assets/images/image-assets.constants';\n\n@Component({\n  selector: 'cqa-table-template',\n  templateUrl: './table-template.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class TableTemplateComponent implements OnInit, OnChanges, OnDestroy {\n  // Search bar inputs\n  @Input() searchPlaceholder: string = 'Search components';\n  @Input() searchValue: string = '';\n  @Input() showClear: boolean = true;\n  @Input() showSearchBar: boolean = true;\n\n  // Search bar outputs\n  @Output() onSearchChange = new EventEmitter<string>();\n\n  // Filter outputs\n  @Output() onApplyFilterClick = new EventEmitter<any>();\n  @Output() onResetFilterClick = new EventEmitter<void>();\n  @Output() onClearAll = new EventEmitter<void>();\n  @Output() removeChip = new EventEmitter<any>();\n\n  // Filter inputs\n  @Input() filterConfig: DynamicFilterItem[] = [];\n  @Input() showFilterPanel: boolean = false;\n  @Input() showFilterButton: boolean = true;\n\n  // Other button input\n  @Input() otherButtonLabel: string = 'Other Button';\n  @Input() otherButtonVariant: 'filled' | 'outlined' | 'text' | 'elevated' | 'tonal' | 'grey-solid' = 'filled';\n  @Input() showOtherButton: boolean = true;\n  \n  // Action menu button (three-dot menu in table rows)\n  @Input() showActionButton: boolean = true;\n\n  // Settings and refresh buttons\n  @Input() showSettingsButton: boolean = true;\n  @Input() showAutoRefreshButton: boolean = true;\n\n  // Data input\n  @Input() data: any[] = [];\n\n  // Empty state inputs\n  @Input() isEmptyState: boolean = false;\n  @Input() emptyStateConfig: EmptyStateConfig = {\n    title: 'No Data Available Yet',\n    description: 'Run or upload your first test to see your analytics and trends here. Watch your quality metrics come to life with real-time insights.',\n    imageUrl: EMPTY_STATE_IMAGES.DASHBOARD,\n    actions: [{ label: 'Run Test Suite', variant: 'filled' }],\n  };\n\n  // Action bar inputs\n  @Input() actions: TableAction[] = [\n    { \n      id: 'delete', \n      label: 'Delete', \n      icon: 'delete', \n      tooltip: 'Delete selected',\n      onClick: (context) => {\n        console.log('Delete action clicked:', context);\n      }\n    },\n    { \n      id: 'edit', \n      label: 'Edit', \n      icon: 'edit', \n      tooltip: 'Edit selected',\n      onClick: (context) => {\n        console.log('Edit action clicked:', context);\n      }\n    },\n    { \n      id: 'add-tag', \n      label: 'Add Tag', \n      icon: 'local_offer', \n      tooltip: 'Add tags',\n      onClick: (context) => {\n        console.log('Add tag action clicked:', context);\n      }\n    },\n    { \n      id: 'remove-tag', \n      label: 'Remove Tag', \n      icon: 'label_off', \n      tooltip: 'Remove tags',\n      onClick: (context) => {\n        console.log('Remove tag action clicked:', context);\n      }\n    },\n  ];\n\n  // Chips inputs\n  @Input() chips: Array<{ key?: string; label?: string; text: string; fullText?: string; hasMore?: boolean }> = [];\n  @Input() filterApplied: boolean = false;\n\n  // Table inputs\n  @Input() columns: DynamicTableColumn[] = [];\n  @Input() selectedAutoRefreshInterval: number = 0;\n  @Input() pageIndex: number = 0;\n  @Input() pageSize: number = 10;\n  // Server-side pagination controls\n  @Input() serverSidePagination: boolean = false;\n  @Input() totalElements: number = 0;\n  @Output() pageChange = new EventEmitter<{ pageIndex: number; pageSize: number }>();\n  // Loader controls (passed down to dynamic table)\n  @Input() isTableLoading?: boolean;\n  @Input() isTableDataLoading?: boolean;\n  // Backward-compatibility flag; if provided, dynamic table will use it when specific flags are undefined\n\n  // Internal state for column visibility\n  private _columnVisibility: ColumnVisibilityConfig = {};\n  private _cachedVisibilityColumns: Array<{ id: string; label: string }> = [];\n  filteredRows: any[] = [];\n  pagedRows: any[] = [];\n  private pendingFilters: any;\n  private appliedFilters: any;\n  private autoRefreshTimer?: ReturnType<typeof setInterval>;\n  @Output() onReload = new EventEmitter<void>();\n  @Output() onAutoRefreshClick = new EventEmitter<void>();\n\n  // Derived columns with visibility applied. Avoid mutating @Input() columns so parent bindings aren't overridden.\n  get computedColumns(): DynamicTableColumn[] {\n    const visibility = this._columnVisibility || {};\n    const source = this.columns || [];\n    return source.map(col => {\n      if (['checkbox', 'actions'].includes(col.fieldId)) {\n        return col;\n      }\n      const show = visibility[col.fieldId];\n      return { ...col, isShow: show !== false };\n    });\n  }\n\n  // Auto-generated visibility columns from columns input (excludes default columns and checkbox/actions)\n  // Cached to avoid creating new arrays on every change detection cycle\n  get visibilityColumns(): Array<{ id: string; label: string }> {\n    return this._cachedVisibilityColumns;\n  }\n\n  // Internal column visibility state\n  get columnVisibility(): ColumnVisibilityConfig {\n    return this._columnVisibility;\n  }\n  ngOnInit(): void {\n    this.initializeComponent();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['data'] || changes['isEmptyState']) {\n      this.initializeComponent();\n    }\n    if (changes['columns']) {\n      this.initializeColumnVisibility();\n    }\n  }\n\n  private initializeComponent(): void {\n    if (this.isEmptyState) {\n      this.filteredRows = [];\n      this.pagedRows = [];\n      return;\n    }\n\n    // In server-side mode, assume incoming data is already a page slice\n    if (this.serverSidePagination) {\n      this.filteredRows = [...this.data];\n      this.pagedRows = [...this.filteredRows];\n      this.initializeColumnVisibility();\n      return;\n    }\n\n    this.filteredRows = [...this.data];\n    this.applyPagination();\n    this.initializeColumnVisibility();\n  }\n\n  private initializeColumnVisibility(): void {\n    // Cache visibility columns to avoid creating new arrays on every change detection\n    this._cachedVisibilityColumns = this.mapVisibilityColumns();\n    // Initialize visibility state for all visibility columns (default to true)\n    for (const col of this._cachedVisibilityColumns) {\n      if (this._columnVisibility[col.id] === undefined) {\n        this._columnVisibility[col.id] = true;\n      }\n    }\n  }\n\n  get anyRowSelected(): boolean {\n    return !!(this.pagedRows && this.pagedRows.some(r => !!(r as any).isSelected));\n  }\n\n  get currentSelectedItems(): any[] {\n    return (this.pagedRows || []).filter(r => !!(r as any).isSelected);\n  }\n\n  actionClick(data: any) {\n    console.log('action toolbar', data);\n  }\n\n  view(id: number | string): void {\n    console.log('View', id);\n  }\n\n  edit(row: any): void {\n    console.log('Edit', row);\n  }\n\n  delete(row: any): void {\n    console.log('Delete', row);\n  }\n\n  toggleFilter(): void {\n    this.showFilterPanel = !this.showFilterPanel;\n  }\n\n  onColumnVisibilityChange(cfg: ColumnVisibilityConfig): void {\n    this._columnVisibility = { ...cfg };\n    // Do not mutate this.columns; computedColumns getter will reflect changes\n  }\n\n  onAutoRefreshChange(intervalMs: number): void {\n    this.selectedAutoRefreshInterval = intervalMs;\n    this.setupAutoRefresh(intervalMs);\n    this.triggerReload();\n  }\n\n  valueChange(value: string): void {\n    console.log('Value changed', value);\n    this.onSearchChange.emit(value);\n  }\n\n  search(value: string): void {\n    console.log('Search', value);\n  }\n\n  cleared(): void {\n    console.log('Cleared');\n  }\n\n  onEmptyAction(action: any) {\n    if (action?.label === 'Show filters') {\n      this.toggleFilter();\n    }\n  }\n\n  onFiltersChanged(current: any): void {\n    // Defer applying filters and updating chips until Apply Filter is clicked\n    this.pendingFilters = current;\n  }\n\n  onFiltersApplied(applied: any): void {\n    const effective = applied ?? this.pendingFilters ?? {};\n    this.appliedFilters = effective;\n    this.filteredRows = this.data.filter(r => this.passFilters(effective, r));\n    this.pageIndex = 0;\n    this.applyPagination();\n    const chips: Array<{ key?: string; label?: string; text: string; fullText?: string; hasMore?: boolean }> = [];\n    if (effective) {\n      for (const key of Object.keys(effective)) {\n        const value = effective[key];\n        if (value == null || value === '' || (Array.isArray(value) && value.length === 0)) continue;\n        let text = '';\n        if (Array.isArray(value)) {\n          text = value.map((v: any) => (v?.name ?? v?.label ?? v?.value ?? v)).join(', ');\n        } else if (typeof value === 'object') {\n          if ('start' in value || 'end' in value) {\n            const s = value.start ? new Date(value.start).toLocaleDateString() : '';\n            const e = value.end ? new Date(value.end).toLocaleDateString() : '';\n            text = [s, e].filter(Boolean).join(' - ');\n          } else {\n            text = (value?.name ?? value?.label ?? value?.value ?? JSON.stringify(value));\n          }\n        } else {\n          text = String(value);\n        }\n        chips.push({ key, text, fullText: text, hasMore: text.length > 30 });\n      }\n    }\n    this.chips = chips;\n    this.filterApplied = this.chips.length > 0;\n  }\n\n  handleResetFilterClick(): void {\n    // Clear chips and reset filtered data when reset is clicked\n    this.pendingFilters = undefined;\n    this.chips = [];\n    this.filterApplied = false;\n    this.filteredRows = [...this.data];\n    this.pageIndex = 0;\n    this.applyPagination();\n    this.onResetFilterClick.emit();\n  }\n\n  handleRefreshClick(): void {\n    this.onAutoRefreshClick.emit();\n    this.triggerReload();\n  }\n\n  private triggerReload(): void {\n    const effective = this.appliedFilters ?? {};\n    this.filteredRows = this.data.filter(r => this.passFilters(effective, r));\n    this.pageIndex = 0;\n    this.applyPagination();\n    this.onReload.emit();\n  }\n\n  private setupAutoRefresh(intervalMs: number): void {\n    this.clearAutoRefresh();\n    if (intervalMs && intervalMs > 0) {\n      this.autoRefreshTimer = setInterval(() => {\n        this.triggerReload();\n      }, intervalMs);\n    }\n  }\n\n  private clearAutoRefresh(): void {\n    if (this.autoRefreshTimer) {\n      clearInterval(this.autoRefreshTimer);\n      this.autoRefreshTimer = undefined;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.clearAutoRefresh();\n  }\n\n  onPaginate(e: { pageIndex: number; pageSize: number }): void {\n    this.pageIndex = e.pageIndex;\n    this.pageSize = e.pageSize;\n    if (this.serverSidePagination) {\n      this.pageChange.emit({ pageIndex: this.pageIndex, pageSize: this.pageSize });\n      return;\n    }\n    this.applyPagination();\n  }\n\n  onPageSizeChange(size: number): void {\n    this.pageSize = size;\n    this.pageIndex = 0;\n    if (this.serverSidePagination) {\n      this.pageChange.emit({ pageIndex: this.pageIndex, pageSize: this.pageSize });\n      return;\n    }\n    this.applyPagination();\n  }\n\n  onRemoveChip(chip: any): void {\n    this.chips = this.chips.filter(c => c !== chip);\n    this.filterApplied = this.chips.length > 0;\n    // Emit event to parent component (same pattern as execution tab)\n    this.removeChip.emit(chip);\n  }\n\n  onClearAllChips(): void {\n    this.chips = [];\n    this.filterApplied = false;\n  }\n\n  private applyPagination(): void {\n    if (this.serverSidePagination) {\n      // Server mode: parent supplies already-paginated data\n      this.pagedRows = [...this.filteredRows];\n      return;\n    }\n    const start = this.pageIndex * this.pageSize;\n    const end = start + this.pageSize;\n    this.pagedRows = this.filteredRows.slice(start, end);\n  }\n\n  private mapVisibilityColumns(): Array<{ id: string; label: string }> {\n    return (this.columns || [])\n      .filter(c => c.isDefault === false)\n      .map(c => ({ id: c.fieldId, label: c.fieldName || c.fieldId }));\n  }\n\n  private normalizeDate(d: any): number | null {\n    if (!d) return null;\n    const ts = Date.parse(d);\n    return isNaN(ts) ? null : ts;\n  }\n\n  private passFilters(filters: any, row: any): boolean {\n    if (!filters) return true;\n    if (filters.status && Array.isArray(filters.status) && filters.status.length) {\n      if (!filters.status.includes(row.status)) return false;\n    }\n    if (filters.priority && Array.isArray(filters.priority) && filters.priority.length) {\n      if (!filters.priority.includes(row.priorityName)) return false;\n    }\n    if (filters.testType && Array.isArray(filters.testType) && filters.testType.length) {\n      if (!filters.testType.includes(row.testType)) return false;\n    }\n    if (filters.created_date && (filters.created_date.start || filters.created_date.end)) {\n      const startTs = this.normalizeDate(filters.created_date.start);\n      const endTs = this.normalizeDate(filters.created_date.end);\n      const rowTs = this.normalizeDate(row.createdAt);\n      if (rowTs != null) {\n        if (startTs != null && rowTs < startTs) return false;\n        if (endTs != null && rowTs > endTs) return false;\n      }\n    }\n    return true;\n  }\n}\n","<div class=\"cqa-ui-root\">\n  <div class=\"cqa-w-full cqa-flex cqa-flex-col cqa-relative\">\n    <div [class]=\"!showSearchBar ? 'cqa-justify-end' : 'cqa-justify-between'\" class=\"cqa-w-full cqa-flex cqa-items-center cqa-gap-3 cqa-flex-wrap cqa-mb-3\">\n      <cqa-search-bar\n        *ngIf=\"showSearchBar\"\n        [placeholder]=\"searchPlaceholder\"\n        [value]=\"searchValue\"\n        [showClear]=\"showClear\"\n        (valueChange)=\"valueChange($event)\"\n        (search)=\"search($event)\"\n        (cleared)=\"cleared()\"\n      ></cqa-search-bar>\n      <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-flex-wrap\">\n        <cqa-button\n          *ngIf=\"showFilterButton\"\n          variant=\"grey-solid\"\n          icon=\"add\"\n          [text]=\"'Filter'\"\n          (clicked)=\"toggleFilter()\"\n        >\n          <span>Filter</span>\n        </cqa-button>\n        <cqa-column-visibility\n          *ngIf=\"showSettingsButton\"\n          [columns]=\"visibilityColumns\"\n          [columnVisibility]=\"columnVisibility\"\n          [selectedAutoRefreshInterval]=\"selectedAutoRefreshInterval\"\n          (columnVisibilityChange)=\"onColumnVisibilityChange($event)\"\n          (autoRefreshChange)=\"onAutoRefreshChange($event)\"\n        ></cqa-column-visibility>\n        <cqa-button\n          *ngIf=\"showAutoRefreshButton\"\n          variant=\"grey-solid\"\n          icon=\"refresh\"\n          (clicked)=\"handleRefreshClick()\"\n          [tooltip]=\"'Refresh'\"\n          tooltipPosition=\"below\"\n        ></cqa-button>\n        <cqa-button *ngIf=\"showOtherButton\" [variant]=\"otherButtonVariant\" [text]=\"otherButtonLabel\"></cqa-button>\n      </div>\n    </div>\n\n    <cqa-selected-filters \n      [filterApplied]=\"filterApplied\"\n      [chips]=\"chips\"\n      (removeChip)=\"onRemoveChip($event)\"\n      (clearAll)=\"onClearAllChips()\"\n      (onClearAll)=\"onClearAll.emit()\"\n    >\n    </cqa-selected-filters>\n\n    <cqa-dynamic-filter\n      *ngIf=\"showFilterPanel\"\n      [config]=\"filterConfig\"\n      [showFilterPanel]=\"showFilterPanel\"\n      (filtersChanged)=\"onFiltersChanged($event)\"\n      (filtersApplied)=\"onFiltersApplied($event)\"\n      (onApplyFilterClick)=\"onApplyFilterClick.emit($event)\"\n      (onResetFilterClick)=\"handleResetFilterClick()\"\n    >\n    </cqa-dynamic-filter>\n\n    <div class=\"cqa-rounded-[7px] cqa-overflow-hidden cqa-border-t cqa-border-l cqa-border-r cqa-border-grey-200 cqa-relative\">\n      <ng-container *ngIf=\"(isTableLoading || isTableDataLoading) || (!isEmptyState && pagedRows && pagedRows.length > 0); else storyEmptyTpl\">\n        <app-dynamic-table\n          [columns]=\"computedColumns\"\n          [data]=\"pagedRows\"\n          [isTableLoading]=\"isTableLoading\"\n          [isTableDataLoading]=\"isTableDataLoading\">\n          <ng-template #emptyTableTpl>\n            <div class=\"cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-py-8\">\n              <img src=\"/assets/illustrations/empty-state.svg\" alt=\"No data\" class=\"cqa-w-32 cqa-h-32 cqa-mb-4\" />\n              <h3 class=\"cqa-text-lg cqa-font-semibold cqa-mb-2\">No test cases</h3>\n              <p class=\"cqa-text-sm cqa-text-neutral-500 cqa-mb-4\">Try adjusting filters or create a new test case.</p>\n              <cqa-button variant=\"filled\" (clicked)=\"toggleFilter()\">Show Filters</cqa-button>\n            </div>\n          </ng-template>\n        </app-dynamic-table>\n      </ng-container>\n\n      <ng-template #storyEmptyTpl>\n        <div class=\"cqa-p-6 cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n          <cqa-empty-state\n            *ngIf=\"isEmptyState\"\n            [title]=\"emptyStateConfig.title\"\n            [description]=\"emptyStateConfig.description\"\n            [imageUrl]=\"emptyStateConfig.imageUrl\"\n            [actions]=\"emptyStateConfig.actions\"\n            (actionClick)=\"onEmptyAction($event)\"\n          >\n          </cqa-empty-state>\n        </div>\n      </ng-template>\n\n    </div>\n\n    <cqa-pagination\n      [totalElements]=\"serverSidePagination ? totalElements : filteredRows.length\"\n      [pageIndex]=\"pageIndex\"\n      [pageSize]=\"pageSize\"\n      [pageItemCount]=\"pagedRows.length\"\n      (paginate)=\"onPaginate($event)\"\n      (pageSizeChange)=\"onPageSizeChange($event)\"\n    >\n    </cqa-pagination>\n\n    <div *ngIf=\"anyRowSelected\" class=\"cqa-absolute cqa-bottom-[18.75px] cqa-left-[50%] cqa-translate-x-[-50%] cqa-w-full lg:cqa-max-w-[68%] cqa-sm:max-w-[75%] cqa-max-w-[90%] cqa-z-[1]\" >\n      <cqa-table-action-toolbar\n        [selectedItems]=\"currentSelectedItems\"\n        [actions]=\"actions\"\n        (actionClick)=\"actionClick($event)\"\n      ></cqa-table-action-toolbar>\n    </div>\n    \n  </div>\n</div>\n\n"]}
|
|
452
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-template.component.js","sourceRoot":"","sources":["../../../../../src/lib/templates/table-template.component.ts","../../../../../src/lib/templates/table-template.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n  OnDestroy,\n} from '@angular/core';\nimport { DynamicFilterItem } from '../filters/dynamic-filter/dynamic-filter.component';\nimport { DynamicTableColumn } from '../table/dynamic-table/dynamic-table.component';\nimport { ColumnVisibilityConfig } from '../column-visibility/column-visibility.component';\nimport { TableAction } from '../table-action-toolbar/table-action-toolbar.component';\nimport { EmptyStateConfig } from '../empty-state/empty-state-config.interface';\nimport { EMPTY_STATE_IMAGES } from '../assets/images/image-assets.constants';\n\n@Component({\n  selector: 'cqa-table-template',\n  templateUrl: './table-template.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class TableTemplateComponent implements OnInit, OnChanges, OnDestroy {\n  // Search bar inputs\n  @Input() searchPlaceholder: string = 'Search components';\n  @Input() searchValue: string = '';\n  @Input() showClear: boolean = true;\n  @Input() showSearchBar: boolean = true;\n\n  // Search bar outputs\n  @Output() onSearchChange = new EventEmitter<string>();\n\n  // Filter outputs\n  @Output() onApplyFilterClick = new EventEmitter<any>();\n  @Output() onResetFilterClick = new EventEmitter<void>();\n  @Output() onClearAll = new EventEmitter<void>();\n  @Output() removeChip = new EventEmitter<any>();\n\n  // Filter inputs\n  @Input() filterConfig: DynamicFilterItem[] = [];\n  @Input() showFilterPanel: boolean = false;\n  @Input() showFilterButton: boolean = true;\n\n  // Other button input\n  @Input() otherButtonLabel: string = 'Other Button';\n  @Input() otherButtonVariant: 'filled' | 'outlined' | 'text' | 'elevated' | 'tonal' | 'grey-solid' = 'filled';\n  @Input() showOtherButton: boolean = true;\n  \n  // Action menu button (three-dot menu in table rows)\n  @Input() showActionButton: boolean = true;\n\n  // Settings and refresh buttons\n  @Input() showSettingsButton: boolean = true;\n  @Input() showAutoRefreshButton: boolean = true;\n\n  // Data input\n  @Input() data: any[] = [];\n\n  // Empty state inputs\n  @Input() isEmptyState: boolean = false;\n  @Input() emptyStateConfig: EmptyStateConfig = {\n    title: 'No Data Available Yet',\n    description: 'Run or upload your first test to see your analytics and trends here. Watch your quality metrics come to life with real-time insights.',\n    imageUrl: EMPTY_STATE_IMAGES.DASHBOARD,\n    actions: [{ label: 'Run Test Suite', variant: 'filled' }],\n  };\n\n  // Action bar inputs\n  @Input() actions: TableAction[] = [\n    { \n      id: 'delete', \n      label: 'Delete', \n      icon: 'delete', \n      tooltip: 'Delete selected',\n      onClick: (context) => {\n        console.log('Delete action clicked:', context);\n      }\n    },\n    { \n      id: 'edit', \n      label: 'Edit', \n      icon: 'edit', \n      tooltip: 'Edit selected',\n      onClick: (context) => {\n        console.log('Edit action clicked:', context);\n      }\n    },\n    { \n      id: 'add-tag', \n      label: 'Add Tag', \n      icon: 'local_offer', \n      tooltip: 'Add tags',\n      onClick: (context) => {\n        console.log('Add tag action clicked:', context);\n      }\n    },\n    { \n      id: 'remove-tag', \n      label: 'Remove Tag', \n      icon: 'label_off', \n      tooltip: 'Remove tags',\n      onClick: (context) => {\n        console.log('Remove tag action clicked:', context);\n      }\n    },\n  ];\n\n  // Chips inputs\n  @Input() chips: Array<{ key?: string; label?: string; text: string; fullText?: string; hasMore?: boolean }> = [];\n  @Input() filterApplied: boolean = false;\n\n  // Table inputs\n  @Input() columns: DynamicTableColumn[] = [];\n  @Input() selectedAutoRefreshInterval: number = 0;\n  @Input() pageIndex: number = 0;\n  @Input() pageSize: number = 10;\n  // Server-side pagination controls\n  @Input() serverSidePagination: boolean = false;\n  @Input() totalElements: number = 0;\n  @Output() pageChange = new EventEmitter<{ pageIndex: number; pageSize: number }>();\n  // Loader controls (passed down to dynamic table)\n  @Input() isTableLoading?: boolean;\n  @Input() isTableDataLoading?: boolean;\n  // Backward-compatibility flag; if provided, dynamic table will use it when specific flags are undefined\n\n  // Internal state for column visibility\n  private _columnVisibility: ColumnVisibilityConfig = {};\n  private _cachedVisibilityColumns: Array<{ id: string; label: string }> = [];\n  filteredRows: any[] = [];\n  pagedRows: any[] = [];\n  private pendingFilters: any;\n  private appliedFilters: any;\n  private autoRefreshTimer?: ReturnType<typeof setInterval>;\n  @Output() onReload = new EventEmitter<void>();\n  @Output() onAutoRefreshClick = new EventEmitter<void>();\n\n  // Derived columns with visibility applied.\n  computedColumns: DynamicTableColumn[] = [];\n  // Auto-generated visibility columns from columns input\n  visibilityColumns: Array<{ id: string; label: string }> = [];\n\n  private updateComputedColumns(): void {\n    const visibility = this._columnVisibility || {};\n    const source = this.columns || [];\n    this.computedColumns = source.map(col => {\n      if (['checkbox', 'actions'].includes(col.fieldId)) {\n        return col;\n      }\n      const show = visibility[col.fieldId];\n      return { ...col, isShow: show !== false };\n    });\n  }\n\n  private updateVisibilityColumns(): void {\n    this._cachedVisibilityColumns = this.mapVisibilityColumns();\n    this.visibilityColumns = this._cachedVisibilityColumns;\n  }\n\n  // Internal column visibility state\n  get columnVisibility(): ColumnVisibilityConfig {\n    return this._columnVisibility;\n  }\n  ngOnInit(): void {\n    this.initializeComponent();\n    this.updateComputedColumns();\n    this.updateVisibilityColumns();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['data'] || changes['isEmptyState']) {\n      this.initializeComponent();\n    }\n    if (changes['columns']) {\n      this.initializeColumnVisibility();\n      this.updateComputedColumns();\n      this.updateVisibilityColumns();\n    }\n  }\n\n  private initializeComponent(): void {\n    if (this.isEmptyState) {\n      this.filteredRows = [];\n      this.pagedRows = [];\n      return;\n    }\n\n    // In server-side mode, assume incoming data is already a page slice\n    if (this.serverSidePagination) {\n      this.filteredRows = [...this.data];\n      this.pagedRows = [...this.filteredRows];\n      this.initializeColumnVisibility();\n      return;\n    }\n\n    this.filteredRows = [...this.data];\n    this.applyPagination();\n    this.initializeColumnVisibility();\n  }\n\n  private initializeColumnVisibility(): void {\n    // Cache visibility columns to avoid creating new arrays on every change detection\n    this._cachedVisibilityColumns = this.mapVisibilityColumns();\n    // Initialize visibility state for all visibility columns (default to true)\n    for (const col of this._cachedVisibilityColumns) {\n      if (this._columnVisibility[col.id] === undefined) {\n        this._columnVisibility[col.id] = true;\n      }\n    }\n  }\n\n  get anyRowSelected(): boolean {\n    return !!(this.pagedRows && this.pagedRows.some(r => !!(r as any).isSelected));\n  }\n\n  get currentSelectedItems(): any[] {\n    return (this.pagedRows || []).filter(r => !!(r as any).isSelected);\n  }\n\n  actionClick(data: any) {\n    console.log('action toolbar', data);\n  }\n\n  view(id: number | string): void {\n    console.log('View', id);\n  }\n\n  edit(row: any): void {\n    console.log('Edit', row);\n  }\n\n  delete(row: any): void {\n    console.log('Delete', row);\n  }\n\n  toggleFilter(): void {\n    this.showFilterPanel = !this.showFilterPanel;\n  }\n\n  onColumnVisibilityChange(cfg: ColumnVisibilityConfig): void {\n    this._columnVisibility = { ...cfg };\n    // Do not mutate this.columns; computedColumns getter will reflect changes\n  }\n\n  onAutoRefreshChange(intervalMs: number): void {\n    this.selectedAutoRefreshInterval = intervalMs;\n    this.setupAutoRefresh(intervalMs);\n    this.triggerReload();\n  }\n\n  valueChange(value: string): void {\n    console.log('Value changed', value);\n    this.onSearchChange.emit(value);\n  }\n\n  search(value: string): void {\n    console.log('Search', value);\n  }\n\n  cleared(): void {\n    console.log('Cleared');\n  }\n\n  onEmptyAction(action: any) {\n    if (action?.label === 'Show filters') {\n      this.toggleFilter();\n    }\n  }\n\n  onFiltersChanged(current: any): void {\n    // Defer applying filters and updating chips until Apply Filter is clicked\n    this.pendingFilters = current;\n  }\n\n  onFiltersApplied(applied: any): void {\n    const effective = applied ?? this.pendingFilters ?? {};\n    this.appliedFilters = effective;\n    this.filteredRows = this.data.filter(r => this.passFilters(effective, r));\n    this.pageIndex = 0;\n    this.applyPagination();\n    const chips: Array<{ key?: string; label?: string; text: string; fullText?: string; hasMore?: boolean }> = [];\n    if (effective) {\n      for (const key of Object.keys(effective)) {\n        const value = effective[key];\n        if (value == null || value === '' || (Array.isArray(value) && value.length === 0)) continue;\n        let text = '';\n        if (Array.isArray(value)) {\n          text = value.map((v: any) => (v?.name ?? v?.label ?? v?.value ?? v)).join(', ');\n        } else if (typeof value === 'object') {\n          if ('start' in value || 'end' in value) {\n            const s = value.start ? new Date(value.start).toLocaleDateString() : '';\n            const e = value.end ? new Date(value.end).toLocaleDateString() : '';\n            text = [s, e].filter(Boolean).join(' - ');\n          } else {\n            text = (value?.name ?? value?.label ?? value?.value ?? JSON.stringify(value));\n          }\n        } else {\n          text = String(value);\n        }\n        chips.push({ key, text, fullText: text, hasMore: text.length > 30 });\n      }\n    }\n    this.chips = chips;\n    this.filterApplied = this.chips.length > 0;\n  }\n\n  handleResetFilterClick(): void {\n    // Clear chips and reset filtered data when reset is clicked\n    this.pendingFilters = undefined;\n    this.chips = [];\n    this.filterApplied = false;\n    this.filteredRows = [...this.data];\n    this.pageIndex = 0;\n    this.applyPagination();\n    this.onResetFilterClick.emit();\n  }\n\n  handleRefreshClick(): void {\n    this.onAutoRefreshClick.emit();\n    this.triggerReload();\n  }\n\n  private triggerReload(): void {\n    const effective = this.appliedFilters ?? {};\n    this.filteredRows = this.data.filter(r => this.passFilters(effective, r));\n    this.pageIndex = 0;\n    this.applyPagination();\n    this.onReload.emit();\n  }\n\n  private setupAutoRefresh(intervalMs: number): void {\n    this.clearAutoRefresh();\n    if (intervalMs && intervalMs > 0) {\n      this.autoRefreshTimer = setInterval(() => {\n        this.triggerReload();\n      }, intervalMs);\n    }\n  }\n\n  private clearAutoRefresh(): void {\n    if (this.autoRefreshTimer) {\n      clearInterval(this.autoRefreshTimer);\n      this.autoRefreshTimer = undefined;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.clearAutoRefresh();\n  }\n\n  onPaginate(e: { pageIndex: number; pageSize: number }): void {\n    this.pageIndex = e.pageIndex;\n    this.pageSize = e.pageSize;\n    if (this.serverSidePagination) {\n      this.pageChange.emit({ pageIndex: this.pageIndex, pageSize: this.pageSize });\n      return;\n    }\n    this.applyPagination();\n  }\n\n  onPageSizeChange(size: number): void {\n    this.pageSize = size;\n    this.pageIndex = 0;\n    if (this.serverSidePagination) {\n      this.pageChange.emit({ pageIndex: this.pageIndex, pageSize: this.pageSize });\n      return;\n    }\n    this.applyPagination();\n  }\n\n  onRemoveChip(chip: any): void {\n    this.chips = this.chips.filter(c => c !== chip);\n    this.filterApplied = this.chips.length > 0;\n    // Emit event to parent component (same pattern as execution tab)\n    this.removeChip.emit(chip);\n  }\n\n  onClearAllChips(): void {\n    this.chips = [];\n    this.filterApplied = false;\n  }\n\n  private applyPagination(): void {\n    if (this.serverSidePagination) {\n      // Server mode: parent supplies already-paginated data\n      this.pagedRows = [...this.filteredRows];\n      return;\n    }\n    const start = this.pageIndex * this.pageSize;\n    const end = start + this.pageSize;\n    this.pagedRows = this.filteredRows.slice(start, end);\n  }\n\n  private mapVisibilityColumns(): Array<{ id: string; label: string }> {\n    return (this.columns || [])\n      .filter(c => c.isDefault === false)\n      .map(c => ({ id: c.fieldId, label: c.fieldName || c.fieldId }));\n  }\n\n  private normalizeDate(d: any): number | null {\n    if (!d) return null;\n    const ts = Date.parse(d);\n    return isNaN(ts) ? null : ts;\n  }\n\n  private passFilters(filters: any, row: any): boolean {\n    if (!filters) return true;\n    if (filters.status && Array.isArray(filters.status) && filters.status.length) {\n      if (!filters.status.includes(row.status)) return false;\n    }\n    if (filters.priority && Array.isArray(filters.priority) && filters.priority.length) {\n      if (!filters.priority.includes(row.priorityName)) return false;\n    }\n    if (filters.testType && Array.isArray(filters.testType) && filters.testType.length) {\n      if (!filters.testType.includes(row.testType)) return false;\n    }\n    if (filters.created_date && (filters.created_date.start || filters.created_date.end)) {\n      const startTs = this.normalizeDate(filters.created_date.start);\n      const endTs = this.normalizeDate(filters.created_date.end);\n      const rowTs = this.normalizeDate(row.createdAt);\n      if (rowTs != null) {\n        if (startTs != null && rowTs < startTs) return false;\n        if (endTs != null && rowTs > endTs) return false;\n      }\n    }\n    return true;\n  }\n}\n","<div class=\"cqa-ui-root\">\n  <div class=\"cqa-w-full cqa-flex cqa-flex-col cqa-relative\">\n    <div [class]=\"!showSearchBar ? 'cqa-justify-end' : 'cqa-justify-between'\" class=\"cqa-w-full cqa-flex cqa-items-center cqa-gap-3 cqa-flex-wrap cqa-mb-3\">\n      <cqa-search-bar\n        *ngIf=\"showSearchBar\"\n        [placeholder]=\"searchPlaceholder\"\n        [value]=\"searchValue\"\n        [showClear]=\"showClear\"\n        (valueChange)=\"valueChange($event)\"\n        (search)=\"search($event)\"\n        (cleared)=\"cleared()\"\n      ></cqa-search-bar>\n      <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-flex-wrap\">\n        <cqa-button\n          *ngIf=\"showFilterButton\"\n          variant=\"grey-solid\"\n          icon=\"add\"\n          [text]=\"'Filter'\"\n          (clicked)=\"toggleFilter()\"\n        >\n          <span>Filter</span>\n        </cqa-button>\n        <cqa-column-visibility\n          *ngIf=\"showSettingsButton\"\n          [columns]=\"visibilityColumns\"\n          [columnVisibility]=\"columnVisibility\"\n          [selectedAutoRefreshInterval]=\"selectedAutoRefreshInterval\"\n          (columnVisibilityChange)=\"onColumnVisibilityChange($event)\"\n          (autoRefreshChange)=\"onAutoRefreshChange($event)\"\n        ></cqa-column-visibility>\n        <cqa-button\n          *ngIf=\"showAutoRefreshButton\"\n          variant=\"grey-solid\"\n          icon=\"refresh\"\n          (clicked)=\"handleRefreshClick()\"\n          [tooltip]=\"'Refresh'\"\n          tooltipPosition=\"below\"\n        ></cqa-button>\n        <cqa-button *ngIf=\"showOtherButton\" [variant]=\"otherButtonVariant\" [text]=\"otherButtonLabel\"></cqa-button>\n      </div>\n    </div>\n\n    <cqa-selected-filters \n      [filterApplied]=\"filterApplied\"\n      [chips]=\"chips\"\n      (removeChip)=\"onRemoveChip($event)\"\n      (clearAll)=\"onClearAllChips()\"\n      (onClearAll)=\"onClearAll.emit()\"\n    >\n    </cqa-selected-filters>\n\n    <cqa-dynamic-filter\n      *ngIf=\"showFilterPanel\"\n      [config]=\"filterConfig\"\n      [showFilterPanel]=\"showFilterPanel\"\n      (filtersChanged)=\"onFiltersChanged($event)\"\n      (filtersApplied)=\"onFiltersApplied($event)\"\n      (onApplyFilterClick)=\"onApplyFilterClick.emit($event)\"\n      (onResetFilterClick)=\"handleResetFilterClick()\"\n    >\n    </cqa-dynamic-filter>\n\n    <div class=\"cqa-rounded-[7px] cqa-overflow-hidden cqa-border-t cqa-border-l cqa-border-r cqa-border-grey-200 cqa-relative\">\n      <ng-container *ngIf=\"(isTableLoading || isTableDataLoading) || (!isEmptyState && pagedRows && pagedRows.length > 0); else storyEmptyTpl\">\n        <app-dynamic-table\n          [columns]=\"computedColumns\"\n          [data]=\"pagedRows\"\n          [isTableLoading]=\"isTableLoading\"\n          [isTableDataLoading]=\"isTableDataLoading\">\n          <ng-template #emptyTableTpl>\n            <div class=\"cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-py-8\">\n              <img src=\"/assets/illustrations/empty-state.svg\" alt=\"No data\" class=\"cqa-w-32 cqa-h-32 cqa-mb-4\" />\n              <h3 class=\"cqa-text-lg cqa-font-semibold cqa-mb-2\">No test cases</h3>\n              <p class=\"cqa-text-sm cqa-text-neutral-500 cqa-mb-4\">Try adjusting filters or create a new test case.</p>\n              <cqa-button variant=\"filled\" (clicked)=\"toggleFilter()\">Show Filters</cqa-button>\n            </div>\n          </ng-template>\n        </app-dynamic-table>\n      </ng-container>\n\n      <ng-template #storyEmptyTpl>\n        <div class=\"cqa-p-6 cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n          <cqa-empty-state\n            *ngIf=\"isEmptyState\"\n            [title]=\"emptyStateConfig.title\"\n            [description]=\"emptyStateConfig.description\"\n            [imageUrl]=\"emptyStateConfig.imageUrl\"\n            [actions]=\"emptyStateConfig.actions\"\n            (actionClick)=\"onEmptyAction($event)\"\n          >\n          </cqa-empty-state>\n        </div>\n      </ng-template>\n\n    </div>\n\n    <cqa-pagination\n      [totalElements]=\"serverSidePagination ? totalElements : filteredRows.length\"\n      [pageIndex]=\"pageIndex\"\n      [pageSize]=\"pageSize\"\n      [pageItemCount]=\"pagedRows.length\"\n      (paginate)=\"onPaginate($event)\"\n      (pageSizeChange)=\"onPageSizeChange($event)\"\n    >\n    </cqa-pagination>\n\n    <div *ngIf=\"anyRowSelected\" class=\"cqa-absolute cqa-bottom-[18.75px] cqa-left-[50%] cqa-translate-x-[-50%] cqa-w-full lg:cqa-max-w-[68%] cqa-sm:max-w-[75%] cqa-max-w-[90%] cqa-z-[1]\" >\n      <cqa-table-action-toolbar\n        [selectedItems]=\"currentSelectedItems\"\n        [actions]=\"actions\"\n        (actionClick)=\"actionClick($event)\"\n      ></cqa-table-action-toolbar>\n    </div>\n    \n  </div>\n</div>\n\n"]}
|
|
@@ -982,8 +982,22 @@ class DynamicTableComponent {
|
|
|
982
982
|
this.sortChange = new EventEmitter();
|
|
983
983
|
// Map to store component references for cleanup: rowIndex_colId -> ComponentRef
|
|
984
984
|
this.componentRefs = new Map();
|
|
985
|
+
// Caches for expensive getters
|
|
986
|
+
this.visibleColumns = [];
|
|
987
|
+
this.computedData = [];
|
|
988
|
+
this.computedGridTemplate = '';
|
|
989
|
+
this.computedColumnWidths = [];
|
|
985
990
|
this._sortDirection = null;
|
|
986
991
|
}
|
|
992
|
+
ngOnChanges(changes) {
|
|
993
|
+
if (changes['columns'] || changes['screenWidth']) {
|
|
994
|
+
this.updateVisibleColumns();
|
|
995
|
+
this.updateGridTemplate();
|
|
996
|
+
}
|
|
997
|
+
if (changes['data'] || changes['columns'] || changes['enableLocalSort']) {
|
|
998
|
+
this.updateComputedData();
|
|
999
|
+
}
|
|
1000
|
+
}
|
|
987
1001
|
ngOnDestroy() {
|
|
988
1002
|
// Clean up all dynamically created components
|
|
989
1003
|
this.componentRefs.forEach(ref => ref.destroy());
|
|
@@ -1001,7 +1015,7 @@ class DynamicTableComponent {
|
|
|
1001
1015
|
const w = this.screenWidth || window.innerWidth;
|
|
1002
1016
|
return w <= 992;
|
|
1003
1017
|
}
|
|
1004
|
-
|
|
1018
|
+
updateVisibleColumns() {
|
|
1005
1019
|
const responsiveFilter = (c) => {
|
|
1006
1020
|
const r = (c === null || c === void 0 ? void 0 : c.responsive) || {};
|
|
1007
1021
|
if (r.xs && this.isXs)
|
|
@@ -1012,7 +1026,7 @@ class DynamicTableComponent {
|
|
|
1012
1026
|
return false;
|
|
1013
1027
|
return true;
|
|
1014
1028
|
};
|
|
1015
|
-
|
|
1029
|
+
this.visibleColumns = (this.columns || []).filter(c => c.isShow !== false).filter(responsiveFilter);
|
|
1016
1030
|
}
|
|
1017
1031
|
getHeaderTemplate(colId) {
|
|
1018
1032
|
var _a;
|
|
@@ -1148,8 +1162,9 @@ class DynamicTableComponent {
|
|
|
1148
1162
|
isHtmlString(value) {
|
|
1149
1163
|
if (typeof value !== 'string')
|
|
1150
1164
|
return false;
|
|
1151
|
-
//
|
|
1152
|
-
|
|
1165
|
+
// Faster check: if it contains an opening tag pattern near the start
|
|
1166
|
+
// or just has any brackets. Wrapping in a regex with [\s\S]* is very slow.
|
|
1167
|
+
return value.includes('<') && /<[a-z/][^>]*>/i.test(value);
|
|
1153
1168
|
}
|
|
1154
1169
|
/**
|
|
1155
1170
|
* Sanitizes HTML string for safe rendering
|
|
@@ -1173,40 +1188,29 @@ class DynamicTableComponent {
|
|
|
1173
1188
|
return index;
|
|
1174
1189
|
}
|
|
1175
1190
|
// Compute grid-template-columns string from column config
|
|
1176
|
-
|
|
1177
|
-
if (this.gridTemplateColumns) {
|
|
1178
|
-
return this.gridTemplateColumns;
|
|
1179
|
-
}
|
|
1180
|
-
const cols = this.visibleColumns;
|
|
1181
|
-
if (!(cols === null || cols === void 0 ? void 0 : cols.length))
|
|
1182
|
-
return '';
|
|
1183
|
-
const fixedPx = cols.reduce((sum, c) => sum + (c.fixedPx || 0), 0);
|
|
1184
|
-
const dynamicCols = cols.filter(c => !c.fixedPx);
|
|
1185
|
-
const totalWeight = dynamicCols.reduce((sum, c) => sum + (c.weight || 1), 0) || 1;
|
|
1186
|
-
const parts = cols.map(c => {
|
|
1187
|
-
if (c.fixedPx && c.fixedPx > 0) {
|
|
1188
|
-
return `${c.fixedPx}px`;
|
|
1189
|
-
}
|
|
1190
|
-
const share = (c.weight || 1) / totalWeight;
|
|
1191
|
-
return `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;
|
|
1192
|
-
});
|
|
1193
|
-
return parts.join(' ');
|
|
1194
|
-
}
|
|
1195
|
-
// Compute per-column widths for use with <colgroup>
|
|
1196
|
-
get computedColumnWidths() {
|
|
1191
|
+
updateGridTemplate() {
|
|
1197
1192
|
const cols = this.visibleColumns;
|
|
1198
|
-
if (!(cols === null || cols === void 0 ? void 0 : cols.length))
|
|
1199
|
-
|
|
1193
|
+
if (!(cols === null || cols === void 0 ? void 0 : cols.length)) {
|
|
1194
|
+
this.computedGridTemplate = '';
|
|
1195
|
+
this.computedColumnWidths = [];
|
|
1196
|
+
return;
|
|
1197
|
+
}
|
|
1200
1198
|
const fixedPx = cols.reduce((sum, c) => sum + (c.fixedPx || 0), 0);
|
|
1201
1199
|
const dynamicCols = cols.filter(c => !c.fixedPx);
|
|
1202
1200
|
const totalWeight = dynamicCols.reduce((sum, c) => sum + (c.weight || 1), 0) || 1;
|
|
1203
|
-
|
|
1201
|
+
this.computedColumnWidths = cols.map(c => {
|
|
1204
1202
|
if (c.fixedPx && c.fixedPx > 0) {
|
|
1205
1203
|
return `${c.fixedPx}px`;
|
|
1206
1204
|
}
|
|
1207
1205
|
const share = (c.weight || 1) / totalWeight;
|
|
1208
1206
|
return `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;
|
|
1209
1207
|
});
|
|
1208
|
+
if (this.gridTemplateColumns) {
|
|
1209
|
+
this.computedGridTemplate = this.gridTemplateColumns;
|
|
1210
|
+
}
|
|
1211
|
+
else {
|
|
1212
|
+
this.computedGridTemplate = this.computedColumnWidths.join(' ');
|
|
1213
|
+
}
|
|
1210
1214
|
}
|
|
1211
1215
|
// Selection helpers
|
|
1212
1216
|
get allSelected() {
|
|
@@ -1238,20 +1242,22 @@ class DynamicTableComponent {
|
|
|
1238
1242
|
}
|
|
1239
1243
|
}
|
|
1240
1244
|
}
|
|
1241
|
-
|
|
1245
|
+
updateComputedData() {
|
|
1242
1246
|
const source = this.data || [];
|
|
1243
1247
|
if (!this.enableLocalSort || !this._sortActive || !this._sortDirection) {
|
|
1244
|
-
|
|
1248
|
+
this.computedData = source;
|
|
1249
|
+
return;
|
|
1245
1250
|
}
|
|
1246
1251
|
const col = this.visibleColumns.find(c => c.fieldId === this._sortActive);
|
|
1247
1252
|
if (!col || !col.fieldValue) {
|
|
1248
|
-
|
|
1253
|
+
this.computedData = source;
|
|
1254
|
+
return;
|
|
1249
1255
|
}
|
|
1250
1256
|
const dir = this._sortDirection === 'asc' ? 1 : -1;
|
|
1251
1257
|
const fieldPath = col.fieldValue;
|
|
1252
1258
|
const out = [...source];
|
|
1253
1259
|
out.sort((a, b) => dir * this.compareValues(this.getCellValue(a, fieldPath), this.getCellValue(b, fieldPath)));
|
|
1254
|
-
|
|
1260
|
+
this.computedData = out;
|
|
1255
1261
|
}
|
|
1256
1262
|
// Computed loading flags to support backward compatibility
|
|
1257
1263
|
get showTableLoading() {
|
|
@@ -1290,6 +1296,7 @@ class DynamicTableComponent {
|
|
|
1290
1296
|
else
|
|
1291
1297
|
this._sortDirection = 'asc';
|
|
1292
1298
|
}
|
|
1299
|
+
this.updateComputedData();
|
|
1293
1300
|
this.sortChange.emit({ fieldId: this._sortActive, fieldValue: col.fieldValue, direction: this._sortDirection });
|
|
1294
1301
|
}
|
|
1295
1302
|
compareValues(a, b) {
|
|
@@ -1331,7 +1338,7 @@ class DynamicTableComponent {
|
|
|
1331
1338
|
}
|
|
1332
1339
|
}
|
|
1333
1340
|
DynamicTableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicTableComponent, deps: [{ token: i1$2.DomSanitizer }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
1334
|
-
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", enableLocalSort: "enableLocalSort", isTableLoading: "isTableLoading", isTableDataLoading: "isTableDataLoading" }, 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 }], 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\" [class.is-loading]=\"true\" *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 [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 [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 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 (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\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: FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: ["label"] }, { type: TableDataLoaderComponent, selector: "cqa-table-data-loader", inputs: ["label", "size"] }], directives: [{ type: i2$1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i2$1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2$1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: DynamicCellContainerDirective, selector: "[cqaCellContainer]", inputs: ["componentConfig", "row", "column", "rowIndex", "colId"], exportAs: ["cqaCellContainer"] }] });
|
|
1341
|
+
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", enableLocalSort: "enableLocalSort", isTableLoading: "isTableLoading", isTableDataLoading: "isTableDataLoading" }, 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\" [class.is-loading]=\"true\" *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 [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 [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 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 (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\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: FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: ["label"] }, { type: TableDataLoaderComponent, selector: "cqa-table-data-loader", inputs: ["label", "size"] }], directives: [{ type: i2$1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i2$1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2$1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: DynamicCellContainerDirective, selector: "[cqaCellContainer]", inputs: ["componentConfig", "row", "column", "rowIndex", "colId"], exportAs: ["cqaCellContainer"] }] });
|
|
1335
1342
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicTableComponent, decorators: [{
|
|
1336
1343
|
type: Component,
|
|
1337
1344
|
args: [{ selector: "app-dynamic-table", host: { class: 'cqa-ui-root' }, 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\" [class.is-loading]=\"true\" *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 [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 [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 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 (change)=\"onRowSelectChange($event, row)\" />\n <label\n class=\"custom-checkbox-label\"\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: [] }]
|
|
@@ -5074,12 +5081,15 @@ class TableTemplateComponent {
|
|
|
5074
5081
|
this.pagedRows = [];
|
|
5075
5082
|
this.onReload = new EventEmitter();
|
|
5076
5083
|
this.onAutoRefreshClick = new EventEmitter();
|
|
5084
|
+
// Derived columns with visibility applied.
|
|
5085
|
+
this.computedColumns = [];
|
|
5086
|
+
// Auto-generated visibility columns from columns input
|
|
5087
|
+
this.visibilityColumns = [];
|
|
5077
5088
|
}
|
|
5078
|
-
|
|
5079
|
-
get computedColumns() {
|
|
5089
|
+
updateComputedColumns() {
|
|
5080
5090
|
const visibility = this._columnVisibility || {};
|
|
5081
5091
|
const source = this.columns || [];
|
|
5082
|
-
|
|
5092
|
+
this.computedColumns = source.map(col => {
|
|
5083
5093
|
if (['checkbox', 'actions'].includes(col.fieldId)) {
|
|
5084
5094
|
return col;
|
|
5085
5095
|
}
|
|
@@ -5087,10 +5097,9 @@ class TableTemplateComponent {
|
|
|
5087
5097
|
return Object.assign(Object.assign({}, col), { isShow: show !== false });
|
|
5088
5098
|
});
|
|
5089
5099
|
}
|
|
5090
|
-
|
|
5091
|
-
|
|
5092
|
-
|
|
5093
|
-
return this._cachedVisibilityColumns;
|
|
5100
|
+
updateVisibilityColumns() {
|
|
5101
|
+
this._cachedVisibilityColumns = this.mapVisibilityColumns();
|
|
5102
|
+
this.visibilityColumns = this._cachedVisibilityColumns;
|
|
5094
5103
|
}
|
|
5095
5104
|
// Internal column visibility state
|
|
5096
5105
|
get columnVisibility() {
|
|
@@ -5098,6 +5107,8 @@ class TableTemplateComponent {
|
|
|
5098
5107
|
}
|
|
5099
5108
|
ngOnInit() {
|
|
5100
5109
|
this.initializeComponent();
|
|
5110
|
+
this.updateComputedColumns();
|
|
5111
|
+
this.updateVisibilityColumns();
|
|
5101
5112
|
}
|
|
5102
5113
|
ngOnChanges(changes) {
|
|
5103
5114
|
if (changes['data'] || changes['isEmptyState']) {
|
|
@@ -5105,6 +5116,8 @@ class TableTemplateComponent {
|
|
|
5105
5116
|
}
|
|
5106
5117
|
if (changes['columns']) {
|
|
5107
5118
|
this.initializeColumnVisibility();
|
|
5119
|
+
this.updateComputedColumns();
|
|
5120
|
+
this.updateVisibilityColumns();
|
|
5108
5121
|
}
|
|
5109
5122
|
}
|
|
5110
5123
|
initializeComponent() {
|
|
@@ -7115,6 +7128,9 @@ class CompareRunsComponent {
|
|
|
7115
7128
|
imageUrl: EMPTY_STATE_IMAGES.COMPARE_RUNS,
|
|
7116
7129
|
actions: []
|
|
7117
7130
|
};
|
|
7131
|
+
this.allOptionsCache = [];
|
|
7132
|
+
this.lastRuns = [];
|
|
7133
|
+
this.runLabelCache = new Map();
|
|
7118
7134
|
}
|
|
7119
7135
|
ngOnInit() {
|
|
7120
7136
|
this.initializeForm();
|
|
@@ -7147,14 +7163,17 @@ class CompareRunsComponent {
|
|
|
7147
7163
|
var _a, _b, _c, _d;
|
|
7148
7164
|
const runAId = (_b = (_a = this.form) === null || _a === void 0 ? void 0 : _a.get('runA')) === null || _b === void 0 ? void 0 : _b.value;
|
|
7149
7165
|
const runBId = (_d = (_c = this.form) === null || _c === void 0 ? void 0 : _c.get('runB')) === null || _d === void 0 ? void 0 : _d.value;
|
|
7150
|
-
|
|
7151
|
-
|
|
7152
|
-
|
|
7153
|
-
|
|
7154
|
-
|
|
7155
|
-
|
|
7156
|
-
|
|
7157
|
-
|
|
7166
|
+
if (this.runs !== this.lastRuns) {
|
|
7167
|
+
this.allOptionsCache = this.runs.map(run => ({
|
|
7168
|
+
id: run.id,
|
|
7169
|
+
value: run.id,
|
|
7170
|
+
label: this.formatRunLabel(run),
|
|
7171
|
+
name: this.formatRunLabel(run)
|
|
7172
|
+
}));
|
|
7173
|
+
this.lastRuns = this.runs;
|
|
7174
|
+
}
|
|
7175
|
+
const runAOptions = this.allOptionsCache.filter(opt => opt.id !== runBId);
|
|
7176
|
+
const runBOptions = this.allOptionsCache.filter(opt => opt.id !== runAId);
|
|
7158
7177
|
this.runASelectConfig = {
|
|
7159
7178
|
key: 'runA',
|
|
7160
7179
|
label: '',
|
|
@@ -7181,6 +7200,9 @@ class CompareRunsComponent {
|
|
|
7181
7200
|
};
|
|
7182
7201
|
}
|
|
7183
7202
|
formatRunLabel(run) {
|
|
7203
|
+
if (this.runLabelCache.has(run.id)) {
|
|
7204
|
+
return this.runLabelCache.get(run.id);
|
|
7205
|
+
}
|
|
7184
7206
|
const date = new Date(run.startTime);
|
|
7185
7207
|
const formattedDate = date.toLocaleString('en-US', {
|
|
7186
7208
|
month: 'short',
|
|
@@ -7189,7 +7211,9 @@ class CompareRunsComponent {
|
|
|
7189
7211
|
minute: '2-digit',
|
|
7190
7212
|
second: '2-digit'
|
|
7191
7213
|
});
|
|
7192
|
-
|
|
7214
|
+
const label = `Run #${run.id} • ${formattedDate}`;
|
|
7215
|
+
this.runLabelCache.set(run.id, label);
|
|
7216
|
+
return label;
|
|
7193
7217
|
}
|
|
7194
7218
|
updateSelectConfigsLoading() {
|
|
7195
7219
|
if (this.runASelectConfig) {
|