@cqa-lib/cqa-ui 1.1.181 → 1.1.183
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/add-prerequisite-cases-section/add-prerequisite-cases-section.component.mjs +176 -0
- package/esm2020/lib/execution-screen/execution-step.models.mjs +1 -1
- package/esm2020/lib/execution-screen/main-step-collapse/main-step-collapse.component.mjs +36 -4
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/lib/utils/tw-overlay-container.mjs +3 -2
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +216 -6
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +210 -6
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/add-prerequisite-cases-section/add-prerequisite-cases-section.component.d.ts +88 -0
- package/lib/execution-screen/execution-step.models.d.ts +2 -0
- package/lib/execution-screen/main-step-collapse/main-step-collapse.component.d.ts +30 -1
- package/lib/ui-kit.module.d.ts +99 -98
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/styles.css +1 -1
package/esm2020/lib/add-prerequisite-cases-section/add-prerequisite-cases-section.component.mjs
ADDED
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, ContentChild, TemplateRef, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "../dynamic-select/dynamic-select-field.component";
|
|
5
|
+
import * as i3 from "../button/button.component";
|
|
6
|
+
import * as i4 from "@angular/common";
|
|
7
|
+
import * as i5 from "@angular/material/tooltip";
|
|
8
|
+
import * as i6 from "ngx-drag-drop";
|
|
9
|
+
export class AddPrerequisiteCasesSectionComponent {
|
|
10
|
+
constructor(cdr) {
|
|
11
|
+
this.cdr = cdr;
|
|
12
|
+
/** Whether the section is expanded (showing the form) */
|
|
13
|
+
this.expanded = false;
|
|
14
|
+
/** Array of row descriptors, each with a unique key */
|
|
15
|
+
this.rows = [];
|
|
16
|
+
/** Section label (e.g. "Prerequisite Cases") */
|
|
17
|
+
this.label = 'Prerequisite Cases';
|
|
18
|
+
/** Tooltip for the info icon */
|
|
19
|
+
this.infoTooltip = 'Add prerequisite test cases that must run before this test case';
|
|
20
|
+
/** Text for the "Add Another" button */
|
|
21
|
+
this.addAnotherText = 'Add Another';
|
|
22
|
+
/** Text for the Update button */
|
|
23
|
+
this.updateText = 'Update';
|
|
24
|
+
/** Text for the Cancel button */
|
|
25
|
+
this.cancelText = 'Cancel';
|
|
26
|
+
/** Text for the collapsed "Add" trigger */
|
|
27
|
+
this.addTriggerText = 'Add Prerequisite';
|
|
28
|
+
/** Validation message shown when Add Another/Update are disabled (not all rows have values) */
|
|
29
|
+
this.selectFirstValidationMessage = 'Please select a prerequisite for each row first, then click Add Another.';
|
|
30
|
+
this.expandedChange = new EventEmitter();
|
|
31
|
+
this.addRow = new EventEmitter();
|
|
32
|
+
this.removeRow = new EventEmitter();
|
|
33
|
+
this.rowsReordered = new EventEmitter();
|
|
34
|
+
/** Emitted when Update button is clicked */
|
|
35
|
+
this.update = new EventEmitter();
|
|
36
|
+
/** Emitted when Cancel button is clicked */
|
|
37
|
+
this.cancel = new EventEmitter();
|
|
38
|
+
/** Emitted when a single row's selection changes */
|
|
39
|
+
this.selectionChange = new EventEmitter();
|
|
40
|
+
/** Emitted when any select changes - returns all current prerequisite selections in order */
|
|
41
|
+
this.prerequisitesChange = new EventEmitter();
|
|
42
|
+
/** Set to true when user clicks Add Another while invalid - shows validation message */
|
|
43
|
+
this.showAddAnotherValidation = false;
|
|
44
|
+
}
|
|
45
|
+
toggleExpanded(show) {
|
|
46
|
+
this.expanded = show;
|
|
47
|
+
this.expandedChange.emit(show);
|
|
48
|
+
}
|
|
49
|
+
/** Called when user clicks "Add Prerequisite" with no rows - expands and adds first row */
|
|
50
|
+
onAddPrerequisiteClick() {
|
|
51
|
+
this.toggleExpanded(true);
|
|
52
|
+
this.addRow.emit();
|
|
53
|
+
}
|
|
54
|
+
onAddRow() {
|
|
55
|
+
if (!this.allRowsHaveValues()) {
|
|
56
|
+
this.showAddAnotherValidation = true;
|
|
57
|
+
this.cdr.markForCheck();
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
this.showAddAnotherValidation = false;
|
|
61
|
+
this.addRow.emit();
|
|
62
|
+
}
|
|
63
|
+
onRemoveRow(index) {
|
|
64
|
+
this.removeRow.emit(index);
|
|
65
|
+
if (this.rows.length <= 1) {
|
|
66
|
+
this.toggleExpanded(false);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
onDndDrop(event) {
|
|
70
|
+
if (event.data == null || event.index == null)
|
|
71
|
+
return;
|
|
72
|
+
const dropRow = event.data;
|
|
73
|
+
let dropIndex = event.index;
|
|
74
|
+
const currentIndex = this.rows.findIndex((r) => r.key === dropRow.key);
|
|
75
|
+
if (currentIndex < 0)
|
|
76
|
+
return;
|
|
77
|
+
if (currentIndex === dropIndex)
|
|
78
|
+
return;
|
|
79
|
+
const newRows = [...this.rows];
|
|
80
|
+
newRows.splice(currentIndex, 1);
|
|
81
|
+
// If moving forward, dropIndex shifts after removal
|
|
82
|
+
if (currentIndex < dropIndex) {
|
|
83
|
+
dropIndex--;
|
|
84
|
+
}
|
|
85
|
+
newRows.splice(dropIndex, 0, dropRow);
|
|
86
|
+
this.rowsReordered.emit(newRows);
|
|
87
|
+
this.cdr.markForCheck();
|
|
88
|
+
}
|
|
89
|
+
trackByKey(_i, row) {
|
|
90
|
+
return row.key;
|
|
91
|
+
}
|
|
92
|
+
onUpdate() {
|
|
93
|
+
this.update.emit();
|
|
94
|
+
}
|
|
95
|
+
onCancel() {
|
|
96
|
+
this.cancel.emit();
|
|
97
|
+
}
|
|
98
|
+
onSelectionChange(event, index) {
|
|
99
|
+
this.selectionChange.emit({ ...event, index });
|
|
100
|
+
this.emitAllPrerequisites();
|
|
101
|
+
if (this.allRowsHaveValues()) {
|
|
102
|
+
this.showAddAnotherValidation = false;
|
|
103
|
+
}
|
|
104
|
+
this.cdr.markForCheck();
|
|
105
|
+
}
|
|
106
|
+
/** True when every row has a non-empty value (no placeholders) */
|
|
107
|
+
allRowsHaveValues() {
|
|
108
|
+
if (!this.form || !this.rows.length)
|
|
109
|
+
return false;
|
|
110
|
+
return this.rows.every((row) => {
|
|
111
|
+
const val = this.form.get(row.key)?.value;
|
|
112
|
+
return val != null && val !== '';
|
|
113
|
+
});
|
|
114
|
+
}
|
|
115
|
+
/** Emits all current prerequisite selections (key + value) in row order */
|
|
116
|
+
emitAllPrerequisites() {
|
|
117
|
+
if (!this.form || !this.rows.length)
|
|
118
|
+
return;
|
|
119
|
+
const selections = this.rows.map((row) => ({
|
|
120
|
+
key: row.key,
|
|
121
|
+
value: this.form.get(row.key)?.value ?? null,
|
|
122
|
+
}));
|
|
123
|
+
this.prerequisitesChange.emit(selections);
|
|
124
|
+
}
|
|
125
|
+
getConfig(index) {
|
|
126
|
+
return this.getSelectConfig?.(index) ?? { key: `row_${index}`, options: [] };
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
AddPrerequisiteCasesSectionComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AddPrerequisiteCasesSectionComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
130
|
+
AddPrerequisiteCasesSectionComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AddPrerequisiteCasesSectionComponent, selector: "cqa-add-prerequisite-cases-section", inputs: { expanded: "expanded", form: "form", rows: "rows", getSelectConfig: "getSelectConfig", label: "label", infoTooltip: "infoTooltip", addAnotherText: "addAnotherText", updateText: "updateText", cancelText: "cancelText", addTriggerText: "addTriggerText", selectFirstValidationMessage: "selectFirstValidationMessage" }, outputs: { expandedChange: "expandedChange", addRow: "addRow", removeRow: "removeRow", rowsReordered: "rowsReordered", update: "update", cancel: "cancel", selectionChange: "selectionChange", prerequisitesChange: "prerequisitesChange" }, host: { classAttribute: "cqa-ui-root cqa-add-prerequisite-cases-section-host" }, queries: [{ propertyName: "selectBodyTpl", first: true, predicate: ["selectBody"], descendants: true, read: TemplateRef }], ngImport: i0, template: "<ng-container *ngIf=\"rows.length === 0\">\n <div\n class=\"cqa-border-2 cqa-border-dashed cqa-border-[#D1D5DC] cqa-rounded-lg cqa-p-1.5 cqa-mt-4 cqa-bg-white cqa-cursor-pointer cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[27px] cqa-hover:cqa-border-[#9ca3af] cqa-hover:cqa-bg-[#f9fafb] cqa-active:cqa-border-[#6366f1] cqa-active:cqa-bg-[#eff6ff]\"\n (click)=\"onAddPrerequisiteClick()\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-center cqa-gap-2 cqa-text-sm cqa-font-medium\">\n <mat-icon class=\"cqa-text-[14px] cqa-w-[14px] cqa-h-[14px] cqa-text-[#4a5565]\">add</mat-icon>\n <span class=\"cqa-text-[#4a5565] cqa-font-normal cqa-font-inter cqa-text-xs\">{{ addTriggerText }}</span>\n </div>\n </div>\n</ng-container>\n\n<!-- Collapsed summary when we have rows but expanded=false -->\n<ng-container *ngIf=\"rows.length > 0 && !expanded\">\n <div\n class=\"cqa-border-2 cqa-border-dashed cqa-border-[#D1D5DC] cqa-rounded-lg cqa-p-1.5 cqa-mt-4 cqa-bg-white cqa-cursor-pointer cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-flex cqa-items-center cqa-justify-between cqa-min-h-[27px] cqa-hover:cqa-border-[#9ca3af] cqa-hover:cqa-bg-[#f9fafb] cqa-active:cqa-border-[#6366f1] cqa-active:cqa-bg-[#eff6ff]\"\n (click)=\"toggleExpanded(true)\">\n <span class=\"cqa-text-[#4a5565] cqa-font-normal cqa-font-inter cqa-text-xs\">{{ rows.length }} prerequisite{{ rows.length !== 1 ? 's' : '' }}</span>\n <mat-icon class=\"cqa-text-[14px] cqa-w-[14px] cqa-h-[14px] cqa-text-[#4a5565]\">expand_more</mat-icon>\n </div>\n</ng-container>\n\n<ng-container *ngIf=\"rows.length > 0 && expanded\">\n <div class=\"cqa-mt-4 cqa-p-3 cqa-bg-[#f9fafb] cqa-border cqa-border-[#e5e7eb] cqa-rounded-lg\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5 cqa-mb-3\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#374151] cqa-m-0 cqa-leading-none\">{{ label }}</label>\n <mat-icon class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-4 cqa-h-4 cqa-text-sm cqa-text-[#6b7280] cqa-cursor-help cqa-flex-shrink-0\" [matTooltip]=\"infoTooltip\" matTooltipPosition=\"above\" matTooltipShowDelay=\"300\">info</mat-icon>\n <button\n type=\"button\"\n class=\"cqa-ml-auto cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-p-0 cqa-bg-transparent cqa-border-none cqa-rounded-md cqa-text-[#6b7280] cqa-cursor-pointer cqa-hover:cqa-bg-[#f3f4f6] cqa-hover:cqa-text-[#374151]\"\n (click)=\"toggleExpanded(false)\"\n matTooltip=\"Close\">\n <mat-icon class=\"cqa-text-xl cqa-w-5 cqa-h-5\">expand_less</mat-icon>\n </button>\n </div>\n\n <div\n class=\"prerequisite-drop-list cqa-flex cqa-flex-col cqa-gap-2 cqa-mb-3\"\n [dndDropzone]=\"['prerequisite-row']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"prerequisite-drag-placeholder\">Drop here</div>\n <div\n *ngFor=\"let row of rows; let i = index; trackBy: trackByKey\"\n class=\"prerequisite-drag-item cqa-flex cqa-items-center cqa-gap-2\"\n [dndDraggable]=\"row\"\n [dndDisableIf]=\"rows.length <= 1\"\n dndEffectAllowed=\"move\"\n dndType=\"prerequisite-row\">\n <!-- 9-dot grid drag handle (only when multiple rows to reorder) -->\n <div *ngIf=\"rows.length > 1\" dndHandle class=\"cqa-flex cqa-flex-shrink-0 cqa-items-center cqa-justify-center cqa-cursor-grab cqa-text-[#6B7280] cqa-hover:cqa-text-[#111827] active:cqa-cursor-grabbing\" matTooltip=\"Drag to reorder\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n </svg>\n </div>\n\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <!-- Custom template (e.g. parent-provided cqa-dynamic-select) -->\n <ng-container *ngIf=\"selectBodyTpl\">\n <ng-container\n *ngTemplateOutlet=\"selectBodyTpl; context: {\n $implicit: row,\n index: i,\n form: form,\n config: getConfig(i)\n }\">\n </ng-container>\n </ng-container>\n <!-- Default: built-in cqa-dynamic-select -->\n <cqa-dynamic-select\n *ngIf=\"!selectBodyTpl && form\"\n [form]=\"form\"\n [config]=\"getConfig(i)\"\n (selectionChange)=\"onSelectionChange($event, i)\">\n </cqa-dynamic-select>\n </div>\n\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-min-w-[32px] cqa-bg-transparent cqa-border-none cqa-rounded-md cqa-text-[#6b7280] cqa-cursor-pointer cqa-p-0 cqa-hover:cqa-bg-[#f3f4f6] cqa-hover:cqa-text-[#374151]\"\n (click)=\"onRemoveRow(i)\"\n matTooltip=\"Cancel\">\n <mat-icon class=\"cqa-text-lg cqa-w-[18px] cqa-h-[18px]\">close</mat-icon>\n </button>\n </div>\n </div>\n\n <div *ngIf=\"showAddAnotherValidation && !allRowsHaveValues()\" class=\"cqa-text-xs cqa-text-[#dc2626] cqa-mb-2\">\n {{ selectFirstValidationMessage }}\n </div>\n <cqa-button\n variant=\"text\"\n icon=\"add\"\n [text]=\"addAnotherText\"\n btnSize=\"md\"\n customClass=\"cqa-text-[#3b82f6] cqa-mb-3 cqa-hover:cqa-bg-[#eff6ff]\"\n (clicked)=\"onAddRow()\">\n </cqa-button>\n <div class=\"cqa-flex cqa-gap-3 cqa-justify-end\">\n <button\n type=\"button\"\n class=\"cqa-py-2.5 cqa-px-4 cqa-bg-white cqa-text-[#374151] cqa-border cqa-border-[#d1d5db] cqa-rounded-md cqa-text-sm cqa-font-semibold cqa-cursor-pointer cqa-hover:cqa-bg-[#f9fafb]\"\n (click)=\"onCancel()\">\n {{ cancelText }}\n </button>\n <button\n type=\"button\"\n class=\"cqa-py-2.5 cqa-px-4 cqa-rounded-md cqa-text-sm cqa-font-semibold cqa-border-none cqa-cursor-pointer disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed cqa-bg-[#1a56db] cqa-text-white cqa-hover:cqa-bg-[#1647b8] disabled:cqa-bg-[#1a56db]\"\n [disabled]=\"!allRowsHaveValues()\"\n (click)=\"onUpdate()\">\n {{ updateText }}\n </button>\n </div>\n </div>\n</ng-container>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i6.DndDropzoneDirective, selector: "[dndDropzone]", inputs: ["dndDropzone", "dndEffectAllowed", "dndAllowExternal", "dndHorizontal", "dndDragoverClass", "dndDropzoneDisabledClass", "dndDisableIf", "dndDisableDropIf"], outputs: ["dndDragover", "dndDrop"] }, { type: i6.DndPlaceholderRefDirective, selector: "[dndPlaceholderRef]" }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i6.DndDraggableDirective, selector: "[dndDraggable]", inputs: ["dndDraggable", "dndEffectAllowed", "dndType", "dndDraggingClass", "dndDraggingSourceClass", "dndDraggableDisabledClass", "dndDragImageOffsetFunction", "dndDisableIf", "dndDisableDragIf"], outputs: ["dndStart", "dndDrag", "dndEnd", "dndMoved", "dndCopied", "dndLinked", "dndCanceled"] }, { type: i6.DndHandleDirective, selector: "[dndHandle]" }, { type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
131
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AddPrerequisiteCasesSectionComponent, decorators: [{
|
|
132
|
+
type: Component,
|
|
133
|
+
args: [{ selector: 'cqa-add-prerequisite-cases-section', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root cqa-add-prerequisite-cases-section-host' }, template: "<ng-container *ngIf=\"rows.length === 0\">\n <div\n class=\"cqa-border-2 cqa-border-dashed cqa-border-[#D1D5DC] cqa-rounded-lg cqa-p-1.5 cqa-mt-4 cqa-bg-white cqa-cursor-pointer cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[27px] cqa-hover:cqa-border-[#9ca3af] cqa-hover:cqa-bg-[#f9fafb] cqa-active:cqa-border-[#6366f1] cqa-active:cqa-bg-[#eff6ff]\"\n (click)=\"onAddPrerequisiteClick()\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-center cqa-gap-2 cqa-text-sm cqa-font-medium\">\n <mat-icon class=\"cqa-text-[14px] cqa-w-[14px] cqa-h-[14px] cqa-text-[#4a5565]\">add</mat-icon>\n <span class=\"cqa-text-[#4a5565] cqa-font-normal cqa-font-inter cqa-text-xs\">{{ addTriggerText }}</span>\n </div>\n </div>\n</ng-container>\n\n<!-- Collapsed summary when we have rows but expanded=false -->\n<ng-container *ngIf=\"rows.length > 0 && !expanded\">\n <div\n class=\"cqa-border-2 cqa-border-dashed cqa-border-[#D1D5DC] cqa-rounded-lg cqa-p-1.5 cqa-mt-4 cqa-bg-white cqa-cursor-pointer cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-flex cqa-items-center cqa-justify-between cqa-min-h-[27px] cqa-hover:cqa-border-[#9ca3af] cqa-hover:cqa-bg-[#f9fafb] cqa-active:cqa-border-[#6366f1] cqa-active:cqa-bg-[#eff6ff]\"\n (click)=\"toggleExpanded(true)\">\n <span class=\"cqa-text-[#4a5565] cqa-font-normal cqa-font-inter cqa-text-xs\">{{ rows.length }} prerequisite{{ rows.length !== 1 ? 's' : '' }}</span>\n <mat-icon class=\"cqa-text-[14px] cqa-w-[14px] cqa-h-[14px] cqa-text-[#4a5565]\">expand_more</mat-icon>\n </div>\n</ng-container>\n\n<ng-container *ngIf=\"rows.length > 0 && expanded\">\n <div class=\"cqa-mt-4 cqa-p-3 cqa-bg-[#f9fafb] cqa-border cqa-border-[#e5e7eb] cqa-rounded-lg\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5 cqa-mb-3\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#374151] cqa-m-0 cqa-leading-none\">{{ label }}</label>\n <mat-icon class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-4 cqa-h-4 cqa-text-sm cqa-text-[#6b7280] cqa-cursor-help cqa-flex-shrink-0\" [matTooltip]=\"infoTooltip\" matTooltipPosition=\"above\" matTooltipShowDelay=\"300\">info</mat-icon>\n <button\n type=\"button\"\n class=\"cqa-ml-auto cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-p-0 cqa-bg-transparent cqa-border-none cqa-rounded-md cqa-text-[#6b7280] cqa-cursor-pointer cqa-hover:cqa-bg-[#f3f4f6] cqa-hover:cqa-text-[#374151]\"\n (click)=\"toggleExpanded(false)\"\n matTooltip=\"Close\">\n <mat-icon class=\"cqa-text-xl cqa-w-5 cqa-h-5\">expand_less</mat-icon>\n </button>\n </div>\n\n <div\n class=\"prerequisite-drop-list cqa-flex cqa-flex-col cqa-gap-2 cqa-mb-3\"\n [dndDropzone]=\"['prerequisite-row']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"prerequisite-drag-placeholder\">Drop here</div>\n <div\n *ngFor=\"let row of rows; let i = index; trackBy: trackByKey\"\n class=\"prerequisite-drag-item cqa-flex cqa-items-center cqa-gap-2\"\n [dndDraggable]=\"row\"\n [dndDisableIf]=\"rows.length <= 1\"\n dndEffectAllowed=\"move\"\n dndType=\"prerequisite-row\">\n <!-- 9-dot grid drag handle (only when multiple rows to reorder) -->\n <div *ngIf=\"rows.length > 1\" dndHandle class=\"cqa-flex cqa-flex-shrink-0 cqa-items-center cqa-justify-center cqa-cursor-grab cqa-text-[#6B7280] cqa-hover:cqa-text-[#111827] active:cqa-cursor-grabbing\" matTooltip=\"Drag to reorder\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n </svg>\n </div>\n\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <!-- Custom template (e.g. parent-provided cqa-dynamic-select) -->\n <ng-container *ngIf=\"selectBodyTpl\">\n <ng-container\n *ngTemplateOutlet=\"selectBodyTpl; context: {\n $implicit: row,\n index: i,\n form: form,\n config: getConfig(i)\n }\">\n </ng-container>\n </ng-container>\n <!-- Default: built-in cqa-dynamic-select -->\n <cqa-dynamic-select\n *ngIf=\"!selectBodyTpl && form\"\n [form]=\"form\"\n [config]=\"getConfig(i)\"\n (selectionChange)=\"onSelectionChange($event, i)\">\n </cqa-dynamic-select>\n </div>\n\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-min-w-[32px] cqa-bg-transparent cqa-border-none cqa-rounded-md cqa-text-[#6b7280] cqa-cursor-pointer cqa-p-0 cqa-hover:cqa-bg-[#f3f4f6] cqa-hover:cqa-text-[#374151]\"\n (click)=\"onRemoveRow(i)\"\n matTooltip=\"Cancel\">\n <mat-icon class=\"cqa-text-lg cqa-w-[18px] cqa-h-[18px]\">close</mat-icon>\n </button>\n </div>\n </div>\n\n <div *ngIf=\"showAddAnotherValidation && !allRowsHaveValues()\" class=\"cqa-text-xs cqa-text-[#dc2626] cqa-mb-2\">\n {{ selectFirstValidationMessage }}\n </div>\n <cqa-button\n variant=\"text\"\n icon=\"add\"\n [text]=\"addAnotherText\"\n btnSize=\"md\"\n customClass=\"cqa-text-[#3b82f6] cqa-mb-3 cqa-hover:cqa-bg-[#eff6ff]\"\n (clicked)=\"onAddRow()\">\n </cqa-button>\n <div class=\"cqa-flex cqa-gap-3 cqa-justify-end\">\n <button\n type=\"button\"\n class=\"cqa-py-2.5 cqa-px-4 cqa-bg-white cqa-text-[#374151] cqa-border cqa-border-[#d1d5db] cqa-rounded-md cqa-text-sm cqa-font-semibold cqa-cursor-pointer cqa-hover:cqa-bg-[#f9fafb]\"\n (click)=\"onCancel()\">\n {{ cancelText }}\n </button>\n <button\n type=\"button\"\n class=\"cqa-py-2.5 cqa-px-4 cqa-rounded-md cqa-text-sm cqa-font-semibold cqa-border-none cqa-cursor-pointer disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed cqa-bg-[#1a56db] cqa-text-white cqa-hover:cqa-bg-[#1647b8] disabled:cqa-bg-[#1a56db]\"\n [disabled]=\"!allRowsHaveValues()\"\n (click)=\"onUpdate()\">\n {{ updateText }}\n </button>\n </div>\n </div>\n</ng-container>\n", styles: [] }]
|
|
134
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { expanded: [{
|
|
135
|
+
type: Input
|
|
136
|
+
}], form: [{
|
|
137
|
+
type: Input
|
|
138
|
+
}], rows: [{
|
|
139
|
+
type: Input
|
|
140
|
+
}], getSelectConfig: [{
|
|
141
|
+
type: Input
|
|
142
|
+
}], label: [{
|
|
143
|
+
type: Input
|
|
144
|
+
}], infoTooltip: [{
|
|
145
|
+
type: Input
|
|
146
|
+
}], addAnotherText: [{
|
|
147
|
+
type: Input
|
|
148
|
+
}], updateText: [{
|
|
149
|
+
type: Input
|
|
150
|
+
}], cancelText: [{
|
|
151
|
+
type: Input
|
|
152
|
+
}], addTriggerText: [{
|
|
153
|
+
type: Input
|
|
154
|
+
}], selectFirstValidationMessage: [{
|
|
155
|
+
type: Input
|
|
156
|
+
}], selectBodyTpl: [{
|
|
157
|
+
type: ContentChild,
|
|
158
|
+
args: ['selectBody', { read: TemplateRef }]
|
|
159
|
+
}], expandedChange: [{
|
|
160
|
+
type: Output
|
|
161
|
+
}], addRow: [{
|
|
162
|
+
type: Output
|
|
163
|
+
}], removeRow: [{
|
|
164
|
+
type: Output
|
|
165
|
+
}], rowsReordered: [{
|
|
166
|
+
type: Output
|
|
167
|
+
}], update: [{
|
|
168
|
+
type: Output
|
|
169
|
+
}], cancel: [{
|
|
170
|
+
type: Output
|
|
171
|
+
}], selectionChange: [{
|
|
172
|
+
type: Output
|
|
173
|
+
}], prerequisitesChange: [{
|
|
174
|
+
type: Output
|
|
175
|
+
}] } });
|
|
176
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"add-prerequisite-cases-section.component.js","sourceRoot":"","sources":["../../../../../src/lib/add-prerequisite-cases-section/add-prerequisite-cases-section.component.ts","../../../../../src/lib/add-prerequisite-cases-section/add-prerequisite-cases-section.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ChangeDetectionStrategy,\n  ContentChild,\n  TemplateRef,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { FormGroup } from '@angular/forms';\nimport { DndDropEvent } from 'ngx-drag-drop';\nimport { DynamicSelectFieldConfig } from '../dynamic-select/dynamic-select-field.component';\n\nexport interface PrerequisiteRow {\n  key: string;\n}\n\n/** Emitted when any prerequisite select changes - contains all current selections */\nexport interface PrerequisiteSelection {\n  key: string;\n  value: unknown;\n}\n\n@Component({\n  selector: 'cqa-add-prerequisite-cases-section',\n  templateUrl: './add-prerequisite-cases-section.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root cqa-add-prerequisite-cases-section-host' },\n})\nexport class AddPrerequisiteCasesSectionComponent {\n  /** Whether the section is expanded (showing the form) */\n  @Input() expanded = false;\n\n  /** Form group containing controls for each row (keyed by row.key) */\n  @Input() form!: FormGroup;\n\n  /** Array of row descriptors, each with a unique key */\n  @Input() rows: PrerequisiteRow[] = [];\n\n  /**\n   * Function to get the dynamic select config for a given row index.\n   * Each row can have different options. To avoid duplicates, exclude options\n   * already selected in other rows (keep current row's selection so it displays).\n   */\n  @Input() getSelectConfig!: (index: number) => DynamicSelectFieldConfig;\n\n  /** Section label (e.g. \"Prerequisite Cases\") */\n  @Input() label = 'Prerequisite Cases';\n\n  /** Tooltip for the info icon */\n  @Input() infoTooltip = 'Add prerequisite test cases that must run before this test case';\n\n  /** Text for the \"Add Another\" button */\n  @Input() addAnotherText = 'Add Another';\n\n  /** Text for the Update button */\n  @Input() updateText = 'Update';\n\n  /** Text for the Cancel button */\n  @Input() cancelText = 'Cancel';\n\n  /** Text for the collapsed \"Add\" trigger */\n  @Input() addTriggerText = 'Add Prerequisite';\n\n  /** Validation message shown when Add Another/Update are disabled (not all rows have values) */\n  @Input() selectFirstValidationMessage = 'Please select a prerequisite for each row first, then click Add Another.';\n\n  /** Optional custom template for the select slot. Use ng-template #selectBody with let-row, let-index, let-form, let-config. */\n  @ContentChild('selectBody', { read: TemplateRef }) selectBodyTpl?: TemplateRef<{\n    $implicit: PrerequisiteRow;\n    index: number;\n    form: FormGroup;\n    config: DynamicSelectFieldConfig;\n  }>;\n\n  @Output() expandedChange = new EventEmitter<boolean>();\n  @Output() addRow = new EventEmitter<void>();\n  @Output() removeRow = new EventEmitter<number>();\n  @Output() rowsReordered = new EventEmitter<PrerequisiteRow[]>();\n  /** Emitted when Update button is clicked */\n  @Output() update = new EventEmitter<void>();\n  /** Emitted when Cancel button is clicked */\n  @Output() cancel = new EventEmitter<void>();\n  /** Emitted when a single row's selection changes */\n  @Output() selectionChange = new EventEmitter<{ key: string; value: unknown; index: number }>();\n  /** Emitted when any select changes - returns all current prerequisite selections in order */\n  @Output() prerequisitesChange = new EventEmitter<PrerequisiteSelection[]>();\n\n  /** Set to true when user clicks Add Another while invalid - shows validation message */\n  showAddAnotherValidation = false;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  toggleExpanded(show: boolean): void {\n    this.expanded = show;\n    this.expandedChange.emit(show);\n  }\n\n  /** Called when user clicks \"Add Prerequisite\" with no rows - expands and adds first row */\n  onAddPrerequisiteClick(): void {\n    this.toggleExpanded(true);\n    this.addRow.emit();\n  }\n\n  onAddRow(): void {\n    if (!this.allRowsHaveValues()) {\n      this.showAddAnotherValidation = true;\n      this.cdr.markForCheck();\n      return;\n    }\n    this.showAddAnotherValidation = false;\n    this.addRow.emit();\n  }\n\n  onRemoveRow(index: number): void {\n    this.removeRow.emit(index);\n    if (this.rows.length <= 1) {\n      this.toggleExpanded(false);\n    }\n  }\n\n  onDndDrop(event: DndDropEvent): void {\n    if (event.data == null || event.index == null) return;\n    const dropRow = event.data as PrerequisiteRow;\n    let dropIndex = event.index;\n    const currentIndex = this.rows.findIndex((r) => r.key === dropRow.key);\n    if (currentIndex < 0) return;\n    if (currentIndex === dropIndex) return;\n\n    const newRows = [...this.rows];\n    newRows.splice(currentIndex, 1);\n    // If moving forward, dropIndex shifts after removal\n    if (currentIndex < dropIndex) {\n      dropIndex--;\n    }\n    newRows.splice(dropIndex, 0, dropRow);\n    this.rowsReordered.emit(newRows);\n    this.cdr.markForCheck();\n  }\n\n  trackByKey(_i: number, row: PrerequisiteRow): string {\n    return row.key;\n  }\n\n  onUpdate(): void {\n    this.update.emit();\n  }\n\n  onCancel(): void {\n    this.cancel.emit();\n  }\n\n  onSelectionChange(event: { key: string; value: unknown }, index: number): void {\n    this.selectionChange.emit({ ...event, index });\n    this.emitAllPrerequisites();\n    if (this.allRowsHaveValues()) {\n      this.showAddAnotherValidation = false;\n    }\n    this.cdr.markForCheck();\n  }\n\n  /** True when every row has a non-empty value (no placeholders) */\n  allRowsHaveValues(): boolean {\n    if (!this.form || !this.rows.length) return false;\n    return this.rows.every((row) => {\n      const val = this.form.get(row.key)?.value;\n      return val != null && val !== '';\n    });\n  }\n\n  /** Emits all current prerequisite selections (key + value) in row order */\n  private emitAllPrerequisites(): void {\n    if (!this.form || !this.rows.length) return;\n    const selections: PrerequisiteSelection[] = this.rows.map((row) => ({\n      key: row.key,\n      value: this.form.get(row.key)?.value ?? null,\n    }));\n    this.prerequisitesChange.emit(selections);\n  }\n\n  getConfig(index: number): DynamicSelectFieldConfig {\n    return this.getSelectConfig?.(index) ?? { key: `row_${index}`, options: [] };\n  }\n}\n","<ng-container *ngIf=\"rows.length === 0\">\n  <div\n    class=\"cqa-border-2 cqa-border-dashed cqa-border-[#D1D5DC] cqa-rounded-lg cqa-p-1.5 cqa-mt-4 cqa-bg-white cqa-cursor-pointer cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[27px] cqa-hover:cqa-border-[#9ca3af] cqa-hover:cqa-bg-[#f9fafb] cqa-active:cqa-border-[#6366f1] cqa-active:cqa-bg-[#eff6ff]\"\n    (click)=\"onAddPrerequisiteClick()\">\n    <div class=\"cqa-flex cqa-items-center cqa-justify-center cqa-gap-2 cqa-text-sm cqa-font-medium\">\n      <mat-icon class=\"cqa-text-[14px] cqa-w-[14px] cqa-h-[14px] cqa-text-[#4a5565]\">add</mat-icon>\n      <span class=\"cqa-text-[#4a5565] cqa-font-normal cqa-font-inter cqa-text-xs\">{{ addTriggerText }}</span>\n    </div>\n  </div>\n</ng-container>\n\n<!-- Collapsed summary when we have rows but expanded=false -->\n<ng-container *ngIf=\"rows.length > 0 && !expanded\">\n  <div\n    class=\"cqa-border-2 cqa-border-dashed cqa-border-[#D1D5DC] cqa-rounded-lg cqa-p-1.5 cqa-mt-4 cqa-bg-white cqa-cursor-pointer cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-flex cqa-items-center cqa-justify-between cqa-min-h-[27px] cqa-hover:cqa-border-[#9ca3af] cqa-hover:cqa-bg-[#f9fafb] cqa-active:cqa-border-[#6366f1] cqa-active:cqa-bg-[#eff6ff]\"\n    (click)=\"toggleExpanded(true)\">\n    <span class=\"cqa-text-[#4a5565] cqa-font-normal cqa-font-inter cqa-text-xs\">{{ rows.length }} prerequisite{{ rows.length !== 1 ? 's' : '' }}</span>\n    <mat-icon class=\"cqa-text-[14px] cqa-w-[14px] cqa-h-[14px] cqa-text-[#4a5565]\">expand_more</mat-icon>\n  </div>\n</ng-container>\n\n<ng-container *ngIf=\"rows.length > 0 && expanded\">\n  <div class=\"cqa-mt-4 cqa-p-3 cqa-bg-[#f9fafb] cqa-border cqa-border-[#e5e7eb] cqa-rounded-lg\">\n    <div class=\"cqa-flex cqa-items-center cqa-gap-1.5 cqa-mb-3\">\n      <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#374151] cqa-m-0 cqa-leading-none\">{{ label }}</label>\n      <mat-icon class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-4 cqa-h-4 cqa-text-sm cqa-text-[#6b7280] cqa-cursor-help cqa-flex-shrink-0\" [matTooltip]=\"infoTooltip\" matTooltipPosition=\"above\" matTooltipShowDelay=\"300\">info</mat-icon>\n      <button\n        type=\"button\"\n        class=\"cqa-ml-auto cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-p-0 cqa-bg-transparent cqa-border-none cqa-rounded-md cqa-text-[#6b7280] cqa-cursor-pointer cqa-hover:cqa-bg-[#f3f4f6] cqa-hover:cqa-text-[#374151]\"\n        (click)=\"toggleExpanded(false)\"\n        matTooltip=\"Close\">\n        <mat-icon class=\"cqa-text-xl cqa-w-5 cqa-h-5\">expand_less</mat-icon>\n      </button>\n    </div>\n\n    <div\n      class=\"prerequisite-drop-list cqa-flex cqa-flex-col cqa-gap-2 cqa-mb-3\"\n      [dndDropzone]=\"['prerequisite-row']\"\n      dndEffectAllowed=\"move\"\n      dndDragoverClass=\"dndDragover\"\n      (dndDrop)=\"onDndDrop($event)\">\n      <div dndPlaceholderRef class=\"prerequisite-drag-placeholder\">Drop here</div>\n      <div\n        *ngFor=\"let row of rows; let i = index; trackBy: trackByKey\"\n        class=\"prerequisite-drag-item cqa-flex cqa-items-center cqa-gap-2\"\n        [dndDraggable]=\"row\"\n        [dndDisableIf]=\"rows.length <= 1\"\n        dndEffectAllowed=\"move\"\n        dndType=\"prerequisite-row\">\n        <!-- 9-dot grid drag handle (only when multiple rows to reorder) -->\n        <div *ngIf=\"rows.length > 1\" dndHandle class=\"cqa-flex cqa-flex-shrink-0 cqa-items-center cqa-justify-center cqa-cursor-grab cqa-text-[#6B7280] cqa-hover:cqa-text-[#111827] active:cqa-cursor-grabbing\" matTooltip=\"Drag to reorder\">\n          <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n            <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n          </svg>\n        </div>\n\n      <div class=\"cqa-flex-1 cqa-min-w-0\">\n        <!-- Custom template (e.g. parent-provided cqa-dynamic-select) -->\n        <ng-container *ngIf=\"selectBodyTpl\">\n          <ng-container\n            *ngTemplateOutlet=\"selectBodyTpl; context: {\n              $implicit: row,\n              index: i,\n              form: form,\n              config: getConfig(i)\n            }\">\n          </ng-container>\n        </ng-container>\n        <!-- Default: built-in cqa-dynamic-select -->\n        <cqa-dynamic-select\n          *ngIf=\"!selectBodyTpl && form\"\n          [form]=\"form\"\n          [config]=\"getConfig(i)\"\n          (selectionChange)=\"onSelectionChange($event, i)\">\n        </cqa-dynamic-select>\n      </div>\n\n      <button\n        type=\"button\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-min-w-[32px] cqa-bg-transparent cqa-border-none cqa-rounded-md cqa-text-[#6b7280] cqa-cursor-pointer cqa-p-0 cqa-hover:cqa-bg-[#f3f4f6] cqa-hover:cqa-text-[#374151]\"\n        (click)=\"onRemoveRow(i)\"\n        matTooltip=\"Cancel\">\n        <mat-icon class=\"cqa-text-lg cqa-w-[18px] cqa-h-[18px]\">close</mat-icon>\n      </button>\n    </div>\n    </div>\n\n    <div *ngIf=\"showAddAnotherValidation && !allRowsHaveValues()\" class=\"cqa-text-xs cqa-text-[#dc2626] cqa-mb-2\">\n      {{ selectFirstValidationMessage }}\n    </div>\n    <cqa-button\n      variant=\"text\"\n      icon=\"add\"\n      [text]=\"addAnotherText\"\n      btnSize=\"md\"\n      customClass=\"cqa-text-[#3b82f6] cqa-mb-3 cqa-hover:cqa-bg-[#eff6ff]\"\n      (clicked)=\"onAddRow()\">\n    </cqa-button>\n    <div class=\"cqa-flex cqa-gap-3 cqa-justify-end\">\n      <button\n        type=\"button\"\n        class=\"cqa-py-2.5 cqa-px-4 cqa-bg-white cqa-text-[#374151] cqa-border cqa-border-[#d1d5db] cqa-rounded-md cqa-text-sm cqa-font-semibold cqa-cursor-pointer cqa-hover:cqa-bg-[#f9fafb]\"\n        (click)=\"onCancel()\">\n        {{ cancelText }}\n      </button>\n      <button\n        type=\"button\"\n        class=\"cqa-py-2.5 cqa-px-4 cqa-rounded-md cqa-text-sm cqa-font-semibold cqa-border-none cqa-cursor-pointer disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed cqa-bg-[#1a56db] cqa-text-white cqa-hover:cqa-bg-[#1647b8] disabled:cqa-bg-[#1a56db]\"\n        [disabled]=\"!allRowsHaveValues()\"\n        (click)=\"onUpdate()\">\n        {{ updateText }}\n      </button>\n    </div>\n  </div>\n</ng-container>\n"]}
|
|
@@ -2,4 +2,4 @@
|
|
|
2
2
|
* Models and interfaces for execution screen step components
|
|
3
3
|
*/
|
|
4
4
|
export {};
|
|
5
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution-step.models.js","sourceRoot":"","sources":["../../../../../src/lib/execution-screen/execution-step.models.ts"],"names":[],"mappings":"AAAA;;GAEG","sourcesContent":["/**\n * Models and interfaces for execution screen step components\n */\n\nexport type StepStatus = 'success' | 'failed' | 'pending' | 'running' | 'skipped' | 'SUCCESS' | 'FAILED' | 'PENDING' | 'RUNNING' | 'SKIPPED';\n\nexport interface TimingBreakdown {\n  app: number; // in seconds\n  tool: number; // in seconds\n}\n\nexport interface SubStep {\n  id: string;\n  description: string;\n  status: StepStatus;\n  duration: number; // in seconds\n  timestamp?: string; // optional timestamp like \"00:12\"\n}\n\nexport interface LoopIteration {\n  id: string;\n  label: string; // e.g., \"Iter 2 (production)\"\n  status: StepStatus;\n  subSteps?: SubStep[];\n}\n\nexport interface ConditionBranch {\n  type: 'if' | 'else' | 'else if';\n  label: string; // e.g., \"Premium user\" or \"Free user\"\n  executed: boolean;\n  subSteps?: SubStep[];\n  branchStepId?: string | number; // ID of the branch step to load children from\n  isIfBranch?: boolean; // Whether this is the main IF branch\n  branchStep: any;\n}\n\nexport interface ScreenshotData {\n  baseline?: string; // URL or base64\n  current?: string; // URL or base64\n  difference?: string; // URL or base64\n}\n\nexport interface LogEntry {\n  level: 'info' | 'warning' | 'error';\n  message: string;\n  timestamp?: string;\n}\n\nexport interface FailureDetails {\n  testStepId: string;\n  expected: string;\n  actual: string;\n  failedAction?: string; // The action that failed\n  aiFixApplied?: boolean;\n  aiFixMessage?: string;\n  screenshots?: ScreenshotData;\n  logs?: LogEntry[];\n}\n\nexport interface BaseStepConfig {\n  id: string;\n  testStepId: number | undefined; // test step id from the test case\n  testStepResultId: string;\n  stepNumber: string; // e.g., \"1\", \"2.1\", \"3.2.1\"\n  title: string;\n  status: StepStatus;\n  duration: number; // total duration in seconds\n  timingBreakdown?: TimingBreakdown;\n  expanded?: boolean; // whether the step is expanded by default\n  failureDetails?: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n  stepDeleted?: boolean;\n  executedResult?: {\n    video_start_time: number;\n    video_end_time: number;\n  };\n  // Additional properties for live execution and runtime use\n  selectedIterationId?: string; // which iteration to display\n}\n\nexport interface SelfHealAnalysisData {\n  originalLocator: string;\n  healedLocator: string;\n  confidence: number; // 0-100\n  healMethod: string; // e.g., \"Semantic Attribute Matching\"\n}\n\nexport interface SelfHealActionEvent {\n  type: SelfHealAction;\n  id: string;\n  healedLocator: string;\n  testStepId?: string; // test step id from the test case\n}\n\nexport type SelfHealAction = 'accept' | 'reject' | 'modify-accept';\n\nexport interface BasicStepConfig extends BaseStepConfig {\n  type: 'basic';\n  subSteps: SubStep[];\n  selfHealAnalysis?: SelfHealAnalysisData;\n  selfHealed?: boolean; // Indicates if self-healing was applied\n  selfHealDuration?: number; // Duration of self-healing process\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface StepGroupConfig extends BaseStepConfig {\n  type: 'step-group';\n  groupName: string; // e.g., \"Navigation\", \"Checkout flow\"\n  steps: ExecutionStepConfig[]; // nested steps\n}\n\nexport interface LoopStepConfig extends BaseStepConfig {\n  type: 'loop';\n  loopType?: 'for' | 'while';\n  iterations: LoopIteration[];\n  selectedIterationId?: string; // which iteration to display\n  defaultIteration?: 'first' | 'last'; // default iteration to show\n  nestedSteps?: ExecutionStepConfig[]; // steps nested within the loop\n  showViewAllIterations?: boolean;\n  subSteps?: SubStep[];\n  iterationData?: any; // data for each iteration in live execution\n}\n\nexport interface ConditionStepConfig extends BaseStepConfig {\n  type: 'condition';\n  conditionText: string; // e.g., \"if user type is 'Premium'\"\n  branches: ConditionBranch[];\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface FailedStepConfig extends BaseStepConfig {\n  type: 'failed';\n  subSteps: SubStep[];\n  failureDetails: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n}\n\nexport interface AIAgentAction {\n  id: string;\n  description: string;\n  type: 'extract' | 'validate' | 'AI_AGENT_ACTION' | 'TYPE' | 'CLICK';\n  status: StepStatus;\n  confidence?: number; // 0-100\n  duration: number;\n  reasoning?: string;\n}\n\nexport interface AIAgentStepConfig extends BaseStepConfig {\n  type: 'ai-agent';\n  prompt: string;\n  optimizedRun?: boolean;\n  actionCount?: number;\n  actions: AIAgentAction[];\n  selectedTab?: 'action-trace' | 'planner-timeline' | 'ai-reasoning';\n}\n\nexport interface AIActionStepConfig extends BaseStepConfig {\n  type: 'ai-action';\n  actionCount?: number;\n  actionName?: 'ai_text_verification' | 'ai_ask' | 'ai_document_ask' | 'ai_verify';\n  actions: AIAgentAction[];\n}\n\nexport interface ApiAssertion {\n  id: string;\n  description: string;\n  status: 'passed' | 'failed';\n  expected: string;\n  actual: string;\n}\n\nexport interface ApiStepConfig extends BaseStepConfig {\n  type: 'api';\n  method: string; // GET, POST, PUT, DELETE, etc.\n  endpoint: string;\n  statusCode: number;\n  responseTime: number; // in milliseconds\n  requestBody?: any;\n  responseBody?: any;\n  requestHeaders?: any;\n  responseHeaders?: any;\n  assertions?: ApiAssertion[];\n  initialActions?: SubStep[]; // Actions before the API call\n}\n\nexport interface FileDownloadStepConfig extends BaseStepConfig {\n  type: 'file-download';\n  fileName: string;\n  fileType: string; // PDF, CSV, etc.\n  fileSize?: string; // e.g., \"2.4 MB\"\n  filePath?: string;\n  downloaded: boolean;\n}\n\nexport interface ExtractedField {\n  label: string;\n  value: string;\n  confidence: number; // 0-100\n}\n\nexport interface VerificationCheck {\n  id: string;\n  description: string;\n  status: 'pass' | 'fail';\n  expected: string;\n  actual: string;\n}\n\nexport interface DocumentVerificationStepConfig extends BaseStepConfig {\n  type: 'document-verification';\n  documentScreenshot?: string; // URL or base64\n  extractedFields: ExtractedField[];\n  verificationChecks: VerificationCheck[];\n}\n\nexport interface LiveSubStep {\n  id: string;\n  text: string; // Text from socket\n  isRunning: boolean; // true = show loader, false = show icon\n  duration?: number; // in seconds\n  status: 'passed' | 'failed' | 'pending' | 'success' | 'failure' | 'skipped';\n}\n\nexport interface LiveExecutionStepConfig extends BaseStepConfig {\n  type: 'live-execution';\n  subSteps: LiveSubStep[];\n}\n\nexport interface DbQueryResult {\n  data: any[];\n  query: string;\n  executedAt?: string;\n  duration?: number; // in seconds\n  [key: string]: any; // For dynamic keys like \"Demo_user\"\n}\n\nexport interface DbAssertionResult {\n  variableName: string;\n  jsonPath: string;\n  expectedValue: any;\n  actualValue: any;\n  verificationType: string;\n  expectedType: string;\n  passed: boolean;\n}\n\nexport interface DbTestResult {\n  success: boolean;\n  results: { [key: string]: DbQueryResult };\n  assertionResults: DbAssertionResult[];\n  allAssertionsPassed: boolean;\n}\n\nexport interface DbVerificationStepConfig extends BaseStepConfig {\n  type: 'db-verification';\n  dbTestResult: DbTestResult;\n  dbConfig?: {\n    name: string; // Environment name\n    dbType: string; // Database type (MySQL, SQLServer, etc.)\n  };\n}\n\nexport interface PrerequisiteItem {\n  id: string;\n  title: string; // e.g., \"Prerequisite 1: Login flow\"\n  status: StepStatus;\n  duration: number; // in seconds\n  stepConfig?: ExecutionStepConfig; // Optional: full step config to view details\n}\n\nexport type ExecutionStepConfig =\n  | BasicStepConfig\n  | StepGroupConfig\n  | LoopStepConfig\n  | ConditionStepConfig\n  | FailedStepConfig\n  | AIAgentStepConfig\n  | AIActionStepConfig\n  | ApiStepConfig\n  | FileDownloadStepConfig\n  | DocumentVerificationStepConfig\n  | LiveExecutionStepConfig\n  | DbVerificationStepConfig;\n"]}
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution-step.models.js","sourceRoot":"","sources":["../../../../../src/lib/execution-screen/execution-step.models.ts"],"names":[],"mappings":"AAAA;;GAEG","sourcesContent":["/**\n * Models and interfaces for execution screen step components\n */\n\nexport type StepStatus = 'success' | 'failed' | 'pending' | 'running' | 'skipped' | 'SUCCESS' | 'FAILED' | 'PENDING' | 'RUNNING' | 'SKIPPED';\n\nexport interface TimingBreakdown {\n  app: number; // in seconds\n  tool: number; // in seconds\n}\n\nexport interface SubStep {\n  id: string;\n  description: string;\n  status: StepStatus;\n  duration: number; // in seconds\n  timestamp?: string; // optional timestamp like \"00:12\"\n}\n\nexport interface LoopIteration {\n  id: string;\n  label: string; // e.g., \"Iter 2 (production)\"\n  status: StepStatus;\n  subSteps?: SubStep[];\n}\n\nexport interface ConditionBranch {\n  type: 'if' | 'else' | 'else if';\n  label: string; // e.g., \"Premium user\" or \"Free user\"\n  executed: boolean;\n  subSteps?: SubStep[];\n  branchStepId?: string | number; // ID of the branch step to load children from\n  isIfBranch?: boolean; // Whether this is the main IF branch\n  branchStep: any;\n}\n\nexport interface ScreenshotData {\n  baseline?: string; // URL or base64\n  current?: string; // URL or base64\n  difference?: string; // URL or base64\n}\n\nexport interface LogEntry {\n  level: 'info' | 'warning' | 'error';\n  message: string;\n  timestamp?: string;\n}\n\nexport interface FailureDetails {\n  testStepId: string;\n  expected: string;\n  actual: string;\n  failedAction?: string; // The action that failed\n  aiFixApplied?: boolean;\n  aiFixMessage?: string;\n  screenshots?: ScreenshotData;\n  logs?: LogEntry[];\n}\n\nexport interface BaseStepConfig {\n  id: string;\n  testStepId: number | undefined; // test step id from the test case\n  testStepResultId: string;\n  stepNumber: string; // e.g., \"1\", \"2.1\", \"3.2.1\"\n  title: string;\n  status: StepStatus;\n  duration: number; // total duration in seconds\n  timingBreakdown?: TimingBreakdown;\n  expanded?: boolean; // whether the step is expanded by default\n  failureDetails?: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n  stepDeleted?: boolean;\n  executedResult?: {\n    video_start_time: number;\n    video_end_time: number;\n  };\n  // Additional properties for live execution and runtime use\n  selectedIterationId?: string; // which iteration to display\n}\n\nexport interface SelfHealAnalysisData {\n  originalLocator: string;\n  healedLocator: string;\n  confidence: number; // 0-100\n  healMethod: string; // e.g., \"Semantic Attribute Matching\"\n}\n\nexport interface SelfHealActionEvent {\n  type: SelfHealAction;\n  id: string;\n  healedLocator: string;\n  testStepId?: string; // test step id from the test case\n}\n\nexport type SelfHealAction = 'accept' | 'reject' | 'modify-accept';\n\nexport interface BasicStepConfig extends BaseStepConfig {\n  type: 'basic';\n  subSteps: SubStep[];\n  selfHealAnalysis?: SelfHealAnalysisData;\n  selfHealed?: boolean; // Indicates if self-healing was applied\n  selfHealDuration?: number; // Duration of self-healing process\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface StepGroupConfig extends BaseStepConfig {\n  type: 'step-group';\n  groupName: string; // e.g., \"Navigation\", \"Checkout flow\"\n  steps: ExecutionStepConfig[]; // nested steps\n}\n\nexport interface LoopStepConfig extends BaseStepConfig {\n  type: 'loop';\n  loopType?: 'for' | 'while';\n  iterations: LoopIteration[];\n  selectedIterationId?: string; // which iteration to display\n  defaultIteration?: 'first' | 'last'; // default iteration to show\n  nestedSteps?: ExecutionStepConfig[]; // steps nested within the loop\n  showViewAllIterations?: boolean;\n  subSteps?: SubStep[];\n  iterationData?: any; // data for each iteration in live execution\n}\n\nexport interface ConditionStepConfig extends BaseStepConfig {\n  type: 'condition';\n  conditionText: string; // e.g., \"if user type is 'Premium'\"\n  branches: ConditionBranch[];\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface FailedStepConfig extends BaseStepConfig {\n  type: 'failed';\n  subSteps: SubStep[];\n  failureDetails: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n}\n\nexport interface AIAgentAction {\n  id: string;\n  description: string;\n  type: 'extract' | 'validate' | 'AI_AGENT_ACTION' | 'TYPE' | 'CLICK';\n  status: StepStatus;\n  confidence?: number; // 0-100\n  duration: number;\n  reasoning?: string;\n}\n\nexport interface AIAgentStepConfig extends BaseStepConfig {\n  type: 'ai-agent';\n  prompt: string;\n  optimizedRun?: boolean;\n  actionCount?: number;\n  actions: AIAgentAction[];\n  selectedTab?: 'action-trace' | 'planner-timeline' | 'ai-reasoning';\n}\n\nexport interface AIActionStepConfig extends BaseStepConfig {\n  type: 'ai-action';\n  actionCount?: number;\n  actionName?: 'ai_text_verification' | 'ai_ask' | 'ai_document_ask' | 'ai_verify';\n  actions: AIAgentAction[];\n}\n\nexport interface ApiAssertion {\n  id: string;\n  description: string;\n  status: 'passed' | 'failed';\n  expected: string;\n  actual: string;\n}\n\nexport interface ApiStepConfig extends BaseStepConfig {\n  type: 'api';\n  method: string; // GET, POST, PUT, DELETE, etc.\n  endpoint: string;\n  statusCode: number;\n  responseTime: number; // in milliseconds\n  requestBody?: any;\n  responseBody?: any;\n  requestHeaders?: any;\n  responseHeaders?: any;\n  assertions?: ApiAssertion[];\n  initialActions?: SubStep[]; // Actions before the API call\n}\n\nexport interface FileDownloadStepConfig extends BaseStepConfig {\n  type: 'file-download';\n  fileName: string;\n  fileType: string; // PDF, CSV, etc.\n  fileSize?: string; // e.g., \"2.4 MB\"\n  filePath?: string;\n  downloaded: boolean;\n}\n\nexport interface ExtractedField {\n  label: string;\n  value: string;\n  confidence: number; // 0-100\n}\n\nexport interface VerificationCheck {\n  id: string;\n  description: string;\n  status: 'pass' | 'fail';\n  expected: string;\n  actual: string;\n}\n\nexport interface DocumentVerificationStepConfig extends BaseStepConfig {\n  type: 'document-verification';\n  documentScreenshot?: string; // URL or base64\n  extractedFields: ExtractedField[];\n  verificationChecks: VerificationCheck[];\n}\n\nexport interface LiveSubStep {\n  id: string;\n  text: string; // Text from socket\n  isRunning: boolean; // true = show loader, false = show icon\n  duration?: number; // in seconds\n  status: 'passed' | 'failed' | 'pending' | 'success' | 'failure' | 'skipped';\n}\n\nexport interface LiveExecutionStepConfig extends BaseStepConfig {\n  type: 'live-execution';\n  subSteps: LiveSubStep[];\n}\n\nexport interface DbQueryResult {\n  data: any[];\n  query: string;\n  executedAt?: string;\n  duration?: number; // in seconds\n  [key: string]: any; // For dynamic keys like \"Demo_user\"\n}\n\nexport interface DbAssertionResult {\n  variableName: string;\n  jsonPath: string;\n  expectedValue: any;\n  actualValue: any;\n  verificationType: string;\n  expectedType: string;\n  passed: boolean;\n}\n\nexport interface DbTestResult {\n  success: boolean;\n  results: { [key: string]: DbQueryResult };\n  assertionResults: DbAssertionResult[];\n  allAssertionsPassed: boolean;\n}\n\nexport interface DbVerificationStepConfig extends BaseStepConfig {\n  type: 'db-verification';\n  dbTestResult: DbTestResult;\n  dbConfig?: {\n    name: string; // Environment name\n    dbType: string; // Database type (MySQL, SQLServer, etc.)\n  };\n}\n\nexport interface PrerequisiteItem {\n  id: string;\n  title: string; // e.g., \"Prerequisite 1: Login flow\"\n  status: StepStatus;\n  duration: number; // in seconds\n  stepConfig?: ExecutionStepConfig; // Optional: full step config to view details\n  /** Optional URL for \"View steps\" redirect. When viewStepsAsRedirect is true, each item can have its own URL. */\n  viewStepsUrl?: string;\n}\n\nexport type ExecutionStepConfig =\n  | BasicStepConfig\n  | StepGroupConfig\n  | LoopStepConfig\n  | ConditionStepConfig\n  | FailedStepConfig\n  | AIAgentStepConfig\n  | AIActionStepConfig\n  | ApiStepConfig\n  | FileDownloadStepConfig\n  | DocumentVerificationStepConfig\n  | LiveExecutionStepConfig\n  | DbVerificationStepConfig;\n"]}
|