@cqa-lib/cqa-ui 1.1.496 → 1.1.497
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/test-case-details/api-mocking-card/api-mocking-card.component.mjs +129 -0
- package/esm2020/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.mjs +66 -5
- package/esm2020/lib/test-case-details/test-case-details.component.mjs +58 -4
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +247 -5
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +247 -5
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/test-case-details/api-mocking-card/api-mocking-card.component.d.ts +78 -0
- package/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.d.ts +36 -1
- package/lib/test-case-details/test-case-details.component.d.ts +25 -1
- package/lib/ui-kit.module.d.ts +40 -39
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "../../custom-toggle/custom-toggle.component";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
/**
|
|
6
|
+
* Dual-mode card summarising the API mocking configuration of a test case.
|
|
7
|
+
*
|
|
8
|
+
* The card surfaces **one boolean choice** — `restoreMock` — as two mutually-exclusive rows
|
|
9
|
+
* labelled "Capture API responses" and "Renew API responses":
|
|
10
|
+
*
|
|
11
|
+
* restoreMock === true → Renew = ON, Capture = OFF
|
|
12
|
+
* restoreMock === false → Renew = OFF, Capture = ON
|
|
13
|
+
*
|
|
14
|
+
* In read-only mode both rows show a value text ("On"/"Off") derived from `restoreMock`.
|
|
15
|
+
* In edit mode both rows render a toggle; interacting with either one flips `restoreMock`
|
|
16
|
+
* and the card emits `restoreMockChange` so the parent can persist the new value.
|
|
17
|
+
*/
|
|
18
|
+
export class ApiMockingCardComponent {
|
|
19
|
+
constructor() {
|
|
20
|
+
this.title = 'API Mocking';
|
|
21
|
+
this.statusLabel = 'Not configured';
|
|
22
|
+
this.captureLabel = 'Capture API responses';
|
|
23
|
+
this.renewLabel = 'Renew API responses';
|
|
24
|
+
/**
|
|
25
|
+
* Single source of truth for the card's dual-toggle state.
|
|
26
|
+
* Defaults to `false` → Capture is ON, Renew is OFF.
|
|
27
|
+
*/
|
|
28
|
+
this.restoreMock = false;
|
|
29
|
+
this.mockedApisCount = 0;
|
|
30
|
+
this.totalApisCount = 0;
|
|
31
|
+
this.progressPercent = 0;
|
|
32
|
+
this.configureButtonLabel = 'Configure';
|
|
33
|
+
this.editing = false;
|
|
34
|
+
/**
|
|
35
|
+
* Controls whether the "Mocked X of Y / progress bar / percent" block is shown.
|
|
36
|
+
* Hidden when the test case has no captured mock summary yet (and always hidden in
|
|
37
|
+
* edit mode regardless of this flag).
|
|
38
|
+
*/
|
|
39
|
+
this.showProgress = false;
|
|
40
|
+
this.configure = new EventEmitter();
|
|
41
|
+
/**
|
|
42
|
+
* Emitted whenever the effective `restoreMock` value changes (either row's toggle was
|
|
43
|
+
* interacted with in edit mode). Parent components should persist this value — e.g. under
|
|
44
|
+
* `testCase.mockApiSettings.restoreMock`.
|
|
45
|
+
*/
|
|
46
|
+
this.restoreMockChange = new EventEmitter();
|
|
47
|
+
}
|
|
48
|
+
// Derived per-row state (template binds these directly so the two rows stay in lock-step).
|
|
49
|
+
get renewEnabled() { return this.restoreMock; }
|
|
50
|
+
get captureEnabled() { return !this.restoreMock; }
|
|
51
|
+
get renewValueText() { return this.restoreMock ? 'On' : 'Off'; }
|
|
52
|
+
get captureValueText() { return this.restoreMock ? 'Off' : 'On'; }
|
|
53
|
+
get computedSummaryText() {
|
|
54
|
+
if (this.summaryText)
|
|
55
|
+
return this.summaryText;
|
|
56
|
+
return `${this.mockedApisCount} of ${this.totalApisCount} APIs set to mock`;
|
|
57
|
+
}
|
|
58
|
+
get computedPercentText() {
|
|
59
|
+
if (this.percentText)
|
|
60
|
+
return this.percentText;
|
|
61
|
+
return `${this.progressPercent}% of calls will use mock responses`;
|
|
62
|
+
}
|
|
63
|
+
get progressWidth() {
|
|
64
|
+
const clamped = Math.max(0, Math.min(100, Number(this.progressPercent) || 0));
|
|
65
|
+
return `${clamped}%`;
|
|
66
|
+
}
|
|
67
|
+
onConfigureClick() {
|
|
68
|
+
this.configure.emit();
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* User interacted with the "Renew API responses" toggle in edit mode.
|
|
72
|
+
* Renew ON → restoreMock = true (Capture OFF).
|
|
73
|
+
* Renew OFF → restoreMock = false (Capture ON — mutual exclusivity).
|
|
74
|
+
*/
|
|
75
|
+
onRenewToggle(value) {
|
|
76
|
+
this.setRestoreMock(value);
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* User interacted with the "Capture API responses" toggle in edit mode.
|
|
80
|
+
* Capture ON → restoreMock = false (Renew OFF).
|
|
81
|
+
* Capture OFF → restoreMock = true (Renew ON — mutual exclusivity).
|
|
82
|
+
*/
|
|
83
|
+
onCaptureToggle(value) {
|
|
84
|
+
this.setRestoreMock(!value);
|
|
85
|
+
}
|
|
86
|
+
setRestoreMock(next) {
|
|
87
|
+
if (this.restoreMock === next)
|
|
88
|
+
return;
|
|
89
|
+
this.restoreMock = next;
|
|
90
|
+
this.restoreMockChange.emit(next);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
ApiMockingCardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ApiMockingCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
94
|
+
ApiMockingCardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ApiMockingCardComponent, selector: "cqa-api-mocking-card", inputs: { title: "title", statusLabel: "statusLabel", captureLabel: "captureLabel", renewLabel: "renewLabel", restoreMock: "restoreMock", mockedApisCount: "mockedApisCount", totalApisCount: "totalApisCount", progressPercent: "progressPercent", summaryText: "summaryText", percentText: "percentText", configureButtonLabel: "configureButtonLabel", editing: "editing", showProgress: "showProgress" }, outputs: { configure: "configure", restoreMockChange: "restoreMockChange" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-amc-card\">\n <div class=\"cqa-amc-header\">\n <span class=\"cqa-amc-title\">{{ title }}</span>\n <span class=\"cqa-amc-status\" *ngIf=\"statusLabel\">{{ statusLabel }}</span>\n </div>\n\n <!-- Read-only rows -->\n <ng-container *ngIf=\"!editing\">\n <div class=\"cqa-amc-row\">\n <span class=\"cqa-amc-label\">{{ captureLabel }}</span>\n <span class=\"cqa-amc-value\">{{ captureValueText }}</span>\n </div>\n <div class=\"cqa-amc-row\">\n <span class=\"cqa-amc-label\">{{ renewLabel }}</span>\n <span class=\"cqa-amc-value\">{{ renewValueText }}</span>\n </div>\n </ng-container>\n\n <!-- Edit rows: toggles -->\n <ng-container *ngIf=\"editing\">\n <div class=\"cqa-amc-row cqa-amc-row--edit\">\n <span class=\"cqa-amc-label\">{{ captureLabel }}</span>\n <cqa-custom-toggle\n [checked]=\"captureEnabled\"\n [ariaLabel]=\"captureLabel\"\n (checkedChange)=\"onCaptureToggle($event)\">\n </cqa-custom-toggle>\n </div>\n <div class=\"cqa-amc-row cqa-amc-row--edit\">\n <span class=\"cqa-amc-label\">{{ renewLabel }}</span>\n <cqa-custom-toggle\n [checked]=\"renewEnabled\"\n [ariaLabel]=\"renewLabel\"\n (checkedChange)=\"onRenewToggle($event)\">\n </cqa-custom-toggle>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"showProgress && !editing\">\n <p class=\"cqa-amc-summary\">{{ computedSummaryText }}</p>\n <div class=\"cqa-amc-progress-track\">\n <div class=\"cqa-amc-progress-fill\" [style.width]=\"progressWidth\"></div>\n </div>\n <p class=\"cqa-amc-percent\">{{ computedPercentText }}</p>\n </ng-container>\n\n <button\n *ngIf=\"!editing\"\n type=\"button\"\n class=\"cqa-amc-configure-btn\"\n (click)=\"onConfigureClick()\">\n {{ configureButtonLabel }}\n </button>\n</div>\n", components: [{ type: i1.CustomToggleComponent, selector: "cqa-custom-toggle", inputs: ["checked", "disabled", "ariaLabel"], outputs: ["checkedChange", "change"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
95
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ApiMockingCardComponent, decorators: [{
|
|
96
|
+
type: Component,
|
|
97
|
+
args: [{ selector: 'cqa-api-mocking-card', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cqa-amc-card\">\n <div class=\"cqa-amc-header\">\n <span class=\"cqa-amc-title\">{{ title }}</span>\n <span class=\"cqa-amc-status\" *ngIf=\"statusLabel\">{{ statusLabel }}</span>\n </div>\n\n <!-- Read-only rows -->\n <ng-container *ngIf=\"!editing\">\n <div class=\"cqa-amc-row\">\n <span class=\"cqa-amc-label\">{{ captureLabel }}</span>\n <span class=\"cqa-amc-value\">{{ captureValueText }}</span>\n </div>\n <div class=\"cqa-amc-row\">\n <span class=\"cqa-amc-label\">{{ renewLabel }}</span>\n <span class=\"cqa-amc-value\">{{ renewValueText }}</span>\n </div>\n </ng-container>\n\n <!-- Edit rows: toggles -->\n <ng-container *ngIf=\"editing\">\n <div class=\"cqa-amc-row cqa-amc-row--edit\">\n <span class=\"cqa-amc-label\">{{ captureLabel }}</span>\n <cqa-custom-toggle\n [checked]=\"captureEnabled\"\n [ariaLabel]=\"captureLabel\"\n (checkedChange)=\"onCaptureToggle($event)\">\n </cqa-custom-toggle>\n </div>\n <div class=\"cqa-amc-row cqa-amc-row--edit\">\n <span class=\"cqa-amc-label\">{{ renewLabel }}</span>\n <cqa-custom-toggle\n [checked]=\"renewEnabled\"\n [ariaLabel]=\"renewLabel\"\n (checkedChange)=\"onRenewToggle($event)\">\n </cqa-custom-toggle>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"showProgress && !editing\">\n <p class=\"cqa-amc-summary\">{{ computedSummaryText }}</p>\n <div class=\"cqa-amc-progress-track\">\n <div class=\"cqa-amc-progress-fill\" [style.width]=\"progressWidth\"></div>\n </div>\n <p class=\"cqa-amc-percent\">{{ computedPercentText }}</p>\n </ng-container>\n\n <button\n *ngIf=\"!editing\"\n type=\"button\"\n class=\"cqa-amc-configure-btn\"\n (click)=\"onConfigureClick()\">\n {{ configureButtonLabel }}\n </button>\n</div>\n", styles: [] }]
|
|
98
|
+
}], propDecorators: { title: [{
|
|
99
|
+
type: Input
|
|
100
|
+
}], statusLabel: [{
|
|
101
|
+
type: Input
|
|
102
|
+
}], captureLabel: [{
|
|
103
|
+
type: Input
|
|
104
|
+
}], renewLabel: [{
|
|
105
|
+
type: Input
|
|
106
|
+
}], restoreMock: [{
|
|
107
|
+
type: Input
|
|
108
|
+
}], mockedApisCount: [{
|
|
109
|
+
type: Input
|
|
110
|
+
}], totalApisCount: [{
|
|
111
|
+
type: Input
|
|
112
|
+
}], progressPercent: [{
|
|
113
|
+
type: Input
|
|
114
|
+
}], summaryText: [{
|
|
115
|
+
type: Input
|
|
116
|
+
}], percentText: [{
|
|
117
|
+
type: Input
|
|
118
|
+
}], configureButtonLabel: [{
|
|
119
|
+
type: Input
|
|
120
|
+
}], editing: [{
|
|
121
|
+
type: Input
|
|
122
|
+
}], showProgress: [{
|
|
123
|
+
type: Input
|
|
124
|
+
}], configure: [{
|
|
125
|
+
type: Output
|
|
126
|
+
}], restoreMockChange: [{
|
|
127
|
+
type: Output
|
|
128
|
+
}] } });
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-mocking-card.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/api-mocking-card/api-mocking-card.component.ts","../../../../../../src/lib/test-case-details/api-mocking-card/api-mocking-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';\n\n/**\n * Legacy toggle-change payload kept for back-compat with consumers that haven't migrated\n * to `restoreMockChange` yet. `key` identifies which row was toggled; `value` is the new\n * checked state of that row. New code should listen to `restoreMockChange` instead.\n */\nexport interface ApiMockingCardToggleChange {\n  key: 'captureApiResponses' | 'renewApiResponses';\n  value: boolean;\n}\n\n/**\n * Dual-mode card summarising the API mocking configuration of a test case.\n *\n * The card surfaces **one boolean choice** — `restoreMock` — as two mutually-exclusive rows\n * labelled \"Capture API responses\" and \"Renew API responses\":\n *\n *   restoreMock === true   →   Renew = ON,  Capture = OFF\n *   restoreMock === false  →   Renew = OFF, Capture = ON\n *\n * In read-only mode both rows show a value text (\"On\"/\"Off\") derived from `restoreMock`.\n * In edit mode both rows render a toggle; interacting with either one flips `restoreMock`\n * and the card emits `restoreMockChange` so the parent can persist the new value.\n */\n@Component({\n  selector: 'cqa-api-mocking-card',\n  templateUrl: './api-mocking-card.component.html',\n  // Styles live in src/lib/styles/tailwind.css (the lib ships a single compiled stylesheet);\n  // per-component .scss files are not supported in the ng-packagr build for this lib.\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ApiMockingCardComponent {\n  @Input() title: string = 'API Mocking';\n  @Input() statusLabel: string = 'Not configured';\n\n  @Input() captureLabel: string = 'Capture API responses';\n  @Input() renewLabel: string = 'Renew API responses';\n\n  /**\n   * Single source of truth for the card's dual-toggle state.\n   * Defaults to `false` → Capture is ON, Renew is OFF.\n   */\n  @Input() restoreMock: boolean = false;\n\n  @Input() mockedApisCount: number = 0;\n  @Input() totalApisCount: number = 0;\n  @Input() progressPercent: number = 0;\n\n  @Input() summaryText?: string;\n  @Input() percentText?: string;\n\n  @Input() configureButtonLabel: string = 'Configure';\n  @Input() editing: boolean = false;\n\n  /**\n   * Controls whether the \"Mocked X of Y / progress bar / percent\" block is shown.\n   * Hidden when the test case has no captured mock summary yet (and always hidden in\n   * edit mode regardless of this flag).\n   */\n  @Input() showProgress: boolean = false;\n\n  @Output() configure = new EventEmitter<void>();\n  /**\n   * Emitted whenever the effective `restoreMock` value changes (either row's toggle was\n   * interacted with in edit mode). Parent components should persist this value — e.g. under\n   * `testCase.mockApiSettings.restoreMock`.\n   */\n  @Output() restoreMockChange = new EventEmitter<boolean>();\n\n  // Derived per-row state (template binds these directly so the two rows stay in lock-step).\n  get renewEnabled(): boolean { return this.restoreMock; }\n  get captureEnabled(): boolean { return !this.restoreMock; }\n  get renewValueText(): string { return this.restoreMock ? 'On' : 'Off'; }\n  get captureValueText(): string { return this.restoreMock ? 'Off' : 'On'; }\n\n  get computedSummaryText(): string {\n    if (this.summaryText) return this.summaryText;\n    return `${this.mockedApisCount} of ${this.totalApisCount} APIs set to mock`;\n  }\n\n  get computedPercentText(): string {\n    if (this.percentText) return this.percentText;\n    return `${this.progressPercent}% of calls will use mock responses`;\n  }\n\n  get progressWidth(): string {\n    const clamped = Math.max(0, Math.min(100, Number(this.progressPercent) || 0));\n    return `${clamped}%`;\n  }\n\n  onConfigureClick(): void {\n    this.configure.emit();\n  }\n\n  /**\n   * User interacted with the \"Renew API responses\" toggle in edit mode.\n   * Renew ON  → restoreMock = true  (Capture OFF).\n   * Renew OFF → restoreMock = false (Capture ON — mutual exclusivity).\n   */\n  onRenewToggle(value: boolean): void {\n    this.setRestoreMock(value);\n  }\n\n  /**\n   * User interacted with the \"Capture API responses\" toggle in edit mode.\n   * Capture ON  → restoreMock = false (Renew OFF).\n   * Capture OFF → restoreMock = true  (Renew ON — mutual exclusivity).\n   */\n  onCaptureToggle(value: boolean): void {\n    this.setRestoreMock(!value);\n  }\n\n  private setRestoreMock(next: boolean): void {\n    if (this.restoreMock === next) return;\n    this.restoreMock = next;\n    this.restoreMockChange.emit(next);\n  }\n}\n","<div class=\"cqa-amc-card\">\n  <div class=\"cqa-amc-header\">\n    <span class=\"cqa-amc-title\">{{ title }}</span>\n    <span class=\"cqa-amc-status\" *ngIf=\"statusLabel\">{{ statusLabel }}</span>\n  </div>\n\n  <!-- Read-only rows -->\n  <ng-container *ngIf=\"!editing\">\n    <div class=\"cqa-amc-row\">\n      <span class=\"cqa-amc-label\">{{ captureLabel }}</span>\n      <span class=\"cqa-amc-value\">{{ captureValueText }}</span>\n    </div>\n    <div class=\"cqa-amc-row\">\n      <span class=\"cqa-amc-label\">{{ renewLabel }}</span>\n      <span class=\"cqa-amc-value\">{{ renewValueText }}</span>\n    </div>\n  </ng-container>\n\n  <!-- Edit rows: toggles -->\n  <ng-container *ngIf=\"editing\">\n    <div class=\"cqa-amc-row cqa-amc-row--edit\">\n      <span class=\"cqa-amc-label\">{{ captureLabel }}</span>\n      <cqa-custom-toggle\n        [checked]=\"captureEnabled\"\n        [ariaLabel]=\"captureLabel\"\n        (checkedChange)=\"onCaptureToggle($event)\">\n      </cqa-custom-toggle>\n    </div>\n    <div class=\"cqa-amc-row cqa-amc-row--edit\">\n      <span class=\"cqa-amc-label\">{{ renewLabel }}</span>\n      <cqa-custom-toggle\n        [checked]=\"renewEnabled\"\n        [ariaLabel]=\"renewLabel\"\n        (checkedChange)=\"onRenewToggle($event)\">\n      </cqa-custom-toggle>\n    </div>\n  </ng-container>\n\n  <ng-container *ngIf=\"showProgress && !editing\">\n    <p class=\"cqa-amc-summary\">{{ computedSummaryText }}</p>\n    <div class=\"cqa-amc-progress-track\">\n      <div class=\"cqa-amc-progress-fill\" [style.width]=\"progressWidth\"></div>\n    </div>\n    <p class=\"cqa-amc-percent\">{{ computedPercentText }}</p>\n  </ng-container>\n\n  <button\n    *ngIf=\"!editing\"\n    type=\"button\"\n    class=\"cqa-amc-configure-btn\"\n    (click)=\"onConfigureClick()\">\n    {{ configureButtonLabel }}\n  </button>\n</div>\n"]}
|