@cqa-lib/cqa-ui 0.1.1 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/action-menu/action-menu.component.mjs +42 -0
- package/esm2020/lib/assets/images/image-assets.constants.mjs +28 -0
- package/esm2020/lib/badge/badge.component.mjs +141 -0
- package/esm2020/lib/button/button.component.mjs +232 -0
- package/esm2020/lib/column-visibility/column-visibility.component.mjs +69 -0
- package/esm2020/lib/dashboards/chart-card/chart-card.component.mjs +22 -0
- package/esm2020/lib/dashboards/coverage-module-card/coverage-module-card.component.mjs +104 -0
- package/esm2020/lib/dashboards/dashboard-header/dashboard-header.component.mjs +82 -0
- package/esm2020/lib/dashboards/failed-test-cases-card/failed-test-cases-card.component.mjs +60 -0
- package/esm2020/lib/dashboards/heat-error-map-cell/heat-error-map-cell.component.mjs +45 -0
- package/esm2020/lib/dashboards/insight-card/insight-card.component.mjs +201 -0
- package/esm2020/lib/dashboards/metrics-card/metrics-block.component.mjs +41 -0
- package/esm2020/lib/dashboards/metrics-card/metrics-card-item.interface.mjs +2 -0
- package/esm2020/lib/dashboards/metrics-card/metrics-card.component.mjs +62 -0
- package/esm2020/lib/dashboards/progress-text-card/progress-text-card.component.mjs +46 -0
- package/esm2020/lib/dashboards/test-distribution-card/test-distribution-card.component.mjs +35 -0
- package/esm2020/lib/dialog/dialog.component.mjs +127 -0
- package/esm2020/lib/dropdown-button/dropdown-button.component.mjs +189 -0
- package/esm2020/lib/dynamic-select/dynamic-select-field.component.mjs +160 -0
- package/esm2020/lib/empty-state/empty-state.component.mjs +37 -0
- package/esm2020/lib/filters/dynamic-filter/dynamic-filter.component.mjs +239 -0
- package/esm2020/lib/full-table-loader/full-table-loader.component.mjs +16 -0
- package/esm2020/lib/inline-sort/inline-sort.component.mjs +58 -0
- package/esm2020/lib/other-button/other-button.component.mjs +76 -0
- package/esm2020/lib/pagination/pagination.component.mjs +102 -0
- package/{dist/cqa-ui/esm2020 → esm2020}/lib/search-bar/search-bar.component.mjs +3 -3
- package/{dist/cqa-ui/esm2020 → esm2020}/lib/segment-control/segment-control.component.mjs +3 -3
- package/esm2020/lib/selected-filters/selected-filters.component.mjs +27 -0
- package/esm2020/lib/table/dynamic-table/dynamic-cell.directive.mjs +35 -0
- package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +258 -0
- package/esm2020/lib/table-action-toolbar/table-action-toolbar.component.mjs +52 -0
- package/esm2020/lib/table-data-loader/table-data-loader.component.mjs +19 -0
- package/esm2020/lib/templates/table-template.component.mjs +365 -0
- package/esm2020/lib/ui-kit.module.mjs +248 -0
- package/esm2020/lib/utils/metadata-colors.util.mjs +100 -0
- package/esm2020/lib/utils/tw-overlay-container.mjs +22 -0
- package/esm2020/public-api.mjs +38 -0
- package/fesm2015/cqa-lib-cqa-ui.mjs +3661 -0
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -0
- package/fesm2020/cqa-lib-cqa-ui.mjs +3615 -0
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -0
- package/lib/action-menu/action-menu.component.d.ts +17 -0
- package/lib/assets/images/image-assets.constants.d.ts +20 -0
- package/lib/badge/badge.component.d.ts +25 -0
- package/{dist/cqa-ui/lib → lib}/button/button.component.d.ts +6 -5
- package/lib/column-visibility/column-visibility.component.d.ts +33 -0
- package/lib/dashboards/chart-card/chart-card.component.d.ts +8 -0
- package/lib/dashboards/coverage-module-card/coverage-module-card.component.d.ts +44 -0
- package/lib/dashboards/dashboard-header/dashboard-header.component.d.ts +30 -0
- package/lib/dashboards/failed-test-cases-card/failed-test-cases-card.component.d.ts +28 -0
- package/lib/dashboards/heat-error-map-cell/heat-error-map-cell.component.d.ts +14 -0
- package/lib/dashboards/insight-card/insight-card.component.d.ts +73 -0
- package/lib/dashboards/metrics-card/metrics-block.component.d.ts +12 -0
- package/lib/dashboards/metrics-card/metrics-card-item.interface.d.ts +12 -0
- package/lib/dashboards/metrics-card/metrics-card.component.d.ts +17 -0
- package/lib/dashboards/progress-text-card/progress-text-card.component.d.ts +13 -0
- package/lib/dashboards/test-distribution-card/test-distribution-card.component.d.ts +29 -0
- package/lib/dropdown-button/dropdown-button.component.d.ts +32 -0
- package/lib/dynamic-select/dynamic-select-field.component.d.ts +43 -0
- package/lib/empty-state/empty-state.component.d.ts +20 -0
- package/lib/filters/dynamic-filter/dynamic-filter.component.d.ts +56 -0
- package/lib/full-table-loader/full-table-loader.component.d.ts +6 -0
- package/lib/inline-sort/inline-sort.component.d.ts +12 -0
- package/lib/other-button/other-button.component.d.ts +37 -0
- package/lib/pagination/pagination.component.d.ts +37 -0
- package/lib/selected-filters/selected-filters.component.d.ts +17 -0
- package/lib/table/dynamic-table/dynamic-cell.directive.d.ts +16 -0
- package/lib/table/dynamic-table/dynamic-table.component.d.ts +72 -0
- package/lib/table-action-toolbar/table-action-toolbar.component.d.ts +34 -0
- package/lib/table-data-loader/table-data-loader.component.d.ts +7 -0
- package/lib/templates/table-template.component.d.ts +90 -0
- package/lib/ui-kit.module.d.ts +52 -0
- package/lib/utils/metadata-colors.util.d.ts +50 -0
- package/lib/utils/tw-overlay-container.d.ts +12 -0
- package/package.json +23 -49
- package/public-api.d.ts +37 -0
- package/src/lib/assets/images/.gitkeep +0 -0
- package/src/lib/assets/images/DashboardIcon.png +0 -0
- package/src/lib/assets/images/FilesIcon.png +0 -0
- package/src/lib/assets/images/README.md +66 -0
- package/src/lib/assets/images/ReportsIcon.png +0 -0
- package/src/lib/assets/images/SearchIcon.png +0 -0
- package/src/lib/assets/images/StepsIcon.png +0 -0
- package/src/lib/assets/images/TestCaseIcon.png +0 -0
- package/src/lib/assets/images/analytics-chart-icon.svg +11 -0
- package/src/lib/assets/images/checklist-add-icon.svg +10 -0
- package/src/lib/assets/images/document-gear-icon.svg +9 -0
- package/src/lib/assets/images/empty-state-default-icon.svg +8 -0
- package/src/lib/assets/images/image-assets.constants.ts +38 -0
- package/src/lib/assets/images/search-debug-icon.svg +8 -0
- package/src/lib/assets/images/test-case-icon.svg +9 -0
- package/src/lib/assets/images/upload-folder-icon.svg +7 -0
- package/src/lib/utils/metadata-colors.constants.js +33 -0
- package/storybook-static/assets/images/README.md +66 -0
- package/styles.css +1 -0
- package/dist/cqa-ui/README.md +0 -226
- package/dist/cqa-ui/esm2020/lib/button/button.component.mjs +0 -257
- package/dist/cqa-ui/esm2020/lib/dialog/dialog.component.mjs +0 -127
- package/dist/cqa-ui/esm2020/lib/ui-kit.module.mjs +0 -69
- package/dist/cqa-ui/esm2020/public-api.mjs +0 -10
- package/dist/cqa-ui/fesm2015/cqa-lib-cqa-ui.mjs +0 -895
- package/dist/cqa-ui/fesm2015/cqa-lib-cqa-ui.mjs.map +0 -1
- package/dist/cqa-ui/fesm2020/cqa-lib-cqa-ui.mjs +0 -881
- package/dist/cqa-ui/fesm2020/cqa-lib-cqa-ui.mjs.map +0 -1
- package/dist/cqa-ui/lib/ui-kit.module.d.ts +0 -15
- package/dist/cqa-ui/package.json +0 -56
- package/dist/cqa-ui/public-api.d.ts +0 -9
- package/dist/cqa-ui/styles.css +0 -1
- /package/{dist/cqa-ui/cqa-lib-cqa-ui.d.ts → cqa-lib-cqa-ui.d.ts} +0 -0
- /package/{dist/cqa-ui/esm2020 → esm2020}/cqa-lib-cqa-ui.mjs +0 -0
- /package/{dist/cqa-ui/esm2020 → esm2020}/lib/dialog/dialog-ref.mjs +0 -0
- /package/{dist/cqa-ui/esm2020 → esm2020}/lib/dialog/dialog.models.mjs +0 -0
- /package/{dist/cqa-ui/esm2020 → esm2020}/lib/dialog/dialog.service.mjs +0 -0
- /package/{dist/cqa-ui/esm2020 → esm2020}/lib/dialog/dialog.tokens.mjs +0 -0
- /package/{dist/cqa-ui/lib → lib}/dialog/dialog-ref.d.ts +0 -0
- /package/{dist/cqa-ui/lib → lib}/dialog/dialog.component.d.ts +0 -0
- /package/{dist/cqa-ui/lib → lib}/dialog/dialog.models.d.ts +0 -0
- /package/{dist/cqa-ui/lib → lib}/dialog/dialog.service.d.ts +0 -0
- /package/{dist/cqa-ui/lib → lib}/dialog/dialog.tokens.d.ts +0 -0
- /package/{dist/cqa-ui/lib → lib}/search-bar/search-bar.component.d.ts +0 -0
- /package/{dist/cqa-ui/lib → lib}/segment-control/segment-control.component.d.ts +0 -0
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "../button/button.component";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
export class EmptyStateComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.title = '';
|
|
8
|
+
this.description = '';
|
|
9
|
+
this.actions = [];
|
|
10
|
+
this.actionClick = new EventEmitter();
|
|
11
|
+
}
|
|
12
|
+
onActionClick(action, event) {
|
|
13
|
+
if (!action.disabled) {
|
|
14
|
+
if (action.onClick) {
|
|
15
|
+
action.onClick();
|
|
16
|
+
}
|
|
17
|
+
this.actionClick.emit(action);
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
EmptyStateComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: EmptyStateComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
22
|
+
EmptyStateComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: EmptyStateComponent, selector: "cqa-empty-state", inputs: { imageUrl: "imageUrl", title: "title", description: "description", actions: "actions" }, outputs: { actionClick: "actionClick" }, ngImport: i0, template: "<div id=\"cqa-ui-root\" style=\"background-color: white; border-radius: 14px; display: flex; flex-direction: column; align-items: center; justify-content: center; padding: 84.63px 33px 84.62px 33px;\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[62px] cqa-items-center\">\n <!-- Icon Container -->\n <div *ngIf=\"imageUrl\" class=\"cqa-relative cqa-shrink-0 cqa-w-32 cqa-h-32\">\n <!-- Main Icon Container with Gradient Background and Shadow -->\n <div class=\"cqa-relative cqa-rounded-3xl cqa-w-32 cqa-h-32 cqa-shadow-sm\">\n <div class=\"cqa-absolute cqa-inset-0 cqa-bg-gradient-to-br cqa-from-indigo-500 cqa-to-violet-950 cqa-rounded-3xl cqa-opacity-10\"></div>\n <!-- Icon/Image centered inside on top layer - fully opaque -->\n <div class=\"cqa-absolute cqa-inset-0 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-3xl\">\n <div class=\"cqa-w-20 cqa-h-20 cqa-flex cqa-items-center cqa-justify-center cqa-relative\">\n <img [src]=\"imageUrl\" alt=\"\" width=\"80px\" height=\"80px\" class=\"cqa-block cqa-max-w-none cqa-w-20 cqa-h-20 cqa-object-contain\" />\n </div>\n </div>\n </div>\n <!-- Decorative Dots -->\n <div class=\"cqa-absolute cqa-rounded-full cqa-bg-primary-300 cqa-opacity-[0.815]\" style=\"left: 120.79px; top: -9.21px; width: 18.416px; height: 18.416px; z-index: 20;\"></div>\n <div class=\"cqa-absolute cqa-rounded-full cqa-bg-primary-300 cqa-opacity-[0.695]\" style=\"left: -9.02px; top: 124.98px; width: 14.044px; height: 14.044px; z-index: 20;\"></div>\n </div>\n\n <!-- Content Container -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-9 cqa-items-center\">\n <!-- Title and Description -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-items-center cqa-w-full\">\n <!-- Title -->\n <div *ngIf=\"title\" class=\"cqa-flex cqa-flex-col cqa-items-center cqa-w-full\">\n <h3 class=\"cqa-font-inter cqa-text-lg cqa-font-medium cqa-leading-[18px] cqa-text-center cqa-text-neutral-900\">\n {{ title }}\n </h3>\n </div>\n <!-- Description -->\n <div *ngIf=\"description\" class=\"cqa-flex cqa-flex-col cqa-items-center cqa-w-full\">\n <p class=\"cqa-font-inter cqa-font-medium cqa-text-sm cqa-leading-[14px] cqa-text-center cqa-text-neutral-500\">\n {{ description }}\n </p>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div\n *ngIf=\"actions && actions.length > 0\"\n class=\"cqa-flex cqa-items-center cqa-justify-center\"\n [ngClass]=\"actions.length > 1 ? 'cqa-flex-row cqa-flex-wrap cqa-gap-4' : 'cqa-flex-col cqa-gap-2'\"\n >\n <cqa-button\n *ngFor=\"let action of actions\"\n [variant]=\"action.variant || 'filled'\"\n [icon]=\"action.icon\"\n [iconPosition]=\"action.iconPosition || 'start'\"\n [disabled]=\"action.disabled\"\n (clicked)=\"onActionClick(action, $event)\"\n >\n {{ action.label }}\n </cqa-button>\n </div>\n </div>\n </div>\n</div>\n\n", components: [{ type: i1.ButtonComponent, selector: "cqa-button", inputs: ["variant", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass"], outputs: ["clicked"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
23
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: EmptyStateComponent, decorators: [{
|
|
24
|
+
type: Component,
|
|
25
|
+
args: [{ selector: 'cqa-empty-state', template: "<div id=\"cqa-ui-root\" style=\"background-color: white; border-radius: 14px; display: flex; flex-direction: column; align-items: center; justify-content: center; padding: 84.63px 33px 84.62px 33px;\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[62px] cqa-items-center\">\n <!-- Icon Container -->\n <div *ngIf=\"imageUrl\" class=\"cqa-relative cqa-shrink-0 cqa-w-32 cqa-h-32\">\n <!-- Main Icon Container with Gradient Background and Shadow -->\n <div class=\"cqa-relative cqa-rounded-3xl cqa-w-32 cqa-h-32 cqa-shadow-sm\">\n <div class=\"cqa-absolute cqa-inset-0 cqa-bg-gradient-to-br cqa-from-indigo-500 cqa-to-violet-950 cqa-rounded-3xl cqa-opacity-10\"></div>\n <!-- Icon/Image centered inside on top layer - fully opaque -->\n <div class=\"cqa-absolute cqa-inset-0 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-3xl\">\n <div class=\"cqa-w-20 cqa-h-20 cqa-flex cqa-items-center cqa-justify-center cqa-relative\">\n <img [src]=\"imageUrl\" alt=\"\" width=\"80px\" height=\"80px\" class=\"cqa-block cqa-max-w-none cqa-w-20 cqa-h-20 cqa-object-contain\" />\n </div>\n </div>\n </div>\n <!-- Decorative Dots -->\n <div class=\"cqa-absolute cqa-rounded-full cqa-bg-primary-300 cqa-opacity-[0.815]\" style=\"left: 120.79px; top: -9.21px; width: 18.416px; height: 18.416px; z-index: 20;\"></div>\n <div class=\"cqa-absolute cqa-rounded-full cqa-bg-primary-300 cqa-opacity-[0.695]\" style=\"left: -9.02px; top: 124.98px; width: 14.044px; height: 14.044px; z-index: 20;\"></div>\n </div>\n\n <!-- Content Container -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-9 cqa-items-center\">\n <!-- Title and Description -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-items-center cqa-w-full\">\n <!-- Title -->\n <div *ngIf=\"title\" class=\"cqa-flex cqa-flex-col cqa-items-center cqa-w-full\">\n <h3 class=\"cqa-font-inter cqa-text-lg cqa-font-medium cqa-leading-[18px] cqa-text-center cqa-text-neutral-900\">\n {{ title }}\n </h3>\n </div>\n <!-- Description -->\n <div *ngIf=\"description\" class=\"cqa-flex cqa-flex-col cqa-items-center cqa-w-full\">\n <p class=\"cqa-font-inter cqa-font-medium cqa-text-sm cqa-leading-[14px] cqa-text-center cqa-text-neutral-500\">\n {{ description }}\n </p>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div\n *ngIf=\"actions && actions.length > 0\"\n class=\"cqa-flex cqa-items-center cqa-justify-center\"\n [ngClass]=\"actions.length > 1 ? 'cqa-flex-row cqa-flex-wrap cqa-gap-4' : 'cqa-flex-col cqa-gap-2'\"\n >\n <cqa-button\n *ngFor=\"let action of actions\"\n [variant]=\"action.variant || 'filled'\"\n [icon]=\"action.icon\"\n [iconPosition]=\"action.iconPosition || 'start'\"\n [disabled]=\"action.disabled\"\n (clicked)=\"onActionClick(action, $event)\"\n >\n {{ action.label }}\n </cqa-button>\n </div>\n </div>\n </div>\n</div>\n\n", styles: [] }]
|
|
26
|
+
}], propDecorators: { imageUrl: [{
|
|
27
|
+
type: Input
|
|
28
|
+
}], title: [{
|
|
29
|
+
type: Input
|
|
30
|
+
}], description: [{
|
|
31
|
+
type: Input
|
|
32
|
+
}], actions: [{
|
|
33
|
+
type: Input
|
|
34
|
+
}], actionClick: [{
|
|
35
|
+
type: Output
|
|
36
|
+
}] } });
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW1wdHktc3RhdGUuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9lbXB0eS1zdGF0ZS9lbXB0eS1zdGF0ZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2VtcHR5LXN0YXRlL2VtcHR5LXN0YXRlLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxZQUFZLEVBQUUsTUFBTSxlQUFlLENBQUM7Ozs7QUFnQnZFLE1BQU0sT0FBTyxtQkFBbUI7SUFMaEM7UUFPVyxVQUFLLEdBQVcsRUFBRSxDQUFDO1FBQ25CLGdCQUFXLEdBQVcsRUFBRSxDQUFDO1FBQ3pCLFlBQU8sR0FBdUIsRUFBRSxDQUFDO1FBRWhDLGdCQUFXLEdBQUcsSUFBSSxZQUFZLEVBQW9CLENBQUM7S0FVOUQ7SUFSQyxhQUFhLENBQUMsTUFBd0IsRUFBRSxLQUFpQjtRQUN2RCxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRTtZQUNwQixJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUU7Z0JBQ2xCLE1BQU0sQ0FBQyxPQUFPLEVBQUUsQ0FBQzthQUNsQjtZQUNELElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1NBQy9CO0lBQ0gsQ0FBQzs7Z0hBZlUsbUJBQW1CO29HQUFuQixtQkFBbUIsa01DaEJoQyx3a0dBMERBOzJGRDFDYSxtQkFBbUI7a0JBTC9CLFNBQVM7K0JBQ0UsaUJBQWlCOzhCQUtsQixRQUFRO3NCQUFoQixLQUFLO2dCQUNHLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxXQUFXO3NCQUFuQixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFFSSxXQUFXO3NCQUFwQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT3V0cHV0LCBFdmVudEVtaXR0ZXIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IGludGVyZmFjZSBFbXB0eVN0YXRlQWN0aW9uIHtcbiAgbGFiZWw6IHN0cmluZztcbiAgdmFyaWFudD86ICdmaWxsZWQnIHwgJ291dGxpbmVkJyB8ICd0ZXh0JyB8ICdlbGV2YXRlZCcgfCAndG9uYWwnO1xuICBpY29uPzogc3RyaW5nO1xuICBpY29uUG9zaXRpb24/OiAnc3RhcnQnIHwgJ2VuZCc7XG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgb25DbGljaz86ICgpID0+IHZvaWQ7XG59XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NxYS1lbXB0eS1zdGF0ZScsXG4gIHRlbXBsYXRlVXJsOiAnLi9lbXB0eS1zdGF0ZS5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogW11cbn0pXG5leHBvcnQgY2xhc3MgRW1wdHlTdGF0ZUNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIGltYWdlVXJsPzogc3RyaW5nOyAvLyBVUkwgZm9yIGltYWdlXG4gIEBJbnB1dCgpIHRpdGxlOiBzdHJpbmcgPSAnJztcbiAgQElucHV0KCkgZGVzY3JpcHRpb246IHN0cmluZyA9ICcnO1xuICBASW5wdXQoKSBhY3Rpb25zOiBFbXB0eVN0YXRlQWN0aW9uW10gPSBbXTtcblxuICBAT3V0cHV0KCkgYWN0aW9uQ2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyPEVtcHR5U3RhdGVBY3Rpb24+KCk7XG5cbiAgb25BY3Rpb25DbGljayhhY3Rpb246IEVtcHR5U3RhdGVBY3Rpb24sIGV2ZW50OiBNb3VzZUV2ZW50KSB7XG4gICAgaWYgKCFhY3Rpb24uZGlzYWJsZWQpIHtcbiAgICAgIGlmIChhY3Rpb24ub25DbGljaykge1xuICAgICAgICBhY3Rpb24ub25DbGljaygpO1xuICAgICAgfVxuICAgICAgdGhpcy5hY3Rpb25DbGljay5lbWl0KGFjdGlvbik7XG4gICAgfVxuICB9XG59XG5cbiIsIjxkaXYgaWQ9XCJjcWEtdWktcm9vdFwiIHN0eWxlPVwiYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7IGJvcmRlci1yYWRpdXM6IDE0cHg7IGRpc3BsYXk6IGZsZXg7IGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47IGFsaWduLWl0ZW1zOiBjZW50ZXI7IGp1c3RpZnktY29udGVudDogY2VudGVyOyBwYWRkaW5nOiA4NC42M3B4IDMzcHggODQuNjJweCAzM3B4O1wiPlxuICA8ZGl2IGNsYXNzPVwiY3FhLWZsZXggY3FhLWZsZXgtY29sIGNxYS1nYXAtWzYycHhdIGNxYS1pdGVtcy1jZW50ZXJcIj5cbiAgICA8IS0tIEljb24gQ29udGFpbmVyIC0tPlxuICAgIDxkaXYgKm5nSWY9XCJpbWFnZVVybFwiIGNsYXNzPVwiY3FhLXJlbGF0aXZlIGNxYS1zaHJpbmstMCBjcWEtdy0zMiBjcWEtaC0zMlwiPlxuICAgICAgPCEtLSBNYWluIEljb24gQ29udGFpbmVyIHdpdGggR3JhZGllbnQgQmFja2dyb3VuZCBhbmQgU2hhZG93IC0tPlxuICAgICAgPGRpdiBjbGFzcz1cImNxYS1yZWxhdGl2ZSBjcWEtcm91bmRlZC0zeGwgY3FhLXctMzIgY3FhLWgtMzIgY3FhLXNoYWRvdy1zbVwiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwiY3FhLWFic29sdXRlIGNxYS1pbnNldC0wIGNxYS1iZy1ncmFkaWVudC10by1iciBjcWEtZnJvbS1pbmRpZ28tNTAwIGNxYS10by12aW9sZXQtOTUwIGNxYS1yb3VuZGVkLTN4bCBjcWEtb3BhY2l0eS0xMFwiPjwvZGl2PlxuICAgICAgICA8IS0tIEljb24vSW1hZ2UgY2VudGVyZWQgaW5zaWRlIG9uIHRvcCBsYXllciAtIGZ1bGx5IG9wYXF1ZSAtLT5cbiAgICAgICAgPGRpdiBjbGFzcz1cImNxYS1hYnNvbHV0ZSBjcWEtaW5zZXQtMCBjcWEtZmxleCBjcWEtaXRlbXMtY2VudGVyIGNxYS1qdXN0aWZ5LWNlbnRlciBjcWEtcm91bmRlZC0zeGxcIj5cbiAgICAgICAgICA8ZGl2IGNsYXNzPVwiY3FhLXctMjAgY3FhLWgtMjAgY3FhLWZsZXggY3FhLWl0ZW1zLWNlbnRlciBjcWEtanVzdGlmeS1jZW50ZXIgY3FhLXJlbGF0aXZlXCI+XG4gICAgICAgICAgICA8aW1nIFtzcmNdPVwiaW1hZ2VVcmxcIiBhbHQ9XCJcIiB3aWR0aD1cIjgwcHhcIiBoZWlnaHQ9XCI4MHB4XCIgY2xhc3M9XCJjcWEtYmxvY2sgY3FhLW1heC13LW5vbmUgY3FhLXctMjAgY3FhLWgtMjAgY3FhLW9iamVjdC1jb250YWluXCIgLz5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICAgIDwhLS0gRGVjb3JhdGl2ZSBEb3RzIC0tPlxuICAgICAgPGRpdiBjbGFzcz1cImNxYS1hYnNvbHV0ZSBjcWEtcm91bmRlZC1mdWxsIGNxYS1iZy1wcmltYXJ5LTMwMCBjcWEtb3BhY2l0eS1bMC44MTVdXCIgc3R5bGU9XCJsZWZ0OiAxMjAuNzlweDsgdG9wOiAtOS4yMXB4OyB3aWR0aDogMTguNDE2cHg7IGhlaWdodDogMTguNDE2cHg7IHotaW5kZXg6IDIwO1wiPjwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cImNxYS1hYnNvbHV0ZSBjcWEtcm91bmRlZC1mdWxsIGNxYS1iZy1wcmltYXJ5LTMwMCBjcWEtb3BhY2l0eS1bMC42OTVdXCIgc3R5bGU9XCJsZWZ0OiAtOS4wMnB4OyB0b3A6IDEyNC45OHB4OyB3aWR0aDogMTQuMDQ0cHg7IGhlaWdodDogMTQuMDQ0cHg7IHotaW5kZXg6IDIwO1wiPjwvZGl2PlxuICAgIDwvZGl2PlxuXG4gICAgPCEtLSBDb250ZW50IENvbnRhaW5lciAtLT5cbiAgICA8ZGl2IGNsYXNzPVwiY3FhLWZsZXggY3FhLWZsZXgtY29sIGNxYS1nYXAtOSBjcWEtaXRlbXMtY2VudGVyXCI+XG4gICAgICA8IS0tIFRpdGxlIGFuZCBEZXNjcmlwdGlvbiAtLT5cbiAgICAgIDxkaXYgY2xhc3M9XCJjcWEtZmxleCBjcWEtZmxleC1jb2wgY3FhLWdhcC0zIGNxYS1pdGVtcy1jZW50ZXIgY3FhLXctZnVsbFwiPlxuICAgICAgICA8IS0tIFRpdGxlIC0tPlxuICAgICAgICA8ZGl2ICpuZ0lmPVwidGl0bGVcIiBjbGFzcz1cImNxYS1mbGV4IGNxYS1mbGV4LWNvbCBjcWEtaXRlbXMtY2VudGVyIGNxYS13LWZ1bGxcIj5cbiAgICAgICAgICA8aDMgY2xhc3M9XCJjcWEtZm9udC1pbnRlciBjcWEtdGV4dC1sZyBjcWEtZm9udC1tZWRpdW0gY3FhLWxlYWRpbmctWzE4cHhdIGNxYS10ZXh0LWNlbnRlciBjcWEtdGV4dC1uZXV0cmFsLTkwMFwiPlxuICAgICAgICAgICAge3sgdGl0bGUgfX1cbiAgICAgICAgICA8L2gzPlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgPCEtLSBEZXNjcmlwdGlvbiAtLT5cbiAgICAgICAgPGRpdiAqbmdJZj1cImRlc2NyaXB0aW9uXCIgY2xhc3M9XCJjcWEtZmxleCBjcWEtZmxleC1jb2wgY3FhLWl0ZW1zLWNlbnRlciBjcWEtdy1mdWxsXCI+XG4gICAgICAgICAgPHAgY2xhc3M9XCJjcWEtZm9udC1pbnRlciBjcWEtZm9udC1tZWRpdW0gY3FhLXRleHQtc20gY3FhLWxlYWRpbmctWzE0cHhdIGNxYS10ZXh0LWNlbnRlciBjcWEtdGV4dC1uZXV0cmFsLTUwMFwiPlxuICAgICAgICAgICAge3sgZGVzY3JpcHRpb24gfX1cbiAgICAgICAgICA8L3A+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG5cbiAgICAgIDwhLS0gQWN0aW9uIEJ1dHRvbnMgLS0+XG4gICAgICA8ZGl2XG4gICAgICAgICpuZ0lmPVwiYWN0aW9ucyAmJiBhY3Rpb25zLmxlbmd0aCA+IDBcIlxuICAgICAgICBjbGFzcz1cImNxYS1mbGV4IGNxYS1pdGVtcy1jZW50ZXIgY3FhLWp1c3RpZnktY2VudGVyXCJcbiAgICAgICAgW25nQ2xhc3NdPVwiYWN0aW9ucy5sZW5ndGggPiAxID8gJ2NxYS1mbGV4LXJvdyBjcWEtZmxleC13cmFwIGNxYS1nYXAtNCcgOiAnY3FhLWZsZXgtY29sIGNxYS1nYXAtMidcIlxuICAgICAgPlxuICAgICAgICA8Y3FhLWJ1dHRvblxuICAgICAgICAgICpuZ0Zvcj1cImxldCBhY3Rpb24gb2YgYWN0aW9uc1wiXG4gICAgICAgICAgW3ZhcmlhbnRdPVwiYWN0aW9uLnZhcmlhbnQgfHwgJ2ZpbGxlZCdcIlxuICAgICAgICAgIFtpY29uXT1cImFjdGlvbi5pY29uXCJcbiAgICAgICAgICBbaWNvblBvc2l0aW9uXT1cImFjdGlvbi5pY29uUG9zaXRpb24gfHwgJ3N0YXJ0J1wiXG4gICAgICAgICAgW2Rpc2FibGVkXT1cImFjdGlvbi5kaXNhYmxlZFwiXG4gICAgICAgICAgKGNsaWNrZWQpPVwib25BY3Rpb25DbGljayhhY3Rpb24sICRldmVudClcIlxuICAgICAgICA+XG4gICAgICAgICAge3sgYWN0aW9uLmxhYmVsIH19XG4gICAgICAgIDwvY3FhLWJ1dHRvbj5cbiAgICAgIDwvZGl2PlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbjwvZGl2PlxuXG4iXX0=
|
|
@@ -0,0 +1,239 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/forms";
|
|
5
|
+
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
6
|
+
import * as i3 from "@angular/material/form-field";
|
|
7
|
+
import * as i4 from "@angular/material/datepicker";
|
|
8
|
+
import * as i5 from "@angular/material/button";
|
|
9
|
+
import * as i6 from "@angular/common";
|
|
10
|
+
export class DynamicFilterComponent {
|
|
11
|
+
constructor(fb) {
|
|
12
|
+
this.fb = fb;
|
|
13
|
+
this.config = [];
|
|
14
|
+
this.model = {};
|
|
15
|
+
this.showFilterPanel = true;
|
|
16
|
+
this.filtersApplied = new EventEmitter();
|
|
17
|
+
this.filtersChanged = new EventEmitter();
|
|
18
|
+
this.resetAction = new EventEmitter();
|
|
19
|
+
this.form = this.fb.group({});
|
|
20
|
+
this.maxDate = new Date();
|
|
21
|
+
this.searchTextByKey = {};
|
|
22
|
+
this.selectOutsideCleanup = new Map();
|
|
23
|
+
}
|
|
24
|
+
ngOnChanges(changes) {
|
|
25
|
+
if (changes['config'] || changes['model']) {
|
|
26
|
+
this.buildForm();
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
onDateChange(event, key) {
|
|
30
|
+
const formGroup = this.getDateGroup(key);
|
|
31
|
+
const start = formGroup.get('start')?.value;
|
|
32
|
+
const end = formGroup.get('end')?.value;
|
|
33
|
+
// When both dates are selected, auto-apply
|
|
34
|
+
// if (start && end) {
|
|
35
|
+
// this.applyDateRange(key, { start, end });
|
|
36
|
+
// }
|
|
37
|
+
}
|
|
38
|
+
buildForm() {
|
|
39
|
+
const ctrls = {};
|
|
40
|
+
(this.config || []).forEach(c => {
|
|
41
|
+
if (c.hidden)
|
|
42
|
+
return;
|
|
43
|
+
if (c.type === 'date-range') {
|
|
44
|
+
ctrls[c.key] = this.fb.group({ start: new FormControl(), end: new FormControl() });
|
|
45
|
+
}
|
|
46
|
+
else {
|
|
47
|
+
const initial = this.model?.[c.key];
|
|
48
|
+
if (c.multiple) {
|
|
49
|
+
const value = Array.isArray(initial) ? initial : (initial != null ? [initial] : []);
|
|
50
|
+
ctrls[c.key] = new FormControl(value);
|
|
51
|
+
}
|
|
52
|
+
else {
|
|
53
|
+
ctrls[c.key] = new FormControl(Array.isArray(initial) ? (initial.length ? initial[0] : undefined) : initial);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
});
|
|
57
|
+
this.form = this.fb.group(ctrls);
|
|
58
|
+
this.form.valueChanges.subscribe(() => this.filtersChanged.emit(this.serialize()));
|
|
59
|
+
}
|
|
60
|
+
onSelectOpenedChange(opened, select) {
|
|
61
|
+
if (opened) {
|
|
62
|
+
setTimeout(() => {
|
|
63
|
+
const onDocDown = (e) => {
|
|
64
|
+
const panel = document.querySelector('.cdk-overlay-pane .mat-select-panel');
|
|
65
|
+
const target = e.target;
|
|
66
|
+
const originEl = select?._elementRef?.nativeElement || null;
|
|
67
|
+
const insidePanel = !!(panel && target && panel.contains(target));
|
|
68
|
+
const insideOrigin = !!(originEl && target && originEl.contains(target));
|
|
69
|
+
if (!insidePanel && !insideOrigin) {
|
|
70
|
+
select.close();
|
|
71
|
+
}
|
|
72
|
+
};
|
|
73
|
+
document.addEventListener('mousedown', onDocDown, true);
|
|
74
|
+
this.selectOutsideCleanup.set(select, () => document.removeEventListener('mousedown', onDocDown, true));
|
|
75
|
+
}, 0);
|
|
76
|
+
}
|
|
77
|
+
else {
|
|
78
|
+
const cleanup = this.selectOutsideCleanup.get(select);
|
|
79
|
+
if (cleanup)
|
|
80
|
+
cleanup();
|
|
81
|
+
this.selectOutsideCleanup.delete(select);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
onSearch(key, text) {
|
|
85
|
+
this.searchTextByKey[key] = (text || '').toLowerCase();
|
|
86
|
+
}
|
|
87
|
+
filteredOptions(item) {
|
|
88
|
+
const options = item?.options || [];
|
|
89
|
+
const q = (this.searchTextByKey[item.key] || '').trim();
|
|
90
|
+
if (!q)
|
|
91
|
+
return options;
|
|
92
|
+
return options.filter(opt => {
|
|
93
|
+
const name = (opt.name ?? opt.label ?? String(opt.value ?? '')).toLowerCase();
|
|
94
|
+
return name.includes(q);
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
getDateGroup(key) {
|
|
98
|
+
return this.form.get(key);
|
|
99
|
+
}
|
|
100
|
+
getSelectConfig(item) {
|
|
101
|
+
return {
|
|
102
|
+
key: item.key,
|
|
103
|
+
label: item.label,
|
|
104
|
+
placeholder: item.placeholder,
|
|
105
|
+
disabled: item.disabled,
|
|
106
|
+
multiple: item.multiple,
|
|
107
|
+
searchable: item.searchable,
|
|
108
|
+
options: item.options || []
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
apply() {
|
|
112
|
+
this.filtersApplied.emit(this.serialize());
|
|
113
|
+
}
|
|
114
|
+
reset() {
|
|
115
|
+
Object.keys(this.form.controls).forEach(key => {
|
|
116
|
+
const ctrl = this.form.get(key);
|
|
117
|
+
if (ctrl instanceof FormGroup) {
|
|
118
|
+
ctrl.get('start')?.setValue(undefined);
|
|
119
|
+
ctrl.get('end')?.setValue(undefined);
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
ctrl?.setValue(undefined);
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
this.resetAction.emit();
|
|
126
|
+
this.filtersChanged.emit(this.serialize());
|
|
127
|
+
}
|
|
128
|
+
serialize() {
|
|
129
|
+
const result = {};
|
|
130
|
+
(this.config || []).forEach(c => {
|
|
131
|
+
const ctrl = this.form.get(c.key);
|
|
132
|
+
if (!ctrl)
|
|
133
|
+
return;
|
|
134
|
+
let val = ctrl instanceof FormGroup ? ctrl.getRawValue() : ctrl.value;
|
|
135
|
+
if (c.type === 'date-range') {
|
|
136
|
+
const start = val?.start;
|
|
137
|
+
const end = val?.end;
|
|
138
|
+
if (start || end)
|
|
139
|
+
result[c.key] = { start, end };
|
|
140
|
+
}
|
|
141
|
+
else {
|
|
142
|
+
if (val !== undefined && val !== null && (Array.isArray(val) ? val.length > 0 : val !== '')) {
|
|
143
|
+
result[c.key] = val;
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
return result;
|
|
148
|
+
}
|
|
149
|
+
// Mat date range picker overlay preset helpers
|
|
150
|
+
applyPresetToGroup(key, presetKey) {
|
|
151
|
+
const dateGroup = this.getDateGroup(key);
|
|
152
|
+
if (!dateGroup)
|
|
153
|
+
return;
|
|
154
|
+
const { start, end } = this.getPresetDates(presetKey);
|
|
155
|
+
dateGroup.patchValue({ start, end });
|
|
156
|
+
this.filtersChanged.emit(this.serialize());
|
|
157
|
+
}
|
|
158
|
+
getPresetDates(presetKey) {
|
|
159
|
+
const today = new Date();
|
|
160
|
+
const clampToEndOfToday = (d) => { const nd = new Date(d); nd.setHours(23, 59, 59, 999); return nd; };
|
|
161
|
+
switch (presetKey) {
|
|
162
|
+
case 'today':
|
|
163
|
+
return { start: today, end: today };
|
|
164
|
+
case 'last7days': {
|
|
165
|
+
const start = new Date();
|
|
166
|
+
start.setDate(start.getDate() - 6);
|
|
167
|
+
return { start, end: today };
|
|
168
|
+
}
|
|
169
|
+
case 'last30days': {
|
|
170
|
+
const start = new Date();
|
|
171
|
+
start.setDate(start.getDate() - 29);
|
|
172
|
+
return { start, end: today };
|
|
173
|
+
}
|
|
174
|
+
case 'last90days': {
|
|
175
|
+
const start = new Date();
|
|
176
|
+
start.setDate(start.getDate() - 89);
|
|
177
|
+
return { start, end: today };
|
|
178
|
+
}
|
|
179
|
+
case 'thismonth': {
|
|
180
|
+
const start = new Date(today.getFullYear(), today.getMonth(), 1);
|
|
181
|
+
return { start, end: today };
|
|
182
|
+
}
|
|
183
|
+
case 'lastmonth': {
|
|
184
|
+
const start = new Date(today.getFullYear(), today.getMonth() - 1, 1);
|
|
185
|
+
const end = new Date(today.getFullYear(), today.getMonth(), 0);
|
|
186
|
+
return { start, end: clampToEndOfToday(end) };
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
getDateValidationError(key) {
|
|
191
|
+
const dateGroup = this.getDateGroup(key);
|
|
192
|
+
if (!dateGroup)
|
|
193
|
+
return null;
|
|
194
|
+
const startCtrl = dateGroup.controls['start'];
|
|
195
|
+
const endCtrl = dateGroup.controls['end'];
|
|
196
|
+
const startVal = startCtrl?.value;
|
|
197
|
+
const endVal = endCtrl?.value;
|
|
198
|
+
const startParseErr = startCtrl?.errors?.['matDatepickerParse']?.text;
|
|
199
|
+
const endParseErr = endCtrl?.errors?.['matDatepickerParse']?.text;
|
|
200
|
+
if (!startVal && !endVal && !startParseErr && !endParseErr)
|
|
201
|
+
return null;
|
|
202
|
+
if (startParseErr)
|
|
203
|
+
return `Invalid start date format`;
|
|
204
|
+
if (endParseErr)
|
|
205
|
+
return `Invalid end date format`;
|
|
206
|
+
if (startVal && !endVal)
|
|
207
|
+
return `Please select an end date`;
|
|
208
|
+
if (!startVal && endVal)
|
|
209
|
+
return `Please select a start date`;
|
|
210
|
+
if (dateGroup.hasError('matStartDateInvalid'))
|
|
211
|
+
return `Start date must be before end date`;
|
|
212
|
+
if (dateGroup.hasError('matEndDateInvalid'))
|
|
213
|
+
return `End date must be after start date`;
|
|
214
|
+
if (dateGroup.invalid) {
|
|
215
|
+
const config = this.config?.find(c => c.key === key);
|
|
216
|
+
return config ? `${config.label} is invalid` : 'Date range is invalid';
|
|
217
|
+
}
|
|
218
|
+
return null;
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
DynamicFilterComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicFilterComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
222
|
+
DynamicFilterComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DynamicFilterComponent, selector: "cqa-dynamic-filter", inputs: { config: "config", model: "model", showFilterPanel: "showFilterPanel" }, outputs: { filtersApplied: "filtersApplied", filtersChanged: "filtersChanged", resetAction: "resetAction" }, usesOnChanges: true, ngImport: i0, template: "<div id=\"cqa-ui-root\">\n <div class=\"cqa-filter cqa-mb-[28px]\" *ngIf=\"showFilterPanel\">\n <form class=\"ts-form cqa-grid lg:cqa-grid-cols-4 md:cqa-grid-cols-2 cqa-gap-4\" [formGroup]=\"form\"\n (keydown.enter)=\"(false)\" novalidate=\"novalidate\">\n <ng-container *ngFor=\"let c of config\">\n <ng-container *ngIf=\"!c.hidden\">\n <div class=\"form-group cqa-flex cqa-flex-col cqa-gap-2\">\n <!-- Select -->\n <ng-container *ngIf=\"c.type === 'select'\">\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(c)\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- Date Range --> \n <ng-container *ngIf=\"c.type === 'date-range'\">\n <label\n class=\"form-label cqa-text-[#374151] cqa-text-[14px] cqa-font-medium cqa-block cqa-leading-[1.4]\">{{\n c.label }}</label>\n <mat-form-field class=\"mat-date-custom\" appearance=\"fill\">\n <mat-date-range-input [rangePicker]=\"picker\" [formGroup]=\"getDateGroup(c.key)\" [max]=\"maxDate\">\n <input matStartDate formControlName=\"start\" placeholder=\"Start date\" [max]=\"maxDate\" required\n [readonly]=\"true\" (focus)=\"picker.open()\" (click)=\"picker.open()\" />\n <input matEndDate formControlName=\"end\" placeholder=\"End date\" [max]=\"maxDate\" required\n [readonly]=\"true\" (focus)=\"picker.open()\" (click)=\"picker.open()\" />\n </mat-date-range-input>\n <mat-date-range-picker #picker (dateChange)=\"onDateChange($event, c.key)\"\n [panelClass]=\"'ctc-date-range-panel'\">\n <mat-datepicker-actions>\n <div class=\"ctc-date-presets\">\n <div class=\"cqa-mb-2 cqa-font-medium\">Quick Presets</div>\n <div class=\"btn-group\">\n <button type=\"button\" class=\"preset-btn today\"\n (click)=\"applyPresetToGroup(c.key, 'today'); picker.close()\">Today</button>\n <button type=\"button\" class=\"preset-btn last7days\"\n (click)=\"applyPresetToGroup(c.key, 'last7days'); picker.close()\">Last 7 days</button>\n <button type=\"button\" class=\"preset-btn last30days\"\n (click)=\"applyPresetToGroup(c.key, 'last30days'); picker.close()\">Last 30 days</button>\n <button type=\"button\" class=\"preset-btn last90days\"\n (click)=\"applyPresetToGroup(c.key, 'last90days'); picker.close()\">Last 90 days</button>\n <button type=\"button\" class=\"preset-btn thismonth\"\n (click)=\"applyPresetToGroup(c.key, 'thismonth'); picker.close()\">This month</button>\n <button type=\"button\" class=\"preset-btn lastmonth\"\n (click)=\"applyPresetToGroup(c.key, 'lastmonth'); picker.close()\">Last month</button>\n </div>\n </div>\n <div class=\"cqa-font-medium cqa-mb-2 cqa-mt-[10px] cqa-w-full\">Custom Range</div>\n <button\n class=\"cqa-font-medium !cqa-mb-2 !cqa-mt-[10px] cqa-w-[calc(100%-32px)] !cqa-absolute cqa-bottom-[2px]\"\n mat-flat-button color=\"primary\" matDatepickerApply>Apply</button>\n </mat-datepicker-actions>\n </mat-date-range-picker>\n <div (click)=\"picker.open()\" class=\"cqa-cursor-pointer\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M5.33398 1.33203V3.9987\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M10.666 1.33203V3.9987\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M12.6667 2.66797H3.33333C2.59695 2.66797 2 3.26492 2 4.0013V13.3346C2 14.071 2.59695 14.668 3.33333 14.668H12.6667C13.403 14.668 14 14.071 14 13.3346V4.0013C14 3.26492 13.403 2.66797 12.6667 2.66797Z\"\n stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 6.66797H14\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </div>\n </mat-form-field>\n <!-- Specific validation messages -->\n <mat-error *ngIf=\"getDateValidationError(c.key) && !picker.opened\">\n {{ getDateValidationError(c.key) }}\n </mat-error>\n\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n </form>\n\n <div class=\"cqa-flex cqa-justify-end cqa-items-stretch cqa-gap-2 cqa-mt-4\">\n <button type=\"button\"\n class=\"cqa-text-[13.33px] cqa-leading-[1] cqa-flex cqa-flex-row cqa-justify-center cqa-items-center cqa-px-[11px] cqa-py-[1px] cqa-gap-[14px] cqa-h-8 cqa-border cqa-border-[#0B0B0C] cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-rounded-lg cqa-text-[#0B0B0C]\"\n (click)=\"reset()\">Reset</button>\n <button type=\"button\"\n class=\"cqa-text-[13.33px] cqa-leading-[1] cqa-flex cqa-flex-row cqa-justify-center cqa-items-center cqa-px-[11px] cqa-py-[1px] cqa-gap-[14px] cqa-h-8 cqa-bg-[#0B0B0C] cqa-border cqa-border-[#E5E5E5] cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-rounded-lg cqa-text-white\"\n (click)=\"apply()\">Apply Filter</button>\n </div>\n </div>\n</div>", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"] }, { type: i3.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i4.MatDateRangeInput, selector: "mat-date-range-input", inputs: ["rangePicker", "required", "dateFilter", "min", "max", "disabled", "separator", "comparisonStart", "comparisonEnd"], exportAs: ["matDateRangeInput"] }, { type: i4.MatDateRangePicker, selector: "mat-date-range-picker", exportAs: ["matDateRangePicker"] }, { type: i4.MatDatepickerActions, selector: "mat-datepicker-actions, mat-date-range-picker-actions" }, { type: i5.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.MatStartDate, selector: "input[matStartDate]", inputs: ["errorStateMatcher"], outputs: ["dateChange", "dateInput"] }, { type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { type: i4.MatEndDate, selector: "input[matEndDate]", inputs: ["errorStateMatcher"], outputs: ["dateChange", "dateInput"] }, { type: i4.MatDatepickerApply, selector: "[matDatepickerApply], [matDateRangePickerApply]" }, { type: i3.MatError, selector: "mat-error", inputs: ["id"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
223
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DynamicFilterComponent, decorators: [{
|
|
224
|
+
type: Component,
|
|
225
|
+
args: [{ selector: 'cqa-dynamic-filter', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div id=\"cqa-ui-root\">\n <div class=\"cqa-filter cqa-mb-[28px]\" *ngIf=\"showFilterPanel\">\n <form class=\"ts-form cqa-grid lg:cqa-grid-cols-4 md:cqa-grid-cols-2 cqa-gap-4\" [formGroup]=\"form\"\n (keydown.enter)=\"(false)\" novalidate=\"novalidate\">\n <ng-container *ngFor=\"let c of config\">\n <ng-container *ngIf=\"!c.hidden\">\n <div class=\"form-group cqa-flex cqa-flex-col cqa-gap-2\">\n <!-- Select -->\n <ng-container *ngIf=\"c.type === 'select'\">\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(c)\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- Date Range --> \n <ng-container *ngIf=\"c.type === 'date-range'\">\n <label\n class=\"form-label cqa-text-[#374151] cqa-text-[14px] cqa-font-medium cqa-block cqa-leading-[1.4]\">{{\n c.label }}</label>\n <mat-form-field class=\"mat-date-custom\" appearance=\"fill\">\n <mat-date-range-input [rangePicker]=\"picker\" [formGroup]=\"getDateGroup(c.key)\" [max]=\"maxDate\">\n <input matStartDate formControlName=\"start\" placeholder=\"Start date\" [max]=\"maxDate\" required\n [readonly]=\"true\" (focus)=\"picker.open()\" (click)=\"picker.open()\" />\n <input matEndDate formControlName=\"end\" placeholder=\"End date\" [max]=\"maxDate\" required\n [readonly]=\"true\" (focus)=\"picker.open()\" (click)=\"picker.open()\" />\n </mat-date-range-input>\n <mat-date-range-picker #picker (dateChange)=\"onDateChange($event, c.key)\"\n [panelClass]=\"'ctc-date-range-panel'\">\n <mat-datepicker-actions>\n <div class=\"ctc-date-presets\">\n <div class=\"cqa-mb-2 cqa-font-medium\">Quick Presets</div>\n <div class=\"btn-group\">\n <button type=\"button\" class=\"preset-btn today\"\n (click)=\"applyPresetToGroup(c.key, 'today'); picker.close()\">Today</button>\n <button type=\"button\" class=\"preset-btn last7days\"\n (click)=\"applyPresetToGroup(c.key, 'last7days'); picker.close()\">Last 7 days</button>\n <button type=\"button\" class=\"preset-btn last30days\"\n (click)=\"applyPresetToGroup(c.key, 'last30days'); picker.close()\">Last 30 days</button>\n <button type=\"button\" class=\"preset-btn last90days\"\n (click)=\"applyPresetToGroup(c.key, 'last90days'); picker.close()\">Last 90 days</button>\n <button type=\"button\" class=\"preset-btn thismonth\"\n (click)=\"applyPresetToGroup(c.key, 'thismonth'); picker.close()\">This month</button>\n <button type=\"button\" class=\"preset-btn lastmonth\"\n (click)=\"applyPresetToGroup(c.key, 'lastmonth'); picker.close()\">Last month</button>\n </div>\n </div>\n <div class=\"cqa-font-medium cqa-mb-2 cqa-mt-[10px] cqa-w-full\">Custom Range</div>\n <button\n class=\"cqa-font-medium !cqa-mb-2 !cqa-mt-[10px] cqa-w-[calc(100%-32px)] !cqa-absolute cqa-bottom-[2px]\"\n mat-flat-button color=\"primary\" matDatepickerApply>Apply</button>\n </mat-datepicker-actions>\n </mat-date-range-picker>\n <div (click)=\"picker.open()\" class=\"cqa-cursor-pointer\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M5.33398 1.33203V3.9987\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M10.666 1.33203V3.9987\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M12.6667 2.66797H3.33333C2.59695 2.66797 2 3.26492 2 4.0013V13.3346C2 14.071 2.59695 14.668 3.33333 14.668H12.6667C13.403 14.668 14 14.071 14 13.3346V4.0013C14 3.26492 13.403 2.66797 12.6667 2.66797Z\"\n stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 6.66797H14\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </div>\n </mat-form-field>\n <!-- Specific validation messages -->\n <mat-error *ngIf=\"getDateValidationError(c.key) && !picker.opened\">\n {{ getDateValidationError(c.key) }}\n </mat-error>\n\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n </form>\n\n <div class=\"cqa-flex cqa-justify-end cqa-items-stretch cqa-gap-2 cqa-mt-4\">\n <button type=\"button\"\n class=\"cqa-text-[13.33px] cqa-leading-[1] cqa-flex cqa-flex-row cqa-justify-center cqa-items-center cqa-px-[11px] cqa-py-[1px] cqa-gap-[14px] cqa-h-8 cqa-border cqa-border-[#0B0B0C] cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-rounded-lg cqa-text-[#0B0B0C]\"\n (click)=\"reset()\">Reset</button>\n <button type=\"button\"\n class=\"cqa-text-[13.33px] cqa-leading-[1] cqa-flex cqa-flex-row cqa-justify-center cqa-items-center cqa-px-[11px] cqa-py-[1px] cqa-gap-[14px] cqa-h-8 cqa-bg-[#0B0B0C] cqa-border cqa-border-[#E5E5E5] cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-rounded-lg cqa-text-white\"\n (click)=\"apply()\">Apply Filter</button>\n </div>\n </div>\n</div>", styles: [] }]
|
|
226
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { config: [{
|
|
227
|
+
type: Input
|
|
228
|
+
}], model: [{
|
|
229
|
+
type: Input
|
|
230
|
+
}], showFilterPanel: [{
|
|
231
|
+
type: Input
|
|
232
|
+
}], filtersApplied: [{
|
|
233
|
+
type: Output
|
|
234
|
+
}], filtersChanged: [{
|
|
235
|
+
type: Output
|
|
236
|
+
}], resetAction: [{
|
|
237
|
+
type: Output
|
|
238
|
+
}] } });
|
|
239
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-filter.component.js","sourceRoot":"","sources":["../../../../../../src/lib/filters/dynamic-filter/dynamic-filter.component.ts","../../../../../../src/lib/filters/dynamic-filter/dynamic-filter.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, OnChanges, Output, SimpleChanges } from '@angular/core';\nimport { FormBuilder, FormControl, FormGroup } from '@angular/forms';\nimport { DynamicSelectFieldConfig } from '../../dynamic-select/dynamic-select-field.component';\n\ntype FilterType = 'select' | 'date-range';\n\nexport interface DynamicFilterOption {\n  id?: number | string;\n  name?: string;\n  value?: any;\n  label?: string;\n}\n\nexport interface DynamicFilterItem {\n  label: string;\n  placeholder?: string;\n  type: FilterType;\n  key: string;\n  options?: DynamicFilterOption[];\n  multiple?: boolean;\n  disabled?: boolean;\n  hidden?: boolean;\n  searchable?: boolean;\n}\n\nexport interface DynamicFilterValues {\n  [key: string]: any;\n}\n\n@Component({\n  selector: 'cqa-dynamic-filter',\n  templateUrl: './dynamic-filter.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DynamicFilterComponent implements OnChanges {\n  @Input() config: DynamicFilterItem[] = [];\n  @Input() model: DynamicFilterValues = {};\n  @Input() showFilterPanel: boolean = true;\n  @Output() filtersApplied = new EventEmitter<DynamicFilterValues>();\n  @Output() filtersChanged = new EventEmitter<DynamicFilterValues>();\n  @Output() resetAction = new EventEmitter<void>();\n\n  form: FormGroup = this.fb.group({});\n  maxDate = new Date();\n  searchTextByKey: Record<string, string> = {};\n  private selectOutsideCleanup = new Map<any, () => void>();\n\n  constructor(private readonly fb: FormBuilder) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['config'] || changes['model']) {\n      this.buildForm();\n    }\n  }\n\n  onDateChange(event: any, key: string) {\n    const formGroup = this.getDateGroup(key);\n    const start = formGroup.get('start')?.value;\n    const end = formGroup.get('end')?.value;\n\n    // When both dates are selected, auto-apply\n    // if (start && end) {\n    //   this.applyDateRange(key, { start, end });\n    // }\n  }\n\n  private buildForm(): void {\n    const ctrls: Record<string, any> = {};\n    (this.config || []).forEach(c => {\n      if (c.hidden) return;\n      if (c.type === 'date-range') {\n        ctrls[c.key] = this.fb.group({ start: new FormControl(), end: new FormControl() });\n      } else {\n        const initial = this.model?.[c.key];\n        if (c.multiple) {\n          const value = Array.isArray(initial) ? initial : (initial != null ? [initial] : []);\n          ctrls[c.key] = new FormControl(value);\n        } else {\n          ctrls[c.key] = new FormControl(Array.isArray(initial) ? (initial.length ? initial[0] : undefined) : initial);\n        }\n      }\n    });\n    this.form = this.fb.group(ctrls);\n    this.form.valueChanges.subscribe(() => this.filtersChanged.emit(this.serialize()));\n  }\n\n  onSelectOpenedChange(opened: boolean, select: any): void {\n    if (opened) {\n      setTimeout(() => {\n        const onDocDown = (e: MouseEvent) => {\n          const panel = document.querySelector('.cdk-overlay-pane .mat-select-panel') as HTMLElement | null;\n          const target = e.target as Node | null;\n          const originEl = (select?._elementRef?.nativeElement as HTMLElement) || null;\n          const insidePanel = !!(panel && target && panel.contains(target));\n          const insideOrigin = !!(originEl && target && originEl.contains(target));\n          if (!insidePanel && !insideOrigin) {\n            select.close();\n          }\n        };\n        document.addEventListener('mousedown', onDocDown, true);\n        this.selectOutsideCleanup.set(select, () => document.removeEventListener('mousedown', onDocDown, true));\n      }, 0);\n    } else {\n      const cleanup = this.selectOutsideCleanup.get(select);\n      if (cleanup) cleanup();\n      this.selectOutsideCleanup.delete(select);\n    }\n  }\n\n  onSearch(key: string, text: string): void {\n    this.searchTextByKey[key] = (text || '').toLowerCase();\n  }\n\n  filteredOptions(item: DynamicFilterItem): DynamicFilterOption[] {\n    const options = item?.options || [];\n    const q = (this.searchTextByKey[item.key] || '').trim();\n    if (!q) return options;\n    return options.filter(opt => {\n      const name = (opt.name ?? opt.label ?? String(opt.value ?? '')).toLowerCase();\n      return name.includes(q);\n    });\n  }\n\n  getDateGroup(key: string): FormGroup {\n    return this.form.get(key) as FormGroup;\n  }\n\n  getSelectConfig(item: DynamicFilterItem): DynamicSelectFieldConfig {\n    return {\n      key: item.key,\n      label: item.label,\n      placeholder: item.placeholder,\n      disabled: item.disabled,\n      multiple: item.multiple,\n      searchable: item.searchable,\n      options: item.options || []\n    };\n  }\n\n  apply(): void {\n    this.filtersApplied.emit(this.serialize());\n  }\n\n  reset(): void {\n    Object.keys(this.form.controls).forEach(key => {\n      const ctrl = this.form.get(key);\n      if (ctrl instanceof FormGroup) {\n        ctrl.get('start')?.setValue(undefined);\n        ctrl.get('end')?.setValue(undefined);\n      } else {\n        ctrl?.setValue(undefined);\n      }\n    });\n    this.resetAction.emit();\n    this.filtersChanged.emit(this.serialize());\n  }\n\n  private serialize(): DynamicFilterValues {\n    const result: DynamicFilterValues = {};\n    (this.config || []).forEach(c => {\n      const ctrl = this.form.get(c.key);\n      if (!ctrl) return;\n      let val: any = ctrl instanceof FormGroup ? (ctrl as FormGroup).getRawValue() : ctrl.value;\n      if (c.type === 'date-range') {\n        const start = val?.start; const end = val?.end;\n        if (start || end) result[c.key] = { start, end };\n      } else {\n        if (val !== undefined && val !== null && (Array.isArray(val) ? val.length > 0 : val !== '')) {\n          result[c.key] = val;\n        }\n      }\n    });\n    return result;\n  }\n\n  // Mat date range picker overlay preset helpers\n  applyPresetToGroup(key: string, presetKey: 'today' | 'last7days' | 'last30days' | 'last90days' | 'thismonth' | 'lastmonth'): void {\n    const dateGroup = this.getDateGroup(key);\n    if (!dateGroup) return;\n    const { start, end } = this.getPresetDates(presetKey);\n    dateGroup.patchValue({ start, end });\n    this.filtersChanged.emit(this.serialize());\n  }\n\n  private getPresetDates(presetKey: 'today' | 'last7days' | 'last30days' | 'last90days' | 'thismonth' | 'lastmonth'): { start: Date; end: Date } {\n    const today = new Date();\n    const clampToEndOfToday = (d: Date) => { const nd = new Date(d); nd.setHours(23,59,59,999); return nd; };\n    switch (presetKey) {\n      case 'today':\n        return { start: today, end: today };\n      case 'last7days': {\n        const start = new Date(); start.setDate(start.getDate() - 6);\n        return { start, end: today };\n      }\n      case 'last30days': {\n        const start = new Date(); start.setDate(start.getDate() - 29);\n        return { start, end: today };\n      }\n      case 'last90days': {\n        const start = new Date(); start.setDate(start.getDate() - 89);\n        return { start, end: today };\n      }\n      case 'thismonth': {\n        const start = new Date(today.getFullYear(), today.getMonth(), 1);\n        return { start, end: today };\n      }\n      case 'lastmonth': {\n        const start = new Date(today.getFullYear(), today.getMonth() - 1, 1);\n        const end = new Date(today.getFullYear(), today.getMonth(), 0);\n        return { start, end: clampToEndOfToday(end) };\n      }\n    }\n  }\n  getDateValidationError(key: string): string | null {\n    const dateGroup = this.getDateGroup(key);\n    if (!dateGroup) return null;\n\n    const startCtrl = dateGroup.controls['start'];\n    const endCtrl = dateGroup.controls['end'];\n    const startVal = startCtrl?.value;\n    const endVal = endCtrl?.value;\n    const startParseErr = startCtrl?.errors?.['matDatepickerParse']?.text;\n    const endParseErr = endCtrl?.errors?.['matDatepickerParse']?.text;\n\n    if (!startVal && !endVal && !startParseErr && !endParseErr) return null;\n\n    if (startParseErr) return `Invalid start date format`;\n    if (endParseErr) return `Invalid end date format`;\n\n    if (startVal && !endVal) return `Please select an end date`;\n    if (!startVal && endVal) return `Please select a start date`;\n\n    if (dateGroup.hasError('matStartDateInvalid')) return `Start date must be before end date`;\n    if (dateGroup.hasError('matEndDateInvalid')) return `End date must be after start date`;\n\n    if (dateGroup.invalid) {\n      const config = this.config?.find(c => c.key === key);\n      return config ? `${config.label} is invalid` : 'Date range is invalid';\n    }\n\n    return null;\n  }\n  // Dynamic Date Picker Integration Methods removed (component no longer used)\n}\n\n\n","<div id=\"cqa-ui-root\">\n  <div class=\"cqa-filter cqa-mb-[28px]\" *ngIf=\"showFilterPanel\">\n    <form class=\"ts-form cqa-grid lg:cqa-grid-cols-4 md:cqa-grid-cols-2 cqa-gap-4\" [formGroup]=\"form\"\n      (keydown.enter)=\"(false)\" novalidate=\"novalidate\">\n      <ng-container *ngFor=\"let c of config\">\n        <ng-container *ngIf=\"!c.hidden\">\n          <div class=\"form-group cqa-flex cqa-flex-col cqa-gap-2\">\n            <!-- Select -->\n            <ng-container *ngIf=\"c.type === 'select'\">\n              <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(c)\"></cqa-dynamic-select>\n            </ng-container>\n\n            <!-- Date Range --> \n            <ng-container *ngIf=\"c.type === 'date-range'\">\n              <label\n                class=\"form-label cqa-text-[#374151] cqa-text-[14px] cqa-font-medium cqa-block cqa-leading-[1.4]\">{{\n                c.label }}</label>\n              <mat-form-field class=\"mat-date-custom\" appearance=\"fill\">\n                <mat-date-range-input [rangePicker]=\"picker\" [formGroup]=\"getDateGroup(c.key)\" [max]=\"maxDate\">\n                  <input matStartDate formControlName=\"start\" placeholder=\"Start date\" [max]=\"maxDate\" required\n                    [readonly]=\"true\" (focus)=\"picker.open()\" (click)=\"picker.open()\" />\n                  <input matEndDate formControlName=\"end\" placeholder=\"End date\" [max]=\"maxDate\" required\n                    [readonly]=\"true\" (focus)=\"picker.open()\" (click)=\"picker.open()\" />\n                </mat-date-range-input>\n                <mat-date-range-picker #picker (dateChange)=\"onDateChange($event, c.key)\"\n                  [panelClass]=\"'ctc-date-range-panel'\">\n                  <mat-datepicker-actions>\n                    <div class=\"ctc-date-presets\">\n                      <div class=\"cqa-mb-2 cqa-font-medium\">Quick Presets</div>\n                      <div class=\"btn-group\">\n                        <button type=\"button\" class=\"preset-btn today\"\n                          (click)=\"applyPresetToGroup(c.key, 'today'); picker.close()\">Today</button>\n                        <button type=\"button\" class=\"preset-btn last7days\"\n                          (click)=\"applyPresetToGroup(c.key, 'last7days'); picker.close()\">Last 7 days</button>\n                        <button type=\"button\" class=\"preset-btn last30days\"\n                          (click)=\"applyPresetToGroup(c.key, 'last30days'); picker.close()\">Last 30 days</button>\n                        <button type=\"button\" class=\"preset-btn last90days\"\n                          (click)=\"applyPresetToGroup(c.key, 'last90days'); picker.close()\">Last 90 days</button>\n                        <button type=\"button\" class=\"preset-btn thismonth\"\n                          (click)=\"applyPresetToGroup(c.key, 'thismonth'); picker.close()\">This month</button>\n                        <button type=\"button\" class=\"preset-btn lastmonth\"\n                          (click)=\"applyPresetToGroup(c.key, 'lastmonth'); picker.close()\">Last month</button>\n                      </div>\n                    </div>\n                    <div class=\"cqa-font-medium cqa-mb-2 cqa-mt-[10px] cqa-w-full\">Custom Range</div>\n                    <button\n                      class=\"cqa-font-medium !cqa-mb-2 !cqa-mt-[10px] cqa-w-[calc(100%-32px)] !cqa-absolute cqa-bottom-[2px]\"\n                      mat-flat-button color=\"primary\" matDatepickerApply>Apply</button>\n                  </mat-datepicker-actions>\n                </mat-date-range-picker>\n                <div (click)=\"picker.open()\" class=\"cqa-cursor-pointer\">\n                  <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n                    <path d=\"M5.33398 1.33203V3.9987\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n                      stroke-linejoin=\"round\" />\n                    <path d=\"M10.666 1.33203V3.9987\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n                      stroke-linejoin=\"round\" />\n                    <path\n                      d=\"M12.6667 2.66797H3.33333C2.59695 2.66797 2 3.26492 2 4.0013V13.3346C2 14.071 2.59695 14.668 3.33333 14.668H12.6667C13.403 14.668 14 14.071 14 13.3346V4.0013C14 3.26492 13.403 2.66797 12.6667 2.66797Z\"\n                      stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n                    <path d=\"M2 6.66797H14\" stroke=\"#0A0A0A\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n                      stroke-linejoin=\"round\" />\n                  </svg>\n                </div>\n              </mat-form-field>\n              <!-- Specific validation messages -->\n              <mat-error *ngIf=\"getDateValidationError(c.key) && !picker.opened\">\n                {{ getDateValidationError(c.key) }}\n              </mat-error>\n\n            </ng-container>\n          </div>\n        </ng-container>\n      </ng-container>\n    </form>\n\n    <div class=\"cqa-flex cqa-justify-end cqa-items-stretch cqa-gap-2 cqa-mt-4\">\n      <button type=\"button\"\n        class=\"cqa-text-[13.33px] cqa-leading-[1] cqa-flex cqa-flex-row cqa-justify-center cqa-items-center cqa-px-[11px] cqa-py-[1px] cqa-gap-[14px] cqa-h-8 cqa-border cqa-border-[#0B0B0C] cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-rounded-lg cqa-text-[#0B0B0C]\"\n        (click)=\"reset()\">Reset</button>\n      <button type=\"button\"\n        class=\"cqa-text-[13.33px] cqa-leading-[1] cqa-flex cqa-flex-row cqa-justify-center cqa-items-center cqa-px-[11px] cqa-py-[1px] cqa-gap-[14px] cqa-h-8 cqa-bg-[#0B0B0C] cqa-border cqa-border-[#E5E5E5] cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-rounded-lg cqa-text-white\"\n        (click)=\"apply()\">Apply Filter</button>\n    </div>\n  </div>\n</div>"]}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class FullTableLoaderComponent {
|
|
4
|
+
constructor() {
|
|
5
|
+
this.label = 'Loading...';
|
|
6
|
+
}
|
|
7
|
+
}
|
|
8
|
+
FullTableLoaderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FullTableLoaderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
9
|
+
FullTableLoaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: { label: "label" }, ngImport: i0, template: "<div\n class=\"table-loading-overlay cqa-absolute cqa-top-0 cqa-bottom-0 cqa-left-0 cqa-right-0 cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-z-[1000]\">\n <div\n class=\"blur-backdrop cqa-absolute cqa-top-0 cqa-bottom-0 cqa-left-0 cqa-right-0 cqa-bg-white/70 cqa-backdrop-blur-[4px] cqa-z-[1]\">\n </div>\n <div\n class=\"loading-spinner cqa-flex cqa-flex-row cqa-items-center cqa-justify-center cqa-gap-3 cqa-text-center cqa-relative cqa-z-20 cqa-bg-white cqa-py-4 cqa-px-6 cqa-rounded-xl cqa-shadow-lg\">\n <svg class=\"cqa-animate-spin cqa-text-primary\" width=\"32\" height=\"32\" xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\"\n 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\"\n 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\">\n </path>\n </svg>\n <span class=\"cqa-text-primary\">{{ label }}</span>\n </div>\n</div>\n\n\n" });
|
|
10
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FullTableLoaderComponent, decorators: [{
|
|
11
|
+
type: Component,
|
|
12
|
+
args: [{ selector: 'cqa-full-table-loader', template: "<div\n class=\"table-loading-overlay cqa-absolute cqa-top-0 cqa-bottom-0 cqa-left-0 cqa-right-0 cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-z-[1000]\">\n <div\n class=\"blur-backdrop cqa-absolute cqa-top-0 cqa-bottom-0 cqa-left-0 cqa-right-0 cqa-bg-white/70 cqa-backdrop-blur-[4px] cqa-z-[1]\">\n </div>\n <div\n class=\"loading-spinner cqa-flex cqa-flex-row cqa-items-center cqa-justify-center cqa-gap-3 cqa-text-center cqa-relative cqa-z-20 cqa-bg-white cqa-py-4 cqa-px-6 cqa-rounded-xl cqa-shadow-lg\">\n <svg class=\"cqa-animate-spin cqa-text-primary\" width=\"32\" height=\"32\" xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\"\n 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\"\n 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\">\n </path>\n </svg>\n <span class=\"cqa-text-primary\">{{ label }}</span>\n </div>\n</div>\n\n\n", styles: [] }]
|
|
13
|
+
}], propDecorators: { label: [{
|
|
14
|
+
type: Input
|
|
15
|
+
}] } });
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZnVsbC10YWJsZS1sb2FkZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9mdWxsLXRhYmxlLWxvYWRlci9mdWxsLXRhYmxlLWxvYWRlci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2Z1bGwtdGFibGUtbG9hZGVyL2Z1bGwtdGFibGUtbG9hZGVyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU9qRCxNQUFNLE9BQU8sd0JBQXdCO0lBTHJDO1FBTVcsVUFBSyxHQUFXLFlBQVksQ0FBQztLQUN2Qzs7cUhBRlksd0JBQXdCO3lHQUF4Qix3QkFBd0IseUZDUHJDLG9tQ0FtQkE7MkZEWmEsd0JBQXdCO2tCQUxwQyxTQUFTOytCQUNFLHVCQUF1Qjs4QkFLeEIsS0FBSztzQkFBYixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjcWEtZnVsbC10YWJsZS1sb2FkZXInLFxuICB0ZW1wbGF0ZVVybDogJy4vZnVsbC10YWJsZS1sb2FkZXIuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFtdXG59KVxuZXhwb3J0IGNsYXNzIEZ1bGxUYWJsZUxvYWRlckNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIGxhYmVsOiBzdHJpbmcgPSAnTG9hZGluZy4uLic7XG59XG5cblxuIiwiPGRpdlxuICBjbGFzcz1cInRhYmxlLWxvYWRpbmctb3ZlcmxheSBjcWEtYWJzb2x1dGUgY3FhLXRvcC0wIGNxYS1ib3R0b20tMCBjcWEtbGVmdC0wIGNxYS1yaWdodC0wIGNxYS13LWZ1bGwgY3FhLWgtZnVsbCBjcWEtZmxleCBjcWEtaXRlbXMtY2VudGVyIGNxYS1qdXN0aWZ5LWNlbnRlciBjcWEtei1bMTAwMF1cIj5cbiAgPGRpdlxuICAgIGNsYXNzPVwiYmx1ci1iYWNrZHJvcCBjcWEtYWJzb2x1dGUgY3FhLXRvcC0wIGNxYS1ib3R0b20tMCBjcWEtbGVmdC0wIGNxYS1yaWdodC0wIGNxYS1iZy13aGl0ZS83MCBjcWEtYmFja2Ryb3AtYmx1ci1bNHB4XSBjcWEtei1bMV1cIj5cbiAgPC9kaXY+XG4gIDxkaXZcbiAgICBjbGFzcz1cImxvYWRpbmctc3Bpbm5lciBjcWEtZmxleCBjcWEtZmxleC1yb3cgY3FhLWl0ZW1zLWNlbnRlciBjcWEtanVzdGlmeS1jZW50ZXIgY3FhLWdhcC0zIGNxYS10ZXh0LWNlbnRlciBjcWEtcmVsYXRpdmUgY3FhLXotMjAgY3FhLWJnLXdoaXRlIGNxYS1weS00IGNxYS1weC02IGNxYS1yb3VuZGVkLXhsIGNxYS1zaGFkb3ctbGdcIj5cbiAgICA8c3ZnIGNsYXNzPVwiY3FhLWFuaW1hdGUtc3BpbiBjcWEtdGV4dC1wcmltYXJ5XCIgd2lkdGg9XCIzMlwiIGhlaWdodD1cIjMyXCIgeG1sbnM9XCJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2Z1wiIGZpbGw9XCJub25lXCJcbiAgICAgIHZpZXdCb3g9XCIwIDAgMjQgMjRcIj5cbiAgICAgIDxjaXJjbGUgY2xhc3M9XCJjcWEtb3BhY2l0eS0yNVwiIGN4PVwiMTJcIiBjeT1cIjEyXCIgcj1cIjEwXCIgc3Ryb2tlPVwiY3VycmVudENvbG9yXCIgc3Ryb2tlLXdpZHRoPVwiNFwiPjwvY2lyY2xlPlxuICAgICAgPHBhdGggY2xhc3M9XCJjcWEtb3BhY2l0eS03NVwiIGZpbGw9XCJjdXJyZW50Q29sb3JcIlxuICAgICAgICBkPVwiTTQgMTJhOCA4IDAgMDE4LThWMEM1LjM3MyAwIDAgNS4zNzMgMCAxMmg0em0yIDUuMjkxQTcuOTYyIDcuOTYyIDAgMDE0IDEySDBjMCAzLjA0MiAxLjEzNSA1LjgyNCAzIDcuOTM4bDMtMi42NDd6XCI+XG4gICAgICA8L3BhdGg+XG4gICAgPC9zdmc+XG4gICAgPHNwYW4gY2xhc3M9XCJjcWEtdGV4dC1wcmltYXJ5XCI+e3sgbGFiZWwgfX08L3NwYW4+XG4gIDwvZGl2PlxuPC9kaXY+XG5cblxuIl19
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { Component, Input, ViewChild } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/tooltip";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
export class InlineSortComponent {
|
|
6
|
+
get getToolTip() {
|
|
7
|
+
if (this.ascending == true)
|
|
8
|
+
return this.heading?.includes('created_at') ? 'message.common.sort_by.old' : 'message.common.sort_by.ascending';
|
|
9
|
+
else if (this.ascending == false)
|
|
10
|
+
return this.heading?.includes('created_at') ? 'message.common.sort_by.new' : 'message.common.sort_by.descending';
|
|
11
|
+
else
|
|
12
|
+
return 'message.common.sort';
|
|
13
|
+
}
|
|
14
|
+
ngOnChanges() {
|
|
15
|
+
if (this.ascending != undefined) {
|
|
16
|
+
this.tooltipDiv?.hide();
|
|
17
|
+
this.tooltipDiv?.show(200);
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
InlineSortComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: InlineSortComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
22
|
+
InlineSortComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: InlineSortComponent, selector: "app-inline-sort, cqa-inline-sort", inputs: { ascending: "ascending", heading: "heading" }, viewQueries: [{ propertyName: "tooltipDiv", first: true, predicate: ["tooltipDiv"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
|
|
23
|
+
<div #tooltipDiv="matTooltip"
|
|
24
|
+
class="w-fit-content sort-header"
|
|
25
|
+
[matTooltip]="getToolTip"
|
|
26
|
+
[matTooltipPosition]="'after'">
|
|
27
|
+
<span>{{ heading }}</span>
|
|
28
|
+
<span *ngIf="ascending" class='fa-down-sort'></span>
|
|
29
|
+
<span *ngIf="ascending==false" class='fa-up-sort'></span>
|
|
30
|
+
<span *ngIf="ascending==undefined" class='fa-down-sort opaque-50'></span>
|
|
31
|
+
</div>
|
|
32
|
+
`, isInline: true, directives: [{ type: i1.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
33
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: InlineSortComponent, decorators: [{
|
|
34
|
+
type: Component,
|
|
35
|
+
args: [{
|
|
36
|
+
selector: 'app-inline-sort, cqa-inline-sort',
|
|
37
|
+
template: `
|
|
38
|
+
<div #tooltipDiv="matTooltip"
|
|
39
|
+
class="w-fit-content sort-header"
|
|
40
|
+
[matTooltip]="getToolTip"
|
|
41
|
+
[matTooltipPosition]="'after'">
|
|
42
|
+
<span>{{ heading }}</span>
|
|
43
|
+
<span *ngIf="ascending" class='fa-down-sort'></span>
|
|
44
|
+
<span *ngIf="ascending==false" class='fa-up-sort'></span>
|
|
45
|
+
<span *ngIf="ascending==undefined" class='fa-down-sort opaque-50'></span>
|
|
46
|
+
</div>
|
|
47
|
+
`,
|
|
48
|
+
styles: []
|
|
49
|
+
}]
|
|
50
|
+
}], propDecorators: { ascending: [{
|
|
51
|
+
type: Input
|
|
52
|
+
}], heading: [{
|
|
53
|
+
type: Input
|
|
54
|
+
}], tooltipDiv: [{
|
|
55
|
+
type: ViewChild,
|
|
56
|
+
args: ['tooltipDiv']
|
|
57
|
+
}] } });
|
|
58
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5saW5lLXNvcnQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9pbmxpbmUtc29ydC9pbmxpbmUtc29ydC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFNBQVMsRUFBRSxLQUFLLEVBQUUsU0FBUyxFQUFDLE1BQU0sZUFBZSxDQUFDOzs7O0FBbUIxRCxNQUFNLE9BQU8sbUJBQW1CO0lBTTlCLElBQUksVUFBVTtRQUNaLElBQUcsSUFBSSxDQUFDLFNBQVMsSUFBRSxJQUFJO1lBQ3JCLE9BQU8sSUFBSSxDQUFDLE9BQU8sRUFBRSxRQUFRLENBQUMsWUFBWSxDQUFDLENBQUEsQ0FBQyxDQUFBLDRCQUE0QixDQUFBLENBQUMsQ0FBQSxrQ0FBa0MsQ0FBQzthQUN6RyxJQUFHLElBQUksQ0FBQyxTQUFTLElBQUUsS0FBSztZQUMzQixPQUFPLElBQUksQ0FBQyxPQUFPLEVBQUUsUUFBUSxDQUFDLFlBQVksQ0FBQyxDQUFBLENBQUMsQ0FBQSw0QkFBNEIsQ0FBQSxDQUFDLENBQUEsbUNBQW1DLENBQUM7O1lBRTdHLE9BQU8scUJBQXFCLENBQUM7SUFDakMsQ0FBQztJQUVELFdBQVc7UUFDVCxJQUFHLElBQUksQ0FBQyxTQUFTLElBQUUsU0FBUyxFQUFFO1lBQzVCLElBQUksQ0FBQyxVQUFVLEVBQUUsSUFBSSxFQUFFLENBQUE7WUFDdkIsSUFBSSxDQUFDLFVBQVUsRUFBRSxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7U0FDNUI7SUFDSCxDQUFDOztnSEFwQlUsbUJBQW1CO29HQUFuQixtQkFBbUIsK1BBZHBCOzs7Ozs7Ozs7O0dBVVQ7MkZBSVUsbUJBQW1CO2tCQWhCL0IsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsa0NBQWtDO29CQUM1QyxRQUFRLEVBQUU7Ozs7Ozs7Ozs7R0FVVDtvQkFDRCxNQUFNLEVBQUUsRUFDUDtpQkFDRjs4QkFFVSxTQUFTO3NCQUFqQixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDa0IsVUFBVTtzQkFBakMsU0FBUzt1QkFBQyxZQUFZIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtDb21wb25lbnQsIElucHV0LCBWaWV3Q2hpbGR9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtNYXRUb29sdGlwfSBmcm9tIFwiQGFuZ3VsYXIvbWF0ZXJpYWwvdG9vbHRpcFwiO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdhcHAtaW5saW5lLXNvcnQsIGNxYS1pbmxpbmUtc29ydCcsXG4gIHRlbXBsYXRlOiBgXG4gICAgPGRpdiAjdG9vbHRpcERpdj1cIm1hdFRvb2x0aXBcIlxuICAgICAgICAgY2xhc3M9XCJ3LWZpdC1jb250ZW50IHNvcnQtaGVhZGVyXCJcbiAgICAgICAgIFttYXRUb29sdGlwXT1cImdldFRvb2xUaXBcIlxuICAgICAgICAgW21hdFRvb2x0aXBQb3NpdGlvbl09XCInYWZ0ZXInXCI+XG4gICAgICA8c3Bhbj57eyBoZWFkaW5nIH19PC9zcGFuPlxuICAgICAgPHNwYW4gKm5nSWY9XCJhc2NlbmRpbmdcIiBjbGFzcz0nZmEtZG93bi1zb3J0Jz48L3NwYW4+XG4gICAgICA8c3BhbiAqbmdJZj1cImFzY2VuZGluZz09ZmFsc2VcIiBjbGFzcz0nZmEtdXAtc29ydCc+PC9zcGFuPlxuICAgICAgPHNwYW4gKm5nSWY9XCJhc2NlbmRpbmc9PXVuZGVmaW5lZFwiIGNsYXNzPSdmYS1kb3duLXNvcnQgb3BhcXVlLTUwJz48L3NwYW4+XG4gICAgPC9kaXY+XG4gIGAsXG4gIHN0eWxlczogW1xuICBdXG59KVxuZXhwb3J0IGNsYXNzIElubGluZVNvcnRDb21wb25lbnR7XG4gIEBJbnB1dCgpIGFzY2VuZGluZz86IGJvb2xlYW47XG4gIEBJbnB1dCgpIGhlYWRpbmc/OiBzdHJpbmc7XG4gIEBWaWV3Q2hpbGQoJ3Rvb2x0aXBEaXYnKXRvb2x0aXBEaXY/OiBNYXRUb29sdGlwO1xuICB0b29sVGlwOiBhbnk7XG5cbiAgZ2V0IGdldFRvb2xUaXAoKXtcbiAgICBpZih0aGlzLmFzY2VuZGluZz09dHJ1ZSlcbiAgICAgIHJldHVybiB0aGlzLmhlYWRpbmc/LmluY2x1ZGVzKCdjcmVhdGVkX2F0Jyk/J21lc3NhZ2UuY29tbW9uLnNvcnRfYnkub2xkJzonbWVzc2FnZS5jb21tb24uc29ydF9ieS5hc2NlbmRpbmcnO1xuICAgIGVsc2UgaWYodGhpcy5hc2NlbmRpbmc9PWZhbHNlKVxuICAgICAgcmV0dXJuIHRoaXMuaGVhZGluZz8uaW5jbHVkZXMoJ2NyZWF0ZWRfYXQnKT8nbWVzc2FnZS5jb21tb24uc29ydF9ieS5uZXcnOidtZXNzYWdlLmNvbW1vbi5zb3J0X2J5LmRlc2NlbmRpbmcnO1xuICAgIGVsc2VcbiAgICAgIHJldHVybiAnbWVzc2FnZS5jb21tb24uc29ydCc7XG4gIH1cblxuICBuZ09uQ2hhbmdlcygpOiB2b2lkIHtcbiAgICBpZih0aGlzLmFzY2VuZGluZyE9dW5kZWZpbmVkKSB7XG4gICAgICB0aGlzLnRvb2x0aXBEaXY/LmhpZGUoKVxuICAgICAgdGhpcy50b29sdGlwRGl2Py5zaG93KDIwMCk7XG4gICAgfVxuICB9XG5cbn1cblxuXG5cbiJdfQ==
|
|
@@ -0,0 +1,76 @@
|
|
|
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 OtherButtonComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
// Single button API (backwards compatible)
|
|
8
|
+
this.icon = '';
|
|
9
|
+
this.label = '';
|
|
10
|
+
this.classes = '';
|
|
11
|
+
this.colorClass = '';
|
|
12
|
+
this.buttonClass = '';
|
|
13
|
+
this.disabled = false;
|
|
14
|
+
this.type = 'button';
|
|
15
|
+
// Group buttons API (new)
|
|
16
|
+
this.buttons = null;
|
|
17
|
+
/** Extra classes for the button group container */
|
|
18
|
+
this.groupClass = '';
|
|
19
|
+
/** Gap utility class; defaults to Tailwind spacing applied in template */
|
|
20
|
+
this.gapClass = 'cqa-gap-2';
|
|
21
|
+
/** When true, allows wrapping to next line on smaller screens */
|
|
22
|
+
this.wrap = true;
|
|
23
|
+
this.clicked = new EventEmitter();
|
|
24
|
+
/** Emits the config of the clicked button in a group, along with the event */
|
|
25
|
+
this.buttonClick = new EventEmitter();
|
|
26
|
+
}
|
|
27
|
+
onClick(event) {
|
|
28
|
+
if (this.disabled) {
|
|
29
|
+
event.preventDefault();
|
|
30
|
+
event.stopPropagation();
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
this.clicked.emit(event);
|
|
34
|
+
}
|
|
35
|
+
onItemClick(event, item) {
|
|
36
|
+
if (item?.disabled) {
|
|
37
|
+
event.preventDefault();
|
|
38
|
+
event.stopPropagation();
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
this.buttonClick.emit({ event, item });
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
OtherButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: OtherButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
45
|
+
OtherButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: OtherButtonComponent, selector: "cqa-other-button", inputs: { icon: "icon", label: "label", classes: "classes", colorClass: "colorClass", buttonClass: "buttonClass", disabled: "disabled", type: "type", buttons: "buttons", groupClass: "groupClass", gapClass: "gapClass", wrap: "wrap" }, outputs: { clicked: "clicked", buttonClick: "buttonClick" }, ngImport: i0, template: "<div id=\"cqa-ui-root\" >\n <!-- Group rendering if buttons are provided -->\n <ng-container *ngIf=\"buttons?.length; else singleButton\">\n <div class=\"cqa-inline-flex cqa-items-center\" [ngClass]=\"[wrap ? 'cqa-flex-wrap' : '', gapClass, groupClass]\">\n <button *ngFor=\"let b of buttons\"\n [type]=\"b.type || 'button'\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-1 cqa-py-[5.75px] cqa-px-[11.5px] cqa-rounded-[5px] cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium cqa-border\"\n [ngClass]=\"[b.classes || classes, buttonClass, b.colorClass || colorClass]\"\n [disabled]=\"b.disabled\"\n (click)=\"onItemClick($event, b)\">\n <mat-icon *ngIf=\"b.icon || icon\" class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ b.icon || icon }}</mat-icon>\n <span>{{ b.label }}</span>\n </button>\n </div>\n </ng-container>\n\n <!-- Single button (backwards compatible) -->\n <ng-template #singleButton>\n <button [type]=\"type\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-1 cqa-py-[5.75px] cqa-px-[11.5px] cqa-rounded-[5px] cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium cqa-border\"\n [ngClass]=\"[classes, buttonClass, colorClass]\" [disabled]=\"disabled\" (click)=\"onClick($event)\">\n <mat-icon *ngIf=\"icon\" class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ icon }}</mat-icon>\n <span>{{ label }}</span>\n </button>\n </ng-template>\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.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
46
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: OtherButtonComponent, decorators: [{
|
|
47
|
+
type: Component,
|
|
48
|
+
args: [{ selector: 'cqa-other-button', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div id=\"cqa-ui-root\" >\n <!-- Group rendering if buttons are provided -->\n <ng-container *ngIf=\"buttons?.length; else singleButton\">\n <div class=\"cqa-inline-flex cqa-items-center\" [ngClass]=\"[wrap ? 'cqa-flex-wrap' : '', gapClass, groupClass]\">\n <button *ngFor=\"let b of buttons\"\n [type]=\"b.type || 'button'\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-1 cqa-py-[5.75px] cqa-px-[11.5px] cqa-rounded-[5px] cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium cqa-border\"\n [ngClass]=\"[b.classes || classes, buttonClass, b.colorClass || colorClass]\"\n [disabled]=\"b.disabled\"\n (click)=\"onItemClick($event, b)\">\n <mat-icon *ngIf=\"b.icon || icon\" class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ b.icon || icon }}</mat-icon>\n <span>{{ b.label }}</span>\n </button>\n </div>\n </ng-container>\n\n <!-- Single button (backwards compatible) -->\n <ng-template #singleButton>\n <button [type]=\"type\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-1 cqa-py-[5.75px] cqa-px-[11.5px] cqa-rounded-[5px] cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium cqa-border\"\n [ngClass]=\"[classes, buttonClass, colorClass]\" [disabled]=\"disabled\" (click)=\"onClick($event)\">\n <mat-icon *ngIf=\"icon\" class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ icon }}</mat-icon>\n <span>{{ label }}</span>\n </button>\n </ng-template>\n</div>", styles: [] }]
|
|
49
|
+
}], propDecorators: { icon: [{
|
|
50
|
+
type: Input
|
|
51
|
+
}], label: [{
|
|
52
|
+
type: Input
|
|
53
|
+
}], classes: [{
|
|
54
|
+
type: Input
|
|
55
|
+
}], colorClass: [{
|
|
56
|
+
type: Input
|
|
57
|
+
}], buttonClass: [{
|
|
58
|
+
type: Input
|
|
59
|
+
}], disabled: [{
|
|
60
|
+
type: Input
|
|
61
|
+
}], type: [{
|
|
62
|
+
type: Input
|
|
63
|
+
}], buttons: [{
|
|
64
|
+
type: Input
|
|
65
|
+
}], groupClass: [{
|
|
66
|
+
type: Input
|
|
67
|
+
}], gapClass: [{
|
|
68
|
+
type: Input
|
|
69
|
+
}], wrap: [{
|
|
70
|
+
type: Input
|
|
71
|
+
}], clicked: [{
|
|
72
|
+
type: Output
|
|
73
|
+
}], buttonClick: [{
|
|
74
|
+
type: Output
|
|
75
|
+
}] } });
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3RoZXItYnV0dG9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvb3RoZXItYnV0dG9uL290aGVyLWJ1dHRvbi5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL290aGVyLWJ1dHRvbi9vdGhlci1idXR0b24uY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQzs7OztBQWtCaEcsTUFBTSxPQUFPLG9CQUFvQjtJQU5qQztRQU9FLDJDQUEyQztRQUNsQyxTQUFJLEdBQVcsRUFBRSxDQUFDO1FBQ2xCLFVBQUssR0FBVyxFQUFFLENBQUM7UUFDbkIsWUFBTyxHQUFnRCxFQUFFLENBQUM7UUFDMUQsZUFBVSxHQUFnRCxFQUFFLENBQUM7UUFDN0QsZ0JBQVcsR0FBVyxFQUFFLENBQUM7UUFDekIsYUFBUSxHQUFZLEtBQUssQ0FBQztRQUMxQixTQUFJLEdBQWtDLFFBQVEsQ0FBQztRQUV4RCwwQkFBMEI7UUFDakIsWUFBTyxHQUE2QixJQUFJLENBQUM7UUFDbEQsbURBQW1EO1FBQzFDLGVBQVUsR0FBZ0QsRUFBRSxDQUFDO1FBQ3RFLDBFQUEwRTtRQUNqRSxhQUFRLEdBQVcsV0FBVyxDQUFDO1FBQ3hDLGlFQUFpRTtRQUN4RCxTQUFJLEdBQVksSUFBSSxDQUFDO1FBRXBCLFlBQU8sR0FBRyxJQUFJLFlBQVksRUFBYyxDQUFDO1FBQ25ELDhFQUE4RTtRQUNwRSxnQkFBVyxHQUFHLElBQUksWUFBWSxFQUFnRCxDQUFDO0tBbUIxRjtJQWpCQyxPQUFPLENBQUMsS0FBaUI7UUFDdkIsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2pCLEtBQUssQ0FBQyxjQUFjLEVBQUUsQ0FBQztZQUN2QixLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7WUFDeEIsT0FBTztTQUNSO1FBQ0QsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDM0IsQ0FBQztJQUVELFdBQVcsQ0FBQyxLQUFpQixFQUFFLElBQXFCO1FBQ2xELElBQUksSUFBSSxFQUFFLFFBQVEsRUFBRTtZQUNsQixLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDdkIsS0FBSyxDQUFDLGVBQWUsRUFBRSxDQUFDO1lBQ3hCLE9BQU87U0FDUjtRQUNELElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7SUFDekMsQ0FBQzs7aUhBdkNVLG9CQUFvQjtxR0FBcEIsb0JBQW9CLCtWQ2xCakMsMDhDQXlCTTsyRkRQTyxvQkFBb0I7a0JBTmhDLFNBQVM7K0JBQ0Usa0JBQWtCLG1CQUdYLHVCQUF1QixDQUFDLE1BQU07OEJBSXRDLElBQUk7c0JBQVosS0FBSztnQkFDRyxLQUFLO3NCQUFiLEtBQUs7Z0JBQ0csT0FBTztzQkFBZixLQUFLO2dCQUNHLFVBQVU7c0JBQWxCLEtBQUs7Z0JBQ0csV0FBVztzQkFBbkIsS0FBSztnQkFDRyxRQUFRO3NCQUFoQixLQUFLO2dCQUNHLElBQUk7c0JBQVosS0FBSztnQkFHRyxPQUFPO3NCQUFmLEtBQUs7Z0JBRUcsVUFBVTtzQkFBbEIsS0FBSztnQkFFRyxRQUFRO3NCQUFoQixLQUFLO2dCQUVHLElBQUk7c0JBQVosS0FBSztnQkFFSSxPQUFPO3NCQUFoQixNQUFNO2dCQUVHLFdBQVc7c0JBQXBCLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIElucHV0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IGludGVyZmFjZSBPdGhlckJ1dHRvbkl0ZW0ge1xuICBsYWJlbDogc3RyaW5nO1xuICBpY29uPzogc3RyaW5nO1xuICBjbGFzc2VzPzogc3RyaW5nIHwgc3RyaW5nW10gfCBSZWNvcmQ8c3RyaW5nLCBib29sZWFuPjtcbiAgY29sb3JDbGFzcz86IHN0cmluZyB8IHN0cmluZ1tdIHwgUmVjb3JkPHN0cmluZywgYm9vbGVhbj47XG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgdHlwZT86ICdidXR0b24nIHwgJ3N1Ym1pdCcgfCAncmVzZXQnO1xuICB2YWx1ZT86IGFueTtcbn1cblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnY3FhLW90aGVyLWJ1dHRvbicsXG4gIHRlbXBsYXRlVXJsOiAnLi9vdGhlci1idXR0b24uY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFtdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaFxufSlcbmV4cG9ydCBjbGFzcyBPdGhlckJ1dHRvbkNvbXBvbmVudCB7XG4gIC8vIFNpbmdsZSBidXR0b24gQVBJIChiYWNrd2FyZHMgY29tcGF0aWJsZSlcbiAgQElucHV0KCkgaWNvbjogc3RyaW5nID0gJyc7XG4gIEBJbnB1dCgpIGxhYmVsOiBzdHJpbmcgPSAnJztcbiAgQElucHV0KCkgY2xhc3Nlczogc3RyaW5nIHwgc3RyaW5nW10gfCBSZWNvcmQ8c3RyaW5nLCBib29sZWFuPiA9ICcnO1xuICBASW5wdXQoKSBjb2xvckNsYXNzOiBzdHJpbmcgfCBzdHJpbmdbXSB8IFJlY29yZDxzdHJpbmcsIGJvb2xlYW4+ID0gJyc7XG4gIEBJbnB1dCgpIGJ1dHRvbkNsYXNzOiBzdHJpbmcgPSAnJztcbiAgQElucHV0KCkgZGlzYWJsZWQ6IGJvb2xlYW4gPSBmYWxzZTtcbiAgQElucHV0KCkgdHlwZTogJ2J1dHRvbicgfCAnc3VibWl0JyB8ICdyZXNldCcgPSAnYnV0dG9uJztcblxuICAvLyBHcm91cCBidXR0b25zIEFQSSAobmV3KVxuICBASW5wdXQoKSBidXR0b25zOiBPdGhlckJ1dHRvbkl0ZW1bXSB8IG51bGwgPSBudWxsO1xuICAvKiogRXh0cmEgY2xhc3NlcyBmb3IgdGhlIGJ1dHRvbiBncm91cCBjb250YWluZXIgKi9cbiAgQElucHV0KCkgZ3JvdXBDbGFzczogc3RyaW5nIHwgc3RyaW5nW10gfCBSZWNvcmQ8c3RyaW5nLCBib29sZWFuPiA9ICcnO1xuICAvKiogR2FwIHV0aWxpdHkgY2xhc3M7IGRlZmF1bHRzIHRvIFRhaWx3aW5kIHNwYWNpbmcgYXBwbGllZCBpbiB0ZW1wbGF0ZSAqL1xuICBASW5wdXQoKSBnYXBDbGFzczogc3RyaW5nID0gJ2NxYS1nYXAtMic7XG4gIC8qKiBXaGVuIHRydWUsIGFsbG93cyB3cmFwcGluZyB0byBuZXh0IGxpbmUgb24gc21hbGxlciBzY3JlZW5zICovXG4gIEBJbnB1dCgpIHdyYXA6IGJvb2xlYW4gPSB0cnVlO1xuXG4gIEBPdXRwdXQoKSBjbGlja2VkID0gbmV3IEV2ZW50RW1pdHRlcjxNb3VzZUV2ZW50PigpO1xuICAvKiogRW1pdHMgdGhlIGNvbmZpZyBvZiB0aGUgY2xpY2tlZCBidXR0b24gaW4gYSBncm91cCwgYWxvbmcgd2l0aCB0aGUgZXZlbnQgKi9cbiAgQE91dHB1dCgpIGJ1dHRvbkNsaWNrID0gbmV3IEV2ZW50RW1pdHRlcjx7IGV2ZW50OiBNb3VzZUV2ZW50OyBpdGVtOiBPdGhlckJ1dHRvbkl0ZW0gfT4oKTtcblxuICBvbkNsaWNrKGV2ZW50OiBNb3VzZUV2ZW50KTogdm9pZCB7XG4gICAgaWYgKHRoaXMuZGlzYWJsZWQpIHtcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgICBldmVudC5zdG9wUHJvcGFnYXRpb24oKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5jbGlja2VkLmVtaXQoZXZlbnQpO1xuICB9XG5cbiAgb25JdGVtQ2xpY2soZXZlbnQ6IE1vdXNlRXZlbnQsIGl0ZW06IE90aGVyQnV0dG9uSXRlbSk6IHZvaWQge1xuICAgIGlmIChpdGVtPy5kaXNhYmxlZCkge1xuICAgICAgZXZlbnQucHJldmVudERlZmF1bHQoKTtcbiAgICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLmJ1dHRvbkNsaWNrLmVtaXQoeyBldmVudCwgaXRlbSB9KTtcbiAgfVxufVxuXG4iLCI8ZGl2IGlkPVwiY3FhLXVpLXJvb3RcIiA+XG4gIDwhLS0gR3JvdXAgcmVuZGVyaW5nIGlmIGJ1dHRvbnMgYXJlIHByb3ZpZGVkIC0tPlxuICA8bmctY29udGFpbmVyICpuZ0lmPVwiYnV0dG9ucz8ubGVuZ3RoOyBlbHNlIHNpbmdsZUJ1dHRvblwiPlxuICAgIDxkaXYgY2xhc3M9XCJjcWEtaW5saW5lLWZsZXggY3FhLWl0ZW1zLWNlbnRlclwiIFtuZ0NsYXNzXT1cIlt3cmFwID8gJ2NxYS1mbGV4LXdyYXAnIDogJycsIGdhcENsYXNzLCBncm91cENsYXNzXVwiPlxuICAgICAgPGJ1dHRvbiAqbmdGb3I9XCJsZXQgYiBvZiBidXR0b25zXCJcbiAgICAgICAgW3R5cGVdPVwiYi50eXBlIHx8ICdidXR0b24nXCJcbiAgICAgICAgY2xhc3M9XCJjcWEtaW5saW5lLWZsZXggY3FhLWl0ZW1zLWNlbnRlciBjcWEtZ2FwLTEgY3FhLXB5LVs1Ljc1cHhdIGNxYS1weC1bMTEuNXB4XSBjcWEtcm91bmRlZC1bNXB4XSBjcWEtdGV4dC1bMTIuM3B4XSBjcWEtbGVhZGluZy1bMTcuNXB4XSBjcWEtZm9udC1tZWRpdW0gY3FhLWJvcmRlclwiXG4gICAgICAgIFtuZ0NsYXNzXT1cIltiLmNsYXNzZXMgfHwgY2xhc3NlcywgYnV0dG9uQ2xhc3MsIGIuY29sb3JDbGFzcyB8fCBjb2xvckNsYXNzXVwiXG4gICAgICAgIFtkaXNhYmxlZF09XCJiLmRpc2FibGVkXCJcbiAgICAgICAgKGNsaWNrKT1cIm9uSXRlbUNsaWNrKCRldmVudCwgYilcIj5cbiAgICAgICAgPG1hdC1pY29uICpuZ0lmPVwiYi5pY29uIHx8IGljb25cIiBjbGFzcz1cIiFjcWEtdy1bMTZweF0gIWNxYS1oLVsxNnB4XSAhY3FhLXRleHQtWzE2cHhdXCI+e3sgYi5pY29uIHx8IGljb24gfX08L21hdC1pY29uPlxuICAgICAgICA8c3Bhbj57eyBiLmxhYmVsIH19PC9zcGFuPlxuICAgICAgPC9idXR0b24+XG4gICAgPC9kaXY+XG4gIDwvbmctY29udGFpbmVyPlxuXG4gIDwhLS0gU2luZ2xlIGJ1dHRvbiAoYmFja3dhcmRzIGNvbXBhdGlibGUpIC0tPlxuICA8bmctdGVtcGxhdGUgI3NpbmdsZUJ1dHRvbj5cbiAgICA8YnV0dG9uIFt0eXBlXT1cInR5cGVcIlxuICAgICAgY2xhc3M9XCJjcWEtaW5saW5lLWZsZXggY3FhLWl0ZW1zLWNlbnRlciBjcWEtZ2FwLTEgY3FhLXB5LVs1Ljc1cHhdIGNxYS1weC1bMTEuNXB4XSBjcWEtcm91bmRlZC1bNXB4XSBjcWEtdGV4dC1bMTIuM3B4XSBjcWEtbGVhZGluZy1bMTcuNXB4XSBjcWEtZm9udC1tZWRpdW0gY3FhLWJvcmRlclwiXG4gICAgICBbbmdDbGFzc109XCJbY2xhc3NlcywgYnV0dG9uQ2xhc3MsIGNvbG9yQ2xhc3NdXCIgW2Rpc2FibGVkXT1cImRpc2FibGVkXCIgKGNsaWNrKT1cIm9uQ2xpY2soJGV2ZW50KVwiPlxuICAgICAgPG1hdC1pY29uICpuZ0lmPVwiaWNvblwiIGNsYXNzPVwiIWNxYS13LVsxNnB4XSAhY3FhLWgtWzE2cHhdICFjcWEtdGV4dC1bMTZweF1cIj57eyBpY29uIH19PC9tYXQtaWNvbj5cbiAgICAgIDxzcGFuPnt7IGxhYmVsIH19PC9zcGFuPlxuICAgIDwvYnV0dG9uPlxuICA8L25nLXRlbXBsYXRlPlxuPC9kaXY+Il19
|