@cqa-lib/cqa-ui 1.1.199 → 1.1.201
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/custom-input/custom-input.component.mjs +3 -3
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +7 -3
- package/esm2020/lib/test-case-details/element-list/element-list.component.mjs +67 -5
- package/esm2020/lib/test-case-details/element-popup/element-popup-data.mjs +1 -1
- package/esm2020/lib/test-case-details/element-popup/element-popup-form-data.mjs +2 -0
- package/esm2020/lib/test-case-details/element-popup/element-popup.component.mjs +289 -99
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +7 -3
- package/esm2020/lib/test-case-details/normal-step/normal-step.component.mjs +26 -27
- package/esm2020/lib/test-case-details/step-group/step-group.component.mjs +1 -1
- package/esm2020/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.mjs +24 -4
- package/esm2020/lib/test-case-details/test-case-step.models.mjs +1 -1
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +431 -140
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +417 -138
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/test-case-details/condition-step/condition-step.component.d.ts +6 -1
- package/lib/test-case-details/element-list/element-list.component.d.ts +15 -3
- package/lib/test-case-details/element-popup/element-popup-data.d.ts +3 -0
- package/lib/test-case-details/element-popup/element-popup-form-data.d.ts +22 -0
- package/lib/test-case-details/element-popup/element-popup.component.d.ts +72 -19
- package/lib/test-case-details/loop-step/loop-step.component.d.ts +6 -1
- package/lib/test-case-details/normal-step/normal-step.component.d.ts +1 -0
- package/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.d.ts +7 -2
- package/lib/test-case-details/test-case-step.models.d.ts +7 -0
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/styles.css +1 -1
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy } from '@angular/core';
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, ViewChild } from '@angular/core';
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
3
|
import * as i1 from "@angular/common";
|
|
4
4
|
export class ElementListComponent {
|
|
@@ -13,8 +13,14 @@ export class ElementListComponent {
|
|
|
13
13
|
this.labelsKey = 'labels';
|
|
14
14
|
/** Maximum height for the scrollable container (default: '200px') */
|
|
15
15
|
this.maxHeight = '200px';
|
|
16
|
+
/** Whether more items can be loaded */
|
|
17
|
+
this.hasMore = false;
|
|
16
18
|
/** Emitted when an item is clicked */
|
|
17
19
|
this.itemClick = new EventEmitter();
|
|
20
|
+
/** Emitted when user scrolls near the bottom and more items should be loaded */
|
|
21
|
+
this.loadMore = new EventEmitter();
|
|
22
|
+
/** Skip the first intersection callback (fires immediately when observe() is called) to prevent duplicate API calls on load */
|
|
23
|
+
this.skipNextIntersection = false;
|
|
18
24
|
}
|
|
19
25
|
/**
|
|
20
26
|
* Get the value from an item using the specified key
|
|
@@ -28,13 +34,65 @@ export class ElementListComponent {
|
|
|
28
34
|
onItemClick(item) {
|
|
29
35
|
this.itemClick.emit(item);
|
|
30
36
|
}
|
|
37
|
+
ngAfterViewInit() {
|
|
38
|
+
if (this.hasMore) {
|
|
39
|
+
setTimeout(() => this.setupScrollObserver(), 0);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
ngOnChanges(changes) {
|
|
43
|
+
if (changes['hasMore'] && this.hasMore && this.scrollContainer) {
|
|
44
|
+
setTimeout(() => this.setupScrollObserver(), 0);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
ngOnDestroy() {
|
|
48
|
+
if (this.scrollObserver) {
|
|
49
|
+
this.scrollObserver.disconnect();
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
setupScrollObserver() {
|
|
53
|
+
if (!this.hasMore || !this.scrollContainer) {
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
// Find the sentinel element (loading indicator)
|
|
57
|
+
const sentinel = this.scrollContainer.nativeElement.querySelector('.load-more-sentinel');
|
|
58
|
+
if (!sentinel) {
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
// Disconnect existing observer if any
|
|
62
|
+
if (this.scrollObserver) {
|
|
63
|
+
this.scrollObserver.disconnect();
|
|
64
|
+
}
|
|
65
|
+
// Skip the first intersection callback - it fires immediately when observe() is called
|
|
66
|
+
// if the sentinel is already in view, causing duplicate API calls (page 0 + page 1) on load
|
|
67
|
+
this.skipNextIntersection = true;
|
|
68
|
+
// Create IntersectionObserver to detect when sentinel comes into view
|
|
69
|
+
this.scrollObserver = new IntersectionObserver((entries) => {
|
|
70
|
+
for (const entry of entries) {
|
|
71
|
+
if (entry.isIntersecting && this.hasMore) {
|
|
72
|
+
if (this.skipNextIntersection) {
|
|
73
|
+
this.skipNextIntersection = false;
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
this.loadMore.emit();
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}, {
|
|
80
|
+
root: this.scrollContainer.nativeElement,
|
|
81
|
+
rootMargin: '50px',
|
|
82
|
+
threshold: 0.1
|
|
83
|
+
});
|
|
84
|
+
this.scrollObserver.observe(sentinel);
|
|
85
|
+
}
|
|
31
86
|
}
|
|
32
87
|
ElementListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
33
|
-
ElementListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ElementListComponent, selector: "cqa-element-list", inputs: { items: "items", titleKey: "titleKey", selectorKey: "selectorKey", labelsKey: "labelsKey", maxHeight: "maxHeight" }, outputs: { itemClick: "itemClick" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div \n class=\"cqa-max-h-[200px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB] cqa-flex cqa-flex-col cqa-gap-2\"\n [style.max-height]=\"maxHeight\">\n <div *ngFor=\"let item of items\" (click)=\"onItemClick(item)\">\n <div class=\"hover:cqa-bg-[#F5F5F5] cqa-flex cqa-items-center cqa-justify-between cqa-border cqa-border-[#FDE68A]\n cqa-rounded-lg cqa-p-3 cqa-cursor-pointer\">\n <div>\n <h6 class=\"cqa-text-[12px] cqa-leading-[100%] cqa-font-[400] cqa-text-[#171717]\">\n {{ getItemValue(item, titleKey) }}\n </h6>\n <span class=\"cqa-text-[10px] cqa-leading-[100%] cqa-text-[#6B7280]\">\n {{ getItemValue(item, selectorKey) }}\n </span>\n </div>\n <div class=\"cqa-flex cqa-gap-1\">\n <span \n *ngFor=\"let tag of getItemValue(item, labelsKey)\" \n class=\"cqa-text-[11px] cqa-text-[#4b74ec] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n {{ tag }}\n </span>\n </div>\n </div>\n </div>\n</div>\n\n", directives: [{ type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
88
|
+
ElementListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ElementListComponent, selector: "cqa-element-list", inputs: { items: "items", titleKey: "titleKey", selectorKey: "selectorKey", labelsKey: "labelsKey", maxHeight: "maxHeight", hasMore: "hasMore" }, outputs: { itemClick: "itemClick", loadMore: "loadMore" }, host: { classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div \n #scrollContainer\n class=\"cqa-max-h-[200px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB] cqa-flex cqa-flex-col cqa-gap-2\"\n [style.max-height]=\"maxHeight\">\n <div *ngFor=\"let item of items\" (click)=\"onItemClick(item)\">\n <div class=\"hover:cqa-bg-[#F5F5F5] cqa-flex cqa-items-center cqa-justify-between cqa-border cqa-border-[#FDE68A]\n cqa-rounded-lg cqa-p-3 cqa-cursor-pointer cqa-gap-1\" style=\"word-break: break-word;\">\n <div>\n <h6 class=\"cqa-text-[12px] cqa-leading-[100%] cqa-font-[400] cqa-text-[#171717]\">\n {{ getItemValue(item, titleKey) }}\n </h6>\n <span class=\"cqa-text-[10px] cqa-leading-[100%] cqa-text-[#6B7280]\">\n {{ getItemValue(item, selectorKey) }}\n </span>\n </div>\n <div class=\"cqa-flex cqa-gap-1\">\n <span \n *ngFor=\"let tag of getItemValue(item, labelsKey)\" \n class=\"cqa-text-[11px] cqa-text-[#4b74ec] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n {{ tag }}\n </span>\n </div>\n </div>\n </div>\n <!-- No elements found message -->\n <div *ngIf=\"items.length === 0 && !hasMore\" class=\"cqa-flex cqa-justify-center cqa-items-center cqa-py-4 cqa-text-[12px] cqa-text-[#6B7280]\">\n No elements found\n </div>\n <!-- Sentinel element for infinite scroll -->\n <div *ngIf=\"hasMore\" class=\"load-more-sentinel cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-gap-3 cqa-py-2\">\n <svg class=\"cqa-animate-spin cqa-h-6 cqa-w-6 cqa-text-[#3F43EE]\" 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-[12px] cqa-text-[#6B7280]\">{{ items.length === 0 ? 'Loading...' : 'Loading more...' }}</span>\n </div>\n</div>\n\n", directives: [{ type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
34
89
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementListComponent, decorators: [{
|
|
35
90
|
type: Component,
|
|
36
|
-
args: [{ selector: 'cqa-element-list', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div \n class=\"cqa-max-h-[200px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB] cqa-flex cqa-flex-col cqa-gap-2\"\n [style.max-height]=\"maxHeight\">\n <div *ngFor=\"let item of items\" (click)=\"onItemClick(item)\">\n <div class=\"hover:cqa-bg-[#F5F5F5] cqa-flex cqa-items-center cqa-justify-between cqa-border cqa-border-[#FDE68A]\n cqa-rounded-lg cqa-p-3 cqa-cursor-pointer\">\n <div>\n <h6 class=\"cqa-text-[12px] cqa-leading-[100%] cqa-font-[400] cqa-text-[#171717]\">\n {{ getItemValue(item, titleKey) }}\n </h6>\n <span class=\"cqa-text-[10px] cqa-leading-[100%] cqa-text-[#6B7280]\">\n {{ getItemValue(item, selectorKey) }}\n </span>\n </div>\n <div class=\"cqa-flex cqa-gap-1\">\n <span \n *ngFor=\"let tag of getItemValue(item, labelsKey)\" \n class=\"cqa-text-[11px] cqa-text-[#4b74ec] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n {{ tag }}\n </span>\n </div>\n </div>\n </div>\n</div>\n\n" }]
|
|
37
|
-
}], propDecorators: {
|
|
91
|
+
args: [{ selector: 'cqa-element-list', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div \n #scrollContainer\n class=\"cqa-max-h-[200px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB] cqa-flex cqa-flex-col cqa-gap-2\"\n [style.max-height]=\"maxHeight\">\n <div *ngFor=\"let item of items\" (click)=\"onItemClick(item)\">\n <div class=\"hover:cqa-bg-[#F5F5F5] cqa-flex cqa-items-center cqa-justify-between cqa-border cqa-border-[#FDE68A]\n cqa-rounded-lg cqa-p-3 cqa-cursor-pointer cqa-gap-1\" style=\"word-break: break-word;\">\n <div>\n <h6 class=\"cqa-text-[12px] cqa-leading-[100%] cqa-font-[400] cqa-text-[#171717]\">\n {{ getItemValue(item, titleKey) }}\n </h6>\n <span class=\"cqa-text-[10px] cqa-leading-[100%] cqa-text-[#6B7280]\">\n {{ getItemValue(item, selectorKey) }}\n </span>\n </div>\n <div class=\"cqa-flex cqa-gap-1\">\n <span \n *ngFor=\"let tag of getItemValue(item, labelsKey)\" \n class=\"cqa-text-[11px] cqa-text-[#4b74ec] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n {{ tag }}\n </span>\n </div>\n </div>\n </div>\n <!-- No elements found message -->\n <div *ngIf=\"items.length === 0 && !hasMore\" class=\"cqa-flex cqa-justify-center cqa-items-center cqa-py-4 cqa-text-[12px] cqa-text-[#6B7280]\">\n No elements found\n </div>\n <!-- Sentinel element for infinite scroll -->\n <div *ngIf=\"hasMore\" class=\"load-more-sentinel cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-gap-3 cqa-py-2\">\n <svg class=\"cqa-animate-spin cqa-h-6 cqa-w-6 cqa-text-[#3F43EE]\" 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-[12px] cqa-text-[#6B7280]\">{{ items.length === 0 ? 'Loading...' : 'Loading more...' }}</span>\n </div>\n</div>\n\n" }]
|
|
92
|
+
}], propDecorators: { scrollContainer: [{
|
|
93
|
+
type: ViewChild,
|
|
94
|
+
args: ['scrollContainer', { static: false }]
|
|
95
|
+
}], items: [{
|
|
38
96
|
type: Input
|
|
39
97
|
}], titleKey: [{
|
|
40
98
|
type: Input
|
|
@@ -44,7 +102,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
44
102
|
type: Input
|
|
45
103
|
}], maxHeight: [{
|
|
46
104
|
type: Input
|
|
105
|
+
}], hasMore: [{
|
|
106
|
+
type: Input
|
|
47
107
|
}], itemClick: [{
|
|
48
108
|
type: Output
|
|
109
|
+
}], loadMore: [{
|
|
110
|
+
type: Output
|
|
49
111
|
}] } });
|
|
50
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxlbWVudC1saXN0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvdGVzdC1jYXNlLWRldGFpbHMvZWxlbWVudC1saXN0L2VsZW1lbnQtbGlzdC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Rlc3QtY2FzZS1kZXRhaWxzL2VsZW1lbnQtbGlzdC9lbGVtZW50LWxpc3QuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLFlBQVksRUFBRSx1QkFBdUIsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBUWhHLE1BQU0sT0FBTyxvQkFBb0I7SUFOakM7UUFPRSxnQ0FBZ0M7UUFDdkIsVUFBSyxHQUFVLEVBQUUsQ0FBQztRQUUzQix5RUFBeUU7UUFDaEUsYUFBUSxHQUFXLE9BQU8sQ0FBQztRQUVwQywrRUFBK0U7UUFDdEUsZ0JBQVcsR0FBVyxVQUFVLENBQUM7UUFFMUMsd0VBQXdFO1FBQy9ELGNBQVMsR0FBVyxRQUFRLENBQUM7UUFFdEMscUVBQXFFO1FBQzVELGNBQVMsR0FBVyxPQUFPLENBQUM7UUFFckMsc0NBQXNDO1FBQzVCLGNBQVMsR0FBRyxJQUFJLFlBQVksRUFBTyxDQUFDO0tBZS9DO0lBYkM7O09BRUc7SUFDSCxZQUFZLENBQUMsSUFBUyxFQUFFLEdBQVc7UUFDakMsT0FBTyxJQUFJLEVBQUUsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNyQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxXQUFXLENBQUMsSUFBUztRQUNuQixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM1QixDQUFDOztpSEEvQlUsb0JBQW9CO3FHQUFwQixvQkFBb0Isb1FDUmpDLDhyQ0F5QkE7MkZEakJhLG9CQUFvQjtrQkFOaEMsU0FBUzsrQkFDRSxrQkFBa0IsUUFFdEIsRUFBRSxLQUFLLEVBQUUsYUFBYSxFQUFFLG1CQUNiLHVCQUF1QixDQUFDLE1BQU07OEJBSXRDLEtBQUs7c0JBQWIsS0FBSztnQkFHRyxRQUFRO3NCQUFoQixLQUFLO2dCQUdHLFdBQVc7c0JBQW5CLEtBQUs7Z0JBR0csU0FBUztzQkFBakIsS0FBSztnQkFHRyxTQUFTO3NCQUFqQixLQUFLO2dCQUdJLFNBQVM7c0JBQWxCLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPdXRwdXQsIEV2ZW50RW1pdHRlciwgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnY3FhLWVsZW1lbnQtbGlzdCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9lbGVtZW50LWxpc3QuY29tcG9uZW50Lmh0bWwnLFxuICBob3N0OiB7IGNsYXNzOiAnY3FhLXVpLXJvb3QnIH0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBFbGVtZW50TGlzdENvbXBvbmVudCB7XG4gIC8qKiBBcnJheSBvZiBpdGVtcyB0byBkaXNwbGF5ICovXG4gIEBJbnB1dCgpIGl0ZW1zOiBhbnlbXSA9IFtdO1xuICBcbiAgLyoqIEtleSB0byBhY2Nlc3MgdGhlIHRpdGxlIHByb3BlcnR5IGZyb20gZWFjaCBpdGVtIChkZWZhdWx0OiAndGl0bGUnKSAqL1xuICBASW5wdXQoKSB0aXRsZUtleTogc3RyaW5nID0gJ3RpdGxlJztcbiAgXG4gIC8qKiBLZXkgdG8gYWNjZXNzIHRoZSBzZWxlY3RvciBwcm9wZXJ0eSBmcm9tIGVhY2ggaXRlbSAoZGVmYXVsdDogJ3NlbGVjdG9yJykgKi9cbiAgQElucHV0KCkgc2VsZWN0b3JLZXk6IHN0cmluZyA9ICdzZWxlY3Rvcic7XG4gIFxuICAvKiogS2V5IHRvIGFjY2VzcyB0aGUgbGFiZWxzIGFycmF5IGZyb20gZWFjaCBpdGVtIChkZWZhdWx0OiAnbGFiZWxzJykgKi9cbiAgQElucHV0KCkgbGFiZWxzS2V5OiBzdHJpbmcgPSAnbGFiZWxzJztcbiAgXG4gIC8qKiBNYXhpbXVtIGhlaWdodCBmb3IgdGhlIHNjcm9sbGFibGUgY29udGFpbmVyIChkZWZhdWx0OiAnMjAwcHgnKSAqL1xuICBASW5wdXQoKSBtYXhIZWlnaHQ6IHN0cmluZyA9ICcyMDBweCc7XG5cbiAgLyoqIEVtaXR0ZWQgd2hlbiBhbiBpdGVtIGlzIGNsaWNrZWQgKi9cbiAgQE91dHB1dCgpIGl0ZW1DbGljayA9IG5ldyBFdmVudEVtaXR0ZXI8YW55PigpO1xuXG4gIC8qKlxuICAgKiBHZXQgdGhlIHZhbHVlIGZyb20gYW4gaXRlbSB1c2luZyB0aGUgc3BlY2lmaWVkIGtleVxuICAgKi9cbiAgZ2V0SXRlbVZhbHVlKGl0ZW06IGFueSwga2V5OiBzdHJpbmcpOiBhbnkge1xuICAgIHJldHVybiBpdGVtPy5ba2V5XTtcbiAgfVxuXG4gIC8qKlxuICAgKiBIYW5kbGUgaXRlbSBjbGlja1xuICAgKi9cbiAgb25JdGVtQ2xpY2soaXRlbTogYW55KTogdm9pZCB7XG4gICAgdGhpcy5pdGVtQ2xpY2suZW1pdChpdGVtKTtcbiAgfVxufVxuXG4iLCI8ZGl2IFxuICBjbGFzcz1cImNxYS1tYXgtaC1bMjAwcHhdIGNxYS1vdmVyZmxvdy15LWF1dG8gY3FhLXNjcm9sbGJhci10aGluIGNxYS1zY3JvbGxiYXItdHJhY2stdHJhbnNwYXJlbnQgY3FhLXNjcm9sbGJhci10aHVtYi1bI0U1RTdFQl0gY3FhLXNjcm9sbGJhci10aHVtYi1yb3VuZGVkLWZ1bGwgY3FhLXNjcm9sbGJhci10aHVtYi1ob3ZlcjpjcWEtYmctWyNEMUQ1REJdIGNxYS1mbGV4IGNxYS1mbGV4LWNvbCBjcWEtZ2FwLTJcIlxuICBbc3R5bGUubWF4LWhlaWdodF09XCJtYXhIZWlnaHRcIj5cbiAgPGRpdiAqbmdGb3I9XCJsZXQgaXRlbSBvZiBpdGVtc1wiIChjbGljayk9XCJvbkl0ZW1DbGljayhpdGVtKVwiPlxuICAgIDxkaXYgY2xhc3M9XCJob3ZlcjpjcWEtYmctWyNGNUY1RjVdIGNxYS1mbGV4IGNxYS1pdGVtcy1jZW50ZXIgY3FhLWp1c3RpZnktYmV0d2VlbiBjcWEtYm9yZGVyIGNxYS1ib3JkZXItWyNGREU2OEFdXG4gICAgICAgICAgICAgY3FhLXJvdW5kZWQtbGcgY3FhLXAtMyBjcWEtY3Vyc29yLXBvaW50ZXJcIj5cbiAgICAgIDxkaXY+XG4gICAgICAgIDxoNiBjbGFzcz1cImNxYS10ZXh0LVsxMnB4XSBjcWEtbGVhZGluZy1bMTAwJV0gY3FhLWZvbnQtWzQwMF0gY3FhLXRleHQtWyMxNzE3MTddXCI+XG4gICAgICAgICAge3sgZ2V0SXRlbVZhbHVlKGl0ZW0sIHRpdGxlS2V5KSB9fVxuICAgICAgICA8L2g2PlxuICAgICAgICA8c3BhbiBjbGFzcz1cImNxYS10ZXh0LVsxMHB4XSBjcWEtbGVhZGluZy1bMTAwJV0gY3FhLXRleHQtWyM2QjcyODBdXCI+XG4gICAgICAgICAge3sgZ2V0SXRlbVZhbHVlKGl0ZW0sIHNlbGVjdG9yS2V5KSB9fVxuICAgICAgICA8L3NwYW4+XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJjcWEtZmxleCBjcWEtZ2FwLTFcIj5cbiAgICAgICAgPHNwYW4gXG4gICAgICAgICAgKm5nRm9yPVwibGV0IHRhZyBvZiBnZXRJdGVtVmFsdWUoaXRlbSwgbGFiZWxzS2V5KVwiIFxuICAgICAgICAgIGNsYXNzPVwiY3FhLXRleHQtWzExcHhdIGNxYS10ZXh0LVsjNGI3NGVjXSBjcWEtcm91bmRlZC1mdWxsIGNxYS1weC0yIGNxYS1iZy1bI2VmZjZmZl0gY3FhLWJvcmRlciBjcWEtYm9yZGVyLXNvbGlkIGNxYS1ib3JkZXItWyNjOGUwZmZdXCI+XG4gICAgICAgICAge3sgdGFnIH19XG4gICAgICAgIDwvc3Bhbj5cbiAgICAgIDwvZGl2PlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbjwvZGl2PlxuXG4iXX0=
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"element-list.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/element-list/element-list.component.ts","../../../../../../src/lib/test-case-details/element-list/element-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, AfterViewInit, OnDestroy, ViewChild, ElementRef, OnChanges, SimpleChanges } from '@angular/core';\n\n@Component({\n  selector: 'cqa-element-list',\n  templateUrl: './element-list.component.html',\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ElementListComponent implements AfterViewInit, OnDestroy, OnChanges {\n  @ViewChild('scrollContainer', { static: false }) scrollContainer?: ElementRef<HTMLDivElement>;\n  \n  /** Array of items to display */\n  @Input() items: any[] = [];\n  \n  /** Key to access the title property from each item (default: 'title') */\n  @Input() titleKey: string = 'title';\n  \n  /** Key to access the selector property from each item (default: 'selector') */\n  @Input() selectorKey: string = 'selector';\n  \n  /** Key to access the labels array from each item (default: 'labels') */\n  @Input() labelsKey: string = 'labels';\n  \n  /** Maximum height for the scrollable container (default: '200px') */\n  @Input() maxHeight: string = '200px';\n\n  /** Whether more items can be loaded */\n  @Input() hasMore: boolean = false;\n\n  /** Emitted when an item is clicked */\n  @Output() itemClick = new EventEmitter<any>();\n\n  /** Emitted when user scrolls near the bottom and more items should be loaded */\n  @Output() loadMore = new EventEmitter<void>();\n\n  private scrollObserver?: IntersectionObserver;\n  /** Skip the first intersection callback (fires immediately when observe() is called) to prevent duplicate API calls on load */\n  private skipNextIntersection = false;\n\n  /**\n   * Get the value from an item using the specified key\n   */\n  getItemValue(item: any, key: string): any {\n    return item?.[key];\n  }\n\n  /**\n   * Handle item click\n   */\n  onItemClick(item: any): void {\n    this.itemClick.emit(item);\n  }\n\n  ngAfterViewInit(): void {\n    if (this.hasMore) {\n      setTimeout(() => this.setupScrollObserver(), 0);\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['hasMore'] && this.hasMore && this.scrollContainer) {\n      setTimeout(() => this.setupScrollObserver(), 0);\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.scrollObserver) {\n      this.scrollObserver.disconnect();\n    }\n  }\n\n  private setupScrollObserver(): void {\n    if (!this.hasMore || !this.scrollContainer) {\n      return;\n    }\n\n    // Find the sentinel element (loading indicator)\n    const sentinel = this.scrollContainer.nativeElement.querySelector('.load-more-sentinel') as HTMLElement | null;\n    if (!sentinel) {\n      return;\n    }\n\n    // Disconnect existing observer if any\n    if (this.scrollObserver) {\n      this.scrollObserver.disconnect();\n    }\n\n    // Skip the first intersection callback - it fires immediately when observe() is called\n    // if the sentinel is already in view, causing duplicate API calls (page 0 + page 1) on load\n    this.skipNextIntersection = true;\n\n    // Create IntersectionObserver to detect when sentinel comes into view\n    this.scrollObserver = new IntersectionObserver(\n      (entries) => {\n        for (const entry of entries) {\n          if (entry.isIntersecting && this.hasMore) {\n            if (this.skipNextIntersection) {\n              this.skipNextIntersection = false;\n              return;\n            }\n            this.loadMore.emit();\n          }\n        }\n      },\n      {\n        root: this.scrollContainer.nativeElement,\n        rootMargin: '50px',\n        threshold: 0.1\n      }\n    );\n\n    this.scrollObserver.observe(sentinel);\n  }\n}\n\n","<div \n  #scrollContainer\n  class=\"cqa-max-h-[200px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB] cqa-flex cqa-flex-col cqa-gap-2\"\n  [style.max-height]=\"maxHeight\">\n  <div *ngFor=\"let item of items\" (click)=\"onItemClick(item)\">\n    <div class=\"hover:cqa-bg-[#F5F5F5] cqa-flex cqa-items-center cqa-justify-between cqa-border cqa-border-[#FDE68A]\n             cqa-rounded-lg cqa-p-3 cqa-cursor-pointer cqa-gap-1\" style=\"word-break: break-word;\">\n      <div>\n        <h6 class=\"cqa-text-[12px] cqa-leading-[100%] cqa-font-[400] cqa-text-[#171717]\">\n          {{ getItemValue(item, titleKey) }}\n        </h6>\n        <span class=\"cqa-text-[10px] cqa-leading-[100%] cqa-text-[#6B7280]\">\n          {{ getItemValue(item, selectorKey) }}\n        </span>\n      </div>\n      <div class=\"cqa-flex cqa-gap-1\">\n        <span \n          *ngFor=\"let tag of getItemValue(item, labelsKey)\" \n          class=\"cqa-text-[11px] cqa-text-[#4b74ec] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n          {{ tag }}\n        </span>\n      </div>\n    </div>\n  </div>\n  <!-- No elements found message -->\n  <div *ngIf=\"items.length === 0 && !hasMore\" class=\"cqa-flex cqa-justify-center cqa-items-center cqa-py-4 cqa-text-[12px] cqa-text-[#6B7280]\">\n    No elements found\n  </div>\n  <!-- Sentinel element for infinite scroll -->\n  <div *ngIf=\"hasMore\" class=\"load-more-sentinel cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-gap-3 cqa-py-2\">\n    <svg class=\"cqa-animate-spin cqa-h-6 cqa-w-6 cqa-text-[#3F43EE]\" 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-[12px] cqa-text-[#6B7280]\">{{ items.length === 0 ? 'Loading...' : 'Loading more...' }}</span>\n  </div>\n</div>\n\n"]}
|
|
@@ -2,4 +2,4 @@ import { InjectionToken } from '@angular/core';
|
|
|
2
2
|
/** Sentinel returned from afterClosed() when user clicked "Edit in depth". */
|
|
3
3
|
export const ELEMENT_POPUP_EDIT_IN_DEPTH = Symbol('ElementPopupEditInDepth');
|
|
4
4
|
export const ELEMENT_POPUP_DATA = new InjectionToken('ELEMENT_POPUP_DATA');
|
|
5
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxlbWVudC1wb3B1cC1kYXRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZXN0LWNhc2UtZGV0YWlscy9lbGVtZW50LXBvcHVwL2VsZW1lbnQtcG9wdXAtZGF0YS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBbUIvQyw4RUFBOEU7QUFDOUUsTUFBTSxDQUFDLE1BQU0sMkJBQTJCLEdBQUcsTUFBTSxDQUFDLHlCQUF5QixDQUFDLENBQUM7QUFFN0UsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsSUFBSSxjQUFjLENBQ2xELG9CQUFvQixDQUNyQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiXG5pbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEVsZW1lbnRQb3B1cERhdGEge1xuICBlbGVtZW50PzogRWxlbWVudFBvcHVwRGF0YUVsZW1lbnRzO1xuICBsYWJlbHM6IHN0cmluZ1tdO1xuICBlbGVtZW50czogRWxlbWVudFBvcHVwRGF0YUVsZW1lbnRzW107XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIGhlbHBVcmw/OiBzdHJpbmc7XG59XG5cblxuZXhwb3J0IGludGVyZmFjZSBFbGVtZW50UG9wdXBEYXRhRWxlbWVudHMge1xuICBsYWJlbHM6IHN0cmluZ1tdO1xuICB0aXRsZTogc3RyaW5nO1xuICBzZWxlY3Rvcj86IHN0cmluZztcbiAgaWQ/OiBudW1iZXI7XG4gIHNjcmVlbk5hbWU/OiBzdHJpbmc7XG4gIHNjcmVlbk5hbWVJZD86IG51bWJlcjtcbn1cbi8qKiBTZW50aW5lbCByZXR1cm5lZCBmcm9tIGFmdGVyQ2xvc2VkKCkgd2hlbiB1c2VyIGNsaWNrZWQgXCJFZGl0IGluIGRlcHRoXCIuICovXG5leHBvcnQgY29uc3QgRUxFTUVOVF9QT1BVUF9FRElUX0lOX0RFUFRIID0gU3ltYm9sKCdFbGVtZW50UG9wdXBFZGl0SW5EZXB0aCcpO1xuXG5leHBvcnQgY29uc3QgRUxFTUVOVF9QT1BVUF9EQVRBID0gbmV3IEluamVjdGlvblRva2VuPEVsZW1lbnRQb3B1cERhdGE+KFxuICAnRUxFTUVOVF9QT1BVUF9EQVRBJ1xuKTtcbiJdfQ==
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxlbWVudC1wb3B1cC1mb3JtLWRhdGEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Rlc3QtY2FzZS1kZXRhaWxzL2VsZW1lbnQtcG9wdXAvZWxlbWVudC1wb3B1cC1mb3JtLWRhdGEudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKiBQYXlsb2FkIGVtaXR0ZWQgd2hlbiB1c2VyIGNyZWF0ZXMgYSBuZXcgZWxlbWVudCAqL1xuZXhwb3J0IGludGVyZmFjZSBFbGVtZW50Q3JlYXRlUGF5bG9hZCB7XG4gIG5hbWU6IHN0cmluZztcbiAgc2NyZWVuTmFtZUlkPzogbnVtYmVyO1xuICBzY3JlZW5OYW1lTmFtZTogc3RyaW5nO1xuICBsb2NhdG9yVmFsdWU6IHN0cmluZztcbiAgbGFiZWxzOiBzdHJpbmdbXTtcbn1cblxuLyoqIFBheWxvYWQgZW1pdHRlZCB3aGVuIHVzZXIgdXBkYXRlcyBhbiBleGlzdGluZyBlbGVtZW50ICovXG5leHBvcnQgaW50ZXJmYWNlIEVsZW1lbnRVcGRhdGVQYXlsb2FkIHtcbiAgZWxlbWVudElkOiBudW1iZXI7XG4gIG5hbWU6IHN0cmluZztcbiAgc2NyZWVuTmFtZUlkPzogbnVtYmVyO1xuICBzY3JlZW5OYW1lTmFtZTogc3RyaW5nO1xuICBsb2NhdG9yVmFsdWU6IHN0cmluZztcbiAgbGFiZWxzOiBzdHJpbmdbXTtcbn1cblxuLyoqIFNjcmVlbiBuYW1lIG9wdGlvbiBmb3IgYXV0b2NvbXBsZXRlICovXG5leHBvcnQgaW50ZXJmYWNlIFNjcmVlbk5hbWVPcHRpb24ge1xuICBpZD86IG51bWJlcjtcbiAgbmFtZTogc3RyaW5nO1xufVxuIl19
|