@cqa-lib/cqa-ui 0.1.2 → 1.0.1

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.
Files changed (125) hide show
  1. package/esm2020/lib/action-menu/action-menu.component.mjs +42 -0
  2. package/esm2020/lib/assets/images/image-assets.constants.mjs +28 -0
  3. package/esm2020/lib/badge/badge.component.mjs +141 -0
  4. package/esm2020/lib/button/button.component.mjs +42 -67
  5. package/esm2020/lib/column-visibility/column-visibility.component.mjs +69 -0
  6. package/esm2020/lib/dashboards/chart-card/chart-card.component.mjs +22 -0
  7. package/esm2020/lib/dashboards/coverage-module-card/coverage-module-card.component.mjs +105 -0
  8. package/esm2020/lib/dashboards/dashboard-header/dashboard-header.component.mjs +82 -0
  9. package/esm2020/lib/dashboards/failed-test-cases-card/failed-test-cases-card.component.mjs +59 -0
  10. package/esm2020/lib/dashboards/heat-error-map-cell/heat-error-map-cell.component.mjs +45 -0
  11. package/esm2020/lib/dashboards/insight-card/insight-card.component.mjs +201 -0
  12. package/esm2020/lib/dashboards/metrics-card/metrics-block.component.mjs +41 -0
  13. package/esm2020/lib/dashboards/metrics-card/metrics-card-item.interface.mjs +2 -0
  14. package/esm2020/lib/dashboards/metrics-card/metrics-card.component.mjs +62 -0
  15. package/esm2020/lib/dashboards/progress-text-card/progress-text-card.component.mjs +46 -0
  16. package/esm2020/lib/dashboards/test-distribution-card/test-distribution-card.component.mjs +35 -0
  17. package/esm2020/lib/dialog/dialog.component.mjs +4 -4
  18. package/esm2020/lib/dropdown-button/dropdown-button.component.mjs +189 -0
  19. package/esm2020/lib/dynamic-select/dynamic-select-field.component.mjs +160 -0
  20. package/esm2020/lib/empty-state/empty-state.component.mjs +37 -0
  21. package/esm2020/lib/filters/dynamic-filter/dynamic-filter.component.mjs +239 -0
  22. package/esm2020/lib/full-table-loader/full-table-loader.component.mjs +16 -0
  23. package/esm2020/lib/inline-sort/inline-sort.component.mjs +58 -0
  24. package/esm2020/lib/other-button/other-button.component.mjs +76 -0
  25. package/esm2020/lib/pagination/pagination.component.mjs +102 -0
  26. package/esm2020/lib/search-bar/search-bar.component.mjs +3 -3
  27. package/esm2020/lib/segment-control/segment-control.component.mjs +3 -3
  28. package/esm2020/lib/selected-filters/selected-filters.component.mjs +27 -0
  29. package/esm2020/lib/table/dynamic-table/dynamic-cell.directive.mjs +35 -0
  30. package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +258 -0
  31. package/esm2020/lib/table-action-toolbar/table-action-toolbar.component.mjs +52 -0
  32. package/esm2020/lib/table-data-loader/table-data-loader.component.mjs +19 -0
  33. package/esm2020/lib/templates/table-template.component.mjs +365 -0
  34. package/esm2020/lib/ui-kit.module.mjs +196 -17
  35. package/esm2020/lib/utils/metadata-colors.util.mjs +100 -0
  36. package/esm2020/lib/utils/tw-overlay-container.mjs +22 -0
  37. package/esm2020/public-api.mjs +29 -1
  38. package/fesm2015/cqa-lib-cqa-ui.mjs +2899 -133
  39. package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
  40. package/fesm2020/cqa-lib-cqa-ui.mjs +2867 -133
  41. package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
  42. package/lib/action-menu/action-menu.component.d.ts +17 -0
  43. package/lib/assets/images/image-assets.constants.d.ts +20 -0
  44. package/lib/badge/badge.component.d.ts +25 -0
  45. package/lib/button/button.component.d.ts +6 -5
  46. package/lib/column-visibility/column-visibility.component.d.ts +33 -0
  47. package/lib/dashboards/chart-card/chart-card.component.d.ts +8 -0
  48. package/lib/dashboards/coverage-module-card/coverage-module-card.component.d.ts +44 -0
  49. package/lib/dashboards/dashboard-header/dashboard-header.component.d.ts +30 -0
  50. package/lib/dashboards/failed-test-cases-card/failed-test-cases-card.component.d.ts +28 -0
  51. package/lib/dashboards/heat-error-map-cell/heat-error-map-cell.component.d.ts +14 -0
  52. package/lib/dashboards/insight-card/insight-card.component.d.ts +73 -0
  53. package/lib/dashboards/metrics-card/metrics-block.component.d.ts +12 -0
  54. package/lib/dashboards/metrics-card/metrics-card-item.interface.d.ts +12 -0
  55. package/lib/dashboards/metrics-card/metrics-card.component.d.ts +17 -0
  56. package/lib/dashboards/progress-text-card/progress-text-card.component.d.ts +13 -0
  57. package/lib/dashboards/test-distribution-card/test-distribution-card.component.d.ts +29 -0
  58. package/lib/dropdown-button/dropdown-button.component.d.ts +32 -0
  59. package/lib/dynamic-select/dynamic-select-field.component.d.ts +43 -0
  60. package/lib/empty-state/empty-state.component.d.ts +20 -0
  61. package/lib/filters/dynamic-filter/dynamic-filter.component.d.ts +56 -0
  62. package/lib/full-table-loader/full-table-loader.component.d.ts +6 -0
  63. package/lib/inline-sort/inline-sort.component.d.ts +12 -0
  64. package/lib/other-button/other-button.component.d.ts +37 -0
  65. package/lib/pagination/pagination.component.d.ts +37 -0
  66. package/lib/selected-filters/selected-filters.component.d.ts +17 -0
  67. package/lib/table/dynamic-table/dynamic-cell.directive.d.ts +16 -0
  68. package/lib/table/dynamic-table/dynamic-table.component.d.ts +72 -0
  69. package/lib/table-action-toolbar/table-action-toolbar.component.d.ts +34 -0
  70. package/lib/table-data-loader/table-data-loader.component.d.ts +7 -0
  71. package/lib/templates/table-template.component.d.ts +90 -0
  72. package/lib/ui-kit.module.d.ts +43 -6
  73. package/lib/utils/metadata-colors.util.d.ts +50 -0
  74. package/lib/utils/tw-overlay-container.d.ts +12 -0
  75. package/package.json +1 -1
  76. package/public-api.d.ts +28 -0
  77. package/src/assets/fonts/SFUIText-Bold/SF-UI-Text-Bold.eot +0 -0
  78. package/src/assets/fonts/SFUIText-Bold/SF-UI-Text-Bold.otf +0 -0
  79. package/src/assets/fonts/SFUIText-Bold/SF-UI-Text-Bold.svg +14730 -0
  80. package/src/assets/fonts/SFUIText-Bold/SF-UI-Text-Bold.ttf +0 -0
  81. package/src/assets/fonts/SFUIText-Bold/SF-UI-Text-Bold.woff +0 -0
  82. package/src/assets/fonts/SFUIText-Bold/SF-UI-Text-Bold.woff2 +0 -0
  83. package/src/assets/fonts/SFUIText-Light/SF-UI-Text-Light.eot +0 -0
  84. package/src/assets/fonts/SFUIText-Light/SF-UI-Text-Light.otf +0 -0
  85. package/src/assets/fonts/SFUIText-Light/SF-UI-Text-Light.svg +21333 -0
  86. package/src/assets/fonts/SFUIText-Light/SF-UI-Text-Light.ttf +0 -0
  87. package/src/assets/fonts/SFUIText-Light/SF-UI-Text-Light.woff +0 -0
  88. package/src/assets/fonts/SFUIText-Light/SF-UI-Text-Light.woff2 +0 -0
  89. package/src/assets/fonts/SFUIText-Medium/SF-UI-Text-Medium.eot +0 -0
  90. package/src/assets/fonts/SFUIText-Medium/SF-UI-Text-Medium.otf +0 -0
  91. package/src/assets/fonts/SFUIText-Medium/SF-UI-Text-Medium.svg +14584 -0
  92. package/src/assets/fonts/SFUIText-Medium/SF-UI-Text-Medium.ttf +0 -0
  93. package/src/assets/fonts/SFUIText-Medium/SF-UI-Text-Medium.woff +0 -0
  94. package/src/assets/fonts/SFUIText-Medium/SF-UI-Text-Medium.woff2 +0 -0
  95. package/src/assets/fonts/SFUIText-Regular/SF-UI-Text-Regular.eot +0 -0
  96. package/src/assets/fonts/SFUIText-Regular/SF-UI-Text-Regular.otf +0 -0
  97. package/src/assets/fonts/SFUIText-Regular/SF-UI-Text-Regular.svg +17870 -0
  98. package/src/assets/fonts/SFUIText-Regular/SF-UI-Text-Regular.ttf +0 -0
  99. package/src/assets/fonts/SFUIText-Regular/SF-UI-Text-Regular.woff +0 -0
  100. package/src/assets/fonts/SFUIText-Regular/SF-UI-Text-Regular.woff2 +0 -0
  101. package/src/assets/fonts/SFUIText-Semibold/SF-UI-Text-Semibold.eot +0 -0
  102. package/src/assets/fonts/SFUIText-Semibold/SF-UI-Text-Semibold.otf +0 -0
  103. package/src/assets/fonts/SFUIText-Semibold/SF-UI-Text-Semibold.svg +14732 -0
  104. package/src/assets/fonts/SFUIText-Semibold/SF-UI-Text-Semibold.ttf +0 -0
  105. package/src/assets/fonts/SFUIText-Semibold/SF-UI-Text-Semibold.woff +0 -0
  106. package/src/assets/fonts/SFUIText-Semibold/SF-UI-Text-Semibold.woff2 +0 -0
  107. package/src/lib/assets/images/.gitkeep +0 -0
  108. package/src/lib/assets/images/DashboardIcon.png +0 -0
  109. package/src/lib/assets/images/FilesIcon.png +0 -0
  110. package/src/lib/assets/images/README.md +66 -0
  111. package/src/lib/assets/images/ReportsIcon.png +0 -0
  112. package/src/lib/assets/images/SearchIcon.png +0 -0
  113. package/src/lib/assets/images/StepsIcon.png +0 -0
  114. package/src/lib/assets/images/TestCaseIcon.png +0 -0
  115. package/src/lib/assets/images/analytics-chart-icon.svg +11 -0
  116. package/src/lib/assets/images/checklist-add-icon.svg +10 -0
  117. package/src/lib/assets/images/document-gear-icon.svg +9 -0
  118. package/src/lib/assets/images/empty-state-default-icon.svg +8 -0
  119. package/src/lib/assets/images/image-assets.constants.ts +38 -0
  120. package/src/lib/assets/images/search-debug-icon.svg +8 -0
  121. package/src/lib/assets/images/test-case-icon.svg +9 -0
  122. package/src/lib/assets/images/upload-folder-icon.svg +7 -0
  123. package/src/lib/utils/metadata-colors.constants.js +33 -0
  124. package/storybook-static/assets/images/README.md +66 -0
  125. package/styles.css +1 -1
@@ -0,0 +1,258 @@
1
+ import { Component, ContentChild, ContentChildren, Input, TemplateRef, EventEmitter, Output } from "@angular/core";
2
+ import { DynamicCellTemplateDirective, DynamicHeaderTemplateDirective } from "./dynamic-cell.directive";
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../../full-table-loader/full-table-loader.component";
5
+ import * as i2 from "../../table-data-loader/table-data-loader.component";
6
+ import * as i3 from "@angular/common";
7
+ export class DynamicTableComponent {
8
+ constructor() {
9
+ this.data = [];
10
+ this.columns = [];
11
+ // Enable built-in select-all behavior for a 'checkbox' column
12
+ this.enableSelectAll = true;
13
+ // Enable simple client-side sorting when clicking sortable headers (no custom header)
14
+ this.enableLocalSort = true;
15
+ // Emit sort changes so parent can perform server-side sort if desired
16
+ this.sortChange = new EventEmitter();
17
+ this._sortDirection = null;
18
+ }
19
+ get isXs() {
20
+ const w = this.screenWidth || window.innerWidth;
21
+ return w <= 480;
22
+ }
23
+ get isSm() {
24
+ const w = this.screenWidth || window.innerWidth;
25
+ return w <= 768;
26
+ }
27
+ get isMd() {
28
+ const w = this.screenWidth || window.innerWidth;
29
+ return w <= 992;
30
+ }
31
+ get visibleColumns() {
32
+ const responsiveFilter = (c) => {
33
+ const r = c?.responsive || {};
34
+ if (r.xs && this.isXs)
35
+ return false;
36
+ if (r.sm && this.isSm)
37
+ return false;
38
+ if (r.md && this.isMd)
39
+ return false;
40
+ return true;
41
+ };
42
+ return (this.columns || []).filter(c => c.isShow !== false).filter(responsiveFilter);
43
+ }
44
+ getHeaderTemplate(colId) {
45
+ const tpl = this.headerTemplates?.find(t => t.name === colId);
46
+ return tpl ? tpl.template : null;
47
+ }
48
+ getCellTemplate(colId) {
49
+ const tpl = this.cellTemplates?.find(t => t.name === colId);
50
+ return tpl ? tpl.template : null;
51
+ }
52
+ getCellValue(row, path) {
53
+ if (!row || !path)
54
+ return "";
55
+ const parts = path.split(".");
56
+ let current = row;
57
+ for (const part of parts) {
58
+ if (current == null)
59
+ return "";
60
+ current = current[part];
61
+ }
62
+ return current ?? "";
63
+ }
64
+ trackByIndex(index) {
65
+ return index;
66
+ }
67
+ // Compute grid-template-columns string from column config
68
+ get computedGridTemplate() {
69
+ if (this.gridTemplateColumns) {
70
+ return this.gridTemplateColumns;
71
+ }
72
+ const cols = this.visibleColumns;
73
+ if (!cols?.length)
74
+ return '';
75
+ const fixedPx = cols.reduce((sum, c) => sum + (c.fixedPx || 0), 0);
76
+ const dynamicCols = cols.filter(c => !c.fixedPx);
77
+ const totalWeight = dynamicCols.reduce((sum, c) => sum + (c.weight || 1), 0) || 1;
78
+ const parts = cols.map(c => {
79
+ if (c.fixedPx && c.fixedPx > 0) {
80
+ return `${c.fixedPx}px`;
81
+ }
82
+ const share = (c.weight || 1) / totalWeight;
83
+ return `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;
84
+ });
85
+ return parts.join(' ');
86
+ }
87
+ // Compute per-column widths for use with <colgroup>
88
+ get computedColumnWidths() {
89
+ const cols = this.visibleColumns;
90
+ if (!cols?.length)
91
+ return [];
92
+ const fixedPx = cols.reduce((sum, c) => sum + (c.fixedPx || 0), 0);
93
+ const dynamicCols = cols.filter(c => !c.fixedPx);
94
+ const totalWeight = dynamicCols.reduce((sum, c) => sum + (c.weight || 1), 0) || 1;
95
+ return cols.map(c => {
96
+ if (c.fixedPx && c.fixedPx > 0) {
97
+ return `${c.fixedPx}px`;
98
+ }
99
+ const share = (c.weight || 1) / totalWeight;
100
+ return `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;
101
+ });
102
+ }
103
+ // Selection helpers
104
+ get allSelected() {
105
+ const rows = this.data || [];
106
+ if (!rows.length)
107
+ return false;
108
+ return rows.every(r => !!r?.isSelected);
109
+ }
110
+ get someSelected() {
111
+ const rows = this.data || [];
112
+ if (!rows.length)
113
+ return false;
114
+ const anySelected = rows.some(r => !!r?.isSelected);
115
+ return anySelected && !this.allSelected;
116
+ }
117
+ onSelectAllChange(event) {
118
+ const target = event.target;
119
+ this.toggleSelectAll(target.checked);
120
+ }
121
+ onRowSelectChange(event, row) {
122
+ const target = event.target;
123
+ row.isSelected = target.checked;
124
+ }
125
+ toggleSelectAll(checked) {
126
+ const rows = this.data || [];
127
+ for (const row of rows) {
128
+ if (row) {
129
+ row.isSelected = checked;
130
+ }
131
+ }
132
+ }
133
+ get computedData() {
134
+ const source = this.data || [];
135
+ if (!this.enableLocalSort || !this._sortActive || !this._sortDirection) {
136
+ return source;
137
+ }
138
+ const col = this.visibleColumns.find(c => c.fieldId === this._sortActive);
139
+ if (!col || !col.fieldValue) {
140
+ return source;
141
+ }
142
+ const dir = this._sortDirection === 'asc' ? 1 : -1;
143
+ const fieldPath = col.fieldValue;
144
+ const out = [...source];
145
+ out.sort((a, b) => dir * this.compareValues(this.getCellValue(a, fieldPath), this.getCellValue(b, fieldPath)));
146
+ return out;
147
+ }
148
+ // Computed loading flags to support backward compatibility
149
+ get showTableLoading() {
150
+ return this.isTableLoading ?? false;
151
+ }
152
+ get showTableDataLoading() {
153
+ return this.isTableDataLoading ?? false;
154
+ }
155
+ // True when table has no data and is not currently loading — used to show an empty state
156
+ get isEmpty() {
157
+ const anyLoading = this.showTableLoading || this.showTableDataLoading;
158
+ return !anyLoading && (!(this.data && this.data.length) || this.data.length === 0);
159
+ }
160
+ isSortedAsc(colId) {
161
+ return this._sortActive === colId && this._sortDirection === 'asc';
162
+ }
163
+ isSortedDesc(colId) {
164
+ return this._sortActive === colId && this._sortDirection === 'desc';
165
+ }
166
+ toggleSort(col) {
167
+ if (!col?.sortable)
168
+ return;
169
+ const colId = col.fieldId;
170
+ if (this._sortActive !== colId) {
171
+ this._sortActive = colId;
172
+ this._sortDirection = 'asc';
173
+ }
174
+ else {
175
+ // cycle asc -> desc -> null -> asc
176
+ if (this._sortDirection === 'asc')
177
+ this._sortDirection = 'desc';
178
+ else if (this._sortDirection === 'desc')
179
+ this._sortDirection = null;
180
+ else
181
+ this._sortDirection = 'asc';
182
+ }
183
+ this.sortChange.emit({ fieldId: this._sortActive, fieldValue: col.fieldValue, direction: this._sortDirection });
184
+ }
185
+ compareValues(a, b) {
186
+ if (a == null && b == null)
187
+ return 0;
188
+ if (a == null)
189
+ return 1; // nulls last in asc (handled by dir multiplier)
190
+ if (b == null)
191
+ return -1;
192
+ const numA = typeof a === 'number' ? a : Number(a);
193
+ const numB = typeof b === 'number' ? b : Number(b);
194
+ const aIsNum = !isNaN(numA) && a !== '' && a !== null && a !== false;
195
+ const bIsNum = !isNaN(numB) && b !== '' && b !== null && b !== false;
196
+ if (aIsNum && bIsNum) {
197
+ if (numA < numB)
198
+ return -1;
199
+ if (numA > numB)
200
+ return 1;
201
+ return 0;
202
+ }
203
+ // Date detection: attempt parse when both values are strings and parseable
204
+ if (typeof a === 'string' && typeof b === 'string') {
205
+ const tsA = Date.parse(a);
206
+ const tsB = Date.parse(b);
207
+ if (!isNaN(tsA) && !isNaN(tsB)) {
208
+ if (tsA < tsB)
209
+ return -1;
210
+ if (tsA > tsB)
211
+ return 1;
212
+ return 0;
213
+ }
214
+ // localeCompare, case-insensitive
215
+ return a.localeCompare(b, undefined, { sensitivity: 'base', numeric: true });
216
+ }
217
+ // Fallback to string comparison
218
+ const sa = String(a);
219
+ const sb = String(b);
220
+ return sa.localeCompare(sb, undefined, { sensitivity: 'base', numeric: true });
221
+ }
222
+ }
223
+ DynamicTableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
224
+ 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" }, queries: [{ propertyName: "emptyTableTpl", first: true, predicate: ["emptyTableTpl"], descendants: true, read: TemplateRef }, { propertyName: "cellTemplates", predicate: DynamicCellTemplateDirective }, { propertyName: "headerTemplates", predicate: DynamicHeaderTemplateDirective }], ngImport: i0, template: "<div id=\"cqa-ui-root\">\n <div class=\"cqa-relative\">\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\" [class.is-loading]=\"true\">\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-left cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n [ngClass]=\"col.fieldId + '-cell'\">\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\"\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]\" [ngClass]=\"col.fieldId + '-cell'\">\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\">\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 <span class=\"cqa-text-xs cqa-text-[#374151]\">{{ getCellValue(row, col.fieldValue) }}</span>\n </ng-template>\n </ng-template>\n </td>\n </ng-container>\n </tr>\n </tbody>\n </table>\n </div>\n</div>", components: [{ type: i1.FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: ["label"] }, { type: i2.TableDataLoaderComponent, selector: "cqa-table-data-loader", inputs: ["label", "size"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }] });
225
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicTableComponent, decorators: [{
226
+ type: Component,
227
+ args: [{ selector: "app-dynamic-table", template: "<div id=\"cqa-ui-root\">\n <div class=\"cqa-relative\">\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\" [class.is-loading]=\"true\">\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-left cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n [ngClass]=\"col.fieldId + '-cell'\">\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\"\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]\" [ngClass]=\"col.fieldId + '-cell'\">\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\">\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 <span class=\"cqa-text-xs cqa-text-[#374151]\">{{ getCellValue(row, col.fieldValue) }}</span>\n </ng-template>\n </ng-template>\n </td>\n </ng-container>\n </tr>\n </tbody>\n </table>\n </div>\n</div>", styles: [] }]
228
+ }], propDecorators: { data: [{
229
+ type: Input
230
+ }], columns: [{
231
+ type: Input
232
+ }], emptyState: [{
233
+ type: Input
234
+ }], gridTemplateColumns: [{
235
+ type: Input
236
+ }], screenWidth: [{
237
+ type: Input
238
+ }], enableSelectAll: [{
239
+ type: Input
240
+ }], enableLocalSort: [{
241
+ type: Input
242
+ }], isTableLoading: [{
243
+ type: Input
244
+ }], isTableDataLoading: [{
245
+ type: Input
246
+ }], sortChange: [{
247
+ type: Output
248
+ }], cellTemplates: [{
249
+ type: ContentChildren,
250
+ args: [DynamicCellTemplateDirective]
251
+ }], headerTemplates: [{
252
+ type: ContentChildren,
253
+ args: [DynamicHeaderTemplateDirective]
254
+ }], emptyTableTpl: [{
255
+ type: ContentChild,
256
+ args: ['emptyTableTpl', { read: TemplateRef }]
257
+ }] } });
258
+ //# 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 } from \"@angular/core\";\nimport { DynamicCellTemplateDirective, DynamicHeaderTemplateDirective } from \"./dynamic-cell.directive\";\n\nexport interface DynamicTableColumn {\n  fieldId: string;\n  fieldName?: string;\n  fieldValue?: string;\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  responsive?: { xs?: boolean; sm?: boolean; md?: boolean }; // hide at specific breakpoints\n}\n\n@Component({\n  selector: \"app-dynamic-table\",\n  templateUrl: \"./dynamic-table.component.html\",\n  styleUrls: []\n})\nexport class DynamicTableComponent {\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  // 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  @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  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  get visibleColumns(): DynamicTableColumn[] {\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    return (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  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  // Compute grid-template-columns string from column config\n  get computedGridTemplate(): string {\n    if (this.gridTemplateColumns) {\n      return this.gridTemplateColumns;\n    }\n    const cols = this.visibleColumns;\n    if (!cols?.length) return '';\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    const parts = cols.map(c => {\n      if (c.fixedPx && c.fixedPx > 0) {\n        return `${c.fixedPx}px`;\n      }\n      const share = (c.weight || 1) / totalWeight;\n      return `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;\n    });\n    return parts.join(' ');\n  }\n\n  // Compute per-column widths for use with <colgroup>\n  get computedColumnWidths(): string[] {\n    const cols = this.visibleColumns;\n    if (!cols?.length) return [];\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    return cols.map(c => {\n      if (c.fixedPx && c.fixedPx > 0) {\n        return `${c.fixedPx}px`;\n      }\n      const share = (c.weight || 1) / totalWeight;\n      return `calc((100% - ${fixedPx}px) * ${share.toFixed(4)})`;\n    });\n  }\n\n  // Selection helpers\n  get allSelected(): boolean {\n    const rows = this.data || [];\n    if (!rows.length) return false;\n    return rows.every(r => !!r?.isSelected);\n  }\n\n  get someSelected(): boolean {\n    const rows = this.data || [];\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    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 = checked;\n      }\n    }\n  }\n\n  // ===== Sorting support =====\n  private _sortActive?: string;\n  private _sortDirection: 'asc' | 'desc' | null = null;\n\n  get computedData(): any[] {\n    const source = this.data || [];\n    if (!this.enableLocalSort || !this._sortActive || !this._sortDirection) {\n      return source;\n    }\n    const col = this.visibleColumns.find(c => c.fieldId === this._sortActive);\n    if (!col || !col.fieldValue) {\n      return source;\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    return 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.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 id=\"cqa-ui-root\">\n  <div class=\"cqa-relative\">\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\" [class.is-loading]=\"true\">\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-left cqa-text-[#374151] cqa-bg-[#eff0f7]\"\n              [ngClass]=\"col.fieldId + '-cell'\">\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\"\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]\" [ngClass]=\"col.fieldId + '-cell'\">\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\">\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                  <span class=\"cqa-text-xs cqa-text-[#374151]\">{{ getCellValue(row, col.fieldValue) }}</span>\n                </ng-template>\n              </ng-template>\n            </td>\n          </ng-container>\n        </tr>\n      </tbody>\n    </table>\n  </div>\n</div>"]}
@@ -0,0 +1,52 @@
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/material/icon";
4
+ import * as i2 from "@angular/common";
5
+ export class TableActionToolbarComponent {
6
+ constructor() {
7
+ this.selectedItems = [];
8
+ this.actions = [];
9
+ this.actionClick = new EventEmitter();
10
+ }
11
+ get hasSelection() {
12
+ return (this.selectedItems?.length || 0) > 0;
13
+ }
14
+ get isSingleSelection() {
15
+ return this.selectedItems?.length === 1;
16
+ }
17
+ get selectionLabel() {
18
+ const n = this.selectedItems?.length || 0;
19
+ return n === 1 ? '1 selected' : `${n} selected`;
20
+ }
21
+ visibleActions() {
22
+ const ctx = { selected: this.selectedItems || [] };
23
+ return (this.actions || []).filter(a => (a.show ? a.show(ctx) : true));
24
+ }
25
+ isDisabled(action) {
26
+ const ctx = { selected: this.selectedItems || [] };
27
+ return action.disabled ? !!action.disabled(ctx) : false;
28
+ }
29
+ onAction(action) {
30
+ if (this.isDisabled(action)) {
31
+ return;
32
+ }
33
+ const context = { id: action.id, selected: this.selectedItems || [] };
34
+ if (action.onClick) {
35
+ action.onClick(context);
36
+ }
37
+ this.actionClick.emit(context);
38
+ }
39
+ }
40
+ TableActionToolbarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TableActionToolbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
41
+ TableActionToolbarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TableActionToolbarComponent, selector: "cqa-table-action-toolbar", inputs: { selectedItems: "selectedItems", actions: "actions" }, outputs: { actionClick: "actionClick" }, ngImport: i0, template: "<div id=\"cqa-ui-root\">\n <div\n class=\"action-toolbar cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-py-[15.5px] sm:cqa-px-[34px] cqa-px-[20px] cqa-bg-primary cqa-text-white cqa-rounded-[7px]\"\n *ngIf=\"hasSelection\">\n <div class=\"action-toolbar-left cqa-text-[14px] cqa-leading-[21px] cqa-font-medium\">\n {{ selectionLabel }}\n </div>\n <div class=\"action-toolbar-right cqa-flex cqa-items-center cqa-gap-[7px]\">\n <ng-container *ngFor=\"let action of visibleActions()\">\n <div [attr.aria-disabled]=\"isDisabled(action)\" [class.cqa-opacity-50]=\"isDisabled(action)\" [class.cqa-cursor-not-allowed]=\"isDisabled(action)\" (click)=\"!isDisabled(action) && onAction(action)\"\n class=\"cqa-flex cqa-items-center cqa-gap-[8.75px] cqa-py-[5px] cqa-px-[8.75px] cqa-cursor-pointer cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium\">\n <mat-icon class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ action.icon }}</mat-icon>\n <span class=\"md:cqa-flex cqa-hidden\">{{ action.label }}</span>\n </div>\n </ng-container>\n </div>\n <!-- Forwards clicks inside toolbar without affecting outer selections -->\n </div>\n</div>", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
42
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TableActionToolbarComponent, decorators: [{
43
+ type: Component,
44
+ args: [{ selector: 'cqa-table-action-toolbar', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div id=\"cqa-ui-root\">\n <div\n class=\"action-toolbar cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-py-[15.5px] sm:cqa-px-[34px] cqa-px-[20px] cqa-bg-primary cqa-text-white cqa-rounded-[7px]\"\n *ngIf=\"hasSelection\">\n <div class=\"action-toolbar-left cqa-text-[14px] cqa-leading-[21px] cqa-font-medium\">\n {{ selectionLabel }}\n </div>\n <div class=\"action-toolbar-right cqa-flex cqa-items-center cqa-gap-[7px]\">\n <ng-container *ngFor=\"let action of visibleActions()\">\n <div [attr.aria-disabled]=\"isDisabled(action)\" [class.cqa-opacity-50]=\"isDisabled(action)\" [class.cqa-cursor-not-allowed]=\"isDisabled(action)\" (click)=\"!isDisabled(action) && onAction(action)\"\n class=\"cqa-flex cqa-items-center cqa-gap-[8.75px] cqa-py-[5px] cqa-px-[8.75px] cqa-cursor-pointer cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium\">\n <mat-icon class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ action.icon }}</mat-icon>\n <span class=\"md:cqa-flex cqa-hidden\">{{ action.label }}</span>\n </div>\n </ng-container>\n </div>\n <!-- Forwards clicks inside toolbar without affecting outer selections -->\n </div>\n</div>", styles: [] }]
45
+ }], propDecorators: { selectedItems: [{
46
+ type: Input
47
+ }], actions: [{
48
+ type: Input
49
+ }], actionClick: [{
50
+ type: Output
51
+ }] } });
52
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUtYWN0aW9uLXRvb2xiYXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90YWJsZS1hY3Rpb24tdG9vbGJhci90YWJsZS1hY3Rpb24tdG9vbGJhci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3RhYmxlLWFjdGlvbi10b29sYmFyL3RhYmxlLWFjdGlvbi10b29sYmFyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7Ozs7QUFrQmhHLE1BQU0sT0FBTywyQkFBMkI7SUFOeEM7UUFPVyxrQkFBYSxHQUFVLEVBQUUsQ0FBQztRQUMxQixZQUFPLEdBQWtCLEVBQUUsQ0FBQztRQUUzQixnQkFBVyxHQUFHLElBQUksWUFBWSxFQUFtQyxDQUFDO0tBbUM3RTtJQWpDQyxJQUFJLFlBQVk7UUFDZCxPQUFPLENBQUMsSUFBSSxDQUFDLGFBQWEsRUFBRSxNQUFNLElBQUksQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQy9DLENBQUM7SUFFRCxJQUFJLGlCQUFpQjtRQUNuQixPQUFPLElBQUksQ0FBQyxhQUFhLEVBQUUsTUFBTSxLQUFLLENBQUMsQ0FBQztJQUMxQyxDQUFDO0lBRUQsSUFBSSxjQUFjO1FBQ2hCLE1BQU0sQ0FBQyxHQUFHLElBQUksQ0FBQyxhQUFhLEVBQUUsTUFBTSxJQUFJLENBQUMsQ0FBQztRQUMxQyxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLFdBQVcsQ0FBQztJQUNsRCxDQUFDO0lBRUQsY0FBYztRQUNaLE1BQU0sR0FBRyxHQUFHLEVBQUUsUUFBUSxFQUFFLElBQUksQ0FBQyxhQUFhLElBQUksRUFBRSxFQUFFLENBQUM7UUFDbkQsT0FBTyxDQUFDLElBQUksQ0FBQyxPQUFPLElBQUksRUFBRSxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO0lBQ3pFLENBQUM7SUFFRCxVQUFVLENBQUMsTUFBbUI7UUFDNUIsTUFBTSxHQUFHLEdBQUcsRUFBRSxRQUFRLEVBQUUsSUFBSSxDQUFDLGFBQWEsSUFBSSxFQUFFLEVBQUUsQ0FBQztRQUNuRCxPQUFPLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUM7SUFDMUQsQ0FBQztJQUVELFFBQVEsQ0FBQyxNQUFtQjtRQUMxQixJQUFJLElBQUksQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLEVBQUU7WUFDM0IsT0FBTztTQUNSO1FBQ0QsTUFBTSxPQUFPLEdBQUcsRUFBRSxFQUFFLEVBQUUsTUFBTSxDQUFDLEVBQUUsRUFBRSxRQUFRLEVBQUUsSUFBSSxDQUFDLGFBQWEsSUFBSSxFQUFFLEVBQUUsQ0FBQztRQUN0RSxJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUU7WUFDbEIsTUFBTSxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztTQUN6QjtRQUNELElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ2pDLENBQUM7O3dIQXRDVSwyQkFBMkI7NEdBQTNCLDJCQUEyQix5S0NsQnhDLDR0Q0FrQk07MkZEQU8sMkJBQTJCO2tCQU52QyxTQUFTOytCQUNFLDBCQUEwQixtQkFHbkIsdUJBQXVCLENBQUMsTUFBTTs4QkFHdEMsYUFBYTtzQkFBckIsS0FBSztnQkFDRyxPQUFPO3NCQUFmLEtBQUs7Z0JBRUksV0FBVztzQkFBcEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgSW5wdXQsIE91dHB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFRhYmxlQWN0aW9uIHtcbiAgaWQ6IHN0cmluZztcbiAgbGFiZWw6IHN0cmluZztcbiAgaWNvbjogc3RyaW5nO1xuICB0b29sdGlwPzogc3RyaW5nO1xuICBzaG93PzogKGNvbnRleHQ6IHsgc2VsZWN0ZWQ6IGFueVtdIH0pID0+IGJvb2xlYW47XG4gIGRpc2FibGVkPzogKGNvbnRleHQ6IHsgc2VsZWN0ZWQ6IGFueVtdIH0pID0+IGJvb2xlYW47XG4gIG9uQ2xpY2s/OiAoY29udGV4dDogeyBpZDogc3RyaW5nOyBzZWxlY3RlZDogYW55W10gfSkgPT4gdm9pZDtcbn1cblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnY3FhLXRhYmxlLWFjdGlvbi10b29sYmFyJyxcbiAgdGVtcGxhdGVVcmw6ICcuL3RhYmxlLWFjdGlvbi10b29sYmFyLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2hcbn0pXG5leHBvcnQgY2xhc3MgVGFibGVBY3Rpb25Ub29sYmFyQ29tcG9uZW50IHtcbiAgQElucHV0KCkgc2VsZWN0ZWRJdGVtczogYW55W10gPSBbXTtcbiAgQElucHV0KCkgYWN0aW9uczogVGFibGVBY3Rpb25bXSA9IFtdO1xuXG4gIEBPdXRwdXQoKSBhY3Rpb25DbGljayA9IG5ldyBFdmVudEVtaXR0ZXI8eyBpZDogc3RyaW5nOyBzZWxlY3RlZDogYW55W10gfT4oKTtcblxuICBnZXQgaGFzU2VsZWN0aW9uKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiAodGhpcy5zZWxlY3RlZEl0ZW1zPy5sZW5ndGggfHwgMCkgPiAwO1xuICB9XG5cbiAgZ2V0IGlzU2luZ2xlU2VsZWN0aW9uKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLnNlbGVjdGVkSXRlbXM/Lmxlbmd0aCA9PT0gMTtcbiAgfVxuXG4gIGdldCBzZWxlY3Rpb25MYWJlbCgpOiBzdHJpbmcge1xuICAgIGNvbnN0IG4gPSB0aGlzLnNlbGVjdGVkSXRlbXM/Lmxlbmd0aCB8fCAwO1xuICAgIHJldHVybiBuID09PSAxID8gJzEgc2VsZWN0ZWQnIDogYCR7bn0gc2VsZWN0ZWRgO1xuICB9XG5cbiAgdmlzaWJsZUFjdGlvbnMoKTogVGFibGVBY3Rpb25bXSB7XG4gICAgY29uc3QgY3R4ID0geyBzZWxlY3RlZDogdGhpcy5zZWxlY3RlZEl0ZW1zIHx8IFtdIH07XG4gICAgcmV0dXJuICh0aGlzLmFjdGlvbnMgfHwgW10pLmZpbHRlcihhID0+IChhLnNob3cgPyBhLnNob3coY3R4KSA6IHRydWUpKTtcbiAgfVxuXG4gIGlzRGlzYWJsZWQoYWN0aW9uOiBUYWJsZUFjdGlvbik6IGJvb2xlYW4ge1xuICAgIGNvbnN0IGN0eCA9IHsgc2VsZWN0ZWQ6IHRoaXMuc2VsZWN0ZWRJdGVtcyB8fCBbXSB9O1xuICAgIHJldHVybiBhY3Rpb24uZGlzYWJsZWQgPyAhIWFjdGlvbi5kaXNhYmxlZChjdHgpIDogZmFsc2U7XG4gIH1cblxuICBvbkFjdGlvbihhY3Rpb246IFRhYmxlQWN0aW9uKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXNEaXNhYmxlZChhY3Rpb24pKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGNvbnN0IGNvbnRleHQgPSB7IGlkOiBhY3Rpb24uaWQsIHNlbGVjdGVkOiB0aGlzLnNlbGVjdGVkSXRlbXMgfHwgW10gfTtcbiAgICBpZiAoYWN0aW9uLm9uQ2xpY2spIHtcbiAgICAgIGFjdGlvbi5vbkNsaWNrKGNvbnRleHQpO1xuICAgIH1cbiAgICB0aGlzLmFjdGlvbkNsaWNrLmVtaXQoY29udGV4dCk7XG4gIH1cbn1cblxuXG4iLCI8ZGl2IGlkPVwiY3FhLXVpLXJvb3RcIj5cbiAgPGRpdlxuICAgIGNsYXNzPVwiYWN0aW9uLXRvb2xiYXIgY3FhLWZsZXggY3FhLWl0ZW1zLWNlbnRlciBjcWEtanVzdGlmeS1iZXR3ZWVuIGNxYS1nYXAtMiBjcWEtZmxleC13cmFwIGNxYS1weS1bMTUuNXB4XSBzbTpjcWEtcHgtWzM0cHhdIGNxYS1weC1bMjBweF0gY3FhLWJnLXByaW1hcnkgY3FhLXRleHQtd2hpdGUgY3FhLXJvdW5kZWQtWzdweF1cIlxuICAgICpuZ0lmPVwiaGFzU2VsZWN0aW9uXCI+XG4gICAgPGRpdiBjbGFzcz1cImFjdGlvbi10b29sYmFyLWxlZnQgY3FhLXRleHQtWzE0cHhdIGNxYS1sZWFkaW5nLVsyMXB4XSBjcWEtZm9udC1tZWRpdW1cIj5cbiAgICAgIHt7IHNlbGVjdGlvbkxhYmVsIH19XG4gICAgPC9kaXY+XG4gICAgPGRpdiBjbGFzcz1cImFjdGlvbi10b29sYmFyLXJpZ2h0IGNxYS1mbGV4IGNxYS1pdGVtcy1jZW50ZXIgY3FhLWdhcC1bN3B4XVwiPlxuICAgICAgPG5nLWNvbnRhaW5lciAqbmdGb3I9XCJsZXQgYWN0aW9uIG9mIHZpc2libGVBY3Rpb25zKClcIj5cbiAgICAgICAgPGRpdiBbYXR0ci5hcmlhLWRpc2FibGVkXT1cImlzRGlzYWJsZWQoYWN0aW9uKVwiIFtjbGFzcy5jcWEtb3BhY2l0eS01MF09XCJpc0Rpc2FibGVkKGFjdGlvbilcIiBbY2xhc3MuY3FhLWN1cnNvci1ub3QtYWxsb3dlZF09XCJpc0Rpc2FibGVkKGFjdGlvbilcIiAoY2xpY2spPVwiIWlzRGlzYWJsZWQoYWN0aW9uKSAmJiBvbkFjdGlvbihhY3Rpb24pXCJcbiAgICAgICAgICBjbGFzcz1cImNxYS1mbGV4IGNxYS1pdGVtcy1jZW50ZXIgY3FhLWdhcC1bOC43NXB4XSBjcWEtcHktWzVweF0gY3FhLXB4LVs4Ljc1cHhdIGNxYS1jdXJzb3ItcG9pbnRlciBjcWEtdGV4dC1bMTIuM3B4XSBjcWEtbGVhZGluZy1bMTcuNXB4XSBjcWEtZm9udC1tZWRpdW1cIj5cbiAgICAgICAgICA8bWF0LWljb24gY2xhc3M9XCIhY3FhLXctWzE2cHhdICFjcWEtaC1bMTZweF0gIWNxYS10ZXh0LVsxNnB4XVwiPnt7IGFjdGlvbi5pY29uIH19PC9tYXQtaWNvbj5cbiAgICAgICAgICA8c3BhbiBjbGFzcz1cIm1kOmNxYS1mbGV4IGNxYS1oaWRkZW5cIj57eyBhY3Rpb24ubGFiZWwgfX08L3NwYW4+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgPC9kaXY+XG4gICAgPCEtLSBGb3J3YXJkcyBjbGlja3MgaW5zaWRlIHRvb2xiYXIgd2l0aG91dCBhZmZlY3Rpbmcgb3V0ZXIgc2VsZWN0aW9ucyAtLT5cbiAgPC9kaXY+XG48L2Rpdj4iXX0=
@@ -0,0 +1,19 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class TableDataLoaderComponent {
4
+ constructor() {
5
+ this.label = 'Loading...';
6
+ this.size = 24;
7
+ }
8
+ }
9
+ TableDataLoaderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TableDataLoaderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
10
+ TableDataLoaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TableDataLoaderComponent, selector: "cqa-table-data-loader", inputs: { label: "label", size: "size" }, ngImport: i0, template: "<div class=\"loading-spinner-simple cqa-py-2 cqa-flex cqa-flex-row cqa-items-center cqa-justify-center cqa-gap-3 cqa-text-center\">\n <svg class=\"cqa-animate-spin cqa-text-primary\" [attr.width]=\"size\" [attr.height]=\"size\" xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\" viewBox=\"0 0 24 24\">\n <circle class=\"cqa-opacity-25\" cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"4\"></circle>\n <path class=\"cqa-opacity-75\" fill=\"currentColor\" d=\"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z\"></path>\n </svg>\n <span class=\"cqa-text-primary\">{{ label }}</span>\n </div>\n\n\n" });
11
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TableDataLoaderComponent, decorators: [{
12
+ type: Component,
13
+ args: [{ selector: 'cqa-table-data-loader', template: "<div class=\"loading-spinner-simple cqa-py-2 cqa-flex cqa-flex-row cqa-items-center cqa-justify-center cqa-gap-3 cqa-text-center\">\n <svg class=\"cqa-animate-spin cqa-text-primary\" [attr.width]=\"size\" [attr.height]=\"size\" xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\" viewBox=\"0 0 24 24\">\n <circle class=\"cqa-opacity-25\" cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"4\"></circle>\n <path class=\"cqa-opacity-75\" fill=\"currentColor\" d=\"M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z\"></path>\n </svg>\n <span class=\"cqa-text-primary\">{{ label }}</span>\n </div>\n\n\n", styles: [] }]
14
+ }], propDecorators: { label: [{
15
+ type: Input
16
+ }], size: [{
17
+ type: Input
18
+ }] } });
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUtZGF0YS1sb2FkZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90YWJsZS1kYXRhLWxvYWRlci90YWJsZS1kYXRhLWxvYWRlci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3RhYmxlLWRhdGEtbG9hZGVyL3RhYmxlLWRhdGEtbG9hZGVyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU9qRCxNQUFNLE9BQU8sd0JBQXdCO0lBTHJDO1FBTVcsVUFBSyxHQUFXLFlBQVksQ0FBQztRQUM3QixTQUFJLEdBQVcsRUFBRSxDQUFDO0tBQzVCOztxSEFIWSx3QkFBd0I7eUdBQXhCLHdCQUF3Qix1R0NQckMsaXJCQVNBOzJGREZhLHdCQUF3QjtrQkFMcEMsU0FBUzsrQkFDRSx1QkFBdUI7OEJBS3hCLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxJQUFJO3NCQUFaLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NxYS10YWJsZS1kYXRhLWxvYWRlcicsXG4gIHRlbXBsYXRlVXJsOiAnLi90YWJsZS1kYXRhLWxvYWRlci5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogW11cbn0pXG5leHBvcnQgY2xhc3MgVGFibGVEYXRhTG9hZGVyQ29tcG9uZW50IHtcbiAgQElucHV0KCkgbGFiZWw6IHN0cmluZyA9ICdMb2FkaW5nLi4uJztcbiAgQElucHV0KCkgc2l6ZTogbnVtYmVyID0gMjQ7XG59XG5cblxuIiwiPGRpdiBjbGFzcz1cImxvYWRpbmctc3Bpbm5lci1zaW1wbGUgY3FhLXB5LTIgY3FhLWZsZXggY3FhLWZsZXgtcm93IGNxYS1pdGVtcy1jZW50ZXIgY3FhLWp1c3RpZnktY2VudGVyIGNxYS1nYXAtMyBjcWEtdGV4dC1jZW50ZXJcIj5cbiAgPHN2ZyBjbGFzcz1cImNxYS1hbmltYXRlLXNwaW4gY3FhLXRleHQtcHJpbWFyeVwiIFthdHRyLndpZHRoXT1cInNpemVcIiBbYXR0ci5oZWlnaHRdPVwic2l6ZVwiIHhtbG5zPVwiaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmdcIiBmaWxsPVwibm9uZVwiIHZpZXdCb3g9XCIwIDAgMjQgMjRcIj5cbiAgICA8Y2lyY2xlIGNsYXNzPVwiY3FhLW9wYWNpdHktMjVcIiBjeD1cIjEyXCIgY3k9XCIxMlwiIHI9XCIxMFwiIHN0cm9rZT1cImN1cnJlbnRDb2xvclwiIHN0cm9rZS13aWR0aD1cIjRcIj48L2NpcmNsZT5cbiAgICA8cGF0aCBjbGFzcz1cImNxYS1vcGFjaXR5LTc1XCIgZmlsbD1cImN1cnJlbnRDb2xvclwiIGQ9XCJNNCAxMmE4IDggMCAwMTgtOFYwQzUuMzczIDAgMCA1LjM3MyAwIDEyaDR6bTIgNS4yOTFBNy45NjIgNy45NjIgMCAwMTQgMTJIMGMwIDMuMDQyIDEuMTM1IDUuODI0IDMgNy45MzhsMy0yLjY0N3pcIj48L3BhdGg+XG4gIDwvc3ZnPlxuICA8c3BhbiBjbGFzcz1cImNxYS10ZXh0LXByaW1hcnlcIj57eyBsYWJlbCB9fTwvc3Bhbj5cbiAgPC9kaXY+XG5cblxuIl19