@cqa-lib/cqa-ui 1.1.0 → 1.1.2
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/ai-debug-alert/ai-debug-alert.component.mjs +20 -0
- package/esm2020/lib/badge/badge.component.mjs +13 -3
- package/esm2020/lib/configuration-card/configuration-card.component.mjs +93 -0
- package/esm2020/lib/console-alert/console-alert.component.mjs +59 -0
- package/esm2020/lib/dashboards/insight-card/insight-card.component.mjs +1 -1
- package/esm2020/lib/network-request/network-request.component.mjs +47 -0
- package/esm2020/lib/run-history-card/run-history-card.component.mjs +149 -0
- package/esm2020/lib/simulator/simulator.component.mjs +186 -0
- package/esm2020/lib/ui-kit.module.mjs +40 -5
- package/esm2020/lib/view-image-modal/view-image-modal.component.mjs +40 -0
- package/esm2020/public-api.mjs +8 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +625 -8
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +617 -8
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/ai-debug-alert/ai-debug-alert.component.d.ts +7 -0
- package/lib/badge/badge.component.d.ts +2 -1
- package/lib/configuration-card/configuration-card.component.d.ts +31 -0
- package/lib/console-alert/console-alert.component.d.ts +12 -0
- package/lib/network-request/network-request.component.d.ts +15 -0
- package/lib/run-history-card/run-history-card.component.d.ts +47 -0
- package/lib/simulator/simulator.component.d.ts +52 -0
- package/lib/ui-kit.module.d.ts +23 -16
- package/lib/view-image-modal/view-image-modal.component.d.ts +13 -0
- package/package.json +1 -1
- package/public-api.d.ts +7 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { Component, Input } from "@angular/core";
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
export class RunHistoryCardComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.size = 'normal';
|
|
8
|
+
}
|
|
9
|
+
get statusBadgeVariant() {
|
|
10
|
+
switch (this.status) {
|
|
11
|
+
case 'passed':
|
|
12
|
+
return 'success';
|
|
13
|
+
case 'failed':
|
|
14
|
+
return 'error';
|
|
15
|
+
case 'aborted':
|
|
16
|
+
return 'warning';
|
|
17
|
+
case 'in-progress':
|
|
18
|
+
return 'info';
|
|
19
|
+
default:
|
|
20
|
+
return 'info';
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
get statusLabel() {
|
|
24
|
+
switch (this.status) {
|
|
25
|
+
case 'passed':
|
|
26
|
+
return 'Passed';
|
|
27
|
+
case 'failed':
|
|
28
|
+
return 'Failed';
|
|
29
|
+
case 'aborted':
|
|
30
|
+
return 'Aborted';
|
|
31
|
+
case 'in-progress':
|
|
32
|
+
return 'In Progress';
|
|
33
|
+
default:
|
|
34
|
+
return '';
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
get statusColor() {
|
|
38
|
+
switch (this.status) {
|
|
39
|
+
case 'passed':
|
|
40
|
+
return '#00C950';
|
|
41
|
+
case 'failed':
|
|
42
|
+
return '#FB2C36';
|
|
43
|
+
case 'aborted':
|
|
44
|
+
return '#F97316';
|
|
45
|
+
case 'in-progress':
|
|
46
|
+
return '#3B82F6';
|
|
47
|
+
default:
|
|
48
|
+
return '#6B7280';
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
get statusIconBgColor() {
|
|
52
|
+
switch (this.status) {
|
|
53
|
+
case 'passed':
|
|
54
|
+
return '#DCFCE7';
|
|
55
|
+
case 'failed':
|
|
56
|
+
return '#ffe2e2';
|
|
57
|
+
case 'aborted':
|
|
58
|
+
return '#FFEDD5';
|
|
59
|
+
case 'in-progress':
|
|
60
|
+
return '#DBEAFE';
|
|
61
|
+
default:
|
|
62
|
+
return '#E5E7EB';
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
get statusIcon() {
|
|
66
|
+
switch (this.status) {
|
|
67
|
+
case 'passed':
|
|
68
|
+
return 'check';
|
|
69
|
+
case 'failed':
|
|
70
|
+
return 'close';
|
|
71
|
+
case 'aborted':
|
|
72
|
+
return 'warning';
|
|
73
|
+
case 'in-progress':
|
|
74
|
+
return 'schedule';
|
|
75
|
+
default:
|
|
76
|
+
return '';
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
get typeIcon() {
|
|
80
|
+
return this.type === 'automated' ? 'settings' : 'person';
|
|
81
|
+
}
|
|
82
|
+
get typeLabel() {
|
|
83
|
+
return this.type === 'automated' ? 'Automated' : 'Manual';
|
|
84
|
+
}
|
|
85
|
+
get isSmall() {
|
|
86
|
+
return this.size === 'small';
|
|
87
|
+
}
|
|
88
|
+
get statusIconContainerStyles() {
|
|
89
|
+
return {
|
|
90
|
+
'background-color': this.statusIconBgColor,
|
|
91
|
+
'width': this.isSmall ? '16px' : '20px',
|
|
92
|
+
'height': this.isSmall ? '16px' : '20px',
|
|
93
|
+
'border-radius': '4px',
|
|
94
|
+
'display': 'inline-flex',
|
|
95
|
+
'align-items': 'center',
|
|
96
|
+
'justify-content': 'center'
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
get statusIconStyles() {
|
|
100
|
+
return {
|
|
101
|
+
'color': this.statusColor,
|
|
102
|
+
'font-size': this.isSmall ? '10px' : '12px',
|
|
103
|
+
'width': this.isSmall ? '10px' : '12px',
|
|
104
|
+
'height': this.isSmall ? '10px' : '12px'
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
get statusBadgeStyles() {
|
|
108
|
+
return {
|
|
109
|
+
'background-color': this.statusColor
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
get runLabelStyles() {
|
|
113
|
+
return {
|
|
114
|
+
'background-color': '#ecedfe',
|
|
115
|
+
'padding': this.isSmall ? '2px 6px' : '2px 8px',
|
|
116
|
+
'border-radius': '4px',
|
|
117
|
+
'font-size': this.isSmall ? '8px' : '10px'
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
get cardPadding() {
|
|
121
|
+
return this.isSmall ? '10px' : '12px';
|
|
122
|
+
}
|
|
123
|
+
get borderLeftWidth() {
|
|
124
|
+
return this.isSmall ? '3px' : '4px';
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
RunHistoryCardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: RunHistoryCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
128
|
+
RunHistoryCardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: RunHistoryCardComponent, selector: "cqa-run-history-card", inputs: { id: "id", status: "status", type: "type", timestamp: "timestamp", duration: "duration", runLabel: "runLabel", errorMessage: "errorMessage", size: "size" }, ngImport: i0, template: "<div class=\"cqa-ui-root\" style=\"display: block; width: 100%; height: 100%; min-width: 180px;\">\n <div class=\"cqa-bg-white cqa-rounded-lg cqa-shadow-sm cqa-flex cqa-flex-col\" [ngStyle]=\"{\n padding: cardPadding,\n border: '1px solid #E4E4E4',\n 'border-left-color': statusColor,\n 'border-left-width': borderLeftWidth\n }\">\n\n <!-- Header: Run ID and Status Badge -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap\" [ngClass]=\"{\n 'cqa-mb-[10px]': isSmall,\n 'cqa-mb-2': !isSmall\n }\">\n <div class=\"cqa-flex cqa-items-center\" [ngClass]=\"{ 'cqa-gap-[6px]': isSmall, 'cqa-gap-2': !isSmall }\">\n <span [ngStyle]=\"statusIconContainerStyles\">\n <mat-icon [ngStyle]=\"statusIconStyles\">\n {{ statusIcon }}\n </mat-icon>\n </span>\n <span class=\"cqa-font-semibold cqa-text-[#3f43ee]\" [ngClass]=\"{\n 'cqa-text-xs': isSmall,\n 'cqa-text-sm': !isSmall\n }\">\n #{{ id }}\n </span>\n </div>\n\n <!-- Status Badge -->\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-rounded-md cqa-font-medium cqa-text-white cqa-px-2\"\n [ngClass]=\"{\n 'cqa-py-[2px] cqa-text-[10px]': isSmall,\n 'cqa-py-1 cqa-text-xs': !isSmall\n }\" [ngStyle]=\"statusBadgeStyles\">\n {{ statusLabel }}\n </span>\n </div>\n\n <!-- Type and Timestamp -->\n <div class=\"cqa-flex cqa-items-center cqa-text-[#636363]\" [ngClass]=\"{\n 'cqa-gap-1': isSmall,\n 'cqa-gap-[6px]': !isSmall,\n 'cqa-mb-2': isSmall,\n 'cqa-mb-[6px]': !isSmall\n }\">\n <mat-icon [ngClass]=\"{\n 'cqa-text-[12px] cqa-w-3 cqa-h-3': isSmall,\n 'cqa-text-[14px] cqa-w-[14px] cqa-h-[14px]': !isSmall }\">\n {{ typeIcon }}\n </mat-icon>\n <span [ngClass]=\"{ 'cqa-text-[10px]': isSmall, 'cqa-text-xs': !isSmall }\">\n {{ typeLabel }}\n </span>\n </div>\n\n <!-- Timestamp -->\n <div class=\"cqa-font-normal cqa-text-[#0B0B0B]\" [ngClass]=\"{\n 'cqa-text-sm': !isSmall,\n 'cqa-text-[11px]': isSmall,\n 'cqa-mb-2': isSmall,\n 'cqa-mb-[6px]': !isSmall\n }\">\n {{ timestamp }}\n </div>\n\n <!-- Duration and Run Label -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n <div class=\"cqa-flex cqa-items-center cqa-text-[#636363]\" [ngClass]=\"{\n 'cqa-gap-1': isSmall,\n 'cqa-gap-[6px]': !isSmall\n }\">\n <mat-icon [ngClass]=\"{\n 'cqa-text-[12px] cqa-w-3 cqa-h-3': isSmall,\n 'cqa-text-[14px] cqa-w-[14px] cqa-h-[14px]': !isSmall }\" style=\"flex-shrink: 0;\">\n schedule\n </mat-icon>\n <span [ngClass]=\"{\n 'cqa-text-[8px]': isSmall,\n 'cqa-text-[10px]': !isSmall\n }\">\n {{ duration }}\n </span>\n </div>\n\n <span *ngIf=\"runLabel\" class=\"cqa-font-medium cqa-text-[#3F43EE] cqa-rounded-md cqa-whitespace-nowrap\"\n [ngStyle]=\"runLabelStyles\">\n {{ runLabel }}\n </span>\n </div>\n\n <!-- Error Message (only for failed status and normal size) -->\n <div *ngIf=\"errorMessage && status === 'failed' && !isSmall\" class=\"cqa-mt-2 cqa-pt-2\"\n style=\"border-top: 1px solid #E4E4E4;\">\n <p class=\"cqa-text-[10px] cqa-text-[#E7000B]\">\n {{ errorMessage }}\n </p>\n </div>\n </div>\n</div>", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
129
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: RunHistoryCardComponent, decorators: [{
|
|
130
|
+
type: Component,
|
|
131
|
+
args: [{ selector: 'cqa-run-history-card', template: "<div class=\"cqa-ui-root\" style=\"display: block; width: 100%; height: 100%; min-width: 180px;\">\n <div class=\"cqa-bg-white cqa-rounded-lg cqa-shadow-sm cqa-flex cqa-flex-col\" [ngStyle]=\"{\n padding: cardPadding,\n border: '1px solid #E4E4E4',\n 'border-left-color': statusColor,\n 'border-left-width': borderLeftWidth\n }\">\n\n <!-- Header: Run ID and Status Badge -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap\" [ngClass]=\"{\n 'cqa-mb-[10px]': isSmall,\n 'cqa-mb-2': !isSmall\n }\">\n <div class=\"cqa-flex cqa-items-center\" [ngClass]=\"{ 'cqa-gap-[6px]': isSmall, 'cqa-gap-2': !isSmall }\">\n <span [ngStyle]=\"statusIconContainerStyles\">\n <mat-icon [ngStyle]=\"statusIconStyles\">\n {{ statusIcon }}\n </mat-icon>\n </span>\n <span class=\"cqa-font-semibold cqa-text-[#3f43ee]\" [ngClass]=\"{\n 'cqa-text-xs': isSmall,\n 'cqa-text-sm': !isSmall\n }\">\n #{{ id }}\n </span>\n </div>\n\n <!-- Status Badge -->\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-rounded-md cqa-font-medium cqa-text-white cqa-px-2\"\n [ngClass]=\"{\n 'cqa-py-[2px] cqa-text-[10px]': isSmall,\n 'cqa-py-1 cqa-text-xs': !isSmall\n }\" [ngStyle]=\"statusBadgeStyles\">\n {{ statusLabel }}\n </span>\n </div>\n\n <!-- Type and Timestamp -->\n <div class=\"cqa-flex cqa-items-center cqa-text-[#636363]\" [ngClass]=\"{\n 'cqa-gap-1': isSmall,\n 'cqa-gap-[6px]': !isSmall,\n 'cqa-mb-2': isSmall,\n 'cqa-mb-[6px]': !isSmall\n }\">\n <mat-icon [ngClass]=\"{\n 'cqa-text-[12px] cqa-w-3 cqa-h-3': isSmall,\n 'cqa-text-[14px] cqa-w-[14px] cqa-h-[14px]': !isSmall }\">\n {{ typeIcon }}\n </mat-icon>\n <span [ngClass]=\"{ 'cqa-text-[10px]': isSmall, 'cqa-text-xs': !isSmall }\">\n {{ typeLabel }}\n </span>\n </div>\n\n <!-- Timestamp -->\n <div class=\"cqa-font-normal cqa-text-[#0B0B0B]\" [ngClass]=\"{\n 'cqa-text-sm': !isSmall,\n 'cqa-text-[11px]': isSmall,\n 'cqa-mb-2': isSmall,\n 'cqa-mb-[6px]': !isSmall\n }\">\n {{ timestamp }}\n </div>\n\n <!-- Duration and Run Label -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n <div class=\"cqa-flex cqa-items-center cqa-text-[#636363]\" [ngClass]=\"{\n 'cqa-gap-1': isSmall,\n 'cqa-gap-[6px]': !isSmall\n }\">\n <mat-icon [ngClass]=\"{\n 'cqa-text-[12px] cqa-w-3 cqa-h-3': isSmall,\n 'cqa-text-[14px] cqa-w-[14px] cqa-h-[14px]': !isSmall }\" style=\"flex-shrink: 0;\">\n schedule\n </mat-icon>\n <span [ngClass]=\"{\n 'cqa-text-[8px]': isSmall,\n 'cqa-text-[10px]': !isSmall\n }\">\n {{ duration }}\n </span>\n </div>\n\n <span *ngIf=\"runLabel\" class=\"cqa-font-medium cqa-text-[#3F43EE] cqa-rounded-md cqa-whitespace-nowrap\"\n [ngStyle]=\"runLabelStyles\">\n {{ runLabel }}\n </span>\n </div>\n\n <!-- Error Message (only for failed status and normal size) -->\n <div *ngIf=\"errorMessage && status === 'failed' && !isSmall\" class=\"cqa-mt-2 cqa-pt-2\"\n style=\"border-top: 1px solid #E4E4E4;\">\n <p class=\"cqa-text-[10px] cqa-text-[#E7000B]\">\n {{ errorMessage }}\n </p>\n </div>\n </div>\n</div>", styles: [] }]
|
|
132
|
+
}], propDecorators: { id: [{
|
|
133
|
+
type: Input
|
|
134
|
+
}], status: [{
|
|
135
|
+
type: Input
|
|
136
|
+
}], type: [{
|
|
137
|
+
type: Input
|
|
138
|
+
}], timestamp: [{
|
|
139
|
+
type: Input
|
|
140
|
+
}], duration: [{
|
|
141
|
+
type: Input
|
|
142
|
+
}], runLabel: [{
|
|
143
|
+
type: Input
|
|
144
|
+
}], errorMessage: [{
|
|
145
|
+
type: Input
|
|
146
|
+
}], size: [{
|
|
147
|
+
type: Input
|
|
148
|
+
}] } });
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"run-history-card.component.js","sourceRoot":"","sources":["../../../../../src/lib/run-history-card/run-history-card.component.ts","../../../../../src/lib/run-history-card/run-history-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from \"@angular/core\";\n\nexport type RunStatus = 'passed' | 'failed' | 'aborted' | 'in-progress';\nexport type RunType = 'automated' | 'manual';\nexport type CardSize = 'normal' | 'small';\n\nexport interface RunHistoryData {\n  id: string | number;\n  status: RunStatus;\n  type: RunType;\n  timestamp: string;\n  duration: string;\n  runLabel?: string;\n  errorMessage?: string;\n}\n\n@Component({\n  selector: 'cqa-run-history-card',\n  templateUrl: './run-history-card.component.html',\n  styleUrls: []\n})\nexport class RunHistoryCardComponent {\n  @Input() id!: string | number;\n  @Input() status!: RunStatus;\n  @Input() type!: RunType;\n  @Input() timestamp!: string;\n  @Input() duration!: string;\n  @Input() runLabel?: string;\n  @Input() errorMessage?: string;\n  @Input() size: CardSize = 'normal';\n\n\n\n  get statusBadgeVariant(): 'success' | 'error' | 'warning' | 'info' {\n    switch (this.status) {\n      case 'passed':\n        return 'success';\n      case 'failed':\n        return 'error';\n      case 'aborted':\n        return 'warning';\n      case 'in-progress':\n        return 'info';\n      default:\n        return 'info';\n    }\n  }\n\n  get statusLabel(): string {\n    switch (this.status) {\n      case 'passed':\n        return 'Passed';\n      case 'failed':\n        return 'Failed';\n      case 'aborted':\n        return 'Aborted';\n      case 'in-progress':\n        return 'In Progress';\n      default:\n        return '';\n    }\n  }\n\n  get statusColor(): string {\n    switch (this.status) {\n      case 'passed':\n        return '#00C950';\n      case 'failed':\n        return '#FB2C36';\n      case 'aborted':\n        return '#F97316';\n      case 'in-progress':\n        return '#3B82F6';\n      default:\n        return '#6B7280';\n    }\n  }\n\n    get statusIconBgColor(): string {\n    switch (this.status) {\n        case 'passed':\n            return '#DCFCE7';\n        case 'failed':\n            return '#ffe2e2';\n        case 'aborted':\n            return '#FFEDD5'; \n        case 'in-progress':\n            return '#DBEAFE';\n        default:\n            return '#E5E7EB';\n    }\n  }\n\n  get statusIcon(): string {\n    switch (this.status) {\n      case 'passed':\n        return 'check';\n      case 'failed':\n        return 'close';\n      case 'aborted':\n        return 'warning';\n      case 'in-progress':\n        return 'schedule';\n      default:\n        return '';\n    }\n  }\n\n  get typeIcon(): string {\n    return this.type === 'automated' ? 'settings' : 'person';\n  }\n\n  get typeLabel(): string {\n    return this.type === 'automated' ? 'Automated' : 'Manual';\n  }\n\n  get isSmall(): boolean {\n    return this.size === 'small';\n  }\n\n  get statusIconContainerStyles(): { [key: string]: string } {\n    return {\n      'background-color': this.statusIconBgColor,\n      'width': this.isSmall ? '16px' : '20px',\n      'height': this.isSmall ? '16px' : '20px',\n      'border-radius': '4px',\n      'display': 'inline-flex',\n      'align-items': 'center',\n      'justify-content': 'center'\n    };\n  }\n\n  get statusIconStyles(): { [key: string]: string } {\n    return {\n      'color': this.statusColor,\n      'font-size': this.isSmall ? '10px' : '12px',\n      'width': this.isSmall ? '10px' : '12px',\n      'height': this.isSmall ? '10px' : '12px'\n    };\n  }\n\n  get statusBadgeStyles(): { [key: string]: string } {\n    return {\n      'background-color': this.statusColor\n    };\n  }\n\n  get runLabelStyles(): { [key: string]: string } {\n    return {\n      'background-color': '#ecedfe',\n      'padding': this.isSmall ? '2px 6px' : '2px 8px',\n      'border-radius': '4px',\n      'font-size' : this.isSmall ? '8px' : '10px'\n    };\n  }\n\n  get cardPadding(): string {\n    return this.isSmall ? '10px' : '12px';\n  }\n\n  get borderLeftWidth(): string {\n    return this.isSmall ? '3px' : '4px';\n  }\n\n}","<div class=\"cqa-ui-root\" style=\"display: block; width: 100%; height: 100%; min-width: 180px;\">\n  <div class=\"cqa-bg-white cqa-rounded-lg cqa-shadow-sm cqa-flex cqa-flex-col\" [ngStyle]=\"{\n      padding: cardPadding,\n      border: '1px solid #E4E4E4',\n      'border-left-color': statusColor,\n      'border-left-width': borderLeftWidth\n    }\">\n\n    <!-- Header: Run ID and Status Badge -->\n    <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap\" [ngClass]=\"{\n        'cqa-mb-[10px]': isSmall,\n        'cqa-mb-2': !isSmall\n      }\">\n      <div class=\"cqa-flex cqa-items-center\" [ngClass]=\"{ 'cqa-gap-[6px]': isSmall, 'cqa-gap-2': !isSmall }\">\n        <span [ngStyle]=\"statusIconContainerStyles\">\n          <mat-icon [ngStyle]=\"statusIconStyles\">\n            {{ statusIcon }}\n          </mat-icon>\n        </span>\n        <span class=\"cqa-font-semibold cqa-text-[#3f43ee]\" [ngClass]=\"{\n            'cqa-text-xs': isSmall,\n            'cqa-text-sm': !isSmall\n          }\">\n          #{{ id }}\n        </span>\n      </div>\n\n      <!-- Status Badge -->\n      <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-rounded-md cqa-font-medium cqa-text-white cqa-px-2\"\n        [ngClass]=\"{\n          'cqa-py-[2px] cqa-text-[10px]': isSmall,\n          'cqa-py-1 cqa-text-xs': !isSmall\n        }\" [ngStyle]=\"statusBadgeStyles\">\n        {{ statusLabel }}\n      </span>\n    </div>\n\n    <!-- Type and Timestamp -->\n    <div class=\"cqa-flex cqa-items-center cqa-text-[#636363]\" [ngClass]=\"{\n          'cqa-gap-1': isSmall,\n          'cqa-gap-[6px]': !isSmall,\n          'cqa-mb-2': isSmall,\n          'cqa-mb-[6px]': !isSmall\n        }\">\n      <mat-icon [ngClass]=\"{\n        'cqa-text-[12px] cqa-w-3 cqa-h-3': isSmall,\n        'cqa-text-[14px] cqa-w-[14px] cqa-h-[14px]': !isSmall }\">\n        {{ typeIcon }}\n      </mat-icon>\n      <span [ngClass]=\"{ 'cqa-text-[10px]': isSmall, 'cqa-text-xs': !isSmall }\">\n        {{ typeLabel }}\n      </span>\n    </div>\n\n    <!-- Timestamp -->\n    <div class=\"cqa-font-normal cqa-text-[#0B0B0B]\" [ngClass]=\"{\n        'cqa-text-sm': !isSmall,\n        'cqa-text-[11px]': isSmall,\n        'cqa-mb-2': isSmall,\n        'cqa-mb-[6px]': !isSmall\n      }\">\n      {{ timestamp }}\n    </div>\n\n    <!-- Duration and Run Label -->\n    <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n      <div class=\"cqa-flex cqa-items-center cqa-text-[#636363]\" [ngClass]=\"{\n        'cqa-gap-1': isSmall,\n        'cqa-gap-[6px]': !isSmall\n      }\">\n        <mat-icon [ngClass]=\"{\n          'cqa-text-[12px] cqa-w-3 cqa-h-3': isSmall,\n          'cqa-text-[14px] cqa-w-[14px] cqa-h-[14px]': !isSmall }\" style=\"flex-shrink: 0;\">\n          schedule\n        </mat-icon>\n        <span [ngClass]=\"{\n          'cqa-text-[8px]': isSmall,\n          'cqa-text-[10px]': !isSmall\n        }\">\n          {{ duration }}\n        </span>\n      </div>\n\n      <span *ngIf=\"runLabel\" class=\"cqa-font-medium cqa-text-[#3F43EE] cqa-rounded-md cqa-whitespace-nowrap\"\n        [ngStyle]=\"runLabelStyles\">\n        {{ runLabel }}\n      </span>\n    </div>\n\n    <!-- Error Message (only for failed status and normal size) -->\n    <div *ngIf=\"errorMessage && status === 'failed' && !isSmall\" class=\"cqa-mt-2 cqa-pt-2\"\n      style=\"border-top: 1px solid #E4E4E4;\">\n      <p class=\"cqa-text-[10px] cqa-text-[#E7000B]\">\n        {{ errorMessage }}\n      </p>\n    </div>\n  </div>\n</div>"]}
|
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ViewChild } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "../segment-control/segment-control.component";
|
|
5
|
+
import * as i3 from "@angular/common";
|
|
6
|
+
export class SimulatorComponent {
|
|
7
|
+
constructor() {
|
|
8
|
+
this.videoUrl = '';
|
|
9
|
+
this.videoCurrentDuration = 0;
|
|
10
|
+
this.stepMarkers = [];
|
|
11
|
+
this.screenShotUrl = '';
|
|
12
|
+
this.platformName = 'Web - Chrome';
|
|
13
|
+
this.platformType = 'browser';
|
|
14
|
+
this.videoTimeUpdate = new EventEmitter();
|
|
15
|
+
this.videoPlay = new EventEmitter();
|
|
16
|
+
this.videoPause = new EventEmitter();
|
|
17
|
+
this.progress = 0;
|
|
18
|
+
this.dragging = false;
|
|
19
|
+
this.isPlaying = false;
|
|
20
|
+
this.isFullScreen = false;
|
|
21
|
+
this.currentView = 'video';
|
|
22
|
+
this.segments = [
|
|
23
|
+
{ label: 'Screenshots', value: 'screenshots', icon: 'videocam' },
|
|
24
|
+
{ label: 'Video', value: 'video', icon: 'photo' },
|
|
25
|
+
];
|
|
26
|
+
this.videoEventListenerCleanup = null;
|
|
27
|
+
this.lastSetDuration = -1;
|
|
28
|
+
}
|
|
29
|
+
ngAfterViewInit() {
|
|
30
|
+
this.attachVideoListeners();
|
|
31
|
+
}
|
|
32
|
+
ngOnChanges(changes) {
|
|
33
|
+
if (changes['videoCurrentDuration'] && !changes['videoCurrentDuration'].firstChange) {
|
|
34
|
+
const newDuration = changes['videoCurrentDuration'].currentValue;
|
|
35
|
+
if (newDuration !== this.lastSetDuration && this.vplayer?.nativeElement) {
|
|
36
|
+
this.seekToTime(newDuration);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
ngOnDestroy() {
|
|
41
|
+
if (this.videoEventListenerCleanup) {
|
|
42
|
+
this.videoEventListenerCleanup();
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
seekToTime(milliseconds) {
|
|
46
|
+
if (!this.vplayer?.nativeElement || !this.vplayer.nativeElement.duration)
|
|
47
|
+
return;
|
|
48
|
+
const seconds = milliseconds / 1000;
|
|
49
|
+
const video = this.vplayer.nativeElement;
|
|
50
|
+
const targetTime = Math.max(0, Math.min(seconds, video.duration));
|
|
51
|
+
video.currentTime = targetTime;
|
|
52
|
+
this.lastSetDuration = milliseconds;
|
|
53
|
+
if (this.isPlaying) {
|
|
54
|
+
video.play().catch(err => console.error('Play failed:', err));
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
togglePlay() {
|
|
58
|
+
if (!this.vplayer?.nativeElement)
|
|
59
|
+
return;
|
|
60
|
+
if (this.isPlaying) {
|
|
61
|
+
this.vplayer.nativeElement.pause();
|
|
62
|
+
this.isPlaying = false;
|
|
63
|
+
this.videoPause.emit();
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
this.vplayer.nativeElement.play();
|
|
67
|
+
this.isPlaying = true;
|
|
68
|
+
this.videoPlay.emit();
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
onTimelineClick(event) {
|
|
72
|
+
if (!this.timelineBar?.nativeElement || !this.vplayer?.nativeElement)
|
|
73
|
+
return;
|
|
74
|
+
const rect = this.timelineBar.nativeElement.getBoundingClientRect();
|
|
75
|
+
const clickX = event.clientX - rect.left;
|
|
76
|
+
const percent = Math.max(0, Math.min(1, clickX / rect.width));
|
|
77
|
+
const targetTimeSeconds = percent * this.vplayer.nativeElement.duration;
|
|
78
|
+
this.vplayer.nativeElement.currentTime = targetTimeSeconds;
|
|
79
|
+
this.lastSetDuration = targetTimeSeconds * 1000;
|
|
80
|
+
}
|
|
81
|
+
startDrag() {
|
|
82
|
+
this.dragging = true;
|
|
83
|
+
}
|
|
84
|
+
onDrag(event) {
|
|
85
|
+
if (!this.dragging || !this.timelineBar?.nativeElement)
|
|
86
|
+
return;
|
|
87
|
+
const rect = this.timelineBar.nativeElement.getBoundingClientRect();
|
|
88
|
+
const x = event.clientX - rect.left;
|
|
89
|
+
const percent = Math.max(0, Math.min(x / rect.width, 1));
|
|
90
|
+
this.progress = percent * 100;
|
|
91
|
+
}
|
|
92
|
+
stopDrag() {
|
|
93
|
+
if (!this.dragging || !this.vplayer?.nativeElement)
|
|
94
|
+
return;
|
|
95
|
+
this.dragging = false;
|
|
96
|
+
const targetTimeSeconds = (this.progress / 100) * this.vplayer.nativeElement.duration;
|
|
97
|
+
this.vplayer.nativeElement.currentTime = targetTimeSeconds;
|
|
98
|
+
this.lastSetDuration = targetTimeSeconds * 1000;
|
|
99
|
+
}
|
|
100
|
+
onVideoMetadataLoaded() {
|
|
101
|
+
this.attachVideoListeners();
|
|
102
|
+
}
|
|
103
|
+
onVideoCanPlay() {
|
|
104
|
+
this.attachVideoListeners();
|
|
105
|
+
}
|
|
106
|
+
attachVideoListeners() {
|
|
107
|
+
if (this.videoEventListenerCleanup) {
|
|
108
|
+
this.videoEventListenerCleanup();
|
|
109
|
+
this.videoEventListenerCleanup = null;
|
|
110
|
+
}
|
|
111
|
+
setTimeout(() => {
|
|
112
|
+
const video = this.vplayer?.nativeElement;
|
|
113
|
+
if (!video)
|
|
114
|
+
return;
|
|
115
|
+
let lastUpdate = 0;
|
|
116
|
+
const handler = () => {
|
|
117
|
+
const now = Date.now();
|
|
118
|
+
if (now - lastUpdate < 100)
|
|
119
|
+
return;
|
|
120
|
+
lastUpdate = now;
|
|
121
|
+
const currentMs = video.currentTime * 1000;
|
|
122
|
+
if (!this.dragging) {
|
|
123
|
+
const percent = (currentMs / (video.duration * 1000)) * 100;
|
|
124
|
+
this.progress = Math.min(100, Math.max(0, percent));
|
|
125
|
+
this.videoTimeUpdate.emit(currentMs);
|
|
126
|
+
}
|
|
127
|
+
};
|
|
128
|
+
video.addEventListener('timeupdate', handler);
|
|
129
|
+
this.videoEventListenerCleanup = () => {
|
|
130
|
+
video.removeEventListener('timeupdate', handler);
|
|
131
|
+
};
|
|
132
|
+
}, 100);
|
|
133
|
+
}
|
|
134
|
+
getStepLeftPosition(step) {
|
|
135
|
+
if (!this.vplayer?.nativeElement?.duration)
|
|
136
|
+
return 0;
|
|
137
|
+
return (step.cumulativeDuration / (this.vplayer.nativeElement.duration * 1000)) * 100;
|
|
138
|
+
}
|
|
139
|
+
getStepColor(step) {
|
|
140
|
+
return step.result === 'SUCCESS' ? '#28a745' : '#dc3545';
|
|
141
|
+
}
|
|
142
|
+
toggleFullScreen() {
|
|
143
|
+
this.isFullScreen = !this.isFullScreen;
|
|
144
|
+
}
|
|
145
|
+
onSegmentChange(value) {
|
|
146
|
+
this.currentView = value;
|
|
147
|
+
}
|
|
148
|
+
formatTime(seconds) {
|
|
149
|
+
if (!seconds || isNaN(seconds))
|
|
150
|
+
return '00:00';
|
|
151
|
+
const mins = Math.floor(seconds / 60);
|
|
152
|
+
const secs = Math.floor(seconds % 60);
|
|
153
|
+
return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
SimulatorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SimulatorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
157
|
+
SimulatorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: SimulatorComponent, selector: "cqa-simulator", inputs: { videoUrl: "videoUrl", videoCurrentDuration: "videoCurrentDuration", stepMarkers: "stepMarkers", screenShotUrl: "screenShotUrl", platformName: "platformName", platformType: "platformType" }, outputs: { videoTimeUpdate: "videoTimeUpdate", videoPlay: "videoPlay", videoPause: "videoPause" }, viewQueries: [{ propertyName: "vplayer", first: true, predicate: ["vplayer"], descendants: true }, { propertyName: "timelineBar", first: true, predicate: ["timelineBar"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-ui-root\" style=\"background-color: #F3F4F6; height: 100%; display: flex; flex-direction: column;\" [ngStyle]=\"{\n position: isFullScreen ? 'fixed' : null,\n inset: isFullScreen ? '1rem' : null,\n zIndex: isFullScreen ? '50' : null,\n boxShadow: isFullScreen ? '0px 13px 25px -12px rgba(0, 0, 0, 0.25)' : null,\n borderRadius: isFullScreen ? '.5rem' : null,\n border: isFullScreen ? '1px solid #E5E7EB' : null,\n width: isFullScreen ? 'calc(100% - 32px)' : null,\n height: isFullScreen ? 'calc(100% - 32px)' : '100%',\n overflow: isFullScreen ? 'hidden' : null\n}\">\n <div class=\"cqa-w-full cqa-py-1 cqa-px-2 cqa-bg-[#FFFFFF]\" style=\"border-bottom: 1px solid #E5E7EB;box-shadow: 0px 1px 2px 0px #0000000D;\">\n <div class=\"cqa-w-full cqa-flex cqa-items-center cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center\">\n <mat-icon *ngIf=\"platformType === 'browser'\" style=\"width: 10px; height: 10px;\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <g clip-path=\"url(#clip0_935_15847)\">\n <path\n d=\"M0.625 5C0.625 6.16032 1.08594 7.27312 1.90641 8.09359C2.72688 8.91406 3.83968 9.375 5 9.375C6.16032 9.375 7.27312 8.91406 8.09359 8.09359C8.91406 7.27312 9.375 6.16032 9.375 5C9.375 3.83968 8.91406 2.72688 8.09359 1.90641C7.27312 1.08594 6.16032 0.625 5 0.625C3.83968 0.625 2.72688 1.08594 1.90641 1.90641C1.08594 2.72688 0.625 3.83968 0.625 5Z\"\n stroke=\"#9CA3AF\" stroke-width=\"0.6\" stroke-linejoin=\"round\" />\n <path\n d=\"M3.125 5C3.125 3.83968 3.32254 2.72688 3.67417 1.90641C4.02581 1.08594 4.50272 0.625 5 0.625C5.49728 0.625 5.97419 1.08594 6.32582 1.90641C6.67746 2.72688 6.875 3.83968 6.875 5C6.875 6.16032 6.67746 7.27312 6.32582 8.09359C5.97419 8.91406 5.49728 9.375 5 9.375C4.50272 9.375 4.02581 8.91406 3.67417 8.09359C3.32254 7.27312 3.125 6.16032 3.125 5Z\"\n stroke=\"#9CA3AF\" stroke-width=\"0.6\" stroke-linejoin=\"round\" />\n <path d=\"M0.9375 6.45866H9.0625M0.9375 3.54199H9.0625\" stroke=\"#9CA3AF\" stroke-width=\"0.6\"\n stroke-linecap=\"round\" />\n </g>\n <defs>\n <clipPath id=\"clip0_935_15847\">\n <rect width=\"10\" height=\"10\" fill=\"white\" />\n </clipPath>\n </defs>\n </svg>\n </mat-icon>\n <mat-icon *ngIf=\"platformType === 'device'\" style=\"width: 10px; height: 10px;\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path d=\"M7.08325 0.833008H2.91659C2.45635 0.833008 2.08325 1.2061 2.08325 1.66634V8.33301C2.08325 8.79324 2.45635 9.16634 2.91659 9.16634H7.08325C7.54349 9.16634 7.91658 8.79324 7.91658 8.33301V1.66634C7.91658 1.2061 7.54349 0.833008 7.08325 0.833008Z\" stroke=\"#6B7280\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M5 7.5H5.00417\" stroke=\"#6B7280\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </mat-icon>\n <p class=\"cqa-text-sm !cqa-text-[10px] cqa-text-[#6B7280] cqa-ml-2\">{{ platformName }}</p>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-segment-control \n [segments]=\"segments\" \n [value]=\"currentView\"\n (valueChange)=\"onSegmentChange($event)\">\n </cqa-segment-control>\n \n <div *ngIf=\"!isFullScreen\" \n class=\"cqa-p-1 cqa-cursor-pointer hover:cqa-bg-gray-100 cqa-rounded-sm cqa-transition-colors\"\n (click)=\"toggleFullScreen()\"\n title=\"Expand\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path d=\"M6.25 1.25H8.75V3.75\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M8.74992 1.25L5.83325 4.16667\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M1.25 8.74967L4.16667 5.83301\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M3.75 8.75H1.25V6.25\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </div>\n\n <div *ngIf=\"isFullScreen\" \n class=\"cqa-p-1 cqa-cursor-pointer hover:cqa-bg-gray-100 cqa-rounded-sm cqa-transition-colors\"\n (click)=\"toggleFullScreen()\"\n title=\"Exit full screen\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path d=\"M8.75 6.25H6.25V8.75\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M6.25008 6.25L9.16675 9.16667\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M0.833252 0.833008L3.74992 3.74967\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M1.25 3.75H3.75V1.25\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </div>\n </div>\n </div>\n </div>\n <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n <div *ngIf=\"currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-35px)]\" *ngIf=\"videoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\">\n <video\n #vplayer\n class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-black\"\n [src]=\"videoUrl\"\n type=\"video/webm\"\n (loadedmetadata)=\"onVideoMetadataLoaded()\"\n (canplay)=\"onVideoCanPlay()\"\n ></video>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm\" *ngIf=\"!videoUrl\">\n No video recording found\n </div>\n \n <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" *ngIf=\"videoUrl\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button \n type=\"button\"\n class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer !cqa-px-0 cqa-flex cqa-items-center cqa-justify-center hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none\"\n style=\"pointer-events: auto;\"\n (click)=\"togglePlay()\">\n <span *ngIf=\"!isPlaying\" class=\"cqa-flex cqa-items-center cqa-justify-center\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M3 2L13 8L3 14V2Z\" fill=\"#374151\"/>\n </svg>\n </span>\n <span *ngIf=\"isPlaying\" class=\"cqa-flex cqa-items-center cqa-justify-center\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <rect x=\"3\" y=\"2\" width=\"3\" height=\"12\" fill=\"#374151\"/>\n <rect x=\"10\" y=\"2\" width=\"3\" height=\"12\" fill=\"#374151\"/>\n </svg>\n </span>\n </button>\n \n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none cqa-mr-[20px]\">\n {{ formatTime(vplayer?.nativeElement?.currentTime || 0) }}\n </span>\n \n <div \n #timelineBar\n class=\"cqa-relative cqa-h-1 cqa-bg-gray-200 cqa-rounded-full cqa-cursor-pointer cqa-flex-1\"\n (click)=\"onTimelineClick($event)\" \n (mousemove)=\"onDrag($event)\"\n (mouseup)=\"stopDrag()\" \n (mouseleave)=\"stopDrag()\">\n \n <div \n *ngFor=\"let step of stepMarkers\" \n class=\"cqa-absolute cqa-w-1 cqa-h-full cqa-top-0 cqa-rounded-sm\"\n [style.left.%]=\"getStepLeftPosition(step)\"\n [style.background]=\"getStepColor(step)\"\n style=\"pointer-events: none; z-index: 20;\">\n </div>\n \n <div \n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-h-full cqa-bg-green-500 cqa-rounded-full cqa-transition-[width] cqa-duration-100\"\n [style.width.%]=\"progress\"\n style=\"pointer-events: none; z-index: 2;\">\n </div>\n \n <div \n class=\"cqa-absolute cqa-top-1/2 cqa-w-3 cqa-h-3 cqa-bg-green-600 cqa-rounded-full cqa-cursor-grab active:cqa-cursor-grabbing cqa-shadow-md\"\n [style.left.%]=\"progress\"\n style=\"transform: translate(-50%, -50%); z-index: 10;\"\n (mousedown)=\"startDrag()\">\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"currentView === 'screenshots'\" class=\"cqa-h-full\">\n <div class=\"cqa-w-full cqa-py-4 cqa-h-full cqa-flex cqa-items-center\" *ngIf=\"screenShotUrl\">\n <img\n [src]=\"screenShotUrl\"\n alt=\"Screenshot\"\n class=\"cqa-object-contain cqa-w-full cqa-max-h-full cqa-block cqa-bg-black cqa-mx-auto\"\n />\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm\" *ngIf=\"!screenShotUrl\">\n No screenshot available\n </div>\n </div>\n </div>\n</div>", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled"], outputs: ["valueChange"] }], directives: [{ type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
158
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SimulatorComponent, decorators: [{
|
|
159
|
+
type: Component,
|
|
160
|
+
args: [{ selector: 'cqa-simulator', template: "<div class=\"cqa-ui-root\" style=\"background-color: #F3F4F6; height: 100%; display: flex; flex-direction: column;\" [ngStyle]=\"{\n position: isFullScreen ? 'fixed' : null,\n inset: isFullScreen ? '1rem' : null,\n zIndex: isFullScreen ? '50' : null,\n boxShadow: isFullScreen ? '0px 13px 25px -12px rgba(0, 0, 0, 0.25)' : null,\n borderRadius: isFullScreen ? '.5rem' : null,\n border: isFullScreen ? '1px solid #E5E7EB' : null,\n width: isFullScreen ? 'calc(100% - 32px)' : null,\n height: isFullScreen ? 'calc(100% - 32px)' : '100%',\n overflow: isFullScreen ? 'hidden' : null\n}\">\n <div class=\"cqa-w-full cqa-py-1 cqa-px-2 cqa-bg-[#FFFFFF]\" style=\"border-bottom: 1px solid #E5E7EB;box-shadow: 0px 1px 2px 0px #0000000D;\">\n <div class=\"cqa-w-full cqa-flex cqa-items-center cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center\">\n <mat-icon *ngIf=\"platformType === 'browser'\" style=\"width: 10px; height: 10px;\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <g clip-path=\"url(#clip0_935_15847)\">\n <path\n d=\"M0.625 5C0.625 6.16032 1.08594 7.27312 1.90641 8.09359C2.72688 8.91406 3.83968 9.375 5 9.375C6.16032 9.375 7.27312 8.91406 8.09359 8.09359C8.91406 7.27312 9.375 6.16032 9.375 5C9.375 3.83968 8.91406 2.72688 8.09359 1.90641C7.27312 1.08594 6.16032 0.625 5 0.625C3.83968 0.625 2.72688 1.08594 1.90641 1.90641C1.08594 2.72688 0.625 3.83968 0.625 5Z\"\n stroke=\"#9CA3AF\" stroke-width=\"0.6\" stroke-linejoin=\"round\" />\n <path\n d=\"M3.125 5C3.125 3.83968 3.32254 2.72688 3.67417 1.90641C4.02581 1.08594 4.50272 0.625 5 0.625C5.49728 0.625 5.97419 1.08594 6.32582 1.90641C6.67746 2.72688 6.875 3.83968 6.875 5C6.875 6.16032 6.67746 7.27312 6.32582 8.09359C5.97419 8.91406 5.49728 9.375 5 9.375C4.50272 9.375 4.02581 8.91406 3.67417 8.09359C3.32254 7.27312 3.125 6.16032 3.125 5Z\"\n stroke=\"#9CA3AF\" stroke-width=\"0.6\" stroke-linejoin=\"round\" />\n <path d=\"M0.9375 6.45866H9.0625M0.9375 3.54199H9.0625\" stroke=\"#9CA3AF\" stroke-width=\"0.6\"\n stroke-linecap=\"round\" />\n </g>\n <defs>\n <clipPath id=\"clip0_935_15847\">\n <rect width=\"10\" height=\"10\" fill=\"white\" />\n </clipPath>\n </defs>\n </svg>\n </mat-icon>\n <mat-icon *ngIf=\"platformType === 'device'\" style=\"width: 10px; height: 10px;\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path d=\"M7.08325 0.833008H2.91659C2.45635 0.833008 2.08325 1.2061 2.08325 1.66634V8.33301C2.08325 8.79324 2.45635 9.16634 2.91659 9.16634H7.08325C7.54349 9.16634 7.91658 8.79324 7.91658 8.33301V1.66634C7.91658 1.2061 7.54349 0.833008 7.08325 0.833008Z\" stroke=\"#6B7280\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M5 7.5H5.00417\" stroke=\"#6B7280\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </mat-icon>\n <p class=\"cqa-text-sm !cqa-text-[10px] cqa-text-[#6B7280] cqa-ml-2\">{{ platformName }}</p>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-segment-control \n [segments]=\"segments\" \n [value]=\"currentView\"\n (valueChange)=\"onSegmentChange($event)\">\n </cqa-segment-control>\n \n <div *ngIf=\"!isFullScreen\" \n class=\"cqa-p-1 cqa-cursor-pointer hover:cqa-bg-gray-100 cqa-rounded-sm cqa-transition-colors\"\n (click)=\"toggleFullScreen()\"\n title=\"Expand\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path d=\"M6.25 1.25H8.75V3.75\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M8.74992 1.25L5.83325 4.16667\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M1.25 8.74967L4.16667 5.83301\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M3.75 8.75H1.25V6.25\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </div>\n\n <div *ngIf=\"isFullScreen\" \n class=\"cqa-p-1 cqa-cursor-pointer hover:cqa-bg-gray-100 cqa-rounded-sm cqa-transition-colors\"\n (click)=\"toggleFullScreen()\"\n title=\"Exit full screen\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path d=\"M8.75 6.25H6.25V8.75\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M6.25008 6.25L9.16675 9.16667\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M0.833252 0.833008L3.74992 3.74967\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M1.25 3.75H3.75V1.25\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </div>\n </div>\n </div>\n </div>\n <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n <div *ngIf=\"currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-35px)]\" *ngIf=\"videoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\">\n <video\n #vplayer\n class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-black\"\n [src]=\"videoUrl\"\n type=\"video/webm\"\n (loadedmetadata)=\"onVideoMetadataLoaded()\"\n (canplay)=\"onVideoCanPlay()\"\n ></video>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm\" *ngIf=\"!videoUrl\">\n No video recording found\n </div>\n \n <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" *ngIf=\"videoUrl\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button \n type=\"button\"\n class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer !cqa-px-0 cqa-flex cqa-items-center cqa-justify-center hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none\"\n style=\"pointer-events: auto;\"\n (click)=\"togglePlay()\">\n <span *ngIf=\"!isPlaying\" class=\"cqa-flex cqa-items-center cqa-justify-center\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M3 2L13 8L3 14V2Z\" fill=\"#374151\"/>\n </svg>\n </span>\n <span *ngIf=\"isPlaying\" class=\"cqa-flex cqa-items-center cqa-justify-center\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <rect x=\"3\" y=\"2\" width=\"3\" height=\"12\" fill=\"#374151\"/>\n <rect x=\"10\" y=\"2\" width=\"3\" height=\"12\" fill=\"#374151\"/>\n </svg>\n </span>\n </button>\n \n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none cqa-mr-[20px]\">\n {{ formatTime(vplayer?.nativeElement?.currentTime || 0) }}\n </span>\n \n <div \n #timelineBar\n class=\"cqa-relative cqa-h-1 cqa-bg-gray-200 cqa-rounded-full cqa-cursor-pointer cqa-flex-1\"\n (click)=\"onTimelineClick($event)\" \n (mousemove)=\"onDrag($event)\"\n (mouseup)=\"stopDrag()\" \n (mouseleave)=\"stopDrag()\">\n \n <div \n *ngFor=\"let step of stepMarkers\" \n class=\"cqa-absolute cqa-w-1 cqa-h-full cqa-top-0 cqa-rounded-sm\"\n [style.left.%]=\"getStepLeftPosition(step)\"\n [style.background]=\"getStepColor(step)\"\n style=\"pointer-events: none; z-index: 20;\">\n </div>\n \n <div \n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-h-full cqa-bg-green-500 cqa-rounded-full cqa-transition-[width] cqa-duration-100\"\n [style.width.%]=\"progress\"\n style=\"pointer-events: none; z-index: 2;\">\n </div>\n \n <div \n class=\"cqa-absolute cqa-top-1/2 cqa-w-3 cqa-h-3 cqa-bg-green-600 cqa-rounded-full cqa-cursor-grab active:cqa-cursor-grabbing cqa-shadow-md\"\n [style.left.%]=\"progress\"\n style=\"transform: translate(-50%, -50%); z-index: 10;\"\n (mousedown)=\"startDrag()\">\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"currentView === 'screenshots'\" class=\"cqa-h-full\">\n <div class=\"cqa-w-full cqa-py-4 cqa-h-full cqa-flex cqa-items-center\" *ngIf=\"screenShotUrl\">\n <img\n [src]=\"screenShotUrl\"\n alt=\"Screenshot\"\n class=\"cqa-object-contain cqa-w-full cqa-max-h-full cqa-block cqa-bg-black cqa-mx-auto\"\n />\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm\" *ngIf=\"!screenShotUrl\">\n No screenshot available\n </div>\n </div>\n </div>\n</div>", styles: [] }]
|
|
161
|
+
}], propDecorators: { videoUrl: [{
|
|
162
|
+
type: Input
|
|
163
|
+
}], videoCurrentDuration: [{
|
|
164
|
+
type: Input
|
|
165
|
+
}], stepMarkers: [{
|
|
166
|
+
type: Input
|
|
167
|
+
}], screenShotUrl: [{
|
|
168
|
+
type: Input
|
|
169
|
+
}], platformName: [{
|
|
170
|
+
type: Input
|
|
171
|
+
}], platformType: [{
|
|
172
|
+
type: Input
|
|
173
|
+
}], videoTimeUpdate: [{
|
|
174
|
+
type: Output
|
|
175
|
+
}], videoPlay: [{
|
|
176
|
+
type: Output
|
|
177
|
+
}], videoPause: [{
|
|
178
|
+
type: Output
|
|
179
|
+
}], vplayer: [{
|
|
180
|
+
type: ViewChild,
|
|
181
|
+
args: ['vplayer']
|
|
182
|
+
}], timelineBar: [{
|
|
183
|
+
type: ViewChild,
|
|
184
|
+
args: ['timelineBar']
|
|
185
|
+
}] } });
|
|
186
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"simulator.component.js","sourceRoot":"","sources":["../../../../../src/lib/simulator/simulator.component.ts","../../../../../src/lib/simulator/simulator.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ViewChild, ElementRef, AfterViewInit, OnDestroy, OnChanges, SimpleChanges } from '@angular/core';\n\nexport interface StepMarker {\n  cumulativeDuration: number;\n  result: 'SUCCESS' | 'FAILURE';\n}\n\ntype SegmentOption = {\n  label: string;\n  value: string;\n  icon?: string;\n};\n@Component({\n  selector: 'cqa-simulator',\n  templateUrl: './simulator.component.html',\n  styleUrls: []\n})\nexport class SimulatorComponent implements AfterViewInit, OnDestroy, OnChanges {\n\n  @Input() videoUrl: string = '';\n  @Input() videoCurrentDuration: number = 0;\n  @Input() stepMarkers: StepMarker[] = [];\n  @Input() screenShotUrl: string = '';\n  @Input() platformName: string = 'Web - Chrome';\n  @Input() platformType: 'browser' | 'device' = 'browser';\n\n  @Output() videoTimeUpdate = new EventEmitter<number>();\n  @Output() videoPlay = new EventEmitter<void>();\n  @Output() videoPause = new EventEmitter<void>();\n\n  @ViewChild('vplayer') vplayer!: ElementRef<HTMLVideoElement>;\n  @ViewChild('timelineBar') timelineBar!: ElementRef<HTMLDivElement>;\n\n  progress = 0;\n  dragging = false;\n  isPlaying: boolean = false;\n  isFullScreen: boolean = false;\n  currentView: string = 'video';\n\n  segments: SegmentOption[] = [\n    { label: 'Screenshots', value: 'screenshots', icon: 'videocam' },\n    { label: 'Video', value: 'video', icon: 'photo' },\n  ];\n\n  private videoEventListenerCleanup: (() => void) | null = null;\n  private lastSetDuration: number = -1;\n\n  ngAfterViewInit(): void {\n    this.attachVideoListeners();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['videoCurrentDuration'] && !changes['videoCurrentDuration'].firstChange) {\n      const newDuration = changes['videoCurrentDuration'].currentValue;\n      if (newDuration !== this.lastSetDuration && this.vplayer?.nativeElement) {\n        this.seekToTime(newDuration);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.videoEventListenerCleanup) {\n      this.videoEventListenerCleanup();\n    }\n  }\n\n  seekToTime(milliseconds: number): void {\n    if (!this.vplayer?.nativeElement || !this.vplayer.nativeElement.duration) return;\n    \n    const seconds = milliseconds / 1000;\n    const video = this.vplayer.nativeElement;\n    \n    const targetTime = Math.max(0, Math.min(seconds, video.duration));\n    video.currentTime = targetTime;\n    this.lastSetDuration = milliseconds;\n\n    if (this.isPlaying) {\n      video.play().catch(err => console.error('Play failed:', err));\n    }\n  }\n\n  togglePlay(): void {\n    if (!this.vplayer?.nativeElement) return;\n\n    if (this.isPlaying) {\n      this.vplayer.nativeElement.pause();\n      this.isPlaying = false;\n      this.videoPause.emit();\n    } else {\n      this.vplayer.nativeElement.play();\n      this.isPlaying = true;\n      this.videoPlay.emit();\n    }\n  }\n\n  onTimelineClick(event: MouseEvent): void {\n    if (!this.timelineBar?.nativeElement || !this.vplayer?.nativeElement) return;\n\n    const rect = this.timelineBar.nativeElement.getBoundingClientRect();\n    const clickX = event.clientX - rect.left;\n    const percent = Math.max(0, Math.min(1, clickX / rect.width));\n    \n    const targetTimeSeconds = percent * this.vplayer.nativeElement.duration;\n    this.vplayer.nativeElement.currentTime = targetTimeSeconds;\n    this.lastSetDuration = targetTimeSeconds * 1000;\n  }\n\n  startDrag(): void {\n    this.dragging = true;\n  }\n\n  onDrag(event: MouseEvent): void {\n    if (!this.dragging || !this.timelineBar?.nativeElement) return;\n\n    const rect = this.timelineBar.nativeElement.getBoundingClientRect();\n    const x = event.clientX - rect.left;\n    const percent = Math.max(0, Math.min(x / rect.width, 1));\n    this.progress = percent * 100;\n  }\n\n  stopDrag(): void {\n    if (!this.dragging || !this.vplayer?.nativeElement) return;\n\n    this.dragging = false;\n    const targetTimeSeconds = (this.progress / 100) * this.vplayer.nativeElement.duration;\n    this.vplayer.nativeElement.currentTime = targetTimeSeconds;\n    this.lastSetDuration = targetTimeSeconds * 1000;\n  }\n\n  onVideoMetadataLoaded(): void {\n    this.attachVideoListeners();\n  }\n\n  onVideoCanPlay(): void {\n    this.attachVideoListeners();\n  }\n\n  private attachVideoListeners(): void {\n    if (this.videoEventListenerCleanup) {\n      this.videoEventListenerCleanup();\n      this.videoEventListenerCleanup = null;\n    }\n\n    setTimeout(() => {\n      const video = this.vplayer?.nativeElement;\n      if (!video) return;\n\n      let lastUpdate = 0;\n      const handler = () => {\n        const now = Date.now();\n        if (now - lastUpdate < 100) return;\n\n        lastUpdate = now;\n        const currentMs = video.currentTime * 1000;\n\n        if (!this.dragging) {\n          const percent = (currentMs / (video.duration * 1000)) * 100;\n          this.progress = Math.min(100, Math.max(0, percent));\n          this.videoTimeUpdate.emit(currentMs);\n        }\n      };\n\n      video.addEventListener('timeupdate', handler);\n\n      this.videoEventListenerCleanup = () => {\n        video.removeEventListener('timeupdate', handler);\n      };\n    }, 100);\n  }\n\n  getStepLeftPosition(step: StepMarker): number {\n    if (!this.vplayer?.nativeElement?.duration) return 0;\n    return (step.cumulativeDuration / (this.vplayer.nativeElement.duration * 1000)) * 100;\n  }\n\n  getStepColor(step: StepMarker): string {\n    return step.result === 'SUCCESS' ? '#28a745' : '#dc3545';\n  }\n\n  toggleFullScreen(): void {\n    this.isFullScreen = !this.isFullScreen;\n  }\n\n  onSegmentChange(value: string): void {\n    this.currentView = value;\n  }\n\n  formatTime(seconds: number): string {\n    if (!seconds || isNaN(seconds)) return '00:00';\n    \n    const mins = Math.floor(seconds / 60);\n    const secs = Math.floor(seconds % 60);\n    \n    return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;\n  }\n}\n\n","<div class=\"cqa-ui-root\" style=\"background-color: #F3F4F6; height: 100%; display: flex; flex-direction: column;\"   [ngStyle]=\"{\n  position: isFullScreen ? 'fixed' : null,\n  inset: isFullScreen ? '1rem' : null,\n  zIndex: isFullScreen ? '50' : null,\n  boxShadow: isFullScreen ? '0px 13px 25px -12px rgba(0, 0, 0, 0.25)' : null,\n  borderRadius: isFullScreen ? '.5rem' : null,\n  border: isFullScreen ? '1px solid #E5E7EB' : null,\n  width: isFullScreen ? 'calc(100% - 32px)' : null,\n  height: isFullScreen ? 'calc(100% - 32px)' : '100%',\n  overflow: isFullScreen ? 'hidden' : null\n}\">\n  <div class=\"cqa-w-full cqa-py-1 cqa-px-2 cqa-bg-[#FFFFFF]\" style=\"border-bottom: 1px solid #E5E7EB;box-shadow: 0px 1px 2px 0px #0000000D;\">\n    <div class=\"cqa-w-full cqa-flex cqa-items-center cqa-justify-between\">\n      <div class=\"cqa-flex cqa-items-center\">\n        <mat-icon *ngIf=\"platformType === 'browser'\" style=\"width: 10px; height: 10px;\">\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n            <g clip-path=\"url(#clip0_935_15847)\">\n              <path\n                d=\"M0.625 5C0.625 6.16032 1.08594 7.27312 1.90641 8.09359C2.72688 8.91406 3.83968 9.375 5 9.375C6.16032 9.375 7.27312 8.91406 8.09359 8.09359C8.91406 7.27312 9.375 6.16032 9.375 5C9.375 3.83968 8.91406 2.72688 8.09359 1.90641C7.27312 1.08594 6.16032 0.625 5 0.625C3.83968 0.625 2.72688 1.08594 1.90641 1.90641C1.08594 2.72688 0.625 3.83968 0.625 5Z\"\n                stroke=\"#9CA3AF\" stroke-width=\"0.6\" stroke-linejoin=\"round\" />\n              <path\n                d=\"M3.125 5C3.125 3.83968 3.32254 2.72688 3.67417 1.90641C4.02581 1.08594 4.50272 0.625 5 0.625C5.49728 0.625 5.97419 1.08594 6.32582 1.90641C6.67746 2.72688 6.875 3.83968 6.875 5C6.875 6.16032 6.67746 7.27312 6.32582 8.09359C5.97419 8.91406 5.49728 9.375 5 9.375C4.50272 9.375 4.02581 8.91406 3.67417 8.09359C3.32254 7.27312 3.125 6.16032 3.125 5Z\"\n                stroke=\"#9CA3AF\" stroke-width=\"0.6\" stroke-linejoin=\"round\" />\n              <path d=\"M0.9375 6.45866H9.0625M0.9375 3.54199H9.0625\" stroke=\"#9CA3AF\" stroke-width=\"0.6\"\n                stroke-linecap=\"round\" />\n            </g>\n            <defs>\n              <clipPath id=\"clip0_935_15847\">\n                <rect width=\"10\" height=\"10\" fill=\"white\" />\n              </clipPath>\n            </defs>\n          </svg>\n        </mat-icon>\n        <mat-icon *ngIf=\"platformType === 'device'\" style=\"width: 10px; height: 10px;\">\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n            <path d=\"M7.08325 0.833008H2.91659C2.45635 0.833008 2.08325 1.2061 2.08325 1.66634V8.33301C2.08325 8.79324 2.45635 9.16634 2.91659 9.16634H7.08325C7.54349 9.16634 7.91658 8.79324 7.91658 8.33301V1.66634C7.91658 1.2061 7.54349 0.833008 7.08325 0.833008Z\" stroke=\"#6B7280\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M5 7.5H5.00417\" stroke=\"#6B7280\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n          </svg>\n        </mat-icon>\n        <p class=\"cqa-text-sm !cqa-text-[10px] cqa-text-[#6B7280] cqa-ml-2\">{{ platformName }}</p>\n      </div>\n      <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n        <cqa-segment-control \n          [segments]=\"segments\" \n          [value]=\"currentView\"\n          (valueChange)=\"onSegmentChange($event)\">\n        </cqa-segment-control>\n        \n        <div *ngIf=\"!isFullScreen\" \n             class=\"cqa-p-1 cqa-cursor-pointer hover:cqa-bg-gray-100 cqa-rounded-sm cqa-transition-colors\"\n             (click)=\"toggleFullScreen()\"\n             title=\"Expand\">\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n            <path d=\"M6.25 1.25H8.75V3.75\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M8.74992 1.25L5.83325 4.16667\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M1.25 8.74967L4.16667 5.83301\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M3.75 8.75H1.25V6.25\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n          </svg>\n        </div>\n\n        <div *ngIf=\"isFullScreen\" \n             class=\"cqa-p-1 cqa-cursor-pointer hover:cqa-bg-gray-100 cqa-rounded-sm cqa-transition-colors\"\n             (click)=\"toggleFullScreen()\"\n             title=\"Exit full screen\">\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\">\n            <path d=\"M8.75 6.25H6.25V8.75\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M6.25008 6.25L9.16675 9.16667\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M0.833252 0.833008L3.74992 3.74967\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            <path d=\"M1.25 3.75H3.75V1.25\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n          </svg>\n        </div>\n      </div>\n    </div>\n  </div>\n  <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n    <div *ngIf=\"currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n      <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-35px)]\" *ngIf=\"videoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\">\n        <video\n          #vplayer\n          class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-black\"\n          [src]=\"videoUrl\"\n          type=\"video/webm\"\n          (loadedmetadata)=\"onVideoMetadataLoaded()\"\n          (canplay)=\"onVideoCanPlay()\"\n        ></video>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm\" *ngIf=\"!videoUrl\">\n        No video recording found\n      </div>\n    \n       <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" *ngIf=\"videoUrl\">\n       <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n         <button \n           type=\"button\"\n           class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer !cqa-px-0 cqa-flex cqa-items-center cqa-justify-center hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none\"\n           style=\"pointer-events: auto;\"\n           (click)=\"togglePlay()\">\n          <span *ngIf=\"!isPlaying\" class=\"cqa-flex cqa-items-center cqa-justify-center\">\n            <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n              <path d=\"M3 2L13 8L3 14V2Z\" fill=\"#374151\"/>\n            </svg>\n          </span>\n          <span *ngIf=\"isPlaying\" class=\"cqa-flex cqa-items-center cqa-justify-center\">\n             <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n               <rect x=\"3\" y=\"2\" width=\"3\" height=\"12\" fill=\"#374151\"/>\n               <rect x=\"10\" y=\"2\" width=\"3\" height=\"12\" fill=\"#374151\"/>\n             </svg>\n           </span>\n         </button>\n   \n         <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none cqa-mr-[20px]\">\n           {{ formatTime(vplayer?.nativeElement?.currentTime || 0) }}\n         </span>\n   \n         <div \n           #timelineBar\n           class=\"cqa-relative cqa-h-1 cqa-bg-gray-200 cqa-rounded-full cqa-cursor-pointer cqa-flex-1\"\n           (click)=\"onTimelineClick($event)\" \n           (mousemove)=\"onDrag($event)\"\n           (mouseup)=\"stopDrag()\" \n           (mouseleave)=\"stopDrag()\">\n           \n           <div \n             *ngFor=\"let step of stepMarkers\" \n             class=\"cqa-absolute cqa-w-1 cqa-h-full cqa-top-0 cqa-rounded-sm\"\n             [style.left.%]=\"getStepLeftPosition(step)\"\n             [style.background]=\"getStepColor(step)\"\n             style=\"pointer-events: none; z-index: 20;\">\n           </div>\n           \n           <div \n             class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-h-full cqa-bg-green-500 cqa-rounded-full cqa-transition-[width] cqa-duration-100\"\n             [style.width.%]=\"progress\"\n             style=\"pointer-events: none; z-index: 2;\">\n           </div>\n           \n           <div \n             class=\"cqa-absolute cqa-top-1/2 cqa-w-3 cqa-h-3 cqa-bg-green-600 cqa-rounded-full cqa-cursor-grab active:cqa-cursor-grabbing cqa-shadow-md\"\n             [style.left.%]=\"progress\"\n             style=\"transform: translate(-50%, -50%); z-index: 10;\"\n             (mousedown)=\"startDrag()\">\n           </div>\n         </div>\n       </div>\n     </div>\n    </div>\n\n    <div *ngIf=\"currentView === 'screenshots'\" class=\"cqa-h-full\">\n      <div class=\"cqa-w-full cqa-py-4 cqa-h-full cqa-flex cqa-items-center\" *ngIf=\"screenShotUrl\">\n        <img\n          [src]=\"screenShotUrl\"\n          alt=\"Screenshot\"\n          class=\"cqa-object-contain cqa-w-full cqa-max-h-full cqa-block cqa-bg-black cqa-mx-auto\"\n        />\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm\" *ngIf=\"!screenShotUrl\">\n        No screenshot available\n      </div>\n    </div>\n  </div>\n</div>"]}
|