@cqa-lib/cqa-ui 1.1.204 → 1.1.205
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/detail-drawer/detail-drawer-tab/detail-drawer-tab.component.mjs +38 -0
- package/esm2020/lib/detail-drawer/detail-drawer-tab-content.directive.mjs +16 -0
- package/esm2020/lib/detail-drawer/detail-drawer.component.mjs +121 -0
- package/esm2020/lib/detail-side-panel/detail-side-panel.component.mjs +211 -0
- package/esm2020/lib/detail-side-panel/detail-side-panel.models.mjs +2 -0
- package/esm2020/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.mjs +634 -0
- package/esm2020/lib/test-case-details/test-case-details.component.mjs +138 -0
- package/esm2020/lib/test-case-details/test-case-details.models.mjs +167 -0
- package/esm2020/lib/ui-kit.module.mjs +31 -1
- package/esm2020/public-api.mjs +9 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +1322 -2
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +1315 -2
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/detail-drawer/detail-drawer-tab/detail-drawer-tab.component.d.ts +15 -0
- package/lib/detail-drawer/detail-drawer-tab-content.directive.d.ts +8 -0
- package/lib/detail-drawer/detail-drawer.component.d.ts +39 -0
- package/lib/detail-side-panel/detail-side-panel.component.d.ts +86 -0
- package/lib/detail-side-panel/detail-side-panel.models.d.ts +20 -0
- package/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.d.ts +175 -0
- package/lib/test-case-details/test-case-details.component.d.ts +62 -0
- package/lib/test-case-details/test-case-details.models.d.ts +118 -0
- package/lib/ui-kit.module.d.ts +32 -26
- package/package.json +1 -1
- package/public-api.d.ts +8 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "./test-case-details-edit/test-case-details-edit.component";
|
|
4
|
+
import * as i2 from "@angular/material/icon";
|
|
5
|
+
import * as i3 from "../badge/badge.component";
|
|
6
|
+
import * as i4 from "../configuration-card/configuration-card.component";
|
|
7
|
+
import * as i5 from "@angular/common";
|
|
8
|
+
export class TestCaseDetailsComponent {
|
|
9
|
+
constructor(cdr) {
|
|
10
|
+
this.cdr = cdr;
|
|
11
|
+
/** Whether the component is in edit mode */
|
|
12
|
+
this.editing = false;
|
|
13
|
+
/** When true, start in edit mode (useful for Storybook). */
|
|
14
|
+
this.startInEditMode = false;
|
|
15
|
+
/** Description section title */
|
|
16
|
+
this.descriptionTitle = 'Description';
|
|
17
|
+
/** Description text content */
|
|
18
|
+
this.descriptionContent = '';
|
|
19
|
+
/** Whether to show the Edit button in the Description header */
|
|
20
|
+
this.showEditButton = false;
|
|
21
|
+
/** Metadata items (createdOn, status, priority, environment, version, testPlanName, etc.) */
|
|
22
|
+
this.metadataItems = [];
|
|
23
|
+
/** Labels/tags (e.g. Automation, API, SDK, UI/UX) */
|
|
24
|
+
this.labels = [];
|
|
25
|
+
/** Configuration section title */
|
|
26
|
+
this.configTitle = 'Configuration';
|
|
27
|
+
/** Configuration sections (e.g. Execution, AI Configuration) */
|
|
28
|
+
this.configSections = [];
|
|
29
|
+
/** Optional config sections displayed in a 2-column row (e.g. Waits & Retries, Device) */
|
|
30
|
+
this.configSectionsRow2 = [];
|
|
31
|
+
/** Platform: 'web' or 'mobile'. Defaults to 'web'. Used for Device Settings fields. */
|
|
32
|
+
this.platform = 'web';
|
|
33
|
+
/** Override config per select for API-driven options, server search, load more. */
|
|
34
|
+
this.selectConfigOverrides = {};
|
|
35
|
+
this.editDescription = new EventEmitter();
|
|
36
|
+
this.saveChanges = new EventEmitter();
|
|
37
|
+
this.metadataLinkClick = new EventEmitter();
|
|
38
|
+
this.selectSearch = new EventEmitter();
|
|
39
|
+
this.selectLoadMore = new EventEmitter();
|
|
40
|
+
this.selectOpened = new EventEmitter();
|
|
41
|
+
this.selectionChange = new EventEmitter();
|
|
42
|
+
}
|
|
43
|
+
ngOnInit() {
|
|
44
|
+
if (this.startInEditMode) {
|
|
45
|
+
this.editing = true;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
onEditClick() {
|
|
49
|
+
this.editing = true;
|
|
50
|
+
this.cdr.detectChanges();
|
|
51
|
+
this.editDescription.emit();
|
|
52
|
+
}
|
|
53
|
+
onSaveChanges(data) {
|
|
54
|
+
this.editing = false;
|
|
55
|
+
this.cdr.detectChanges();
|
|
56
|
+
this.saveChanges.emit(data);
|
|
57
|
+
}
|
|
58
|
+
onCancelEdit() {
|
|
59
|
+
this.editing = false;
|
|
60
|
+
this.cdr.detectChanges();
|
|
61
|
+
}
|
|
62
|
+
trackByConfigTitle(_i, section) {
|
|
63
|
+
return section.title;
|
|
64
|
+
}
|
|
65
|
+
trackByMetadataLabel(_i, item) {
|
|
66
|
+
return item.label;
|
|
67
|
+
}
|
|
68
|
+
onMetadataLinkClick(item) {
|
|
69
|
+
if (item.link) {
|
|
70
|
+
this.metadataLinkClick.emit(item);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
getStatusDotClass(item) {
|
|
74
|
+
if (!item.statusColor)
|
|
75
|
+
return '';
|
|
76
|
+
switch (item.statusColor) {
|
|
77
|
+
case 'yellow':
|
|
78
|
+
return 'cqa-bg-[#EAB308]';
|
|
79
|
+
case 'red':
|
|
80
|
+
return 'cqa-bg-[#DC2626]';
|
|
81
|
+
case 'green':
|
|
82
|
+
return 'cqa-bg-[#16A34A]';
|
|
83
|
+
case 'gray':
|
|
84
|
+
default:
|
|
85
|
+
return 'cqa-bg-[#94A3B8]';
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
/** Text color for metadata value (e.g. red for critical priority) */
|
|
89
|
+
getValueTextClass(item) {
|
|
90
|
+
if (item.statusColor === 'red') {
|
|
91
|
+
return 'cqa-text-[#DC2626]';
|
|
92
|
+
}
|
|
93
|
+
return 'cqa-text-[#111827]';
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
TestCaseDetailsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseDetailsComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
97
|
+
TestCaseDetailsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TestCaseDetailsComponent, selector: "cqa-test-case-details", inputs: { startInEditMode: "startInEditMode", descriptionTitle: "descriptionTitle", descriptionContent: "descriptionContent", showEditButton: "showEditButton", metadataItems: "metadataItems", labels: "labels", configTitle: "configTitle", configSections: "configSections", configSectionsRow2: "configSectionsRow2", platform: "platform", selectConfigOverrides: "selectConfigOverrides" }, outputs: { editDescription: "editDescription", saveChanges: "saveChanges", metadataLinkClick: "metadataLinkClick", selectSearch: "selectSearch", selectLoadMore: "selectLoadMore", selectOpened: "selectOpened", selectionChange: "selectionChange" }, ngImport: i0, template: "<!-- Edit mode: show edit form (Figma design) -->\n<cqa-test-case-details-edit\n *ngIf=\"editing\"\n [descriptionTitle]=\"descriptionTitle\"\n [descriptionContent]=\"descriptionContent\"\n [metadataItems]=\"metadataItems\"\n [labels]=\"labels\"\n [configTitle]=\"configTitle\"\n [configSections]=\"configSections\"\n [configSectionsRow2]=\"configSectionsRow2\"\n [platform]=\"platform\"\n [selectConfigOverrides]=\"selectConfigOverrides\"\n (save)=\"onSaveChanges($event)\"\n (cancel)=\"onCancelEdit()\"\n (selectSearch)=\"selectSearch.emit($event)\"\n (selectLoadMore)=\"selectLoadMore.emit($event)\"\n (selectOpened)=\"selectOpened.emit($event)\"\n (selectionChange)=\"selectionChange.emit($event)\">\n</cqa-test-case-details-edit>\n\n<!-- View mode: read-only details -->\n<div *ngIf=\"!editing\" class=\"cqa-self-stretch cqa-py-4 cqa-px-0 cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-4 cqa-overflow-hidden cqa-text-sm cqa-leading-[19.6px]\">\n <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-4 cqa-text-sm cqa-leading-[19.6px]\">\n <!-- Description Section -->\n <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-inline-flex cqa-justify-between cqa-items-center\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-w-5 cqa-h-5 cqa-p-1 cqa-flex-shrink-0 cqa-bg-[#3F43EE1A] cqa-rounded-md cqa-inline-flex cqa-justify-center cqa-items-center\">\n <mat-icon class=\"cqa-text-[#1B1FEB] cqa-text-[12px] cqa-w-3 cqa-h-3 cqa-leading-none cqa-block\">description</mat-icon>\n </div>\n <span class=\"cqa-text-[#374151] cqa-text-sm cqa-font-semibold cqa-leading-[19.6px]\">{{ descriptionTitle }}</span>\n </div>\n <button\n *ngIf=\"showEditButton\"\n type=\"button\"\n class=\"cqa-flex cqa-justify-end cqa-items-center cqa-gap-2 cqa-text-[#A3A3A3] cqa-text-xs cqa-font-semibold hover:cqa-text-[#737373] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2\"\n (click)=\"onEditClick()\">\n <mat-icon class=\"cqa-text-[14px] cqa-w-4 cqa-h-4\">edit</mat-icon>\n Edit\n </button>\n </div>\n <div *ngIf=\"descriptionContent\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start\">\n <div class=\"cqa-self-stretch cqa-text-[#111827] cqa-text-sm cqa-leading-[19.6px] cqa-font-normal\">{{ descriptionContent }}</div>\n </div>\n </div>\n\n <!-- Metadata Section -->\n <div *ngIf=\"metadataItems.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-grid cqa-grid-cols-3 cqa-gap-x-0 cqa-gap-y-4\">\n <ng-container *ngFor=\"let item of metadataItems; trackBy: trackByMetadataLabel\">\n <div class=\"cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-0.5\">\n <div class=\"cqa-text-[#6B7280] cqa-text-xs cqa-leading-[15px] cqa-font-medium\">{{ item.label }}</div>\n <div class=\"cqa-flex cqa-justify-start cqa-items-center cqa-gap-2.5 cqa-min-w-0 cqa-w-full\">\n <mat-icon *ngIf=\"item.icon && (!item.iconLibrary || item.iconLibrary === 'mat')\" class=\"cqa-text-[#6B7280] cqa-text-sm cqa-w-4 cqa-h-4\">\n {{ item.icon }}\n </mat-icon>\n <span *ngIf=\"item.statusColor\" class=\"cqa-w-3.5 cqa-h-3.5 cqa-rounded-full cqa-flex-shrink-0\" [ngClass]=\"getStatusDotClass(item)\"></span>\n <a\n *ngIf=\"item.link\"\n href=\"javascript:void(0)\"\n class=\"cqa-text-[#2563EB] cqa-text-xs cqa-leading-[12px] cqa-font-normal hover:cqa-underline cqa-cursor-pointer cqa-block cqa-truncate cqa-min-w-0\"\n (click)=\"onMetadataLinkClick(item); $event.preventDefault()\">\n {{ item.value }}\n </a>\n <span *ngIf=\"!item.link\" class=\"cqa-text-xs cqa-leading-[12px] cqa-font-normal\" [ngClass]=\"getValueTextClass(item)\">\n {{ item.value }}\n </span>\n </div>\n </div>\n </ng-container>\n </div>\n </div>\n\n <!-- Labels (Figma: Labels title, cqa-badge chips with tag icon) -->\n <div *ngIf=\"labels.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2 cqa-text-sm cqa-leading-[19.6px]\">\n <span class=\"cqa-text-[#111827] cqa-text-sm cqa-font-medium cqa-leading-5\">Labels</span>\n <div class=\"cqa-flex cqa-flex-wrap cqa-gap-2\">\n <cqa-badge\n *ngFor=\"let label of labels\"\n [label]=\"label\"\n icon=\"label\"\n variant=\"outline\"\n size=\"small\"\n backgroundColor=\"#ffffff\"\n textColor=\"#475569\"\n borderColor=\"#E2E8F0\"\n iconColor=\"#94A3B8\">\n </cqa-badge>\n </div>\n </div>\n\n <!-- Configuration Section -->\n <div *ngIf=\"configSections.length || configSectionsRow2.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-inline-flex cqa-justify-start cqa-items-center\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-w-5 cqa-h-5 cqa-p-1 cqa-flex-shrink-0 cqa-bg-[#3F43EE1A] cqa-rounded-md cqa-inline-flex cqa-justify-center cqa-items-center\">\n <mat-icon class=\"cqa-text-[#1B1FEB] cqa-text-[12px] cqa-w-3 cqa-h-3 cqa-leading-none cqa-block\">settings</mat-icon>\n </div>\n <span class=\"cqa-text-[#374151] cqa-text-sm cqa-font-semibold cqa-leading-[19.6px]\">{{ configTitle }}</span>\n </div>\n </div>\n <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-grid cqa-grid-cols-1 cqa-gap-2.5\">\n <ng-container *ngFor=\"let section of configSections; trackBy: trackByConfigTitle\">\n <cqa-configuration-card\n [icon]=\"section.icon || 'tune'\"\n [title]=\"section.title\"\n [data]=\"section.items\">\n </cqa-configuration-card>\n </ng-container>\n </div>\n <div *ngIf=\"configSectionsRow2.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-gap-4 cqa-text-sm cqa-leading-[19.6px]\">\n <cqa-configuration-card\n *ngFor=\"let section of configSectionsRow2; trackBy: trackByConfigTitle\"\n [icon]=\"section.icon || 'tune'\"\n [title]=\"section.title\"\n [data]=\"section.items\">\n </cqa-configuration-card>\n </div>\n </div>\n </div>\n </div>\n</div>\n", components: [{ type: i1.TestCaseDetailsEditComponent, selector: "cqa-test-case-details-edit", inputs: ["descriptionTitle", "descriptionContent", "metadataItems", "labels", "configTitle", "configSections", "configSectionsRow2", "prerequisiteCaseOptions", "platform", "selectConfigOverrides"], outputs: ["save", "cancel", "selectSearch", "selectLoadMore", "selectOpened", "selectionChange"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { type: i4.ConfigurationCardComponent, selector: "cqa-configuration-card", inputs: ["icon", "title", "data"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
98
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseDetailsComponent, decorators: [{
|
|
99
|
+
type: Component,
|
|
100
|
+
args: [{ selector: 'cqa-test-case-details', changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- Edit mode: show edit form (Figma design) -->\n<cqa-test-case-details-edit\n *ngIf=\"editing\"\n [descriptionTitle]=\"descriptionTitle\"\n [descriptionContent]=\"descriptionContent\"\n [metadataItems]=\"metadataItems\"\n [labels]=\"labels\"\n [configTitle]=\"configTitle\"\n [configSections]=\"configSections\"\n [configSectionsRow2]=\"configSectionsRow2\"\n [platform]=\"platform\"\n [selectConfigOverrides]=\"selectConfigOverrides\"\n (save)=\"onSaveChanges($event)\"\n (cancel)=\"onCancelEdit()\"\n (selectSearch)=\"selectSearch.emit($event)\"\n (selectLoadMore)=\"selectLoadMore.emit($event)\"\n (selectOpened)=\"selectOpened.emit($event)\"\n (selectionChange)=\"selectionChange.emit($event)\">\n</cqa-test-case-details-edit>\n\n<!-- View mode: read-only details -->\n<div *ngIf=\"!editing\" class=\"cqa-self-stretch cqa-py-4 cqa-px-0 cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-4 cqa-overflow-hidden cqa-text-sm cqa-leading-[19.6px]\">\n <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-4 cqa-text-sm cqa-leading-[19.6px]\">\n <!-- Description Section -->\n <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-inline-flex cqa-justify-between cqa-items-center\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-w-5 cqa-h-5 cqa-p-1 cqa-flex-shrink-0 cqa-bg-[#3F43EE1A] cqa-rounded-md cqa-inline-flex cqa-justify-center cqa-items-center\">\n <mat-icon class=\"cqa-text-[#1B1FEB] cqa-text-[12px] cqa-w-3 cqa-h-3 cqa-leading-none cqa-block\">description</mat-icon>\n </div>\n <span class=\"cqa-text-[#374151] cqa-text-sm cqa-font-semibold cqa-leading-[19.6px]\">{{ descriptionTitle }}</span>\n </div>\n <button\n *ngIf=\"showEditButton\"\n type=\"button\"\n class=\"cqa-flex cqa-justify-end cqa-items-center cqa-gap-2 cqa-text-[#A3A3A3] cqa-text-xs cqa-font-semibold hover:cqa-text-[#737373] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2\"\n (click)=\"onEditClick()\">\n <mat-icon class=\"cqa-text-[14px] cqa-w-4 cqa-h-4\">edit</mat-icon>\n Edit\n </button>\n </div>\n <div *ngIf=\"descriptionContent\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start\">\n <div class=\"cqa-self-stretch cqa-text-[#111827] cqa-text-sm cqa-leading-[19.6px] cqa-font-normal\">{{ descriptionContent }}</div>\n </div>\n </div>\n\n <!-- Metadata Section -->\n <div *ngIf=\"metadataItems.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-grid cqa-grid-cols-3 cqa-gap-x-0 cqa-gap-y-4\">\n <ng-container *ngFor=\"let item of metadataItems; trackBy: trackByMetadataLabel\">\n <div class=\"cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-0.5\">\n <div class=\"cqa-text-[#6B7280] cqa-text-xs cqa-leading-[15px] cqa-font-medium\">{{ item.label }}</div>\n <div class=\"cqa-flex cqa-justify-start cqa-items-center cqa-gap-2.5 cqa-min-w-0 cqa-w-full\">\n <mat-icon *ngIf=\"item.icon && (!item.iconLibrary || item.iconLibrary === 'mat')\" class=\"cqa-text-[#6B7280] cqa-text-sm cqa-w-4 cqa-h-4\">\n {{ item.icon }}\n </mat-icon>\n <span *ngIf=\"item.statusColor\" class=\"cqa-w-3.5 cqa-h-3.5 cqa-rounded-full cqa-flex-shrink-0\" [ngClass]=\"getStatusDotClass(item)\"></span>\n <a\n *ngIf=\"item.link\"\n href=\"javascript:void(0)\"\n class=\"cqa-text-[#2563EB] cqa-text-xs cqa-leading-[12px] cqa-font-normal hover:cqa-underline cqa-cursor-pointer cqa-block cqa-truncate cqa-min-w-0\"\n (click)=\"onMetadataLinkClick(item); $event.preventDefault()\">\n {{ item.value }}\n </a>\n <span *ngIf=\"!item.link\" class=\"cqa-text-xs cqa-leading-[12px] cqa-font-normal\" [ngClass]=\"getValueTextClass(item)\">\n {{ item.value }}\n </span>\n </div>\n </div>\n </ng-container>\n </div>\n </div>\n\n <!-- Labels (Figma: Labels title, cqa-badge chips with tag icon) -->\n <div *ngIf=\"labels.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2 cqa-text-sm cqa-leading-[19.6px]\">\n <span class=\"cqa-text-[#111827] cqa-text-sm cqa-font-medium cqa-leading-5\">Labels</span>\n <div class=\"cqa-flex cqa-flex-wrap cqa-gap-2\">\n <cqa-badge\n *ngFor=\"let label of labels\"\n [label]=\"label\"\n icon=\"label\"\n variant=\"outline\"\n size=\"small\"\n backgroundColor=\"#ffffff\"\n textColor=\"#475569\"\n borderColor=\"#E2E8F0\"\n iconColor=\"#94A3B8\">\n </cqa-badge>\n </div>\n </div>\n\n <!-- Configuration Section -->\n <div *ngIf=\"configSections.length || configSectionsRow2.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-inline-flex cqa-justify-start cqa-items-center\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-w-5 cqa-h-5 cqa-p-1 cqa-flex-shrink-0 cqa-bg-[#3F43EE1A] cqa-rounded-md cqa-inline-flex cqa-justify-center cqa-items-center\">\n <mat-icon class=\"cqa-text-[#1B1FEB] cqa-text-[12px] cqa-w-3 cqa-h-3 cqa-leading-none cqa-block\">settings</mat-icon>\n </div>\n <span class=\"cqa-text-[#374151] cqa-text-sm cqa-font-semibold cqa-leading-[19.6px]\">{{ configTitle }}</span>\n </div>\n </div>\n <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n <div class=\"cqa-self-stretch cqa-grid cqa-grid-cols-1 cqa-gap-2.5\">\n <ng-container *ngFor=\"let section of configSections; trackBy: trackByConfigTitle\">\n <cqa-configuration-card\n [icon]=\"section.icon || 'tune'\"\n [title]=\"section.title\"\n [data]=\"section.items\">\n </cqa-configuration-card>\n </ng-container>\n </div>\n <div *ngIf=\"configSectionsRow2.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-gap-4 cqa-text-sm cqa-leading-[19.6px]\">\n <cqa-configuration-card\n *ngFor=\"let section of configSectionsRow2; trackBy: trackByConfigTitle\"\n [icon]=\"section.icon || 'tune'\"\n [title]=\"section.title\"\n [data]=\"section.items\">\n </cqa-configuration-card>\n </div>\n </div>\n </div>\n </div>\n</div>\n", styles: [] }]
|
|
101
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { startInEditMode: [{
|
|
102
|
+
type: Input
|
|
103
|
+
}], descriptionTitle: [{
|
|
104
|
+
type: Input
|
|
105
|
+
}], descriptionContent: [{
|
|
106
|
+
type: Input
|
|
107
|
+
}], showEditButton: [{
|
|
108
|
+
type: Input
|
|
109
|
+
}], metadataItems: [{
|
|
110
|
+
type: Input
|
|
111
|
+
}], labels: [{
|
|
112
|
+
type: Input
|
|
113
|
+
}], configTitle: [{
|
|
114
|
+
type: Input
|
|
115
|
+
}], configSections: [{
|
|
116
|
+
type: Input
|
|
117
|
+
}], configSectionsRow2: [{
|
|
118
|
+
type: Input
|
|
119
|
+
}], platform: [{
|
|
120
|
+
type: Input
|
|
121
|
+
}], selectConfigOverrides: [{
|
|
122
|
+
type: Input
|
|
123
|
+
}], editDescription: [{
|
|
124
|
+
type: Output
|
|
125
|
+
}], saveChanges: [{
|
|
126
|
+
type: Output
|
|
127
|
+
}], metadataLinkClick: [{
|
|
128
|
+
type: Output
|
|
129
|
+
}], selectSearch: [{
|
|
130
|
+
type: Output
|
|
131
|
+
}], selectLoadMore: [{
|
|
132
|
+
type: Output
|
|
133
|
+
}], selectOpened: [{
|
|
134
|
+
type: Output
|
|
135
|
+
}], selectionChange: [{
|
|
136
|
+
type: Output
|
|
137
|
+
}] } });
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-case-details.component.js","sourceRoot":"","sources":["../../../../../src/lib/test-case-details/test-case-details.component.ts","../../../../../src/lib/test-case-details/test-case-details.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  OnInit,\n} from '@angular/core';\nimport {\n  TestCaseDetailsConfigSection,\n  TestCaseDetailsMetadataItem,\n} from './test-case-details.models';\nimport {\n  TestCaseDetailsEditFormData,\n  SelectConfigOverrides,\n} from './test-case-details-edit/test-case-details-edit.component';\n\n@Component({\n  selector: 'cqa-test-case-details',\n  templateUrl: './test-case-details.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TestCaseDetailsComponent implements OnInit {\n  /** Whether the component is in edit mode */\n  editing = false;\n\n  /** When true, start in edit mode (useful for Storybook). */\n  @Input() startInEditMode = false;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    if (this.startInEditMode) {\n      this.editing = true;\n    }\n  }\n  /** Description section title */\n  @Input() descriptionTitle = 'Description';\n\n  /** Description text content */\n  @Input() descriptionContent = '';\n\n  /** Whether to show the Edit button in the Description header */\n  @Input() showEditButton = false;\n\n  /** Metadata items (createdOn, status, priority, environment, version, testPlanName, etc.) */\n  @Input() metadataItems: TestCaseDetailsMetadataItem[] = [];\n\n  /** Labels/tags (e.g. Automation, API, SDK, UI/UX) */\n  @Input() labels: string[] = [];\n\n  /** Configuration section title */\n  @Input() configTitle = 'Configuration';\n\n  /** Configuration sections (e.g. Execution, AI Configuration) */\n  @Input() configSections: TestCaseDetailsConfigSection[] = [];\n\n  /** Optional config sections displayed in a 2-column row (e.g. Waits & Retries, Device) */\n  @Input() configSectionsRow2: TestCaseDetailsConfigSection[] = [];\n  /** Platform: 'web' or 'mobile'. Defaults to 'web'. Used for Device Settings fields. */\n  @Input() platform: 'web' | 'mobile' = 'web';\n\n  /** Override config per select for API-driven options, server search, load more. */\n  @Input() selectConfigOverrides: SelectConfigOverrides = {};\n\n  @Output() editDescription = new EventEmitter<void>();\n  @Output() saveChanges = new EventEmitter<TestCaseDetailsEditFormData>();\n  @Output() metadataLinkClick = new EventEmitter<TestCaseDetailsMetadataItem>();\n  @Output() selectSearch = new EventEmitter<{ key: string; query: string }>();\n  @Output() selectLoadMore = new EventEmitter<{ key: string; query: string }>();\n  @Output() selectOpened = new EventEmitter<{ key: string }>();\n  @Output() selectionChange = new EventEmitter<{ key: string; value: unknown }>();\n\n  onEditClick(): void {\n    this.editing = true;\n    this.cdr.detectChanges();\n    this.editDescription.emit();\n  }\n\n  onSaveChanges(data: TestCaseDetailsEditFormData): void {\n    this.editing = false;\n    this.cdr.detectChanges();\n    this.saveChanges.emit(data);\n  }\n\n  onCancelEdit(): void {\n    this.editing = false;\n    this.cdr.detectChanges();\n  }\n\n  trackByConfigTitle(_i: number, section: TestCaseDetailsConfigSection): string {\n    return section.title;\n  }\n\n  trackByMetadataLabel(_i: number, item: TestCaseDetailsMetadataItem): string {\n    return item.label;\n  }\n\n\n  onMetadataLinkClick(item: TestCaseDetailsMetadataItem): void {\n    if (item.link) {\n      this.metadataLinkClick.emit(item);\n    }\n  }\n\n  getStatusDotClass(item: TestCaseDetailsMetadataItem): string {\n    if (!item.statusColor) return '';\n    switch (item.statusColor) {\n      case 'yellow':\n        return 'cqa-bg-[#EAB308]';\n      case 'red':\n        return 'cqa-bg-[#DC2626]';\n      case 'green':\n        return 'cqa-bg-[#16A34A]';\n      case 'gray':\n      default:\n        return 'cqa-bg-[#94A3B8]';\n    }\n  }\n\n  /** Text color for metadata value (e.g. red for critical priority) */\n  getValueTextClass(item: TestCaseDetailsMetadataItem): string {\n    if (item.statusColor === 'red') {\n      return 'cqa-text-[#DC2626]';\n    }\n    return 'cqa-text-[#111827]';\n  }\n}\n","<!-- Edit mode: show edit form (Figma design) -->\n<cqa-test-case-details-edit\n  *ngIf=\"editing\"\n  [descriptionTitle]=\"descriptionTitle\"\n  [descriptionContent]=\"descriptionContent\"\n  [metadataItems]=\"metadataItems\"\n  [labels]=\"labels\"\n  [configTitle]=\"configTitle\"\n  [configSections]=\"configSections\"\n  [configSectionsRow2]=\"configSectionsRow2\"\n  [platform]=\"platform\"\n  [selectConfigOverrides]=\"selectConfigOverrides\"\n  (save)=\"onSaveChanges($event)\"\n  (cancel)=\"onCancelEdit()\"\n  (selectSearch)=\"selectSearch.emit($event)\"\n  (selectLoadMore)=\"selectLoadMore.emit($event)\"\n  (selectOpened)=\"selectOpened.emit($event)\"\n  (selectionChange)=\"selectionChange.emit($event)\">\n</cqa-test-case-details-edit>\n\n<!-- View mode: read-only details -->\n<div *ngIf=\"!editing\" class=\"cqa-self-stretch cqa-py-4 cqa-px-0 cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-4 cqa-overflow-hidden cqa-text-sm cqa-leading-[19.6px]\">\n  <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-4 cqa-text-sm cqa-leading-[19.6px]\">\n    <!-- Description Section -->\n    <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n      <div class=\"cqa-self-stretch cqa-inline-flex cqa-justify-between cqa-items-center\">\n        <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n          <div class=\"cqa-w-5 cqa-h-5 cqa-p-1 cqa-flex-shrink-0 cqa-bg-[#3F43EE1A] cqa-rounded-md cqa-inline-flex cqa-justify-center cqa-items-center\">\n            <mat-icon class=\"cqa-text-[#1B1FEB] cqa-text-[12px] cqa-w-3 cqa-h-3 cqa-leading-none cqa-block\">description</mat-icon>\n          </div>\n          <span class=\"cqa-text-[#374151] cqa-text-sm cqa-font-semibold cqa-leading-[19.6px]\">{{ descriptionTitle }}</span>\n        </div>\n        <button\n          *ngIf=\"showEditButton\"\n          type=\"button\"\n          class=\"cqa-flex cqa-justify-end cqa-items-center cqa-gap-2 cqa-text-[#A3A3A3] cqa-text-xs cqa-font-semibold hover:cqa-text-[#737373] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2\"\n          (click)=\"onEditClick()\">\n          <mat-icon class=\"cqa-text-[14px] cqa-w-4 cqa-h-4\">edit</mat-icon>\n          Edit\n        </button>\n      </div>\n      <div *ngIf=\"descriptionContent\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start\">\n        <div class=\"cqa-self-stretch cqa-text-[#111827] cqa-text-sm cqa-leading-[19.6px] cqa-font-normal\">{{ descriptionContent }}</div>\n      </div>\n    </div>\n\n    <!-- Metadata Section -->\n    <div *ngIf=\"metadataItems.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n      <div class=\"cqa-self-stretch cqa-grid cqa-grid-cols-3 cqa-gap-x-0 cqa-gap-y-4\">\n        <ng-container *ngFor=\"let item of metadataItems; trackBy: trackByMetadataLabel\">\n          <div class=\"cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-0.5\">\n            <div class=\"cqa-text-[#6B7280] cqa-text-xs cqa-leading-[15px] cqa-font-medium\">{{ item.label }}</div>\n            <div class=\"cqa-flex cqa-justify-start cqa-items-center cqa-gap-2.5 cqa-min-w-0 cqa-w-full\">\n              <mat-icon *ngIf=\"item.icon && (!item.iconLibrary || item.iconLibrary === 'mat')\" class=\"cqa-text-[#6B7280] cqa-text-sm cqa-w-4 cqa-h-4\">\n                {{ item.icon }}\n              </mat-icon>\n              <span *ngIf=\"item.statusColor\" class=\"cqa-w-3.5 cqa-h-3.5 cqa-rounded-full cqa-flex-shrink-0\" [ngClass]=\"getStatusDotClass(item)\"></span>\n              <a\n                *ngIf=\"item.link\"\n                href=\"javascript:void(0)\"\n                class=\"cqa-text-[#2563EB] cqa-text-xs cqa-leading-[12px] cqa-font-normal hover:cqa-underline cqa-cursor-pointer cqa-block cqa-truncate cqa-min-w-0\"\n                (click)=\"onMetadataLinkClick(item); $event.preventDefault()\">\n                {{ item.value }}\n              </a>\n              <span *ngIf=\"!item.link\" class=\"cqa-text-xs cqa-leading-[12px] cqa-font-normal\" [ngClass]=\"getValueTextClass(item)\">\n                {{ item.value }}\n              </span>\n            </div>\n          </div>\n        </ng-container>\n      </div>\n    </div>\n\n    <!-- Labels (Figma: Labels title, cqa-badge chips with tag icon) -->\n    <div *ngIf=\"labels.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2 cqa-text-sm cqa-leading-[19.6px]\">\n      <span class=\"cqa-text-[#111827] cqa-text-sm cqa-font-medium cqa-leading-5\">Labels</span>\n      <div class=\"cqa-flex cqa-flex-wrap cqa-gap-2\">\n        <cqa-badge\n          *ngFor=\"let label of labels\"\n          [label]=\"label\"\n          icon=\"label\"\n          variant=\"outline\"\n          size=\"small\"\n          backgroundColor=\"#ffffff\"\n          textColor=\"#475569\"\n          borderColor=\"#E2E8F0\"\n          iconColor=\"#94A3B8\">\n        </cqa-badge>\n      </div>\n    </div>\n\n    <!-- Configuration Section -->\n    <div *ngIf=\"configSections.length || configSectionsRow2.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n      <div class=\"cqa-self-stretch cqa-inline-flex cqa-justify-start cqa-items-center\">\n        <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n          <div class=\"cqa-w-5 cqa-h-5 cqa-p-1 cqa-flex-shrink-0 cqa-bg-[#3F43EE1A] cqa-rounded-md cqa-inline-flex cqa-justify-center cqa-items-center\">\n            <mat-icon class=\"cqa-text-[#1B1FEB] cqa-text-[12px] cqa-w-3 cqa-h-3 cqa-leading-none cqa-block\">settings</mat-icon>\n          </div>\n          <span class=\"cqa-text-[#374151] cqa-text-sm cqa-font-semibold cqa-leading-[19.6px]\">{{ configTitle }}</span>\n        </div>\n      </div>\n      <div class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-justify-start cqa-items-start cqa-gap-2.5\">\n        <div class=\"cqa-self-stretch cqa-grid cqa-grid-cols-1 cqa-gap-2.5\">\n          <ng-container *ngFor=\"let section of configSections; trackBy: trackByConfigTitle\">\n            <cqa-configuration-card\n              [icon]=\"section.icon || 'tune'\"\n              [title]=\"section.title\"\n              [data]=\"section.items\">\n            </cqa-configuration-card>\n          </ng-container>\n        </div>\n        <div *ngIf=\"configSectionsRow2.length\" class=\"cqa-self-stretch cqa-flex cqa-flex-col cqa-gap-4 cqa-text-sm cqa-leading-[19.6px]\">\n          <cqa-configuration-card\n            *ngFor=\"let section of configSectionsRow2; trackBy: trackByConfigTitle\"\n            [icon]=\"section.icon || 'tune'\"\n            [title]=\"section.title\"\n            [data]=\"section.items\">\n          </cqa-configuration-card>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
/** Default status value -> dot color. Override via BuildTestCaseDetailsOptions. */
|
|
2
|
+
export const DEFAULT_STATUS_COLOR_CONFIG = {
|
|
3
|
+
active: 'green',
|
|
4
|
+
inactive: 'gray',
|
|
5
|
+
pending: 'yellow',
|
|
6
|
+
completed: 'green',
|
|
7
|
+
'in review': 'yellow',
|
|
8
|
+
draft: 'gray',
|
|
9
|
+
approved: 'green',
|
|
10
|
+
rejected: 'red',
|
|
11
|
+
blocked: 'red',
|
|
12
|
+
'in progress': 'yellow',
|
|
13
|
+
done: 'green',
|
|
14
|
+
};
|
|
15
|
+
/** Default priority value -> dot color. Override via BuildTestCaseDetailsOptions. */
|
|
16
|
+
export const DEFAULT_PRIORITY_COLOR_CONFIG = {
|
|
17
|
+
critical: 'red',
|
|
18
|
+
major: 'red',
|
|
19
|
+
high: 'red',
|
|
20
|
+
medium: 'yellow',
|
|
21
|
+
low: 'green',
|
|
22
|
+
minor: 'green',
|
|
23
|
+
'not set': 'gray',
|
|
24
|
+
p0: 'red',
|
|
25
|
+
p1: 'yellow',
|
|
26
|
+
p2: 'green',
|
|
27
|
+
p3: 'gray',
|
|
28
|
+
};
|
|
29
|
+
/**
|
|
30
|
+
* API field names for mapping test case data to cqa-test-case-details.
|
|
31
|
+
* Use these when building metadataItems and configSections from your API response.
|
|
32
|
+
*/
|
|
33
|
+
export const TEST_CASE_DETAILS_FIELD_MAP = {
|
|
34
|
+
/** Metadata (description section) */
|
|
35
|
+
metadata: {
|
|
36
|
+
createdOn: 'createdOn',
|
|
37
|
+
status: 'status',
|
|
38
|
+
priority: 'priority',
|
|
39
|
+
environment: 'environment',
|
|
40
|
+
version: 'version',
|
|
41
|
+
platform: 'platform',
|
|
42
|
+
labels: 'labels',
|
|
43
|
+
},
|
|
44
|
+
/** Execution config */
|
|
45
|
+
execution: {
|
|
46
|
+
prerequisiteCase: 'prerequisiteCase',
|
|
47
|
+
defaultBrowser: 'defaultBrowser',
|
|
48
|
+
videoRecording: 'videoRecording',
|
|
49
|
+
testCaseTimeout: 'testCaseTimeout',
|
|
50
|
+
waitTimeoutForLocators: 'waitTimeoutForLocators',
|
|
51
|
+
webBrowser: 'webBrowser',
|
|
52
|
+
viewport: 'viewport',
|
|
53
|
+
},
|
|
54
|
+
/** AI Configuration */
|
|
55
|
+
aiConfig: {
|
|
56
|
+
enableAiSmartness: 'enableAiSmartness',
|
|
57
|
+
defaultAiAction: 'defaultAiAction',
|
|
58
|
+
knowledgeBaseDefaultTestCase: 'knowledgeBaseDefaultTestCase',
|
|
59
|
+
useAiMetadata: 'useAiMetadata',
|
|
60
|
+
},
|
|
61
|
+
/** Waits & Retries */
|
|
62
|
+
waitsRetries: {
|
|
63
|
+
autoWait: 'autoWait',
|
|
64
|
+
retryFailedSteps: 'retryFailedSteps',
|
|
65
|
+
},
|
|
66
|
+
/** Device */
|
|
67
|
+
device: {
|
|
68
|
+
platform: 'platform',
|
|
69
|
+
deviceType: 'deviceType',
|
|
70
|
+
deviceOS: 'deviceOS',
|
|
71
|
+
defaultBrowser: 'defaultBrowser',
|
|
72
|
+
defaultViewport: 'defaultViewport',
|
|
73
|
+
viewport: 'viewport',
|
|
74
|
+
},
|
|
75
|
+
};
|
|
76
|
+
/**
|
|
77
|
+
* Build metadataItems and configSections from API data for cqa-test-case-details.
|
|
78
|
+
* Use in ts-portal when mapping test case API response to component inputs.
|
|
79
|
+
*/
|
|
80
|
+
export function buildTestCaseDetailsFromApi(data, options) {
|
|
81
|
+
const formatDate = (v) => v ? (typeof v === 'string' ? v : v.toISOString?.() ?? String(v)) : '';
|
|
82
|
+
const statusConfig = { ...DEFAULT_STATUS_COLOR_CONFIG, ...options?.statusColorConfig };
|
|
83
|
+
const priorityConfig = { ...DEFAULT_PRIORITY_COLOR_CONFIG, ...options?.priorityColorConfig };
|
|
84
|
+
const resolveColor = (value, config) => config[value.toLowerCase().trim()] ?? 'gray';
|
|
85
|
+
const metadataItems = [];
|
|
86
|
+
if (data.createdOn != null)
|
|
87
|
+
metadataItems.push({ label: 'Created on', value: formatDate(data.createdOn), icon: 'calendar_today' });
|
|
88
|
+
if (data.status != null)
|
|
89
|
+
metadataItems.push({ label: 'Status', value: data.status, statusColor: resolveColor(data.status, statusConfig) });
|
|
90
|
+
if (data.priority != null)
|
|
91
|
+
metadataItems.push({ label: 'Priority', value: data.priority, statusColor: resolveColor(data.priority, priorityConfig) });
|
|
92
|
+
if (data.environment != null)
|
|
93
|
+
metadataItems.push({ label: 'Environment', value: data.environment, icon: 'storage' });
|
|
94
|
+
if (data.version != null)
|
|
95
|
+
metadataItems.push({ label: 'Version', value: data.version, icon: 'label' });
|
|
96
|
+
if (data.platform != null)
|
|
97
|
+
metadataItems.push({
|
|
98
|
+
label: 'Platform',
|
|
99
|
+
value: data.platform === 'web' ? 'web' : 'iOS',
|
|
100
|
+
icon: 'devices',
|
|
101
|
+
});
|
|
102
|
+
const configSections = [
|
|
103
|
+
{
|
|
104
|
+
title: 'Execution',
|
|
105
|
+
icon: 'play_circle',
|
|
106
|
+
items: [
|
|
107
|
+
{ label: 'Prerequisite Case', value: data.prerequisiteCase ?? null },
|
|
108
|
+
{ label: 'Video Recording', value: data.videoRecording ?? '' },
|
|
109
|
+
],
|
|
110
|
+
},
|
|
111
|
+
{
|
|
112
|
+
title: 'AI Configuration',
|
|
113
|
+
icon: 'smart_toy',
|
|
114
|
+
items: [
|
|
115
|
+
{ label: 'Enable AI Smartness', value: data.enableAiSmartness ?? '' },
|
|
116
|
+
{ label: 'Default AI Action', value: data.defaultAiAction ?? '' },
|
|
117
|
+
{ label: 'Knowledge Base Default Test Case', value: data.knowledgeBaseDefaultTestCase ?? null },
|
|
118
|
+
{ label: 'Enable AI metadata collection', value: data.useAiMetadata ?? '' },
|
|
119
|
+
],
|
|
120
|
+
},
|
|
121
|
+
];
|
|
122
|
+
const configSectionsRow2 = [
|
|
123
|
+
{
|
|
124
|
+
title: 'Wait and retries',
|
|
125
|
+
icon: 'timer',
|
|
126
|
+
items: [
|
|
127
|
+
{ label: 'Enable Avoid Auto wait for steps', value: data.autoWaitEnabled ?? false },
|
|
128
|
+
{ label: 'Retry Failed Steps', value: data.retryFailedSteps ?? '' },
|
|
129
|
+
{ label: 'Test Case Timeout (minutes)', value: data.testCaseTimeout ?? '' },
|
|
130
|
+
{ label: 'Wait Timeout for Locators (secs)', value: data.waitTimeoutForLocators ?? '' },
|
|
131
|
+
],
|
|
132
|
+
},
|
|
133
|
+
{
|
|
134
|
+
title: 'Device Setting',
|
|
135
|
+
icon: 'devices',
|
|
136
|
+
items: (() => {
|
|
137
|
+
const platform = data.platform ?? 'web';
|
|
138
|
+
if (platform === 'web') {
|
|
139
|
+
return [
|
|
140
|
+
{ label: 'Default Browser', value: data.defaultBrowser ?? data.webBrowser ?? '' },
|
|
141
|
+
{ label: 'Default Viewport', value: data.defaultViewport ?? data.viewport ?? '' },
|
|
142
|
+
];
|
|
143
|
+
}
|
|
144
|
+
return [
|
|
145
|
+
{ label: 'Default OS', value: data.deviceOS ?? '' },
|
|
146
|
+
{ label: 'Default Phone', value: data.deviceType ?? '' },
|
|
147
|
+
];
|
|
148
|
+
})(),
|
|
149
|
+
},
|
|
150
|
+
{
|
|
151
|
+
title: 'Key Flags',
|
|
152
|
+
icon: 'flag',
|
|
153
|
+
items: [
|
|
154
|
+
{ label: 'Mobile Testing', value: data.mobileTesting ?? '' },
|
|
155
|
+
{ label: 'Extension Use', value: data.extensionUse ?? '' },
|
|
156
|
+
{ label: 'Data Driven', value: data.dataDriven ?? '' },
|
|
157
|
+
],
|
|
158
|
+
},
|
|
159
|
+
];
|
|
160
|
+
return {
|
|
161
|
+
metadataItems,
|
|
162
|
+
labels: data.labels ?? [],
|
|
163
|
+
configSections,
|
|
164
|
+
configSectionsRow2,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-case-details.models.js","sourceRoot":"","sources":["../../../../../src/lib/test-case-details/test-case-details.models.ts"],"names":[],"mappings":"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","sourcesContent":["import { ConfigurationItem } from '../configuration-card/configuration-card.component';\n\nexport interface TestCaseDetailsConfigSection {\n  title: string;\n  icon?: string;\n  items: ConfigurationItem[];\n}\n\nexport type StatusDotColor = 'yellow' | 'red' | 'green' | 'gray';\n\nexport interface TestCaseDetailsMetadataItem {\n  label: string;\n  value: string;\n  icon?: string;\n  iconLibrary?: 'mat' | 'fa';\n  link?: string;\n  statusColor?: StatusDotColor;\n}\n\n/** Maps status/priority value (lowercase) to dot color. Override to customize. */\nexport type StatusColorConfig = Partial<Record<string, StatusDotColor>>;\n\n/** Default status value -> dot color. Override via BuildTestCaseDetailsOptions. */\nexport const DEFAULT_STATUS_COLOR_CONFIG: Record<string, StatusDotColor> = {\n  active: 'green',\n  inactive: 'gray',\n  pending: 'yellow',\n  completed: 'green',\n  'in review': 'yellow',\n  draft: 'gray',\n  approved: 'green',\n  rejected: 'red',\n  blocked: 'red',\n  'in progress': 'yellow',\n  done: 'green',\n};\n\n/** Default priority value -> dot color. Override via BuildTestCaseDetailsOptions. */\nexport const DEFAULT_PRIORITY_COLOR_CONFIG: Record<string, StatusDotColor> = {\n  critical: 'red',\n  major: 'red',\n  high: 'red',\n  medium: 'yellow',\n  low: 'green',\n  minor: 'green',\n  'not set': 'gray',\n  p0: 'red',\n  p1: 'yellow',\n  p2: 'green',\n  p3: 'gray',\n};\n\n/**\n * API field names for mapping test case data to cqa-test-case-details.\n * Use these when building metadataItems and configSections from your API response.\n */\nexport const TEST_CASE_DETAILS_FIELD_MAP = {\n  /** Metadata (description section) */\n  metadata: {\n    createdOn: 'createdOn',\n    status: 'status',\n    priority: 'priority',\n    environment: 'environment',\n    version: 'version',\n    platform: 'platform',\n    labels: 'labels',\n  },\n  /** Execution config */\n  execution: {\n    prerequisiteCase: 'prerequisiteCase',\n    defaultBrowser: 'defaultBrowser',\n    videoRecording: 'videoRecording',\n    testCaseTimeout: 'testCaseTimeout',\n    waitTimeoutForLocators: 'waitTimeoutForLocators',\n    webBrowser: 'webBrowser',\n    viewport: 'viewport',\n  },\n  /** AI Configuration */\n  aiConfig: {\n    enableAiSmartness: 'enableAiSmartness',\n    defaultAiAction: 'defaultAiAction',\n    knowledgeBaseDefaultTestCase: 'knowledgeBaseDefaultTestCase',\n    useAiMetadata: 'useAiMetadata',\n  },\n  /** Waits & Retries */\n  waitsRetries: {\n    autoWait: 'autoWait',\n    retryFailedSteps: 'retryFailedSteps',\n  },\n  /** Device */\n  device: {\n    platform: 'platform',\n    deviceType: 'deviceType',\n    deviceOS: 'deviceOS',\n    defaultBrowser: 'defaultBrowser',\n    defaultViewport: 'defaultViewport',\n    viewport: 'viewport',\n  },\n} as const;\n\n/** API response shape for test case details. Map your API fields to this structure. */\nexport interface TestCaseDetailsApiData {\n  createdOn?: string | Date;\n  status?: string;\n  priority?: string;\n  environment?: string;\n  version?: string;\n  testPlanName?: string;\n  testPlanRedirectUrl?: string;\n  labels?: string[];\n  prerequisiteCase?: string | null;\n  defaultBrowser?: string;\n  videoRecording?: boolean | string;\n  testCaseTimeout?: string;\n  waitTimeoutForLocators?: string;\n  webBrowser?: string;\n  viewport?: string;\n  enableAiSmartness?: string;\n  defaultAiAction?: string;\n  knowledgeBaseDefaultTestCase?: string | null;\n  useAiMetadata?: string;\n  autoWait?: string;\n  autoWaitEnabled?: boolean;\n  retryFailedSteps?: string;\n  platform?: 'web' | 'mobile';\n  deviceType?: string;\n  deviceOS?: string;\n  defaultViewport?: string;\n  /** Key Flags */\n  mobileTesting?: string | boolean;\n  extensionUse?: string | boolean;\n  dataDriven?: string | boolean;\n}\n\n/** Options for buildTestCaseDetailsFromApi */\nexport interface BuildTestCaseDetailsOptions {\n  /** Override status value -> dot color. Merged with defaults. */\n  statusColorConfig?: StatusColorConfig;\n  /** Override priority value -> dot color. Merged with defaults. */\n  priorityColorConfig?: StatusColorConfig;\n}\n\n/**\n * Build metadataItems and configSections from API data for cqa-test-case-details.\n * Use in ts-portal when mapping test case API response to component inputs.\n */\nexport function buildTestCaseDetailsFromApi(\n  data: TestCaseDetailsApiData,\n  options?: BuildTestCaseDetailsOptions\n): {\n  metadataItems: TestCaseDetailsMetadataItem[];\n  labels: string[];\n  configSections: TestCaseDetailsConfigSection[];\n  configSectionsRow2: TestCaseDetailsConfigSection[];\n} {\n  const formatDate = (v: string | Date | undefined) =>\n    v ? (typeof v === 'string' ? v : (v as Date).toISOString?.() ?? String(v)) : '';\n\n  const statusConfig = { ...DEFAULT_STATUS_COLOR_CONFIG, ...options?.statusColorConfig };\n  const priorityConfig = { ...DEFAULT_PRIORITY_COLOR_CONFIG, ...options?.priorityColorConfig };\n  const resolveColor = (value: string, config: Partial<Record<string, StatusDotColor>>): StatusDotColor =>\n    config[value.toLowerCase().trim()] ?? 'gray';\n\n  const metadataItems: TestCaseDetailsMetadataItem[] = [];\n  if (data.createdOn != null)\n    metadataItems.push({ label: 'Created on', value: formatDate(data.createdOn), icon: 'calendar_today' });\n  if (data.status != null)\n    metadataItems.push({ label: 'Status', value: data.status, statusColor: resolveColor(data.status, statusConfig) });\n  if (data.priority != null)\n    metadataItems.push({ label: 'Priority', value: data.priority, statusColor: resolveColor(data.priority, priorityConfig) });\n  if (data.environment != null)\n    metadataItems.push({ label: 'Environment', value: data.environment, icon: 'storage' });\n  if (data.version != null)\n    metadataItems.push({ label: 'Version', value: data.version, icon: 'label' });\n  if (data.platform != null)\n    metadataItems.push({\n      label: 'Platform',\n      value: data.platform === 'web' ? 'web' : 'iOS',\n      icon: 'devices',\n    });\n\n  const configSections: TestCaseDetailsConfigSection[] = [\n    {\n      title: 'Execution',\n      icon: 'play_circle',\n      items: [\n        { label: 'Prerequisite Case', value: data.prerequisiteCase ?? null },\n        { label: 'Video Recording', value: data.videoRecording ?? '' },\n      ],\n    },\n    {\n      title: 'AI Configuration',\n      icon: 'smart_toy',\n      items: [\n        { label: 'Enable AI Smartness', value: data.enableAiSmartness ?? '' },\n        { label: 'Default AI Action', value: data.defaultAiAction ?? '' },\n        { label: 'Knowledge Base Default Test Case', value: data.knowledgeBaseDefaultTestCase ?? null },\n        { label: 'Enable AI metadata collection', value: data.useAiMetadata ?? '' },\n      ],\n    },\n  ];\n\n  const configSectionsRow2: TestCaseDetailsConfigSection[] = [\n    {\n      title: 'Wait and retries',\n      icon: 'timer',\n      items: [\n        { label: 'Enable Avoid Auto wait for steps', value: data.autoWaitEnabled ?? false },\n        { label: 'Retry Failed Steps', value: data.retryFailedSteps ?? '' },\n        { label: 'Test Case Timeout (minutes)', value: data.testCaseTimeout ?? '' },\n        { label: 'Wait Timeout for Locators (secs)', value: data.waitTimeoutForLocators ?? '' },\n      ],\n    },\n    {\n      title: 'Device Setting',\n      icon: 'devices',\n      items: (() => {\n        const platform = data.platform ?? 'web';\n        if (platform === 'web') {\n          return [\n            { label: 'Default Browser', value: data.defaultBrowser ?? data.webBrowser ?? '' },\n            { label: 'Default Viewport', value: data.defaultViewport ?? data.viewport ?? '' },\n          ];\n        }\n        return [\n          { label: 'Default OS', value: data.deviceOS ?? '' },\n          { label: 'Default Phone', value: data.deviceType ?? '' },\n        ];\n      })(),\n    },\n    {\n      title: 'Key Flags',\n      icon: 'flag',\n      items: [\n        { label: 'Mobile Testing', value: data.mobileTesting ?? '' },\n        { label: 'Extension Use', value: data.extensionUse ?? '' },\n        { label: 'Data Driven', value: data.dataDriven ?? '' },\n      ],\n    },\n  ];\n\n  return {\n    metadataItems,\n    labels: data.labels ?? [],\n    configSections,\n    configSectionsRow2,\n  };\n}\n"]}
|