@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,38 @@
|
|
|
1
|
+
import { Component, Input, ContentChild } from '@angular/core';
|
|
2
|
+
import { DetailDrawerTabContentDirective } from '../detail-drawer-tab-content.directive';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class DetailDrawerTabComponent {
|
|
5
|
+
constructor() {
|
|
6
|
+
/** Tab label (shown in tooltip on icon button) */
|
|
7
|
+
this.label = '';
|
|
8
|
+
/** Tab value (unique identifier) */
|
|
9
|
+
this.value = '';
|
|
10
|
+
/** Material icon name for the tab button */
|
|
11
|
+
this.icon = 'folder';
|
|
12
|
+
}
|
|
13
|
+
get contentTemplate() {
|
|
14
|
+
return this.contentDirective?.templateRef ?? null;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
DetailDrawerTabComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailDrawerTabComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
18
|
+
DetailDrawerTabComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DetailDrawerTabComponent, selector: "cqa-detail-drawer-tab", inputs: { label: "label", value: "value", icon: "icon" }, host: { styleAttribute: "display: contents" }, queries: [{ propertyName: "contentDirective", first: true, predicate: DetailDrawerTabContentDirective, descendants: true }], ngImport: i0, template: '', isInline: true });
|
|
19
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailDrawerTabComponent, decorators: [{
|
|
20
|
+
type: Component,
|
|
21
|
+
args: [{
|
|
22
|
+
selector: 'cqa-detail-drawer-tab',
|
|
23
|
+
template: '',
|
|
24
|
+
host: {
|
|
25
|
+
style: 'display: contents',
|
|
26
|
+
},
|
|
27
|
+
}]
|
|
28
|
+
}], propDecorators: { label: [{
|
|
29
|
+
type: Input
|
|
30
|
+
}], value: [{
|
|
31
|
+
type: Input
|
|
32
|
+
}], icon: [{
|
|
33
|
+
type: Input
|
|
34
|
+
}], contentDirective: [{
|
|
35
|
+
type: ContentChild,
|
|
36
|
+
args: [DetailDrawerTabContentDirective]
|
|
37
|
+
}] } });
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLWRyYXdlci10YWIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9kZXRhaWwtZHJhd2VyL2RldGFpbC1kcmF3ZXItdGFiL2RldGFpbC1kcmF3ZXItdGFiLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxZQUFZLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUFFLCtCQUErQixFQUFFLE1BQU0sd0NBQXdDLENBQUM7O0FBU3pGLE1BQU0sT0FBTyx3QkFBd0I7SUFQckM7UUFRRSxrREFBa0Q7UUFDekMsVUFBSyxHQUFHLEVBQUUsQ0FBQztRQUVwQixvQ0FBb0M7UUFDM0IsVUFBSyxHQUFHLEVBQUUsQ0FBQztRQUVwQiw0Q0FBNEM7UUFDbkMsU0FBSSxHQUFHLFFBQVEsQ0FBQztLQVExQjtJQUhDLElBQUksZUFBZTtRQUNqQixPQUFPLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxXQUFXLElBQUksSUFBSSxDQUFDO0lBQ3BELENBQUM7O3FIQWZVLHdCQUF3Qjt5R0FBeEIsd0JBQXdCLG9OQVdyQiwrQkFBK0IsZ0RBaEJuQyxFQUFFOzJGQUtELHdCQUF3QjtrQkFQcEMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsdUJBQXVCO29CQUNqQyxRQUFRLEVBQUUsRUFBRTtvQkFDWixJQUFJLEVBQUU7d0JBQ0osS0FBSyxFQUFFLG1CQUFtQjtxQkFDM0I7aUJBQ0Y7OEJBR1UsS0FBSztzQkFBYixLQUFLO2dCQUdHLEtBQUs7c0JBQWIsS0FBSztnQkFHRyxJQUFJO3NCQUFaLEtBQUs7Z0JBR3lDLGdCQUFnQjtzQkFBOUQsWUFBWTt1QkFBQywrQkFBK0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBDb250ZW50Q2hpbGQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IERldGFpbERyYXdlclRhYkNvbnRlbnREaXJlY3RpdmUgfSBmcm9tICcuLi9kZXRhaWwtZHJhd2VyLXRhYi1jb250ZW50LmRpcmVjdGl2ZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NxYS1kZXRhaWwtZHJhd2VyLXRhYicsXG4gIHRlbXBsYXRlOiAnJyxcbiAgaG9zdDoge1xuICAgIHN0eWxlOiAnZGlzcGxheTogY29udGVudHMnLFxuICB9LFxufSlcbmV4cG9ydCBjbGFzcyBEZXRhaWxEcmF3ZXJUYWJDb21wb25lbnQge1xuICAvKiogVGFiIGxhYmVsIChzaG93biBpbiB0b29sdGlwIG9uIGljb24gYnV0dG9uKSAqL1xuICBASW5wdXQoKSBsYWJlbCA9ICcnO1xuXG4gIC8qKiBUYWIgdmFsdWUgKHVuaXF1ZSBpZGVudGlmaWVyKSAqL1xuICBASW5wdXQoKSB2YWx1ZSA9ICcnO1xuXG4gIC8qKiBNYXRlcmlhbCBpY29uIG5hbWUgZm9yIHRoZSB0YWIgYnV0dG9uICovXG4gIEBJbnB1dCgpIGljb24gPSAnZm9sZGVyJztcblxuICAvKiogVGVtcGxhdGUgZm9yIHRhYiBjb250ZW50IC0gdXNlIHdpdGggbmctdGVtcGxhdGUgY3FhVGFiQ29udGVudCAqL1xuICBAQ29udGVudENoaWxkKERldGFpbERyYXdlclRhYkNvbnRlbnREaXJlY3RpdmUpIGNvbnRlbnREaXJlY3RpdmU/OiBEZXRhaWxEcmF3ZXJUYWJDb250ZW50RGlyZWN0aXZlO1xuXG4gIGdldCBjb250ZW50VGVtcGxhdGUoKSB7XG4gICAgcmV0dXJuIHRoaXMuY29udGVudERpcmVjdGl2ZT8udGVtcGxhdGVSZWYgPz8gbnVsbDtcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { Directive } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class DetailDrawerTabContentDirective {
|
|
4
|
+
constructor(templateRef) {
|
|
5
|
+
this.templateRef = templateRef;
|
|
6
|
+
}
|
|
7
|
+
}
|
|
8
|
+
DetailDrawerTabContentDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailDrawerTabContentDirective, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
9
|
+
DetailDrawerTabContentDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.4.0", type: DetailDrawerTabContentDirective, selector: "[cqaTabContent]", ngImport: i0 });
|
|
10
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailDrawerTabContentDirective, decorators: [{
|
|
11
|
+
type: Directive,
|
|
12
|
+
args: [{
|
|
13
|
+
selector: '[cqaTabContent]',
|
|
14
|
+
}]
|
|
15
|
+
}], ctorParameters: function () { return [{ type: i0.TemplateRef }]; } });
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLWRyYXdlci10YWItY29udGVudC5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2RldGFpbC1kcmF3ZXIvZGV0YWlsLWRyYXdlci10YWItY29udGVudC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBZSxNQUFNLGVBQWUsQ0FBQzs7QUFLdkQsTUFBTSxPQUFPLCtCQUErQjtJQUMxQyxZQUFtQixXQUFpQztRQUFqQyxnQkFBVyxHQUFYLFdBQVcsQ0FBc0I7SUFBRyxDQUFDOzs0SEFEN0MsK0JBQStCO2dIQUEvQiwrQkFBK0I7MkZBQS9CLCtCQUErQjtrQkFIM0MsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsaUJBQWlCO2lCQUM1QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2NxYVRhYkNvbnRlbnRdJyxcbn0pXG5leHBvcnQgY2xhc3MgRGV0YWlsRHJhd2VyVGFiQ29udGVudERpcmVjdGl2ZSB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyB0ZW1wbGF0ZVJlZjogVGVtcGxhdGVSZWY8dW5rbm93bj4pIHt9XG59XG4iXX0=
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, ContentChildren, HostBinding, } from '@angular/core';
|
|
2
|
+
import { DetailDrawerTabComponent } from './detail-drawer-tab/detail-drawer-tab.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/material/icon";
|
|
5
|
+
import * as i2 from "@angular/material/tooltip";
|
|
6
|
+
import * as i3 from "@angular/common";
|
|
7
|
+
export class DetailDrawerComponent {
|
|
8
|
+
constructor() {
|
|
9
|
+
/** Currently active tab value */
|
|
10
|
+
this.activeTab = '';
|
|
11
|
+
/** Whether to show the close button */
|
|
12
|
+
this.showCloseButton = true;
|
|
13
|
+
/** Whether the drawer is expanded */
|
|
14
|
+
this.expanded = true;
|
|
15
|
+
/** Panel width when expanded */
|
|
16
|
+
this.expandedWidth = '280px';
|
|
17
|
+
/** Maximum width when expanded (e.g. '600px', '30vw'). Default: 30% of viewport */
|
|
18
|
+
this.maxExpandedWidth = '30vw';
|
|
19
|
+
/** Panel width when collapsed */
|
|
20
|
+
this.collapsedWidth = '56px';
|
|
21
|
+
this.expandTooltip = 'Expand';
|
|
22
|
+
this.collapseTooltip = 'Collapse';
|
|
23
|
+
this.closeTooltip = 'Close';
|
|
24
|
+
this.activeTabChange = new EventEmitter();
|
|
25
|
+
this.expandToggle = new EventEmitter();
|
|
26
|
+
this.close = new EventEmitter();
|
|
27
|
+
}
|
|
28
|
+
get hostWidth() {
|
|
29
|
+
return this.expanded ? this.expandedWidth : this.collapsedWidth;
|
|
30
|
+
}
|
|
31
|
+
get hostMinWidth() {
|
|
32
|
+
return this.expanded && this.minExpandedWidth ? this.minExpandedWidth : null;
|
|
33
|
+
}
|
|
34
|
+
get hostMaxWidth() {
|
|
35
|
+
return this.expanded ? this.maxExpandedWidth : null;
|
|
36
|
+
}
|
|
37
|
+
ngAfterContentInit() {
|
|
38
|
+
this.ensureActiveTab();
|
|
39
|
+
}
|
|
40
|
+
ngAfterContentChecked() {
|
|
41
|
+
this.ensureActiveTab();
|
|
42
|
+
}
|
|
43
|
+
ensureActiveTab() {
|
|
44
|
+
const tabs = this.tabComponents?.toArray() ?? [];
|
|
45
|
+
if (tabs.length > 0 && !this.activeTab) {
|
|
46
|
+
this.activeTab = tabs[0].value;
|
|
47
|
+
this.activeTabChange.emit(this.activeTab);
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
onTabClick(tab) {
|
|
51
|
+
// If drawer is collapsed, open it (but never close on tab click)
|
|
52
|
+
if (!this.expanded) {
|
|
53
|
+
this.expandToggle.emit();
|
|
54
|
+
}
|
|
55
|
+
// Select the tab
|
|
56
|
+
if (tab.value !== this.activeTab) {
|
|
57
|
+
this.activeTab = tab.value;
|
|
58
|
+
this.activeTabChange.emit(this.activeTab);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
onExpandToggle() {
|
|
62
|
+
this.expandToggle.emit();
|
|
63
|
+
}
|
|
64
|
+
onClose() {
|
|
65
|
+
this.close.emit();
|
|
66
|
+
}
|
|
67
|
+
trackByValue(_i, tab) {
|
|
68
|
+
return tab.value;
|
|
69
|
+
}
|
|
70
|
+
isTabActive(tab) {
|
|
71
|
+
return tab.value === this.activeTab;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
DetailDrawerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailDrawerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
75
|
+
DetailDrawerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DetailDrawerComponent, selector: "cqa-detail-drawer", inputs: { activeTab: "activeTab", showCloseButton: "showCloseButton", expanded: "expanded", expandedWidth: "expandedWidth", minExpandedWidth: "minExpandedWidth", maxExpandedWidth: "maxExpandedWidth", collapsedWidth: "collapsedWidth", expandTooltip: "expandTooltip", collapseTooltip: "collapseTooltip", closeTooltip: "closeTooltip" }, outputs: { activeTabChange: "activeTabChange", expandToggle: "expandToggle", close: "close" }, host: { properties: { "style.width": "this.hostWidth", "style.min-width": "this.hostMinWidth", "style.max-width": "this.hostMaxWidth" }, styleAttribute: "transition: width 0.3s ease-in-out", classAttribute: "cqa-ui-root cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-shadow-[-4px_0_6px_-1px_rgba(0,0,0,0.05)]" }, queries: [{ propertyName: "tabComponents", predicate: DetailDrawerTabComponent }], ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white\">\n <div\n class=\"cqa-grid cqa-flex-1 cqa-min-h-0 cqa-overflow-hidden\"\n [style.grid-template-columns]=\"expanded ? 'auto 1fr' : 'auto 0fr'\"\n style=\"transition: grid-template-columns 0.3s ease-in-out\">\n <!-- Left vertical icon bar: one button per tab -->\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-min-h-0 cqa-items-center cqa-py-4 cqa-px-2 cqa-border-r cqa-border-[#E2E8F0] cqa-bg-[#FAFAFA] cqa-flex-shrink-0\">\n <!-- Expand / Collapse button (always visible) -->\n <button\n type=\"button\"\n [matTooltip]=\"expanded ? collapseTooltip : expandTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n (click)=\"onExpandToggle()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ expanded ? 'unfold_less' : 'unfold_more' }}</mat-icon>\n </button>\n <!-- Close button -->\n <button\n *ngIf=\"showCloseButton\"\n type=\"button\"\n [matTooltip]=\"closeTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n (click)=\"onClose()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">close</mat-icon>\n </button>\n <!-- Tab buttons: equally distributed in remaining space -->\n <div class=\"cqa-flex cqa-flex-1 cqa-flex-col cqa-justify-evenly cqa-items-center cqa-w-full\">\n <button\n *ngFor=\"let tab of tabComponents; trackBy: trackByValue\"\n type=\"button\"\n role=\"tab\"\n [attr.aria-selected]=\"isTabActive(tab)\"\n [matTooltip]=\"tab.label\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n [ngClass]=\"{\n 'cqa-bg-[#3F43EE] cqa-text-white': isTabActive(tab),\n 'cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155]': !isTabActive(tab)\n }\"\n (click)=\"onTabClick(tab)\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ tab.icon }}</mat-icon>\n </button>\n </div>\n </div>\n\n <!-- Content area: show only the active tab's content -->\n <div class=\"cqa-min-w-0 cqa-overflow-hidden\">\n <div class=\"cqa-h-full cqa-overflow-y-auto cqa-p-4 cqa-min-w-[280px]\">\n <ng-container *ngFor=\"let tab of tabComponents\">\n <ng-container *ngIf=\"tab.value === activeTab && tab.contentTemplate\" [ngTemplateOutlet]=\"tab.contentTemplate\"></ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
76
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailDrawerComponent, decorators: [{
|
|
77
|
+
type: Component,
|
|
78
|
+
args: [{ selector: 'cqa-detail-drawer', changeDetection: ChangeDetectionStrategy.OnPush, host: {
|
|
79
|
+
class: 'cqa-ui-root cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-shadow-[-4px_0_6px_-1px_rgba(0,0,0,0.05)]',
|
|
80
|
+
style: 'transition: width 0.3s ease-in-out',
|
|
81
|
+
}, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white\">\n <div\n class=\"cqa-grid cqa-flex-1 cqa-min-h-0 cqa-overflow-hidden\"\n [style.grid-template-columns]=\"expanded ? 'auto 1fr' : 'auto 0fr'\"\n style=\"transition: grid-template-columns 0.3s ease-in-out\">\n <!-- Left vertical icon bar: one button per tab -->\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-min-h-0 cqa-items-center cqa-py-4 cqa-px-2 cqa-border-r cqa-border-[#E2E8F0] cqa-bg-[#FAFAFA] cqa-flex-shrink-0\">\n <!-- Expand / Collapse button (always visible) -->\n <button\n type=\"button\"\n [matTooltip]=\"expanded ? collapseTooltip : expandTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n (click)=\"onExpandToggle()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ expanded ? 'unfold_less' : 'unfold_more' }}</mat-icon>\n </button>\n <!-- Close button -->\n <button\n *ngIf=\"showCloseButton\"\n type=\"button\"\n [matTooltip]=\"closeTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n (click)=\"onClose()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">close</mat-icon>\n </button>\n <!-- Tab buttons: equally distributed in remaining space -->\n <div class=\"cqa-flex cqa-flex-1 cqa-flex-col cqa-justify-evenly cqa-items-center cqa-w-full\">\n <button\n *ngFor=\"let tab of tabComponents; trackBy: trackByValue\"\n type=\"button\"\n role=\"tab\"\n [attr.aria-selected]=\"isTabActive(tab)\"\n [matTooltip]=\"tab.label\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n [ngClass]=\"{\n 'cqa-bg-[#3F43EE] cqa-text-white': isTabActive(tab),\n 'cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155]': !isTabActive(tab)\n }\"\n (click)=\"onTabClick(tab)\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ tab.icon }}</mat-icon>\n </button>\n </div>\n </div>\n\n <!-- Content area: show only the active tab's content -->\n <div class=\"cqa-min-w-0 cqa-overflow-hidden\">\n <div class=\"cqa-h-full cqa-overflow-y-auto cqa-p-4 cqa-min-w-[280px]\">\n <ng-container *ngFor=\"let tab of tabComponents\">\n <ng-container *ngIf=\"tab.value === activeTab && tab.contentTemplate\" [ngTemplateOutlet]=\"tab.contentTemplate\"></ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n</div>\n", styles: [] }]
|
|
82
|
+
}], propDecorators: { tabComponents: [{
|
|
83
|
+
type: ContentChildren,
|
|
84
|
+
args: [DetailDrawerTabComponent]
|
|
85
|
+
}], activeTab: [{
|
|
86
|
+
type: Input
|
|
87
|
+
}], showCloseButton: [{
|
|
88
|
+
type: Input
|
|
89
|
+
}], expanded: [{
|
|
90
|
+
type: Input
|
|
91
|
+
}], expandedWidth: [{
|
|
92
|
+
type: Input
|
|
93
|
+
}], minExpandedWidth: [{
|
|
94
|
+
type: Input
|
|
95
|
+
}], maxExpandedWidth: [{
|
|
96
|
+
type: Input
|
|
97
|
+
}], collapsedWidth: [{
|
|
98
|
+
type: Input
|
|
99
|
+
}], hostWidth: [{
|
|
100
|
+
type: HostBinding,
|
|
101
|
+
args: ['style.width']
|
|
102
|
+
}], hostMinWidth: [{
|
|
103
|
+
type: HostBinding,
|
|
104
|
+
args: ['style.min-width']
|
|
105
|
+
}], hostMaxWidth: [{
|
|
106
|
+
type: HostBinding,
|
|
107
|
+
args: ['style.max-width']
|
|
108
|
+
}], expandTooltip: [{
|
|
109
|
+
type: Input
|
|
110
|
+
}], collapseTooltip: [{
|
|
111
|
+
type: Input
|
|
112
|
+
}], closeTooltip: [{
|
|
113
|
+
type: Input
|
|
114
|
+
}], activeTabChange: [{
|
|
115
|
+
type: Output
|
|
116
|
+
}], expandToggle: [{
|
|
117
|
+
type: Output
|
|
118
|
+
}], close: [{
|
|
119
|
+
type: Output
|
|
120
|
+
}] } });
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detail-drawer.component.js","sourceRoot":"","sources":["../../../../../src/lib/detail-drawer/detail-drawer.component.ts","../../../../../src/lib/detail-drawer/detail-drawer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ChangeDetectionStrategy,\n  ContentChildren,\n  QueryList,\n  AfterContentInit,\n  AfterContentChecked,\n  HostBinding,\n} from '@angular/core';\nimport { DetailDrawerTabComponent } from './detail-drawer-tab/detail-drawer-tab.component';\n\n@Component({\n  selector: 'cqa-detail-drawer',\n  templateUrl: './detail-drawer.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: {\n    class:\n      'cqa-ui-root cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-shadow-[-4px_0_6px_-1px_rgba(0,0,0,0.05)]',\n    style: 'transition: width 0.3s ease-in-out',\n  },\n})\nexport class DetailDrawerComponent implements AfterContentInit, AfterContentChecked {\n  @ContentChildren(DetailDrawerTabComponent) tabComponents!: QueryList<DetailDrawerTabComponent>;\n\n  /** Currently active tab value */\n  @Input() activeTab = '';\n\n  /** Whether to show the close button */\n  @Input() showCloseButton = true;\n\n  /** Whether the drawer is expanded */\n  @Input() expanded = true;\n\n  /** Panel width when expanded */\n  @Input() expandedWidth = '280px';\n\n  /** Minimum width when expanded (e.g. '280px') */\n  @Input() minExpandedWidth?: string;\n\n  /** Maximum width when expanded (e.g. '600px', '30vw'). Default: 30% of viewport */\n  @Input() maxExpandedWidth = '30vw';\n\n  /** Panel width when collapsed */\n  @Input() collapsedWidth = '56px';\n\n  @HostBinding('style.width') get hostWidth(): string {\n    return this.expanded ? this.expandedWidth : this.collapsedWidth;\n  }\n\n  @HostBinding('style.min-width') get hostMinWidth(): string | null {\n    return this.expanded && this.minExpandedWidth ? this.minExpandedWidth : null;\n  }\n\n  @HostBinding('style.max-width') get hostMaxWidth(): string | null {\n    return this.expanded ? this.maxExpandedWidth : null;\n  }\n\n  @Input() expandTooltip = 'Expand';\n  @Input() collapseTooltip = 'Collapse';\n  @Input() closeTooltip = 'Close';\n\n  @Output() activeTabChange = new EventEmitter<string>();\n  @Output() expandToggle = new EventEmitter<void>();\n  @Output() close = new EventEmitter<void>();\n\n  ngAfterContentInit(): void {\n    this.ensureActiveTab();\n  }\n\n  ngAfterContentChecked(): void {\n    this.ensureActiveTab();\n  }\n\n  private ensureActiveTab(): void {\n    const tabs = this.tabComponents?.toArray() ?? [];\n    if (tabs.length > 0 && !this.activeTab) {\n      this.activeTab = tabs[0].value;\n      this.activeTabChange.emit(this.activeTab);\n    }\n  }\n\n  onTabClick(tab: DetailDrawerTabComponent): void {\n    // If drawer is collapsed, open it (but never close on tab click)\n    if (!this.expanded) {\n      this.expandToggle.emit();\n    }\n    // Select the tab\n    if (tab.value !== this.activeTab) {\n      this.activeTab = tab.value;\n      this.activeTabChange.emit(this.activeTab);\n    }\n  }\n\n  onExpandToggle(): void {\n    this.expandToggle.emit();\n  }\n\n  onClose(): void {\n    this.close.emit();\n  }\n\n  trackByValue(_i: number, tab: DetailDrawerTabComponent): string {\n    return tab.value;\n  }\n\n  isTabActive(tab: DetailDrawerTabComponent): boolean {\n    return tab.value === this.activeTab;\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white\">\n  <div\n    class=\"cqa-grid cqa-flex-1 cqa-min-h-0 cqa-overflow-hidden\"\n    [style.grid-template-columns]=\"expanded ? 'auto 1fr' : 'auto 0fr'\"\n    style=\"transition: grid-template-columns 0.3s ease-in-out\">\n    <!-- Left vertical icon bar: one button per tab -->\n    <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-min-h-0 cqa-items-center cqa-py-4 cqa-px-2 cqa-border-r cqa-border-[#E2E8F0] cqa-bg-[#FAFAFA] cqa-flex-shrink-0\">\n      <!-- Expand / Collapse button (always visible) -->\n      <button\n        type=\"button\"\n        [matTooltip]=\"expanded ? collapseTooltip : expandTooltip\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n        (click)=\"onExpandToggle()\">\n        <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ expanded ? 'unfold_less' : 'unfold_more' }}</mat-icon>\n      </button>\n      <!-- Close button -->\n      <button\n        *ngIf=\"showCloseButton\"\n        type=\"button\"\n        [matTooltip]=\"closeTooltip\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n        (click)=\"onClose()\">\n        <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">close</mat-icon>\n      </button>\n      <!-- Tab buttons: equally distributed in remaining space -->\n      <div class=\"cqa-flex cqa-flex-1 cqa-flex-col cqa-justify-evenly cqa-items-center cqa-w-full\">\n        <button\n          *ngFor=\"let tab of tabComponents; trackBy: trackByValue\"\n          type=\"button\"\n          role=\"tab\"\n          [attr.aria-selected]=\"isTabActive(tab)\"\n          [matTooltip]=\"tab.label\"\n          class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-transition-colors focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2 cqa-flex-shrink-0\"\n          [ngClass]=\"{\n            'cqa-bg-[#3F43EE] cqa-text-white': isTabActive(tab),\n            'cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155]': !isTabActive(tab)\n          }\"\n          (click)=\"onTabClick(tab)\">\n          <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ tab.icon }}</mat-icon>\n        </button>\n      </div>\n    </div>\n\n    <!-- Content area: show only the active tab's content -->\n    <div class=\"cqa-min-w-0 cqa-overflow-hidden\">\n      <div class=\"cqa-h-full cqa-overflow-y-auto cqa-p-4 cqa-min-w-[280px]\">\n        <ng-container *ngFor=\"let tab of tabComponents\">\n          <ng-container *ngIf=\"tab.value === activeTab && tab.contentTemplate\" [ngTemplateOutlet]=\"tab.contentTemplate\"></ng-container>\n        </ng-container>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -0,0 +1,211 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, HostBinding, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "../test-case-details/test-case-details.component";
|
|
5
|
+
import * as i3 from "@angular/material/tooltip";
|
|
6
|
+
import * as i4 from "@angular/common";
|
|
7
|
+
export class DetailSidePanelComponent {
|
|
8
|
+
constructor() {
|
|
9
|
+
/** Tabs - each tab has a side panel icon button; tabs and buttons are 1:1 */
|
|
10
|
+
this.tabs = [
|
|
11
|
+
{ label: 'Test Case', value: 'test-case', icon: 'description' },
|
|
12
|
+
{ label: 'Data Library', value: 'data-library', icon: 'folder' },
|
|
13
|
+
{ label: 'Variables', value: 'variables', icon: 'code' },
|
|
14
|
+
];
|
|
15
|
+
/** Currently active tab value */
|
|
16
|
+
this.activeTab = 'test-case';
|
|
17
|
+
/** Description section title */
|
|
18
|
+
this.descriptionTitle = 'Description';
|
|
19
|
+
/** Description text content */
|
|
20
|
+
this.descriptionContent = '';
|
|
21
|
+
/** Whether to show the Edit button in the Description header */
|
|
22
|
+
this.showEditButton = true;
|
|
23
|
+
/** Metadata items (Created on, Status, Priority, etc.) */
|
|
24
|
+
this.metadataItems = [];
|
|
25
|
+
/** Labels/tags (e.g. Automation, API, SDK, UI/UX) */
|
|
26
|
+
this.labels = [];
|
|
27
|
+
/** Configuration sections - full width (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. */
|
|
32
|
+
this.platform = 'web';
|
|
33
|
+
/** Configuration section title */
|
|
34
|
+
this.configTitle = 'Configuration';
|
|
35
|
+
/** Whether to show the close button in the side menu */
|
|
36
|
+
this.showCloseButton = false;
|
|
37
|
+
/** When true, test case details start in edit mode (useful for Storybook). */
|
|
38
|
+
this.startInEditMode = false;
|
|
39
|
+
/** Override config per select for API-driven options, server search, load more. */
|
|
40
|
+
this.selectConfigOverrides = {};
|
|
41
|
+
/** Whether the panel is expanded (affects expand button icon and panel width) */
|
|
42
|
+
this.expanded = true;
|
|
43
|
+
/** Panel width when expanded (e.g. '480px', '25%') */
|
|
44
|
+
this.expandedWidth = '380px';
|
|
45
|
+
/** Panel width when collapsed (e.g. '56px' - fits icon bar + back button) */
|
|
46
|
+
this.collapsedWidth = '56px';
|
|
47
|
+
this.hostOverflow = 'hidden';
|
|
48
|
+
/** Tooltip for expand button when panel is collapsed */
|
|
49
|
+
this.expandTooltip = 'Expand';
|
|
50
|
+
/** Tooltip for expand button when panel is expanded (collapse) */
|
|
51
|
+
this.collapseTooltip = 'Collapse';
|
|
52
|
+
/** Tooltip for close button */
|
|
53
|
+
this.closeTooltip = 'Close';
|
|
54
|
+
this.back = new EventEmitter();
|
|
55
|
+
this.tabChange = new EventEmitter();
|
|
56
|
+
this.editDescription = new EventEmitter();
|
|
57
|
+
this.saveChanges = new EventEmitter();
|
|
58
|
+
this.metadataLinkClick = new EventEmitter();
|
|
59
|
+
this.selectSearch = new EventEmitter();
|
|
60
|
+
this.selectLoadMore = new EventEmitter();
|
|
61
|
+
this.selectOpened = new EventEmitter();
|
|
62
|
+
this.selectionChange = new EventEmitter();
|
|
63
|
+
this.expandToggle = new EventEmitter();
|
|
64
|
+
this.close = new EventEmitter();
|
|
65
|
+
}
|
|
66
|
+
get hostWidth() {
|
|
67
|
+
return this.expanded ? this.expandedWidth : this.collapsedWidth;
|
|
68
|
+
}
|
|
69
|
+
get hostMinWidth() {
|
|
70
|
+
return this.expanded ? this.expandedWidth : this.collapsedWidth;
|
|
71
|
+
}
|
|
72
|
+
get hostMaxWidth() {
|
|
73
|
+
return this.expanded ? this.expandedWidth : this.collapsedWidth;
|
|
74
|
+
}
|
|
75
|
+
trackByTabValue(_i, tab) {
|
|
76
|
+
return tab.value;
|
|
77
|
+
}
|
|
78
|
+
trackByMetadataLabel(_i, item) {
|
|
79
|
+
return item.label;
|
|
80
|
+
}
|
|
81
|
+
trackByConfigTitle(_i, section) {
|
|
82
|
+
return section.title;
|
|
83
|
+
}
|
|
84
|
+
onBack() {
|
|
85
|
+
this.back.emit();
|
|
86
|
+
}
|
|
87
|
+
onTabClick(tab) {
|
|
88
|
+
if (!this.expanded) {
|
|
89
|
+
this.expandToggle.emit();
|
|
90
|
+
}
|
|
91
|
+
if (tab.value !== this.activeTab) {
|
|
92
|
+
this.tabChange.emit(tab.value);
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
onEditDescription() {
|
|
96
|
+
this.editDescription.emit();
|
|
97
|
+
}
|
|
98
|
+
onSaveChanges(data) {
|
|
99
|
+
this.saveChanges.emit(data);
|
|
100
|
+
}
|
|
101
|
+
onExpandToggle() {
|
|
102
|
+
this.expandToggle.emit();
|
|
103
|
+
}
|
|
104
|
+
onClose() {
|
|
105
|
+
this.close.emit();
|
|
106
|
+
}
|
|
107
|
+
onMetadataLinkClick(item) {
|
|
108
|
+
if (item.link) {
|
|
109
|
+
this.metadataLinkClick.emit(item);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
getStatusDotClass(item) {
|
|
113
|
+
if (!item.statusColor)
|
|
114
|
+
return '';
|
|
115
|
+
switch (item.statusColor) {
|
|
116
|
+
case 'yellow':
|
|
117
|
+
return 'cqa-bg-[#EAB308]';
|
|
118
|
+
case 'red':
|
|
119
|
+
return 'cqa-bg-[#DC2626]';
|
|
120
|
+
case 'green':
|
|
121
|
+
return 'cqa-bg-[#16A34A]';
|
|
122
|
+
case 'gray':
|
|
123
|
+
default:
|
|
124
|
+
return 'cqa-bg-[#94A3B8]';
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
DetailSidePanelComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailSidePanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
129
|
+
DetailSidePanelComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DetailSidePanelComponent, selector: "cqa-detail-side-panel", inputs: { tabs: "tabs", activeTab: "activeTab", descriptionTitle: "descriptionTitle", descriptionContent: "descriptionContent", showEditButton: "showEditButton", metadataItems: "metadataItems", labels: "labels", configSections: "configSections", configSectionsRow2: "configSectionsRow2", platform: "platform", configTitle: "configTitle", showCloseButton: "showCloseButton", startInEditMode: "startInEditMode", selectConfigOverrides: "selectConfigOverrides", expanded: "expanded", expandedWidth: "expandedWidth", collapsedWidth: "collapsedWidth", expandTooltip: "expandTooltip", collapseTooltip: "collapseTooltip", closeTooltip: "closeTooltip" }, outputs: { back: "back", tabChange: "tabChange", editDescription: "editDescription", saveChanges: "saveChanges", metadataLinkClick: "metadataLinkClick", selectSearch: "selectSearch", selectLoadMore: "selectLoadMore", selectOpened: "selectOpened", selectionChange: "selectionChange", expandToggle: "expandToggle", close: "close" }, host: { properties: { "style.width": "this.hostWidth", "style.min-width": "this.hostMinWidth", "style.max-width": "this.hostMaxWidth", "style.overflow": "this.hostOverflow" }, styleAttribute: "transition: width 0.3s ease-in-out", classAttribute: "cqa-ui-root cqa-flex cqa-flex-col cqa-h-full cqa-flex-shrink-0 cqa-flex-grow-0 cqa-bg-white cqa-shadow-[-4px_0_6px_-1px_rgba(0,0,0,0.05)]" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-min-w-0 cqa-bg-white\">\n <!-- Main content: Side menu + Scrollable content -->\n <div\n class=\"cqa-grid cqa-flex-1 cqa-min-h-0 cqa-min-w-0 cqa-overflow-hidden\"\n [style.grid-template-columns]=\"expanded ? 'auto 1fr' : 'auto 0fr'\"\n style=\"transition: grid-template-columns 0.3s ease-in-out\">\n <!-- Left vertical icon menu -->\n <div class=\"cqa-flex cqa-flex-col cqa-items-center cqa-gap-2 cqa-py-4 cqa-px-2 cqa-border-r cqa-border-[#E2E8F0] cqa-bg-[#FAFAFA] cqa-flex-shrink-0\">\n <!-- Expand / Collapse button (always visible) -->\n <button\n type=\"button\"\n [matTooltip]=\"expanded ? collapseTooltip : expandTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onExpandToggle()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ expanded ? 'chevron_left' : 'chevron_right' }}</mat-icon>\n </button>\n <!-- Close button -->\n <button\n *ngIf=\"showCloseButton\"\n type=\"button\"\n [matTooltip]=\"closeTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onClose()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">close</mat-icon>\n </button>\n <!-- Tab buttons (1:1 with tabs) -->\n <button\n *ngFor=\"let tab of tabs; trackBy: trackByTabValue\"\n type=\"button\"\n [matTooltip]=\"tab.label\"\n [class.cqa-bg-[#3F43EE]]=\"activeTab === tab.value\"\n [class.cqa-text-white]=\"activeTab === tab.value\"\n [class.cqa-text-[#64748B]]=\"activeTab !== tab.value\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onTabClick(tab)\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ tab.icon || 'circle' }}</mat-icon>\n </button>\n </div>\n\n <!-- Scrollable content area (collapses with animation when expanded is false) -->\n <div class=\"cqa-min-w-0 cqa-overflow-hidden cqa-w-full\">\n <div class=\"cqa-h-full cqa-min-w-0 cqa-overflow-y-auto cqa-overflow-x-hidden cqa-p-4 cqa-space-y-4\">\n <!-- Tabs: equally distributed (Figma: purple active, gray inactive) -->\n <div class=\"cqa-flex cqa-gap-[3.5px] cqa-p-[3.5px] cqa-bg-[#F1F5F9] cqa-rounded-lg cqa-w-full\" role=\"tablist\">\n <button\n *ngFor=\"let tab of tabs; trackBy: trackByTabValue\"\n type=\"button\"\n role=\"tab\"\n [attr.aria-selected]=\"activeTab === tab.value\"\n [class.cqa-bg-[#3F43EE]]=\"activeTab === tab.value\"\n [class.cqa-text-white]=\"activeTab === tab.value\"\n [class.cqa-text-[#64748B]]=\"activeTab !== tab.value\"\n class=\"cqa-flex-1 cqa-flex cqa-justify-center cqa-items-center cqa-py-2 cqa-rounded-lg cqa-text-xs cqa-font-medium cqa-transition-colors hover:cqa-opacity-90 focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2\"\n (click)=\"onTabClick(tab)\">\n {{ tab.label }}\n </button>\n </div>\n\n <!-- Test Case tab: use cqa-test-case-details (Figma design) -->\n <cqa-test-case-details\n *ngIf=\"activeTab === 'test-case'\"\n [descriptionTitle]=\"descriptionTitle\"\n [descriptionContent]=\"descriptionContent\"\n [showEditButton]=\"showEditButton\"\n [startInEditMode]=\"startInEditMode\"\n [selectConfigOverrides]=\"selectConfigOverrides\"\n [metadataItems]=\"metadataItems\"\n [labels]=\"labels\"\n [configTitle]=\"configTitle\"\n [configSections]=\"configSections\"\n [configSectionsRow2]=\"configSectionsRow2\"\n [platform]=\"platform\"\n (editDescription)=\"onEditDescription()\"\n (saveChanges)=\"onSaveChanges($event)\"\n (metadataLinkClick)=\"onMetadataLinkClick($event)\"\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>\n\n <!-- Placeholder for other tabs (Data Library, Variables) -->\n <div *ngIf=\"activeTab !== 'test-case'\" class=\"cqa-p-4 cqa-text-[#64748B] cqa-text-sm\">\n {{ activeTab === 'data-library' ? 'Data Library content' : 'Variables content' }} \u2013 coming soon\n </div>\n </div>\n </div>\n </div>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.TestCaseDetailsComponent, selector: "cqa-test-case-details", inputs: ["startInEditMode", "descriptionTitle", "descriptionContent", "showEditButton", "metadataItems", "labels", "configTitle", "configSections", "configSectionsRow2", "platform", "selectConfigOverrides"], outputs: ["editDescription", "saveChanges", "metadataLinkClick", "selectSearch", "selectLoadMore", "selectOpened", "selectionChange"] }], directives: [{ type: i3.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
130
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DetailSidePanelComponent, decorators: [{
|
|
131
|
+
type: Component,
|
|
132
|
+
args: [{ selector: 'cqa-detail-side-panel', changeDetection: ChangeDetectionStrategy.OnPush, host: {
|
|
133
|
+
class: 'cqa-ui-root cqa-flex cqa-flex-col cqa-h-full cqa-flex-shrink-0 cqa-flex-grow-0 cqa-bg-white cqa-shadow-[-4px_0_6px_-1px_rgba(0,0,0,0.05)]',
|
|
134
|
+
style: 'transition: width 0.3s ease-in-out',
|
|
135
|
+
}, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-min-w-0 cqa-bg-white\">\n <!-- Main content: Side menu + Scrollable content -->\n <div\n class=\"cqa-grid cqa-flex-1 cqa-min-h-0 cqa-min-w-0 cqa-overflow-hidden\"\n [style.grid-template-columns]=\"expanded ? 'auto 1fr' : 'auto 0fr'\"\n style=\"transition: grid-template-columns 0.3s ease-in-out\">\n <!-- Left vertical icon menu -->\n <div class=\"cqa-flex cqa-flex-col cqa-items-center cqa-gap-2 cqa-py-4 cqa-px-2 cqa-border-r cqa-border-[#E2E8F0] cqa-bg-[#FAFAFA] cqa-flex-shrink-0\">\n <!-- Expand / Collapse button (always visible) -->\n <button\n type=\"button\"\n [matTooltip]=\"expanded ? collapseTooltip : expandTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onExpandToggle()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ expanded ? 'chevron_left' : 'chevron_right' }}</mat-icon>\n </button>\n <!-- Close button -->\n <button\n *ngIf=\"showCloseButton\"\n type=\"button\"\n [matTooltip]=\"closeTooltip\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onClose()\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">close</mat-icon>\n </button>\n <!-- Tab buttons (1:1 with tabs) -->\n <button\n *ngFor=\"let tab of tabs; trackBy: trackByTabValue\"\n type=\"button\"\n [matTooltip]=\"tab.label\"\n [class.cqa-bg-[#3F43EE]]=\"activeTab === tab.value\"\n [class.cqa-text-white]=\"activeTab === tab.value\"\n [class.cqa-text-[#64748B]]=\"activeTab !== tab.value\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onTabClick(tab)\">\n <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ tab.icon || 'circle' }}</mat-icon>\n </button>\n </div>\n\n <!-- Scrollable content area (collapses with animation when expanded is false) -->\n <div class=\"cqa-min-w-0 cqa-overflow-hidden cqa-w-full\">\n <div class=\"cqa-h-full cqa-min-w-0 cqa-overflow-y-auto cqa-overflow-x-hidden cqa-p-4 cqa-space-y-4\">\n <!-- Tabs: equally distributed (Figma: purple active, gray inactive) -->\n <div class=\"cqa-flex cqa-gap-[3.5px] cqa-p-[3.5px] cqa-bg-[#F1F5F9] cqa-rounded-lg cqa-w-full\" role=\"tablist\">\n <button\n *ngFor=\"let tab of tabs; trackBy: trackByTabValue\"\n type=\"button\"\n role=\"tab\"\n [attr.aria-selected]=\"activeTab === tab.value\"\n [class.cqa-bg-[#3F43EE]]=\"activeTab === tab.value\"\n [class.cqa-text-white]=\"activeTab === tab.value\"\n [class.cqa-text-[#64748B]]=\"activeTab !== tab.value\"\n class=\"cqa-flex-1 cqa-flex cqa-justify-center cqa-items-center cqa-py-2 cqa-rounded-lg cqa-text-xs cqa-font-medium cqa-transition-colors hover:cqa-opacity-90 focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2\"\n (click)=\"onTabClick(tab)\">\n {{ tab.label }}\n </button>\n </div>\n\n <!-- Test Case tab: use cqa-test-case-details (Figma design) -->\n <cqa-test-case-details\n *ngIf=\"activeTab === 'test-case'\"\n [descriptionTitle]=\"descriptionTitle\"\n [descriptionContent]=\"descriptionContent\"\n [showEditButton]=\"showEditButton\"\n [startInEditMode]=\"startInEditMode\"\n [selectConfigOverrides]=\"selectConfigOverrides\"\n [metadataItems]=\"metadataItems\"\n [labels]=\"labels\"\n [configTitle]=\"configTitle\"\n [configSections]=\"configSections\"\n [configSectionsRow2]=\"configSectionsRow2\"\n [platform]=\"platform\"\n (editDescription)=\"onEditDescription()\"\n (saveChanges)=\"onSaveChanges($event)\"\n (metadataLinkClick)=\"onMetadataLinkClick($event)\"\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>\n\n <!-- Placeholder for other tabs (Data Library, Variables) -->\n <div *ngIf=\"activeTab !== 'test-case'\" class=\"cqa-p-4 cqa-text-[#64748B] cqa-text-sm\">\n {{ activeTab === 'data-library' ? 'Data Library content' : 'Variables content' }} \u2013 coming soon\n </div>\n </div>\n </div>\n </div>\n</div>\n" }]
|
|
136
|
+
}], propDecorators: { tabs: [{
|
|
137
|
+
type: Input
|
|
138
|
+
}], activeTab: [{
|
|
139
|
+
type: Input
|
|
140
|
+
}], descriptionTitle: [{
|
|
141
|
+
type: Input
|
|
142
|
+
}], descriptionContent: [{
|
|
143
|
+
type: Input
|
|
144
|
+
}], showEditButton: [{
|
|
145
|
+
type: Input
|
|
146
|
+
}], metadataItems: [{
|
|
147
|
+
type: Input
|
|
148
|
+
}], labels: [{
|
|
149
|
+
type: Input
|
|
150
|
+
}], configSections: [{
|
|
151
|
+
type: Input
|
|
152
|
+
}], configSectionsRow2: [{
|
|
153
|
+
type: Input
|
|
154
|
+
}], platform: [{
|
|
155
|
+
type: Input
|
|
156
|
+
}], configTitle: [{
|
|
157
|
+
type: Input
|
|
158
|
+
}], showCloseButton: [{
|
|
159
|
+
type: Input
|
|
160
|
+
}], startInEditMode: [{
|
|
161
|
+
type: Input
|
|
162
|
+
}], selectConfigOverrides: [{
|
|
163
|
+
type: Input
|
|
164
|
+
}], expanded: [{
|
|
165
|
+
type: Input
|
|
166
|
+
}], expandedWidth: [{
|
|
167
|
+
type: Input
|
|
168
|
+
}], collapsedWidth: [{
|
|
169
|
+
type: Input
|
|
170
|
+
}], hostWidth: [{
|
|
171
|
+
type: HostBinding,
|
|
172
|
+
args: ['style.width']
|
|
173
|
+
}], hostMinWidth: [{
|
|
174
|
+
type: HostBinding,
|
|
175
|
+
args: ['style.min-width']
|
|
176
|
+
}], hostMaxWidth: [{
|
|
177
|
+
type: HostBinding,
|
|
178
|
+
args: ['style.max-width']
|
|
179
|
+
}], hostOverflow: [{
|
|
180
|
+
type: HostBinding,
|
|
181
|
+
args: ['style.overflow']
|
|
182
|
+
}], expandTooltip: [{
|
|
183
|
+
type: Input
|
|
184
|
+
}], collapseTooltip: [{
|
|
185
|
+
type: Input
|
|
186
|
+
}], closeTooltip: [{
|
|
187
|
+
type: Input
|
|
188
|
+
}], back: [{
|
|
189
|
+
type: Output
|
|
190
|
+
}], tabChange: [{
|
|
191
|
+
type: Output
|
|
192
|
+
}], editDescription: [{
|
|
193
|
+
type: Output
|
|
194
|
+
}], saveChanges: [{
|
|
195
|
+
type: Output
|
|
196
|
+
}], metadataLinkClick: [{
|
|
197
|
+
type: Output
|
|
198
|
+
}], selectSearch: [{
|
|
199
|
+
type: Output
|
|
200
|
+
}], selectLoadMore: [{
|
|
201
|
+
type: Output
|
|
202
|
+
}], selectOpened: [{
|
|
203
|
+
type: Output
|
|
204
|
+
}], selectionChange: [{
|
|
205
|
+
type: Output
|
|
206
|
+
}], expandToggle: [{
|
|
207
|
+
type: Output
|
|
208
|
+
}], close: [{
|
|
209
|
+
type: Output
|
|
210
|
+
}] } });
|
|
211
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detail-side-panel.component.js","sourceRoot":"","sources":["../../../../../src/lib/detail-side-panel/detail-side-panel.component.ts","../../../../../src/lib/detail-side-panel/detail-side-panel.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ChangeDetectionStrategy,\n  HostBinding,\n} from '@angular/core';\nimport { ConfigurationItem } from '../configuration-card/configuration-card.component';\nimport {\n  DetailSidePanelTab,\n  DetailSidePanelMetadataItem,\n  DetailSidePanelConfigSection,\n} from './detail-side-panel.models';\nimport {\n  TestCaseDetailsEditFormData,\n  SelectConfigOverrides,\n} from '../test-case-details/test-case-details-edit/test-case-details-edit.component';\n\nexport type { DetailSidePanelTab, DetailSidePanelMetadataItem, DetailSidePanelConfigSection } from './detail-side-panel.models';\n\n@Component({\n  selector: 'cqa-detail-side-panel',\n  templateUrl: './detail-side-panel.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: {\n    class:\n      'cqa-ui-root cqa-flex cqa-flex-col cqa-h-full cqa-flex-shrink-0 cqa-flex-grow-0 cqa-bg-white cqa-shadow-[-4px_0_6px_-1px_rgba(0,0,0,0.05)]',\n    style: 'transition: width 0.3s ease-in-out',\n  },\n})\nexport class DetailSidePanelComponent {\n  /** Tabs - each tab has a side panel icon button; tabs and buttons are 1:1 */\n  @Input() tabs: DetailSidePanelTab[] = [\n    { label: 'Test Case', value: 'test-case', icon: 'description' },\n    { label: 'Data Library', value: 'data-library', icon: 'folder' },\n    { label: 'Variables', value: 'variables', icon: 'code' },\n  ];\n\n  /** Currently active tab value */\n  @Input() activeTab = 'test-case';\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 = true;\n\n  /** Metadata items (Created on, Status, Priority, etc.) */\n  @Input() metadataItems: DetailSidePanelMetadataItem[] = [];\n\n  /** Labels/tags (e.g. Automation, API, SDK, UI/UX) */\n  @Input() labels: string[] = [];\n\n  /** Configuration sections - full width (e.g. Execution, AI Configuration) */\n  @Input() configSections: DetailSidePanelConfigSection[] = [];\n\n  /** Optional config sections displayed in a 2-column row (e.g. Waits & Retries, Device) */\n  @Input() configSectionsRow2: DetailSidePanelConfigSection[] = [];\n\n  /** Platform: 'web' or 'mobile'. Defaults to 'web'. Used for Device Settings. */\n  @Input() platform: 'web' | 'mobile' = 'web';\n\n  /** Configuration section title */\n  @Input() configTitle = 'Configuration';\n\n  /** Whether to show the close button in the side menu */\n  @Input() showCloseButton = false;\n\n  /** When true, test case details start in edit mode (useful for Storybook). */\n  @Input() startInEditMode = false;\n\n  /** Override config per select for API-driven options, server search, load more. */\n  @Input() selectConfigOverrides: SelectConfigOverrides = {};\n\n  /** Whether the panel is expanded (affects expand button icon and panel width) */\n  @Input() expanded = true;\n\n  /** Panel width when expanded (e.g. '480px', '25%') */\n  @Input() expandedWidth = '380px';\n\n  /** Panel width when collapsed (e.g. '56px' - fits icon bar + back button) */\n  @Input() collapsedWidth = '56px';\n\n  @HostBinding('style.width') get hostWidth(): string {\n    return this.expanded ? this.expandedWidth : this.collapsedWidth;\n  }\n\n  @HostBinding('style.min-width') get hostMinWidth(): string {\n    return this.expanded ? this.expandedWidth : this.collapsedWidth;\n  }\n\n  @HostBinding('style.max-width') get hostMaxWidth(): string {\n    return this.expanded ? this.expandedWidth : this.collapsedWidth;\n  }\n\n  @HostBinding('style.overflow') hostOverflow = 'hidden';\n\n  /** Tooltip for expand button when panel is collapsed */\n  @Input() expandTooltip = 'Expand';\n\n  /** Tooltip for expand button when panel is expanded (collapse) */\n  @Input() collapseTooltip = 'Collapse';\n\n  /** Tooltip for close button */\n  @Input() closeTooltip = 'Close';\n\n  @Output() back = new EventEmitter<void>();\n  @Output() tabChange = new EventEmitter<string>();\n  @Output() editDescription = new EventEmitter<void>();\n  @Output() saveChanges = new EventEmitter<TestCaseDetailsEditFormData>();\n  @Output() metadataLinkClick = new EventEmitter<DetailSidePanelMetadataItem>();\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  @Output() expandToggle = new EventEmitter<void>();\n  @Output() close = new EventEmitter<void>();\n\n  trackByTabValue(_i: number, tab: DetailSidePanelTab): string {\n    return tab.value;\n  }\n\n  trackByMetadataLabel(_i: number, item: DetailSidePanelMetadataItem): string {\n    return item.label;\n  }\n\n  trackByConfigTitle(_i: number, section: DetailSidePanelConfigSection): string {\n    return section.title;\n  }\n\n  onBack(): void {\n    this.back.emit();\n  }\n\n  onTabClick(tab: DetailSidePanelTab): void {\n    if (!this.expanded) {\n      this.expandToggle.emit();\n    }\n    if (tab.value !== this.activeTab) {\n      this.tabChange.emit(tab.value);\n    }\n  }\n\n  onEditDescription(): void {\n    this.editDescription.emit();\n  }\n\n  onSaveChanges(data: TestCaseDetailsEditFormData): void {\n    this.saveChanges.emit(data);\n  }\n\n  onExpandToggle(): void {\n    this.expandToggle.emit();\n  }\n\n  onClose(): void {\n    this.close.emit();\n  }\n\n  onMetadataLinkClick(item: DetailSidePanelMetadataItem): void {\n    if (item.link) {\n      this.metadataLinkClick.emit(item);\n    }\n  }\n\n  getStatusDotClass(item: DetailSidePanelMetadataItem): 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","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-min-w-0 cqa-bg-white\">\n  <!-- Main content: Side menu + Scrollable content -->\n  <div\n    class=\"cqa-grid cqa-flex-1 cqa-min-h-0 cqa-min-w-0 cqa-overflow-hidden\"\n    [style.grid-template-columns]=\"expanded ? 'auto 1fr' : 'auto 0fr'\"\n    style=\"transition: grid-template-columns 0.3s ease-in-out\">\n    <!-- Left vertical icon menu -->\n    <div class=\"cqa-flex cqa-flex-col cqa-items-center cqa-gap-2 cqa-py-4 cqa-px-2 cqa-border-r cqa-border-[#E2E8F0] cqa-bg-[#FAFAFA] cqa-flex-shrink-0\">\n      <!-- Expand / Collapse button (always visible) -->\n      <button\n        type=\"button\"\n        [matTooltip]=\"expanded ? collapseTooltip : expandTooltip\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onExpandToggle()\">\n        <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ expanded ? 'chevron_left' : 'chevron_right' }}</mat-icon>\n      </button>\n      <!-- Close button -->\n      <button\n        *ngIf=\"showCloseButton\"\n        type=\"button\"\n        [matTooltip]=\"closeTooltip\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg cqa-text-[#64748B] hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onClose()\">\n        <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">close</mat-icon>\n      </button>\n      <!-- Tab buttons (1:1 with tabs) -->\n      <button\n        *ngFor=\"let tab of tabs; trackBy: trackByTabValue\"\n        type=\"button\"\n        [matTooltip]=\"tab.label\"\n        [class.cqa-bg-[#3F43EE]]=\"activeTab === tab.value\"\n        [class.cqa-text-white]=\"activeTab === tab.value\"\n        [class.cqa-text-[#64748B]]=\"activeTab !== tab.value\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-9 cqa-h-9 cqa-rounded-lg hover:cqa-bg-[#E2E8F0] hover:cqa-text-[#334155] 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)=\"onTabClick(tab)\">\n        <mat-icon class=\"cqa-text-[20px] cqa-w-5 cqa-h-5\">{{ tab.icon || 'circle' }}</mat-icon>\n      </button>\n    </div>\n\n    <!-- Scrollable content area (collapses with animation when expanded is false) -->\n    <div class=\"cqa-min-w-0 cqa-overflow-hidden cqa-w-full\">\n      <div class=\"cqa-h-full cqa-min-w-0 cqa-overflow-y-auto cqa-overflow-x-hidden cqa-p-4 cqa-space-y-4\">\n      <!-- Tabs: equally distributed (Figma: purple active, gray inactive) -->\n      <div class=\"cqa-flex cqa-gap-[3.5px] cqa-p-[3.5px] cqa-bg-[#F1F5F9] cqa-rounded-lg cqa-w-full\" role=\"tablist\">\n        <button\n          *ngFor=\"let tab of tabs; trackBy: trackByTabValue\"\n          type=\"button\"\n          role=\"tab\"\n          [attr.aria-selected]=\"activeTab === tab.value\"\n          [class.cqa-bg-[#3F43EE]]=\"activeTab === tab.value\"\n          [class.cqa-text-white]=\"activeTab === tab.value\"\n          [class.cqa-text-[#64748B]]=\"activeTab !== tab.value\"\n          class=\"cqa-flex-1 cqa-flex cqa-justify-center cqa-items-center cqa-py-2 cqa-rounded-lg cqa-text-xs cqa-font-medium cqa-transition-colors hover:cqa-opacity-90 focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] focus-visible:cqa-ring-offset-2\"\n          (click)=\"onTabClick(tab)\">\n          {{ tab.label }}\n        </button>\n      </div>\n\n      <!-- Test Case tab: use cqa-test-case-details (Figma design) -->\n      <cqa-test-case-details\n        *ngIf=\"activeTab === 'test-case'\"\n        [descriptionTitle]=\"descriptionTitle\"\n        [descriptionContent]=\"descriptionContent\"\n        [showEditButton]=\"showEditButton\"\n        [startInEditMode]=\"startInEditMode\"\n        [selectConfigOverrides]=\"selectConfigOverrides\"\n        [metadataItems]=\"metadataItems\"\n        [labels]=\"labels\"\n        [configTitle]=\"configTitle\"\n        [configSections]=\"configSections\"\n        [configSectionsRow2]=\"configSectionsRow2\"\n        [platform]=\"platform\"\n        (editDescription)=\"onEditDescription()\"\n        (saveChanges)=\"onSaveChanges($event)\"\n        (metadataLinkClick)=\"onMetadataLinkClick($event)\"\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>\n\n      <!-- Placeholder for other tabs (Data Library, Variables) -->\n      <div *ngIf=\"activeTab !== 'test-case'\" class=\"cqa-p-4 cqa-text-[#64748B] cqa-text-sm\">\n        {{ activeTab === 'data-library' ? 'Data Library content' : 'Variables content' }} – coming soon\n      </div>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLXNpZGUtcGFuZWwubW9kZWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9kZXRhaWwtc2lkZS1wYW5lbC9kZXRhaWwtc2lkZS1wYW5lbC5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbmZpZ3VyYXRpb25JdGVtIH0gZnJvbSAnLi4vY29uZmlndXJhdGlvbi1jYXJkL2NvbmZpZ3VyYXRpb24tY2FyZC5jb21wb25lbnQnO1xuXG5leHBvcnQgaW50ZXJmYWNlIERldGFpbFNpZGVQYW5lbFRhYiB7XG4gIGxhYmVsOiBzdHJpbmc7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIC8qKiBNYXRlcmlhbCBpY29uIG5hbWUgZm9yIHRoZSBzaWRlIHBhbmVsIGJ1dHRvbiAqL1xuICBpY29uPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERldGFpbFNpZGVQYW5lbE1ldGFkYXRhSXRlbSB7XG4gIGxhYmVsOiBzdHJpbmc7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIGljb24/OiBzdHJpbmc7XG4gIGljb25MaWJyYXJ5PzogJ21hdCcgfCAnZmEnO1xuICBsaW5rPzogc3RyaW5nO1xuICBzdGF0dXNDb2xvcj86ICd5ZWxsb3cnIHwgJ3JlZCcgfCAnZ3JlZW4nIHwgJ2dyYXknO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERldGFpbFNpZGVQYW5lbENvbmZpZ1NlY3Rpb24ge1xuICB0aXRsZTogc3RyaW5nO1xuICBpY29uPzogc3RyaW5nO1xuICBpdGVtczogQ29uZmlndXJhdGlvbkl0ZW1bXTtcbn1cbiJdfQ==
|