@cqa-lib/cqa-ui 1.1.176 → 1.1.178
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/compare-runs/compare-runs.component.mjs +1 -1
- package/esm2020/lib/execution-screen/ai-action-step/ai-action-step.component.mjs +23 -7
- package/esm2020/lib/execution-screen/ai-agent-step/ai-agent-step.component.mjs +23 -8
- package/esm2020/lib/execution-screen/api-step/api-step.component.mjs +24 -7
- package/esm2020/lib/execution-screen/base-step.component.mjs +12 -1
- package/esm2020/lib/execution-screen/basic-step/basic-step.component.mjs +25 -8
- package/esm2020/lib/execution-screen/condition-step/condition-step.component.mjs +25 -8
- package/esm2020/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.mjs +70 -5
- package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +40 -11
- package/esm2020/lib/execution-screen/document-verification-step/document-verification-step.component.mjs +22 -4
- package/esm2020/lib/execution-screen/execution-step.models.mjs +1 -1
- package/esm2020/lib/execution-screen/file-download-step/file-download-step.component.mjs +23 -5
- package/esm2020/lib/execution-screen/loop-step/loop-step.component.mjs +26 -9
- package/esm2020/lib/execution-screen/step-group/step-group.component.mjs +22 -5
- package/esm2020/lib/execution-screen/step-renderer/step-renderer.component.mjs +6 -2
- package/esm2020/lib/iterations-loop/iterations-loop.component.mjs +1 -1
- package/esm2020/lib/simulator/simulator.component.mjs +1202 -63
- package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +35 -3
- package/esm2020/lib/templates/table-template.component.mjs +7 -4
- package/fesm2015/cqa-lib-cqa-ui.mjs +1649 -195
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +1599 -184
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/execution-screen/ai-action-step/ai-action-step.component.d.ts +3 -1
- package/lib/execution-screen/ai-agent-step/ai-agent-step.component.d.ts +3 -1
- package/lib/execution-screen/api-step/api-step.component.d.ts +3 -1
- package/lib/execution-screen/base-step.component.d.ts +1 -0
- package/lib/execution-screen/basic-step/basic-step.component.d.ts +3 -1
- package/lib/execution-screen/condition-step/condition-step.component.d.ts +3 -1
- package/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.d.ts +18 -3
- package/lib/execution-screen/db-verification-step/db-verification-step.component.d.ts +9 -1
- package/lib/execution-screen/document-verification-step/document-verification-step.component.d.ts +4 -1
- package/lib/execution-screen/execution-step.models.d.ts +4 -0
- package/lib/execution-screen/file-download-step/file-download-step.component.d.ts +4 -1
- package/lib/execution-screen/loop-step/loop-step.component.d.ts +3 -1
- package/lib/execution-screen/step-group/step-group.component.d.ts +3 -1
- package/lib/execution-screen/step-renderer/step-renderer.component.d.ts +2 -1
- package/lib/simulator/simulator.component.d.ts +122 -6
- package/lib/table/dynamic-table/dynamic-table.component.d.ts +7 -1
- package/lib/templates/table-template.component.d.ts +6 -1
- package/package.json +1 -1
- package/styles.css +1 -1
|
@@ -33,6 +33,8 @@ export class SimulatorComponent {
|
|
|
33
33
|
this.videoTimeUpdate = new EventEmitter();
|
|
34
34
|
this.videoPlay = new EventEmitter();
|
|
35
35
|
this.videoPause = new EventEmitter();
|
|
36
|
+
this.markerHit = new EventEmitter();
|
|
37
|
+
this.isVideoPlayingChange = new EventEmitter();
|
|
36
38
|
this.progress = 0;
|
|
37
39
|
this.dragging = false;
|
|
38
40
|
this.isPlaying = false;
|
|
@@ -41,6 +43,13 @@ export class SimulatorComponent {
|
|
|
41
43
|
this.currentVideoIndex = 0;
|
|
42
44
|
this.currentSpeed = '1x';
|
|
43
45
|
this.isSpeedControlOpen = false;
|
|
46
|
+
this.detectedVideoDurations = []; // Actual durations detected from video metadata
|
|
47
|
+
this.wasPlayingBeforeDrag = false; // Track if video was playing before drag started
|
|
48
|
+
this.playPromise = null; // Track the current play promise
|
|
49
|
+
this.hitMarkers = new Set(); // Track which markers have been hit (by testStepId)
|
|
50
|
+
// State machine for robust media operations
|
|
51
|
+
this.playerState = 'idle';
|
|
52
|
+
this.operationQueue = Promise.resolve(); // Serialize all media operations
|
|
44
53
|
this.segments = [
|
|
45
54
|
{ label: 'Screenshots', value: 'screenshots', icon: 'photo' },
|
|
46
55
|
{ label: 'Video', value: 'video', icon: 'videocam' },
|
|
@@ -55,6 +64,9 @@ export class SimulatorComponent {
|
|
|
55
64
|
this.dragMouseMoveHandler = null;
|
|
56
65
|
this.dragMouseUpHandler = null;
|
|
57
66
|
this.speedControlClickOutsideHandler = null;
|
|
67
|
+
this.preloadVideoElement = null;
|
|
68
|
+
this.preloadAllVideoElement = null;
|
|
69
|
+
this.targetGlobalTimeDuringSwitch = null;
|
|
58
70
|
this.traceViewerLoading = false;
|
|
59
71
|
this.traceViewerError = false;
|
|
60
72
|
this.isCorrectDeviceFrameAvailable = false;
|
|
@@ -171,9 +183,28 @@ export class SimulatorComponent {
|
|
|
171
183
|
get vplayer() {
|
|
172
184
|
return this._vplayer;
|
|
173
185
|
}
|
|
186
|
+
set timelineBarRef(ref) {
|
|
187
|
+
if (!ref)
|
|
188
|
+
return;
|
|
189
|
+
this._timelineBar = ref;
|
|
190
|
+
}
|
|
191
|
+
get timelineBar() {
|
|
192
|
+
return this._timelineBar;
|
|
193
|
+
}
|
|
194
|
+
set speedControlContainerRef(ref) {
|
|
195
|
+
if (!ref)
|
|
196
|
+
return;
|
|
197
|
+
this._speedControlContainer = ref;
|
|
198
|
+
}
|
|
199
|
+
get speedControlContainer() {
|
|
200
|
+
return this._speedControlContainer;
|
|
201
|
+
}
|
|
174
202
|
get hasDeviceFrame() {
|
|
175
203
|
return !!this.deviceFrameConfig;
|
|
176
204
|
}
|
|
205
|
+
get isPlayerSwitching() {
|
|
206
|
+
return this.playerState === 'switching';
|
|
207
|
+
}
|
|
177
208
|
get effectiveBrowserViewPort() {
|
|
178
209
|
const defaultViewport = { width: 1280, height: 720 };
|
|
179
210
|
if (!this.browserViewPort) {
|
|
@@ -234,12 +265,49 @@ export class SimulatorComponent {
|
|
|
234
265
|
const arr = changes['videoUrls'].currentValue;
|
|
235
266
|
if (arr && arr.length > 0) {
|
|
236
267
|
this.currentVideoIndex = 0;
|
|
268
|
+
this.detectVideoDurations(arr);
|
|
269
|
+
this.schedulePreloadAllSegments(arr);
|
|
237
270
|
}
|
|
238
271
|
}
|
|
239
272
|
if (changes['videoCurrentDuration'] && !changes['videoCurrentDuration'].firstChange) {
|
|
240
273
|
const newDuration = changes['videoCurrentDuration'].currentValue;
|
|
241
|
-
|
|
242
|
-
|
|
274
|
+
// Ignore sentinel value (-1) used to force change detection
|
|
275
|
+
if (newDuration === -1) {
|
|
276
|
+
return;
|
|
277
|
+
}
|
|
278
|
+
if (this.isPlaying) {
|
|
279
|
+
this.pauseVideo();
|
|
280
|
+
console.log('[Simulator] ngOnChanges videoCurrentDuration: paused video before seeking');
|
|
281
|
+
}
|
|
282
|
+
if (this.hasMultipleVideos) {
|
|
283
|
+
const targetVideoIndex = this.findVideoIndexForTimestamp(newDuration);
|
|
284
|
+
if (targetVideoIndex !== null) {
|
|
285
|
+
const boundaries = this.getVideoDurationBoundaries();
|
|
286
|
+
const targetBoundary = boundaries[targetVideoIndex];
|
|
287
|
+
const relativeTimestamp = newDuration - targetBoundary.start;
|
|
288
|
+
const needsSwitch = this.currentVideoIndex !== targetVideoIndex;
|
|
289
|
+
if (this.vplayer?.nativeElement) {
|
|
290
|
+
const currentVideoTimeMs = this.vplayer.nativeElement.currentTime * 1000;
|
|
291
|
+
const timeDifference = Math.abs(relativeTimestamp - currentVideoTimeMs);
|
|
292
|
+
if (needsSwitch || newDuration !== this.lastSetDuration || timeDifference > 100) {
|
|
293
|
+
this.switchToVideoAndSeek(newDuration);
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
else {
|
|
297
|
+
if (needsSwitch) {
|
|
298
|
+
this.currentVideoIndex = targetVideoIndex;
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
else {
|
|
304
|
+
if (this.vplayer?.nativeElement) {
|
|
305
|
+
const currentVideoTimeMs = this.vplayer.nativeElement.currentTime * 1000;
|
|
306
|
+
const timeDifference = Math.abs(newDuration - currentVideoTimeMs);
|
|
307
|
+
if (newDuration !== this.lastSetDuration || timeDifference > 100) {
|
|
308
|
+
this.seekToTime(newDuration);
|
|
309
|
+
}
|
|
310
|
+
}
|
|
243
311
|
}
|
|
244
312
|
}
|
|
245
313
|
if (changes['traceViewUrl']) {
|
|
@@ -260,46 +328,485 @@ export class SimulatorComponent {
|
|
|
260
328
|
if (this.videoEventListenerCleanup) {
|
|
261
329
|
this.videoEventListenerCleanup();
|
|
262
330
|
}
|
|
331
|
+
if (this.preloadVideoElement) {
|
|
332
|
+
this.preloadVideoElement.src = '';
|
|
333
|
+
this.preloadVideoElement.remove();
|
|
334
|
+
this.preloadVideoElement = null;
|
|
335
|
+
}
|
|
336
|
+
if (this.preloadAllVideoElement) {
|
|
337
|
+
this.preloadAllVideoElement.src = '';
|
|
338
|
+
this.preloadAllVideoElement.remove();
|
|
339
|
+
this.preloadAllVideoElement = null;
|
|
340
|
+
}
|
|
263
341
|
this.removeDragListeners();
|
|
264
342
|
this.removeSpeedControlClickOutsideListener();
|
|
343
|
+
console.log('[Simulator] ngOnDestroy: cleaned up listeners');
|
|
265
344
|
}
|
|
345
|
+
/**
|
|
346
|
+
* Seek video - public method (enqueues operation)
|
|
347
|
+
*/
|
|
266
348
|
seekToTime(milliseconds) {
|
|
267
|
-
if (!this.vplayer?.nativeElement
|
|
349
|
+
if (!this.vplayer?.nativeElement)
|
|
350
|
+
return;
|
|
351
|
+
this.enqueueOperation(() => this.seekToTimeInternal(milliseconds));
|
|
352
|
+
}
|
|
353
|
+
/**
|
|
354
|
+
* Seek to a global time on the combined timeline (converts to segment + local time)
|
|
355
|
+
*/
|
|
356
|
+
async seekToGlobalTime(globalTimeMs) {
|
|
357
|
+
const total = this.totalDuration;
|
|
358
|
+
if (total <= 0)
|
|
359
|
+
return;
|
|
360
|
+
const clampedGlobal = Math.max(0, Math.min(globalTimeMs, total));
|
|
361
|
+
const boundaries = this.segmentBoundaries;
|
|
362
|
+
let segmentIndex = 0;
|
|
363
|
+
let segmentStart = 0;
|
|
364
|
+
for (let i = 0; i < boundaries.length; i++) {
|
|
365
|
+
if (clampedGlobal < boundaries[i]) {
|
|
366
|
+
segmentIndex = i;
|
|
367
|
+
segmentStart = i === 0 ? 0 : boundaries[i - 1];
|
|
368
|
+
break;
|
|
369
|
+
}
|
|
370
|
+
segmentIndex = i;
|
|
371
|
+
segmentStart = i === 0 ? 0 : boundaries[i - 1];
|
|
372
|
+
}
|
|
373
|
+
if (boundaries.length > 0 && clampedGlobal >= boundaries[boundaries.length - 1]) {
|
|
374
|
+
segmentIndex = boundaries.length - 1;
|
|
375
|
+
segmentStart = segmentIndex === 0 ? 0 : boundaries[segmentIndex - 1];
|
|
376
|
+
}
|
|
377
|
+
const localTimeMs = clampedGlobal - segmentStart;
|
|
378
|
+
if (this.currentVideoIndex === segmentIndex) {
|
|
379
|
+
await this.seekToTimeInternal(localTimeMs);
|
|
380
|
+
// Update progress after seek in same segment
|
|
381
|
+
if (this.totalDuration > 0) {
|
|
382
|
+
this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / this.totalDuration) * 100));
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
else {
|
|
386
|
+
const cumulativeTimestamp = segmentStart + localTimeMs;
|
|
387
|
+
await this.switchToVideoAndSeekInternal(cumulativeTimestamp);
|
|
388
|
+
// Progress is already updated inside switchToVideoAndSeekInternal
|
|
389
|
+
}
|
|
390
|
+
}
|
|
391
|
+
/**
|
|
392
|
+
* Seek video internal implementation (no enqueue - called from queue)
|
|
393
|
+
* This prevents pipeline read errors during rapid seek operations
|
|
394
|
+
*/
|
|
395
|
+
async seekToTimeInternal(milliseconds) {
|
|
396
|
+
const video = this.vplayer?.nativeElement;
|
|
397
|
+
if (!video || !video.duration)
|
|
268
398
|
return;
|
|
269
399
|
const seconds = milliseconds / 1000;
|
|
270
|
-
const video = this.vplayer.nativeElement;
|
|
271
400
|
const targetTime = Math.max(0, Math.min(seconds, video.duration));
|
|
401
|
+
console.log('[Simulator] seekToTimeInternal: seeking', {
|
|
402
|
+
milliseconds,
|
|
403
|
+
targetTime,
|
|
404
|
+
playerState: this.playerState,
|
|
405
|
+
currentTimeBefore: video.currentTime
|
|
406
|
+
});
|
|
407
|
+
// Set seeking state
|
|
408
|
+
this.playerState = 'seeking';
|
|
409
|
+
// CRITICAL: Pause before seeking to avoid pipeline errors
|
|
410
|
+
video.pause();
|
|
411
|
+
// Update state
|
|
412
|
+
this.isPlaying = false;
|
|
413
|
+
this.playPromise = null;
|
|
414
|
+
this.videoPause.emit();
|
|
415
|
+
this.isVideoPlayingChange.emit(false);
|
|
416
|
+
// Perform seek
|
|
272
417
|
video.currentTime = targetTime;
|
|
273
418
|
this.lastSetDuration = milliseconds;
|
|
274
|
-
|
|
275
|
-
|
|
419
|
+
// Wait briefly for seek to complete (prevents rapid seek issues)
|
|
420
|
+
await new Promise(resolve => setTimeout(resolve, 50));
|
|
421
|
+
const total = this.totalDuration;
|
|
422
|
+
if (total > 0) {
|
|
423
|
+
this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / total) * 100));
|
|
276
424
|
}
|
|
425
|
+
this.videoTimeUpdate.emit(this.globalCurrentTime);
|
|
426
|
+
// Set to paused state (user must explicitly play)
|
|
427
|
+
this.playerState = 'paused';
|
|
428
|
+
console.log('[Simulator] seekToTimeInternal: seek complete', {
|
|
429
|
+
currentTimeAfter: video.currentTime,
|
|
430
|
+
playerState: this.playerState
|
|
431
|
+
});
|
|
277
432
|
}
|
|
433
|
+
/**
|
|
434
|
+
* Toggle play/pause (now trivial with state machine)
|
|
435
|
+
*/
|
|
278
436
|
togglePlay() {
|
|
279
437
|
const video = this.vplayer?.nativeElement;
|
|
280
|
-
if (!video)
|
|
438
|
+
if (!video) {
|
|
439
|
+
console.error('[Simulator] togglePlay: video element not found', {
|
|
440
|
+
playerState: this.playerState,
|
|
441
|
+
isFullScreen: this.isFullScreen
|
|
442
|
+
});
|
|
281
443
|
return;
|
|
282
|
-
|
|
444
|
+
}
|
|
445
|
+
// Check for valid source
|
|
283
446
|
if (!this.currentVideoUrl || video.networkState === HTMLMediaElement.NETWORK_NO_SOURCE || video.error) {
|
|
447
|
+
console.error('[Simulator] togglePlay: no valid source', {
|
|
448
|
+
currentVideoUrl: this.currentVideoUrl,
|
|
449
|
+
networkState: video.networkState,
|
|
450
|
+
error: video.error
|
|
451
|
+
});
|
|
284
452
|
return;
|
|
285
453
|
}
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
this.
|
|
289
|
-
|
|
454
|
+
// Recover from stuck 'loading' state when video is actually ready (e.g. after jump-to-segment)
|
|
455
|
+
if (this.playerState === 'loading' && video.readyState >= HTMLMediaElement.HAVE_METADATA) {
|
|
456
|
+
this.playerState = 'paused';
|
|
457
|
+
}
|
|
458
|
+
// Don't allow play during error state
|
|
459
|
+
if (this.playerState === 'error') {
|
|
460
|
+
console.warn('[Simulator] togglePlay: player in error state', { readyState: video.readyState });
|
|
461
|
+
this.playerState = 'idle';
|
|
462
|
+
}
|
|
463
|
+
// If still loading and video not ready, wait and retry
|
|
464
|
+
if (this.playerState === 'loading') {
|
|
465
|
+
console.warn('[Simulator] togglePlay: player not ready', {
|
|
466
|
+
playerState: this.playerState,
|
|
467
|
+
readyState: video.readyState
|
|
468
|
+
});
|
|
469
|
+
setTimeout(() => {
|
|
470
|
+
if (video.readyState >= HTMLMediaElement.HAVE_METADATA && this.playerState !== 'playing') {
|
|
471
|
+
this.togglePlay();
|
|
472
|
+
}
|
|
473
|
+
}, 50);
|
|
474
|
+
return;
|
|
475
|
+
}
|
|
476
|
+
console.log('[Simulator] togglePlay:', {
|
|
477
|
+
playerState: this.playerState,
|
|
478
|
+
isPlaying: this.isPlaying,
|
|
479
|
+
readyState: video.readyState
|
|
480
|
+
});
|
|
481
|
+
// Simple toggle based on state
|
|
482
|
+
if (this.playerState === 'playing') {
|
|
483
|
+
this.pauseVideo();
|
|
290
484
|
}
|
|
291
485
|
else {
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
486
|
+
this.playVideo();
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
/**
|
|
490
|
+
* Play video - public method (enqueues operation)
|
|
491
|
+
*/
|
|
492
|
+
playVideo() {
|
|
493
|
+
this.enqueueOperation(() => this.playVideoInternal());
|
|
494
|
+
}
|
|
495
|
+
/**
|
|
496
|
+
* Play video internal implementation (no enqueue - called from queue)
|
|
497
|
+
* This eliminates FFmpegDemuxer errors and play promise interruptions
|
|
498
|
+
*/
|
|
499
|
+
async playVideoInternal() {
|
|
500
|
+
const video = this.vplayer?.nativeElement;
|
|
501
|
+
if (!video) {
|
|
502
|
+
console.error('[Simulator] playVideoInternal: video element not found');
|
|
503
|
+
return;
|
|
504
|
+
}
|
|
505
|
+
// Already playing - skip
|
|
506
|
+
if (this.playerState === 'playing') {
|
|
507
|
+
console.log('[Simulator] playVideoInternal: already playing, skipping');
|
|
508
|
+
return;
|
|
509
|
+
}
|
|
510
|
+
if (this.hasMultipleVideos && this.videoUrls) {
|
|
511
|
+
const total = this.totalDuration;
|
|
512
|
+
const currentGlobalTime = this.globalCurrentTime;
|
|
513
|
+
const isAtEnd = total > 0 && (this.progress >= 100 || currentGlobalTime >= total);
|
|
514
|
+
const isLastVideo = this.currentVideoIndex >= this.videoUrls.length - 1;
|
|
515
|
+
const isLastVideoAtEnd = isLastVideo && video.duration && video.currentTime >= video.duration - 0.1;
|
|
516
|
+
if (isAtEnd || isLastVideoAtEnd) {
|
|
517
|
+
console.log('[Simulator] playVideoInternal: all videos completed, resetting to first video', {
|
|
518
|
+
progress: this.progress,
|
|
519
|
+
currentGlobalTime,
|
|
520
|
+
total,
|
|
521
|
+
currentVideoIndex: this.currentVideoIndex,
|
|
522
|
+
isAtEnd,
|
|
523
|
+
isLastVideoAtEnd
|
|
524
|
+
});
|
|
525
|
+
// Reset to first video at 0 progress
|
|
526
|
+
this.currentVideoIndex = 0;
|
|
527
|
+
this.progress = 0;
|
|
528
|
+
this.targetGlobalTimeDuringSwitch = 0;
|
|
529
|
+
this.hitMarkers.clear();
|
|
530
|
+
// Switch to first video and seek to 0
|
|
531
|
+
await this.switchToVideoAndSeekInternal(0);
|
|
532
|
+
// Update video reference after switch
|
|
533
|
+
const videoAfterSwitch = this.vplayer?.nativeElement;
|
|
534
|
+
if (!videoAfterSwitch) {
|
|
535
|
+
console.error('[Simulator] playVideoInternal: video element not found after switch');
|
|
536
|
+
return;
|
|
537
|
+
}
|
|
538
|
+
// Continue with normal play logic below
|
|
539
|
+
// The video element reference has changed, so we'll use videoAfterSwitch
|
|
540
|
+
// But we need to wait for metadata again
|
|
541
|
+
if (videoAfterSwitch.readyState < HTMLMediaElement.HAVE_METADATA) {
|
|
542
|
+
await new Promise((resolve, reject) => {
|
|
543
|
+
const onLoadedMetadata = () => {
|
|
544
|
+
videoAfterSwitch.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
545
|
+
videoAfterSwitch.removeEventListener('error', onError);
|
|
546
|
+
resolve();
|
|
547
|
+
};
|
|
548
|
+
const onError = () => {
|
|
549
|
+
videoAfterSwitch.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
550
|
+
videoAfterSwitch.removeEventListener('error', onError);
|
|
551
|
+
reject(new Error('Failed to load video metadata'));
|
|
552
|
+
};
|
|
553
|
+
videoAfterSwitch.addEventListener('loadedmetadata', onLoadedMetadata, { once: true });
|
|
554
|
+
videoAfterSwitch.addEventListener('error', onError, { once: true });
|
|
555
|
+
setTimeout(() => {
|
|
556
|
+
videoAfterSwitch.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
557
|
+
videoAfterSwitch.removeEventListener('error', onError);
|
|
558
|
+
if (videoAfterSwitch.readyState >= HTMLMediaElement.HAVE_METADATA) {
|
|
559
|
+
resolve();
|
|
560
|
+
}
|
|
561
|
+
else {
|
|
562
|
+
reject(new Error('Timeout waiting for metadata'));
|
|
563
|
+
}
|
|
564
|
+
}, 500);
|
|
565
|
+
});
|
|
566
|
+
}
|
|
567
|
+
// Set state to loading
|
|
568
|
+
this.playerState = 'loading';
|
|
569
|
+
try {
|
|
570
|
+
await videoAfterSwitch.play();
|
|
571
|
+
this.playerState = 'playing';
|
|
572
|
+
this.isPlaying = true;
|
|
573
|
+
this.playPromise = null;
|
|
574
|
+
this.videoPlay.emit();
|
|
575
|
+
this.isVideoPlayingChange.emit(true);
|
|
576
|
+
this.targetGlobalTimeDuringSwitch = null;
|
|
577
|
+
console.log('[Simulator] playVideoInternal: restarted from first video');
|
|
578
|
+
return;
|
|
579
|
+
}
|
|
580
|
+
catch (err) {
|
|
581
|
+
this.playerState = 'error';
|
|
582
|
+
this.isPlaying = false;
|
|
583
|
+
this.playPromise = null;
|
|
584
|
+
this.isVideoPlayingChange.emit(false);
|
|
585
|
+
if (err.name === 'AbortError') {
|
|
586
|
+
this.playerState = 'paused';
|
|
587
|
+
}
|
|
588
|
+
else {
|
|
589
|
+
console.error('[Simulator] playVideoInternal: failed after reset', err);
|
|
590
|
+
}
|
|
591
|
+
return;
|
|
592
|
+
}
|
|
593
|
+
}
|
|
594
|
+
}
|
|
595
|
+
console.log('[Simulator] playVideoInternal: starting', {
|
|
596
|
+
playerState: this.playerState,
|
|
597
|
+
currentTime: video.currentTime,
|
|
598
|
+
readyState: video.readyState
|
|
599
|
+
});
|
|
600
|
+
// Wait for video to be ready if needed
|
|
601
|
+
if (video.readyState < HTMLMediaElement.HAVE_METADATA) {
|
|
602
|
+
console.log('[Simulator] playVideoInternal: waiting for metadata');
|
|
603
|
+
await new Promise((resolve, reject) => {
|
|
604
|
+
const onLoadedMetadata = () => {
|
|
605
|
+
video.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
606
|
+
video.removeEventListener('error', onError);
|
|
607
|
+
resolve();
|
|
608
|
+
};
|
|
609
|
+
const onError = () => {
|
|
610
|
+
video.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
611
|
+
video.removeEventListener('error', onError);
|
|
612
|
+
reject(new Error('Failed to load video metadata'));
|
|
613
|
+
};
|
|
614
|
+
video.addEventListener('loadedmetadata', onLoadedMetadata, { once: true });
|
|
615
|
+
video.addEventListener('error', onError, { once: true });
|
|
616
|
+
// Timeout fallback
|
|
617
|
+
setTimeout(() => {
|
|
618
|
+
video.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
619
|
+
video.removeEventListener('error', onError);
|
|
620
|
+
if (video.readyState >= HTMLMediaElement.HAVE_METADATA) {
|
|
621
|
+
resolve();
|
|
622
|
+
}
|
|
623
|
+
else {
|
|
624
|
+
reject(new Error('Timeout waiting for metadata'));
|
|
625
|
+
}
|
|
626
|
+
}, 500);
|
|
297
627
|
});
|
|
298
628
|
}
|
|
629
|
+
// Clear hit markers if starting from beginning
|
|
630
|
+
const currentTimeMs = video.currentTime * 1000;
|
|
631
|
+
if (currentTimeMs < 100) {
|
|
632
|
+
this.hitMarkers.clear();
|
|
633
|
+
}
|
|
634
|
+
// Set state to loading
|
|
635
|
+
this.playerState = 'loading';
|
|
636
|
+
try {
|
|
637
|
+
// Verify video element is still available before playing
|
|
638
|
+
const currentVideo = this.vplayer?.nativeElement;
|
|
639
|
+
if (!currentVideo || currentVideo !== video) {
|
|
640
|
+
console.warn('[Simulator] playVideoInternal: video element changed or unavailable');
|
|
641
|
+
this.playerState = 'paused';
|
|
642
|
+
this.isPlaying = false;
|
|
643
|
+
this.playPromise = null;
|
|
644
|
+
return;
|
|
645
|
+
}
|
|
646
|
+
// Await play promise - this is critical
|
|
647
|
+
await video.play();
|
|
648
|
+
// Verify video element is still available after play promise resolves
|
|
649
|
+
const videoAfterPlay = this.vplayer?.nativeElement;
|
|
650
|
+
if (!videoAfterPlay || videoAfterPlay !== video) {
|
|
651
|
+
console.warn('[Simulator] playVideoInternal: video element changed during play');
|
|
652
|
+
this.playerState = 'paused';
|
|
653
|
+
this.isPlaying = false;
|
|
654
|
+
this.playPromise = null;
|
|
655
|
+
return;
|
|
656
|
+
}
|
|
657
|
+
// Play succeeded
|
|
658
|
+
this.playerState = 'playing';
|
|
659
|
+
this.isPlaying = true;
|
|
660
|
+
this.playPromise = null;
|
|
661
|
+
this.videoPlay.emit();
|
|
662
|
+
this.isVideoPlayingChange.emit(true);
|
|
663
|
+
console.log('[Simulator] playVideoInternal: playing successfully', {
|
|
664
|
+
currentTime: video.currentTime,
|
|
665
|
+
playerState: this.playerState
|
|
666
|
+
});
|
|
667
|
+
// Check early markers (non-blocking, doesn't need to be queued)
|
|
668
|
+
setTimeout(() => {
|
|
669
|
+
if (this.isPlaying) {
|
|
670
|
+
this.checkMarkerHit(this.globalCurrentTime);
|
|
671
|
+
}
|
|
672
|
+
}, 50);
|
|
673
|
+
}
|
|
674
|
+
catch (err) {
|
|
675
|
+
// Play failed
|
|
676
|
+
// Check if video element is still available
|
|
677
|
+
const videoAfterError = this.vplayer?.nativeElement;
|
|
678
|
+
if (!videoAfterError || videoAfterError !== video) {
|
|
679
|
+
console.warn('[Simulator] playVideoInternal: video element changed during error');
|
|
680
|
+
this.playerState = 'paused';
|
|
681
|
+
}
|
|
682
|
+
else {
|
|
683
|
+
this.playerState = 'error';
|
|
684
|
+
}
|
|
685
|
+
this.isPlaying = false;
|
|
686
|
+
this.playPromise = null;
|
|
687
|
+
this.isVideoPlayingChange.emit(false);
|
|
688
|
+
if (err.name === 'AbortError') {
|
|
689
|
+
console.log('[Simulator] playVideoInternal: aborted (expected during pause)');
|
|
690
|
+
// AbortError is expected, reset to paused state instead of error
|
|
691
|
+
if (videoAfterError && videoAfterError === video) {
|
|
692
|
+
this.playerState = 'paused';
|
|
693
|
+
}
|
|
694
|
+
}
|
|
695
|
+
else {
|
|
696
|
+
console.error('[Simulator] playVideoInternal: failed', err);
|
|
697
|
+
}
|
|
698
|
+
}
|
|
699
|
+
}
|
|
700
|
+
/**
|
|
701
|
+
* Pause video - public method (enqueues operation)
|
|
702
|
+
*/
|
|
703
|
+
pauseVideo() {
|
|
704
|
+
this.enqueueOperation(() => this.pauseVideoInternal());
|
|
705
|
+
}
|
|
706
|
+
/**
|
|
707
|
+
* Pause video internal implementation (no enqueue - called from queue)
|
|
708
|
+
*/
|
|
709
|
+
async pauseVideoInternal() {
|
|
710
|
+
const video = this.vplayer?.nativeElement;
|
|
711
|
+
if (!video)
|
|
712
|
+
return;
|
|
713
|
+
// Already paused - skip
|
|
714
|
+
if (this.playerState === 'paused') {
|
|
715
|
+
console.log('[Simulator] pauseVideoInternal: already paused, skipping');
|
|
716
|
+
return;
|
|
717
|
+
}
|
|
718
|
+
console.log('[Simulator] pauseVideoInternal: pausing', {
|
|
719
|
+
playerState: this.playerState,
|
|
720
|
+
currentTime: video.currentTime
|
|
721
|
+
});
|
|
722
|
+
video.pause();
|
|
723
|
+
this.playerState = 'paused';
|
|
724
|
+
this.isPlaying = false;
|
|
725
|
+
this.playPromise = null;
|
|
726
|
+
this.videoPause.emit();
|
|
727
|
+
this.isVideoPlayingChange.emit(false);
|
|
728
|
+
console.log('[Simulator] pauseVideoInternal: paused successfully');
|
|
729
|
+
}
|
|
730
|
+
/**
|
|
731
|
+
* Enqueue media operation to prevent race conditions
|
|
732
|
+
* This is the CRITICAL fix for FFmpegDemuxer errors and play promise interruptions
|
|
733
|
+
* All media operations (play, pause, seek, switch) go through this queue
|
|
734
|
+
*/
|
|
735
|
+
enqueueOperation(operation) {
|
|
736
|
+
this.operationQueue = this.operationQueue
|
|
737
|
+
.then(() => operation())
|
|
738
|
+
.catch(err => {
|
|
739
|
+
console.error('[Simulator] Media operation failed:', err);
|
|
740
|
+
this.playerState = 'error';
|
|
741
|
+
});
|
|
299
742
|
}
|
|
300
743
|
get hasMultipleVideos() {
|
|
301
744
|
return Array.isArray(this.videoUrls) && this.videoUrls.length > 1;
|
|
302
745
|
}
|
|
746
|
+
/** Cumulative duration boundaries [ms]. segmentBoundaries[i] = sum of durations [0..i] */
|
|
747
|
+
get segmentBoundaries() {
|
|
748
|
+
const durations = this.videoDurations;
|
|
749
|
+
if (durations.length === 0)
|
|
750
|
+
return [];
|
|
751
|
+
const boundaries = [];
|
|
752
|
+
let sum = 0;
|
|
753
|
+
for (const d of durations) {
|
|
754
|
+
sum += d;
|
|
755
|
+
boundaries.push(sum);
|
|
756
|
+
}
|
|
757
|
+
return boundaries;
|
|
758
|
+
}
|
|
759
|
+
/** Total duration across all segments in ms */
|
|
760
|
+
get totalDuration() {
|
|
761
|
+
const durations = this.videoDurations;
|
|
762
|
+
if (durations.length === 0)
|
|
763
|
+
return 0;
|
|
764
|
+
return durations.reduce((a, b) => a + b, 0);
|
|
765
|
+
}
|
|
766
|
+
/** Video durations in ms (detected or from single video element) */
|
|
767
|
+
get videoDurations() {
|
|
768
|
+
if (this.hasMultipleVideos && this.detectedVideoDurations.length > 0) {
|
|
769
|
+
return this.detectedVideoDurations;
|
|
770
|
+
}
|
|
771
|
+
const video = this.vplayer?.nativeElement;
|
|
772
|
+
if (video?.duration && isFinite(video.duration)) {
|
|
773
|
+
return [video.duration * 1000];
|
|
774
|
+
}
|
|
775
|
+
return [];
|
|
776
|
+
}
|
|
777
|
+
get globalCurrentTime() {
|
|
778
|
+
if (this.targetGlobalTimeDuringSwitch !== null && !this.isPlaying && this.playerState !== 'playing') {
|
|
779
|
+
return this.targetGlobalTimeDuringSwitch;
|
|
780
|
+
}
|
|
781
|
+
const video = this.vplayer?.nativeElement;
|
|
782
|
+
if (!video?.duration || !isFinite(video.duration)) {
|
|
783
|
+
// Fallback to override if video not ready yet
|
|
784
|
+
if (this.targetGlobalTimeDuringSwitch !== null) {
|
|
785
|
+
return this.targetGlobalTimeDuringSwitch;
|
|
786
|
+
}
|
|
787
|
+
return 0;
|
|
788
|
+
}
|
|
789
|
+
const segmentStart = this.currentVideoIndex === 0 ? 0 : (this.segmentBoundaries[this.currentVideoIndex - 1] ?? 0);
|
|
790
|
+
return segmentStart + video.currentTime * 1000;
|
|
791
|
+
}
|
|
792
|
+
getActualGlobalTime() {
|
|
793
|
+
const video = this.vplayer?.nativeElement;
|
|
794
|
+
if (!video?.duration || !isFinite(video.duration))
|
|
795
|
+
return 0;
|
|
796
|
+
const segmentStart = this.currentVideoIndex === 0 ? 0 : (this.segmentBoundaries[this.currentVideoIndex - 1] ?? 0);
|
|
797
|
+
return segmentStart + video.currentTime * 1000;
|
|
798
|
+
}
|
|
799
|
+
/** Global step markers for rendering on the combined timeline */
|
|
800
|
+
get globalStepMarkers() {
|
|
801
|
+
const allMarkers = this.flattenMarkers(this.stepMarkers);
|
|
802
|
+
return allMarkers.map(m => ({
|
|
803
|
+
globalTime: m.cumulativeDuration,
|
|
804
|
+
result: m.result,
|
|
805
|
+
testStepId: m.testStepId,
|
|
806
|
+
title: m.title,
|
|
807
|
+
level: m.level
|
|
808
|
+
}));
|
|
809
|
+
}
|
|
303
810
|
get currentVideoUrl() {
|
|
304
811
|
if (Array.isArray(this.videoUrls) && this.videoUrls.length > 0) {
|
|
305
812
|
const idx = Math.min(Math.max(this.currentVideoIndex, 0), this.videoUrls.length - 1);
|
|
@@ -307,14 +814,30 @@ export class SimulatorComponent {
|
|
|
307
814
|
}
|
|
308
815
|
return this.videoUrl;
|
|
309
816
|
}
|
|
817
|
+
/**
|
|
818
|
+
* Reset video state - public method (enqueues operation)
|
|
819
|
+
*/
|
|
310
820
|
resetVideoState() {
|
|
821
|
+
this.enqueueOperation(() => this.resetVideoStateInternal());
|
|
822
|
+
}
|
|
823
|
+
/**
|
|
824
|
+
* Reset video state internal implementation (no enqueue - called from queue)
|
|
825
|
+
*/
|
|
826
|
+
async resetVideoStateInternal() {
|
|
827
|
+
const video = this.vplayer?.nativeElement;
|
|
828
|
+
if (!video)
|
|
829
|
+
return;
|
|
830
|
+
console.log('[Simulator] resetVideoStateInternal: resetting');
|
|
831
|
+
video.pause();
|
|
832
|
+
video.currentTime = 0;
|
|
833
|
+
this.playerState = 'paused';
|
|
311
834
|
this.isPlaying = false;
|
|
312
835
|
this.progress = 0;
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
this.vplayer.nativeElement.currentTime = 0;
|
|
316
|
-
}
|
|
836
|
+
this.playPromise = null;
|
|
837
|
+
this.hitMarkers.clear();
|
|
317
838
|
this.lastSetDuration = -1;
|
|
839
|
+
this.isVideoPlayingChange.emit(false);
|
|
840
|
+
console.log('[Simulator] resetVideoStateInternal: complete');
|
|
318
841
|
}
|
|
319
842
|
prevVideo() {
|
|
320
843
|
if (!this.hasMultipleVideos)
|
|
@@ -333,22 +856,45 @@ export class SimulatorComponent {
|
|
|
333
856
|
}
|
|
334
857
|
}
|
|
335
858
|
onTimelineClick(event) {
|
|
336
|
-
if (!this.timelineBar?.nativeElement
|
|
859
|
+
if (!this.timelineBar?.nativeElement)
|
|
337
860
|
return;
|
|
338
861
|
const rect = this.timelineBar.nativeElement.getBoundingClientRect();
|
|
339
862
|
const clickX = event.clientX - rect.left;
|
|
340
863
|
const percent = Math.max(0, Math.min(1, clickX / rect.width));
|
|
341
|
-
const
|
|
342
|
-
if (
|
|
864
|
+
const total = this.totalDuration;
|
|
865
|
+
if (total <= 0)
|
|
343
866
|
return;
|
|
344
|
-
|
|
345
|
-
const
|
|
346
|
-
|
|
347
|
-
|
|
867
|
+
const globalTimeMs = percent * total;
|
|
868
|
+
const shouldResumePlaying = this.isPlaying;
|
|
869
|
+
console.log('[Simulator] onTimelineClick', {
|
|
870
|
+
percent,
|
|
871
|
+
globalTimeMs,
|
|
872
|
+
shouldResumePlaying
|
|
873
|
+
});
|
|
874
|
+
this.enqueueOperation(async () => {
|
|
875
|
+
await this.seekToGlobalTime(globalTimeMs);
|
|
876
|
+
if (shouldResumePlaying) {
|
|
877
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
878
|
+
await this.playVideoInternal();
|
|
879
|
+
}
|
|
880
|
+
});
|
|
348
881
|
}
|
|
349
882
|
startDrag(event) {
|
|
350
883
|
event.preventDefault();
|
|
351
884
|
this.dragging = true;
|
|
885
|
+
// CRITICAL: Capture playing state BEFORE pausing
|
|
886
|
+
// Use both isPlaying flag and playerState for reliability
|
|
887
|
+
this.wasPlayingBeforeDrag = this.isPlaying || this.playerState === 'playing';
|
|
888
|
+
console.log('[Simulator] startDrag', {
|
|
889
|
+
clientX: event.clientX,
|
|
890
|
+
playerState: this.playerState,
|
|
891
|
+
isPlaying: this.isPlaying,
|
|
892
|
+
wasPlayingBeforeDrag: this.wasPlayingBeforeDrag
|
|
893
|
+
});
|
|
894
|
+
// Pause video during drag to avoid conflicts
|
|
895
|
+
if (this.isPlaying || this.playerState === 'playing') {
|
|
896
|
+
this.pauseVideo();
|
|
897
|
+
}
|
|
352
898
|
this.addDragListeners();
|
|
353
899
|
}
|
|
354
900
|
onDrag(event) {
|
|
@@ -357,22 +903,64 @@ export class SimulatorComponent {
|
|
|
357
903
|
const rect = this.timelineBar.nativeElement.getBoundingClientRect();
|
|
358
904
|
const x = event.clientX - rect.left;
|
|
359
905
|
const percent = Math.max(0, Math.min(x / rect.width, 1));
|
|
360
|
-
this.progress = percent * 100;
|
|
906
|
+
this.progress = percent * 100; // Global timeline position
|
|
361
907
|
}
|
|
362
908
|
stopDrag() {
|
|
363
909
|
if (!this.dragging)
|
|
364
910
|
return;
|
|
365
911
|
this.dragging = false;
|
|
366
912
|
this.removeDragListeners();
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
913
|
+
// CRITICAL: Capture the playing state BEFORE any async operations
|
|
914
|
+
const shouldResumePlaying = this.wasPlayingBeforeDrag;
|
|
915
|
+
console.log('[Simulator] stopDrag: drag ended', {
|
|
916
|
+
progress: this.progress,
|
|
917
|
+
wasPlayingBeforeDrag: this.wasPlayingBeforeDrag,
|
|
918
|
+
shouldResumePlaying: shouldResumePlaying,
|
|
919
|
+
currentPlayerState: this.playerState
|
|
920
|
+
});
|
|
921
|
+
const total = this.totalDuration;
|
|
922
|
+
if (total <= 0)
|
|
923
|
+
return;
|
|
924
|
+
const globalTimeMs = (this.progress / 100) * total;
|
|
925
|
+
// CRITICAL FIX: Queue both seek and resume play together
|
|
926
|
+
// This ensures seek completes before attempting to play
|
|
927
|
+
this.enqueueOperation(async () => {
|
|
928
|
+
// Perform seek (this will pause the video)
|
|
929
|
+
await this.seekToGlobalTime(globalTimeMs);
|
|
930
|
+
// If video was playing before drag, resume it
|
|
931
|
+
if (shouldResumePlaying) {
|
|
932
|
+
console.log('[Simulator] stopDrag: resuming playback after seek', {
|
|
933
|
+
globalTimeMs,
|
|
934
|
+
playerState: this.playerState
|
|
935
|
+
});
|
|
936
|
+
// Wait a bit longer to ensure seek is fully complete
|
|
937
|
+
await new Promise(resolve => setTimeout(resolve, 150));
|
|
938
|
+
// Verify state is 'paused' after seek (seekToTimeInternal sets it to 'paused')
|
|
939
|
+
// If state is still 'seeking', wait a bit more
|
|
940
|
+
if (this.playerState === 'seeking') {
|
|
941
|
+
console.log('[Simulator] stopDrag: seek still in progress, waiting more');
|
|
942
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
943
|
+
}
|
|
944
|
+
// Now resume playback (state should be 'paused' at this point)
|
|
945
|
+
if (this.playerState === 'paused') {
|
|
946
|
+
await this.playVideoInternal();
|
|
947
|
+
console.log('[Simulator] stopDrag: playback resumed successfully', {
|
|
948
|
+
playerState: this.playerState,
|
|
949
|
+
isPlaying: this.isPlaying
|
|
950
|
+
});
|
|
951
|
+
}
|
|
952
|
+
else {
|
|
953
|
+
console.warn('[Simulator] stopDrag: unexpected state after seek, attempting to resume anyway', {
|
|
954
|
+
playerState: this.playerState
|
|
955
|
+
});
|
|
956
|
+
// Try to resume anyway - better to try than to leave it stuck
|
|
957
|
+
await this.playVideoInternal();
|
|
958
|
+
}
|
|
371
959
|
}
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
}
|
|
960
|
+
else {
|
|
961
|
+
console.log('[Simulator] stopDrag: video was paused, not resuming');
|
|
962
|
+
}
|
|
963
|
+
});
|
|
376
964
|
}
|
|
377
965
|
addDragListeners() {
|
|
378
966
|
this.removeDragListeners();
|
|
@@ -396,14 +984,33 @@ export class SimulatorComponent {
|
|
|
396
984
|
}
|
|
397
985
|
}
|
|
398
986
|
onVideoMetadataLoaded() {
|
|
987
|
+
console.log('[Simulator] onVideoMetadataLoaded');
|
|
988
|
+
// Ensure playback speed is consistent across all videos
|
|
989
|
+
this.applyCurrentPlaybackRate();
|
|
399
990
|
this.attachVideoListeners();
|
|
400
991
|
}
|
|
401
992
|
onVideoCanPlay() {
|
|
993
|
+
console.log('[Simulator] onVideoCanPlay');
|
|
994
|
+
// Ensure playback speed is consistent across all videos
|
|
995
|
+
this.applyCurrentPlaybackRate();
|
|
402
996
|
this.attachVideoListeners();
|
|
403
997
|
}
|
|
404
998
|
onVideoEnded() {
|
|
405
999
|
this.isPlaying = false;
|
|
1000
|
+
this.playerState = 'paused';
|
|
406
1001
|
this.videoPause.emit();
|
|
1002
|
+
this.isVideoPlayingChange.emit(false);
|
|
1003
|
+
// Seamlessly load and play next segment
|
|
1004
|
+
if (this.hasMultipleVideos && this.videoUrls && this.currentVideoIndex < this.videoUrls.length - 1) {
|
|
1005
|
+
const nextIndex = this.currentVideoIndex + 1;
|
|
1006
|
+
const segmentStart = nextIndex === 0 ? 0 : (this.segmentBoundaries[nextIndex - 1] ?? 0);
|
|
1007
|
+
this.enqueueOperation(async () => {
|
|
1008
|
+
await this.switchToVideoAndSeekInternal(segmentStart);
|
|
1009
|
+
if (this.vplayer?.nativeElement && this.currentVideoUrl) {
|
|
1010
|
+
await this.playVideoInternal();
|
|
1011
|
+
}
|
|
1012
|
+
});
|
|
1013
|
+
}
|
|
407
1014
|
}
|
|
408
1015
|
attachVideoListeners() {
|
|
409
1016
|
if (this.videoEventListenerCleanup) {
|
|
@@ -421,10 +1028,26 @@ export class SimulatorComponent {
|
|
|
421
1028
|
return;
|
|
422
1029
|
lastUpdate = now;
|
|
423
1030
|
const currentMs = video.currentTime * 1000;
|
|
424
|
-
if (!this.dragging) {
|
|
425
|
-
const
|
|
426
|
-
|
|
427
|
-
|
|
1031
|
+
if (!this.dragging && this.playerState !== 'switching') {
|
|
1032
|
+
const total = this.totalDuration;
|
|
1033
|
+
// Clear targetGlobalTimeDuringSwitch if we've reached the target position
|
|
1034
|
+
// When playing, be more aggressive about clearing to allow live updates
|
|
1035
|
+
if (this.targetGlobalTimeDuringSwitch !== null) {
|
|
1036
|
+
const currentGlobal = this.getActualGlobalTime();
|
|
1037
|
+
const tolerance = this.isPlaying ? 1000 : 500; // More tolerance when playing
|
|
1038
|
+
if (Math.abs(currentGlobal - this.targetGlobalTimeDuringSwitch) < tolerance) {
|
|
1039
|
+
this.targetGlobalTimeDuringSwitch = null;
|
|
1040
|
+
}
|
|
1041
|
+
}
|
|
1042
|
+
// globalCurrentTime now handles override logic - use it directly
|
|
1043
|
+
if (total > 0) {
|
|
1044
|
+
this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / total) * 100));
|
|
1045
|
+
}
|
|
1046
|
+
this.videoTimeUpdate.emit(this.globalCurrentTime);
|
|
1047
|
+
// Check if current time hits any marker
|
|
1048
|
+
this.checkMarkerHit(this.globalCurrentTime);
|
|
1049
|
+
// Preload next segment when ~5-10 seconds from ending
|
|
1050
|
+
this.maybePreloadNextSegment();
|
|
428
1051
|
}
|
|
429
1052
|
};
|
|
430
1053
|
video.addEventListener('timeupdate', handler);
|
|
@@ -433,28 +1056,323 @@ export class SimulatorComponent {
|
|
|
433
1056
|
};
|
|
434
1057
|
}, 100);
|
|
435
1058
|
}
|
|
436
|
-
|
|
437
|
-
if (!
|
|
438
|
-
return
|
|
439
|
-
|
|
1059
|
+
schedulePreloadAllSegments(videoUrls) {
|
|
1060
|
+
if (!videoUrls || videoUrls.length === 0)
|
|
1061
|
+
return;
|
|
1062
|
+
this.cancelPreloadAllSegments();
|
|
1063
|
+
setTimeout(() => this.preloadAllSegments(videoUrls), 500);
|
|
1064
|
+
}
|
|
1065
|
+
cancelPreloadAllSegments() {
|
|
1066
|
+
if (this.preloadAllVideoElement) {
|
|
1067
|
+
this.preloadAllVideoElement.src = '';
|
|
1068
|
+
this.preloadAllVideoElement.remove();
|
|
1069
|
+
this.preloadAllVideoElement = null;
|
|
1070
|
+
}
|
|
1071
|
+
}
|
|
1072
|
+
/**
|
|
1073
|
+
* Preload all video URLs in sequence (first, then second, ...) so they are cached.
|
|
1074
|
+
* When user jumps to any segment, that video is likely already loaded.
|
|
1075
|
+
*/
|
|
1076
|
+
preloadAllSegments(videoUrls) {
|
|
1077
|
+
if (!videoUrls || videoUrls.length === 0)
|
|
1078
|
+
return;
|
|
1079
|
+
this.cancelPreloadAllSegments();
|
|
1080
|
+
const video = document.createElement('video');
|
|
1081
|
+
video.preload = 'auto';
|
|
1082
|
+
video.muted = true;
|
|
1083
|
+
video.playsInline = true;
|
|
1084
|
+
video.style.display = 'none';
|
|
1085
|
+
document.body.appendChild(video);
|
|
1086
|
+
this.preloadAllVideoElement = video;
|
|
1087
|
+
let index = 0;
|
|
1088
|
+
const loadNext = () => {
|
|
1089
|
+
if (this.preloadAllVideoElement !== video || index >= videoUrls.length) {
|
|
1090
|
+
video.remove();
|
|
1091
|
+
if (this.preloadAllVideoElement === video)
|
|
1092
|
+
this.preloadAllVideoElement = null;
|
|
1093
|
+
return;
|
|
1094
|
+
}
|
|
1095
|
+
const url = videoUrls[index];
|
|
1096
|
+
video.src = url;
|
|
1097
|
+
index += 1;
|
|
1098
|
+
const onDone = () => {
|
|
1099
|
+
clearTimeout(timeoutId);
|
|
1100
|
+
video.removeEventListener('loadeddata', onDone);
|
|
1101
|
+
video.removeEventListener('error', onDone);
|
|
1102
|
+
loadNext();
|
|
1103
|
+
};
|
|
1104
|
+
video.addEventListener('loadeddata', onDone, { once: true });
|
|
1105
|
+
video.addEventListener('error', onDone, { once: true });
|
|
1106
|
+
const timeoutId = setTimeout(() => onDone(), 15000);
|
|
1107
|
+
};
|
|
1108
|
+
loadNext();
|
|
1109
|
+
}
|
|
1110
|
+
/** Preload next segment when ~5-10 seconds from ending to minimize switching delay */
|
|
1111
|
+
maybePreloadNextSegment() {
|
|
1112
|
+
if (!this.hasMultipleVideos || !this.videoUrls || this.currentVideoIndex >= this.videoUrls.length - 1) {
|
|
1113
|
+
return;
|
|
1114
|
+
}
|
|
1115
|
+
const video = this.vplayer?.nativeElement;
|
|
1116
|
+
if (!video?.duration)
|
|
1117
|
+
return;
|
|
1118
|
+
const timeLeftInSegment = (video.duration - video.currentTime) * 1000; // ms
|
|
1119
|
+
if (timeLeftInSegment > 10000 || timeLeftInSegment < 5000)
|
|
1120
|
+
return; // Preload when 5-10s from end
|
|
1121
|
+
const nextUrl = this.videoUrls[this.currentVideoIndex + 1];
|
|
1122
|
+
if (!nextUrl)
|
|
1123
|
+
return;
|
|
1124
|
+
if (this.preloadVideoElement?.src === nextUrl)
|
|
1125
|
+
return; // Already preloading
|
|
1126
|
+
this.preloadNextSegment(nextUrl);
|
|
1127
|
+
}
|
|
1128
|
+
preloadNextSegment(url) {
|
|
1129
|
+
if (this.preloadVideoElement) {
|
|
1130
|
+
this.preloadVideoElement.src = '';
|
|
1131
|
+
this.preloadVideoElement.remove();
|
|
1132
|
+
this.preloadVideoElement = null;
|
|
1133
|
+
}
|
|
1134
|
+
const video = document.createElement('video');
|
|
1135
|
+
video.preload = 'auto';
|
|
1136
|
+
video.src = url;
|
|
1137
|
+
video.style.display = 'none';
|
|
1138
|
+
document.body.appendChild(video);
|
|
1139
|
+
this.preloadVideoElement = video;
|
|
1140
|
+
video.addEventListener('loadeddata', () => {
|
|
1141
|
+
video.remove();
|
|
1142
|
+
if (this.preloadVideoElement === video)
|
|
1143
|
+
this.preloadVideoElement = null;
|
|
1144
|
+
}, { once: true });
|
|
1145
|
+
}
|
|
1146
|
+
/**
|
|
1147
|
+
* Detect actual video durations by loading video metadata
|
|
1148
|
+
* This creates temporary video elements to read duration without playing
|
|
1149
|
+
*/
|
|
1150
|
+
detectVideoDurations(videoUrls) {
|
|
1151
|
+
if (!videoUrls || videoUrls.length === 0) {
|
|
1152
|
+
this.detectedVideoDurations = [];
|
|
1153
|
+
return;
|
|
1154
|
+
}
|
|
1155
|
+
// Reset detected durations
|
|
1156
|
+
this.detectedVideoDurations = [];
|
|
1157
|
+
let loadedCount = 0;
|
|
1158
|
+
videoUrls.forEach((url, index) => {
|
|
1159
|
+
const tempVideo = document.createElement('video');
|
|
1160
|
+
tempVideo.preload = 'metadata';
|
|
1161
|
+
const onLoadedMetadata = () => {
|
|
1162
|
+
const durationMs = tempVideo.duration * 1000;
|
|
1163
|
+
this.detectedVideoDurations[index] = durationMs;
|
|
1164
|
+
loadedCount++;
|
|
1165
|
+
// Clean up
|
|
1166
|
+
tempVideo.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
1167
|
+
tempVideo.removeEventListener('error', onError);
|
|
1168
|
+
tempVideo.src = '';
|
|
1169
|
+
tempVideo.remove();
|
|
1170
|
+
if (loadedCount === videoUrls.length) {
|
|
1171
|
+
console.log('[Simulator] All video durations detected:', this.detectedVideoDurations);
|
|
1172
|
+
}
|
|
1173
|
+
};
|
|
1174
|
+
const onError = () => {
|
|
1175
|
+
console.warn(`[Simulator] Failed to load metadata for video ${index + 1}`);
|
|
1176
|
+
this.detectedVideoDurations[index] = 0;
|
|
1177
|
+
loadedCount++;
|
|
1178
|
+
tempVideo.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
1179
|
+
tempVideo.removeEventListener('error', onError);
|
|
1180
|
+
tempVideo.src = '';
|
|
1181
|
+
tempVideo.remove();
|
|
1182
|
+
};
|
|
1183
|
+
tempVideo.addEventListener('loadedmetadata', onLoadedMetadata);
|
|
1184
|
+
tempVideo.addEventListener('error', onError);
|
|
1185
|
+
tempVideo.src = url;
|
|
1186
|
+
});
|
|
1187
|
+
}
|
|
1188
|
+
/**
|
|
1189
|
+
* Get cumulative duration boundaries for each video
|
|
1190
|
+
* Returns an array where each element is [startDuration, endDuration] for that video
|
|
1191
|
+
* Uses detected video durations if available, otherwise falls back to videoDurations input
|
|
1192
|
+
*/
|
|
1193
|
+
getVideoDurationBoundaries() {
|
|
1194
|
+
if (!this.hasMultipleVideos) {
|
|
1195
|
+
return [];
|
|
1196
|
+
}
|
|
1197
|
+
// Use detected durations if available, otherwise use provided durations
|
|
1198
|
+
const durations = this.detectedVideoDurations.length > 0
|
|
1199
|
+
? this.detectedVideoDurations
|
|
1200
|
+
: [];
|
|
1201
|
+
if (durations.length === 0) {
|
|
1202
|
+
return [];
|
|
1203
|
+
}
|
|
1204
|
+
const boundaries = [];
|
|
1205
|
+
let cumulativeStart = 0;
|
|
1206
|
+
for (const duration of durations) {
|
|
1207
|
+
boundaries.push({
|
|
1208
|
+
start: cumulativeStart,
|
|
1209
|
+
end: cumulativeStart + duration
|
|
1210
|
+
});
|
|
1211
|
+
cumulativeStart += duration;
|
|
1212
|
+
}
|
|
1213
|
+
return boundaries;
|
|
440
1214
|
}
|
|
441
|
-
|
|
442
|
-
|
|
1215
|
+
/**
|
|
1216
|
+
* Flatten all markers recursively including childSteps
|
|
1217
|
+
*/
|
|
1218
|
+
flattenMarkers(markers) {
|
|
1219
|
+
const flattened = [];
|
|
1220
|
+
const flattenRecursive = (markers) => {
|
|
1221
|
+
markers.forEach(marker => {
|
|
1222
|
+
// Add the marker itself
|
|
1223
|
+
flattened.push(marker);
|
|
1224
|
+
// Recursively add child markers
|
|
1225
|
+
if (marker.childSteps && marker.childSteps.length > 0) {
|
|
1226
|
+
flattenRecursive(marker.childSteps);
|
|
1227
|
+
}
|
|
1228
|
+
});
|
|
1229
|
+
};
|
|
1230
|
+
flattenRecursive(markers);
|
|
1231
|
+
return flattened;
|
|
1232
|
+
}
|
|
1233
|
+
checkMarkerHit(globalTimeMs) {
|
|
1234
|
+
if (!this.isPlaying || this.dragging)
|
|
1235
|
+
return;
|
|
1236
|
+
// Check all markers (including nested ones) for a match
|
|
1237
|
+
const allMarkers = this.flattenMarkers(this.stepMarkers);
|
|
1238
|
+
const tolerance = 200; // 200ms tolerance for matching
|
|
1239
|
+
for (const marker of allMarkers) {
|
|
1240
|
+
// Skip if marker doesn't have testStepId
|
|
1241
|
+
if (!marker.testStepId)
|
|
1242
|
+
continue;
|
|
1243
|
+
// Skip if already hit
|
|
1244
|
+
if (this.hitMarkers.has(marker.testStepId))
|
|
1245
|
+
continue;
|
|
1246
|
+
// Check if current time matches marker time (within tolerance)
|
|
1247
|
+
const timeDifference = Math.abs(globalTimeMs - marker.cumulativeDuration);
|
|
1248
|
+
if (timeDifference <= tolerance) {
|
|
1249
|
+
// Mark as hit and emit event
|
|
1250
|
+
this.hitMarkers.add(marker.testStepId);
|
|
1251
|
+
this.markerHit.emit(marker);
|
|
1252
|
+
console.log('[Simulator] Marker hit detected', {
|
|
1253
|
+
testStepId: marker.testStepId,
|
|
1254
|
+
cumulativeDuration: marker.cumulativeDuration,
|
|
1255
|
+
currentTime: globalTimeMs,
|
|
1256
|
+
timeDifference,
|
|
1257
|
+
title: marker.title,
|
|
1258
|
+
level: marker.level
|
|
1259
|
+
});
|
|
1260
|
+
break; // Only emit one marker per check
|
|
1261
|
+
}
|
|
1262
|
+
}
|
|
1263
|
+
}
|
|
1264
|
+
getGlobalMarkerResultColor(result) {
|
|
1265
|
+
switch (result) {
|
|
1266
|
+
case 'SUCCESS':
|
|
1267
|
+
return '#12B76A';
|
|
1268
|
+
case 'FAILURE':
|
|
1269
|
+
return '#B91C1C';
|
|
1270
|
+
case 'ABORTED':
|
|
1271
|
+
return '#F79009';
|
|
1272
|
+
case 'SKIPPED':
|
|
1273
|
+
return '#667085';
|
|
1274
|
+
default:
|
|
1275
|
+
return '#6366F1';
|
|
1276
|
+
}
|
|
1277
|
+
}
|
|
1278
|
+
getGlobalMarkerColor(level) {
|
|
1279
|
+
return level === 1 ? '#000000' : '#FFA500';
|
|
1280
|
+
}
|
|
1281
|
+
onMarkerClick(event, marker) {
|
|
1282
|
+
event.stopPropagation();
|
|
1283
|
+
event.preventDefault();
|
|
1284
|
+
this.enqueueOperation(async () => {
|
|
1285
|
+
await this.seekToGlobalTime(marker.globalTime);
|
|
1286
|
+
if (marker.testStepId != null) {
|
|
1287
|
+
this.hitMarkers.add(marker.testStepId);
|
|
1288
|
+
const fullMarker = this.flattenMarkers(this.stepMarkers).find(m => m.testStepId === marker.testStepId);
|
|
1289
|
+
if (fullMarker)
|
|
1290
|
+
this.markerHit.emit(fullMarker);
|
|
1291
|
+
}
|
|
1292
|
+
});
|
|
443
1293
|
}
|
|
444
1294
|
toggleFullScreen() {
|
|
1295
|
+
const wasPlaying = this.isPlaying;
|
|
1296
|
+
const currentTime = this.vplayer?.nativeElement?.currentTime || 0;
|
|
1297
|
+
// Pause video before fullscreen transition to avoid state issues
|
|
1298
|
+
if (wasPlaying) {
|
|
1299
|
+
this.pauseVideo();
|
|
1300
|
+
}
|
|
1301
|
+
// Toggle fullscreen
|
|
445
1302
|
this.isFullScreen = !this.isFullScreen;
|
|
1303
|
+
// Reset player state to allow re-initialization after DOM update
|
|
1304
|
+
// The state might be stuck in 'loading' or 'error' after fullscreen transition
|
|
1305
|
+
if (this.playerState === 'loading' || this.playerState === 'error') {
|
|
1306
|
+
this.playerState = 'idle';
|
|
1307
|
+
}
|
|
1308
|
+
// Wait for Angular to update the DOM, then re-initialize video
|
|
1309
|
+
setTimeout(() => {
|
|
1310
|
+
const video = this.vplayer?.nativeElement;
|
|
1311
|
+
if (video) {
|
|
1312
|
+
// Restore video position if it was set
|
|
1313
|
+
if (currentTime > 0 && video.duration > currentTime) {
|
|
1314
|
+
video.currentTime = currentTime;
|
|
1315
|
+
}
|
|
1316
|
+
// Re-apply playback rate
|
|
1317
|
+
this.applyCurrentPlaybackRate();
|
|
1318
|
+
// Re-attach listeners
|
|
1319
|
+
this.attachVideoListeners();
|
|
1320
|
+
// Reset state to 'paused' (ready to play)
|
|
1321
|
+
this.playerState = 'paused';
|
|
1322
|
+
this.isPlaying = false;
|
|
1323
|
+
console.log('[Simulator] toggleFullScreen: video re-initialized', {
|
|
1324
|
+
isFullScreen: this.isFullScreen,
|
|
1325
|
+
currentTime: video.currentTime,
|
|
1326
|
+
playerState: this.playerState
|
|
1327
|
+
});
|
|
1328
|
+
}
|
|
1329
|
+
else {
|
|
1330
|
+
console.warn('[Simulator] toggleFullScreen: video element not found after transition');
|
|
1331
|
+
// Try again after a bit more time
|
|
1332
|
+
setTimeout(() => {
|
|
1333
|
+
const videoRetry = this.vplayer?.nativeElement;
|
|
1334
|
+
if (videoRetry) {
|
|
1335
|
+
this.applyCurrentPlaybackRate();
|
|
1336
|
+
this.attachVideoListeners();
|
|
1337
|
+
this.playerState = 'paused';
|
|
1338
|
+
this.isPlaying = false;
|
|
1339
|
+
}
|
|
1340
|
+
}, 100);
|
|
1341
|
+
}
|
|
1342
|
+
}, 100);
|
|
446
1343
|
}
|
|
447
1344
|
onSegmentChange(value) {
|
|
448
1345
|
this.currentView = value;
|
|
449
|
-
|
|
450
|
-
this.
|
|
451
|
-
|
|
1346
|
+
// Pause video when switching away from video view
|
|
1347
|
+
if (this.currentView !== 'video') {
|
|
1348
|
+
this.pauseVideo();
|
|
1349
|
+
this.progress = 0;
|
|
1350
|
+
}
|
|
1351
|
+
else {
|
|
1352
|
+
// Reset everything when switching back to video view
|
|
1353
|
+
this.enqueueOperation(async () => {
|
|
1354
|
+
await this.resetVideoStateInternal();
|
|
1355
|
+
// If multiple videos, reset to first video
|
|
1356
|
+
if (this.hasMultipleVideos && this.videoUrls) {
|
|
1357
|
+
this.currentVideoIndex = 0;
|
|
1358
|
+
this.targetGlobalTimeDuringSwitch = 0;
|
|
1359
|
+
// Switch to first video and seek to 0
|
|
1360
|
+
await this.switchToVideoAndSeekInternal(0);
|
|
1361
|
+
}
|
|
1362
|
+
});
|
|
1363
|
+
}
|
|
452
1364
|
}
|
|
453
1365
|
formatTime(seconds) {
|
|
454
1366
|
if (!seconds || isNaN(seconds))
|
|
455
1367
|
return '00:00';
|
|
456
|
-
const
|
|
457
|
-
const
|
|
1368
|
+
const totalSeconds = Math.floor(seconds);
|
|
1369
|
+
const hours = Math.floor(totalSeconds / 3600);
|
|
1370
|
+
const mins = Math.floor((totalSeconds % 3600) / 60);
|
|
1371
|
+
const secs = totalSeconds % 60;
|
|
1372
|
+
// If hours > 0, show HH:MM:SS format, otherwise show MM:SS format
|
|
1373
|
+
if (hours > 0) {
|
|
1374
|
+
return `${hours.toString().padStart(2, '0')}:${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;
|
|
1375
|
+
}
|
|
458
1376
|
return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;
|
|
459
1377
|
}
|
|
460
1378
|
getStatusBadgeClass() {
|
|
@@ -531,11 +1449,8 @@ export class SimulatorComponent {
|
|
|
531
1449
|
}
|
|
532
1450
|
onSpeedChange(value) {
|
|
533
1451
|
this.currentSpeed = value;
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
if (this.vplayer?.nativeElement) {
|
|
537
|
-
this.vplayer.nativeElement.playbackRate = rate;
|
|
538
|
-
}
|
|
1452
|
+
// Whenever speed changes, apply it to the current video
|
|
1453
|
+
this.applyCurrentPlaybackRate();
|
|
539
1454
|
this.isSpeedControlOpen = false;
|
|
540
1455
|
}
|
|
541
1456
|
toggleSpeedControl() {
|
|
@@ -586,19 +1501,239 @@ export class SimulatorComponent {
|
|
|
586
1501
|
}
|
|
587
1502
|
onVideoElementReady() {
|
|
588
1503
|
const video = this._vplayer?.nativeElement;
|
|
1504
|
+
if (!video)
|
|
1505
|
+
return;
|
|
1506
|
+
console.log('[Simulator] onVideoElementReady: video element ready', {
|
|
1507
|
+
playerState: this.playerState,
|
|
1508
|
+
isFullScreen: this.isFullScreen,
|
|
1509
|
+
readyState: video.readyState
|
|
1510
|
+
});
|
|
1511
|
+
// Reset error/loading states when video element is recreated (e.g., during fullscreen toggle)
|
|
1512
|
+
if (this.playerState === 'error' || this.playerState === 'loading') {
|
|
1513
|
+
console.log('[Simulator] onVideoElementReady: resetting error/loading state');
|
|
1514
|
+
this.playerState = 'idle';
|
|
1515
|
+
this.isPlaying = false;
|
|
1516
|
+
this.playPromise = null;
|
|
1517
|
+
}
|
|
1518
|
+
// Apply current playback speed when the video element is first ready
|
|
1519
|
+
this.applyCurrentPlaybackRate();
|
|
1520
|
+
this.attachVideoListeners();
|
|
1521
|
+
}
|
|
1522
|
+
applyCurrentPlaybackRate() {
|
|
1523
|
+
const video = this.vplayer?.nativeElement;
|
|
589
1524
|
if (!video)
|
|
590
1525
|
return;
|
|
591
1526
|
const numeric = parseFloat(this.currentSpeed.replace('x', ''));
|
|
592
1527
|
const rate = !isNaN(numeric) && numeric > 0 ? numeric : 1;
|
|
593
1528
|
video.playbackRate = rate;
|
|
594
|
-
|
|
1529
|
+
}
|
|
1530
|
+
findVideoIndexForTimestamp(cumulativeTimestamp) {
|
|
1531
|
+
if (!this.hasMultipleVideos) {
|
|
1532
|
+
return null;
|
|
1533
|
+
}
|
|
1534
|
+
const boundaries = this.getVideoDurationBoundaries();
|
|
1535
|
+
if (boundaries.length === 0) {
|
|
1536
|
+
return null;
|
|
1537
|
+
}
|
|
1538
|
+
// Find which video contains this timestamp
|
|
1539
|
+
for (let i = 0; i < boundaries.length; i++) {
|
|
1540
|
+
const boundary = boundaries[i];
|
|
1541
|
+
if (cumulativeTimestamp >= boundary.start && cumulativeTimestamp < boundary.end) {
|
|
1542
|
+
return i;
|
|
1543
|
+
}
|
|
1544
|
+
}
|
|
1545
|
+
// If timestamp is exactly at the end of the last video, return the last video index
|
|
1546
|
+
if (boundaries.length > 0) {
|
|
1547
|
+
const lastBoundary = boundaries[boundaries.length - 1];
|
|
1548
|
+
if (cumulativeTimestamp === lastBoundary.end) {
|
|
1549
|
+
return boundaries.length - 1;
|
|
1550
|
+
}
|
|
1551
|
+
}
|
|
1552
|
+
return null;
|
|
1553
|
+
}
|
|
1554
|
+
switchToVideoAndSeek(cumulativeTimestamp) {
|
|
1555
|
+
this.enqueueOperation(() => this.switchToVideoAndSeekInternal(cumulativeTimestamp));
|
|
1556
|
+
}
|
|
1557
|
+
async switchToVideoAndSeekInternal(cumulativeTimestamp) {
|
|
1558
|
+
if (!this.hasMultipleVideos) {
|
|
1559
|
+
await this.seekToTimeInternal(cumulativeTimestamp);
|
|
1560
|
+
return;
|
|
1561
|
+
}
|
|
1562
|
+
const targetVideoIndex = this.findVideoIndexForTimestamp(cumulativeTimestamp);
|
|
1563
|
+
if (targetVideoIndex === null) {
|
|
1564
|
+
console.warn('[Simulator] switchToVideoAndSeek: Could not find video for timestamp', cumulativeTimestamp);
|
|
1565
|
+
return;
|
|
1566
|
+
}
|
|
1567
|
+
const boundaries = this.getVideoDurationBoundaries();
|
|
1568
|
+
const targetBoundary = boundaries[targetVideoIndex];
|
|
1569
|
+
const relativeTimestamp = cumulativeTimestamp - targetBoundary.start;
|
|
1570
|
+
if (this.currentVideoIndex === targetVideoIndex) {
|
|
1571
|
+
await this.seekToTimeInternal(relativeTimestamp);
|
|
1572
|
+
this.lastSetDuration = cumulativeTimestamp;
|
|
1573
|
+
return;
|
|
1574
|
+
}
|
|
1575
|
+
const video = this.vplayer?.nativeElement;
|
|
1576
|
+
if (!video)
|
|
1577
|
+
return;
|
|
1578
|
+
const total = this.totalDuration;
|
|
1579
|
+
this.targetGlobalTimeDuringSwitch = cumulativeTimestamp;
|
|
1580
|
+
if (total > 0) {
|
|
1581
|
+
this.progress = Math.min(100, Math.max(0, (cumulativeTimestamp / total) * 100));
|
|
1582
|
+
}
|
|
1583
|
+
this.playerState = 'switching';
|
|
1584
|
+
try {
|
|
1585
|
+
// CRITICAL: Pause before switching source
|
|
1586
|
+
video.pause();
|
|
1587
|
+
this.isPlaying = false;
|
|
1588
|
+
this.playPromise = null;
|
|
1589
|
+
this.videoPause.emit();
|
|
1590
|
+
this.isVideoPlayingChange.emit(false);
|
|
1591
|
+
// Switch video source
|
|
1592
|
+
this.currentVideoIndex = targetVideoIndex;
|
|
1593
|
+
await new Promise((resolve, reject) => {
|
|
1594
|
+
let resolved = false;
|
|
1595
|
+
const cleanup = () => {
|
|
1596
|
+
if (resolved)
|
|
1597
|
+
return;
|
|
1598
|
+
resolved = true;
|
|
1599
|
+
video.removeEventListener('loadedmetadata', onLoadedMetadata);
|
|
1600
|
+
video.removeEventListener('error', onError);
|
|
1601
|
+
clearTimeout(timeoutId);
|
|
1602
|
+
};
|
|
1603
|
+
const onLoadedMetadata = () => {
|
|
1604
|
+
cleanup();
|
|
1605
|
+
console.log('[Simulator] switchToVideoAndSeek: loadedmetadata', {
|
|
1606
|
+
readyState: video.readyState,
|
|
1607
|
+
duration: video.duration,
|
|
1608
|
+
currentTime: video.currentTime
|
|
1609
|
+
});
|
|
1610
|
+
resolve();
|
|
1611
|
+
};
|
|
1612
|
+
const onError = () => {
|
|
1613
|
+
cleanup();
|
|
1614
|
+
console.error('[Simulator] switchToVideoAndSeek: video error');
|
|
1615
|
+
reject(new Error('Failed to load video'));
|
|
1616
|
+
};
|
|
1617
|
+
setTimeout(() => {
|
|
1618
|
+
if (resolved)
|
|
1619
|
+
return;
|
|
1620
|
+
video.addEventListener('loadedmetadata', onLoadedMetadata, { once: true });
|
|
1621
|
+
video.addEventListener('error', onError, { once: true });
|
|
1622
|
+
}, 100);
|
|
1623
|
+
const timeoutId = setTimeout(() => {
|
|
1624
|
+
cleanup();
|
|
1625
|
+
if (video.readyState >= HTMLMediaElement.HAVE_METADATA) {
|
|
1626
|
+
console.log('[Simulator] switchToVideoAndSeek: metadata timeout but video ready', {
|
|
1627
|
+
readyState: video.readyState,
|
|
1628
|
+
duration: video.duration
|
|
1629
|
+
});
|
|
1630
|
+
resolve();
|
|
1631
|
+
}
|
|
1632
|
+
else {
|
|
1633
|
+
console.error('[Simulator] switchToVideoAndSeek: metadata timeout and video not ready');
|
|
1634
|
+
reject(new Error('Timeout waiting for video to load'));
|
|
1635
|
+
}
|
|
1636
|
+
}, 2000);
|
|
1637
|
+
});
|
|
1638
|
+
// Seek to the target position using 'seeked' event for reliability
|
|
1639
|
+
const seekSeconds = relativeTimestamp / 1000;
|
|
1640
|
+
const targetTime = Math.max(0, Math.min(seekSeconds, video.duration || seekSeconds));
|
|
1641
|
+
console.log('[Simulator] switchToVideoAndSeek: setting currentTime', {
|
|
1642
|
+
targetTime,
|
|
1643
|
+
seekSeconds,
|
|
1644
|
+
relativeTimestamp,
|
|
1645
|
+
videoDuration: video.duration
|
|
1646
|
+
});
|
|
1647
|
+
await new Promise((resolve) => {
|
|
1648
|
+
let resolved = false;
|
|
1649
|
+
const onSeeked = () => {
|
|
1650
|
+
if (resolved)
|
|
1651
|
+
return;
|
|
1652
|
+
resolved = true;
|
|
1653
|
+
video.removeEventListener('seeked', onSeeked);
|
|
1654
|
+
clearTimeout(timeoutId);
|
|
1655
|
+
console.log('[Simulator] switchToVideoAndSeek: seeked event fired', {
|
|
1656
|
+
currentTime: video.currentTime,
|
|
1657
|
+
targetTime,
|
|
1658
|
+
diff: Math.abs(video.currentTime - targetTime)
|
|
1659
|
+
});
|
|
1660
|
+
// Small delay to ensure browser has processed the seek
|
|
1661
|
+
setTimeout(() => resolve(), 100);
|
|
1662
|
+
};
|
|
1663
|
+
video.addEventListener('seeked', onSeeked, { once: true });
|
|
1664
|
+
video.currentTime = targetTime;
|
|
1665
|
+
// Fallback timeout
|
|
1666
|
+
const timeoutId = setTimeout(() => {
|
|
1667
|
+
if (resolved)
|
|
1668
|
+
return;
|
|
1669
|
+
resolved = true;
|
|
1670
|
+
video.removeEventListener('seeked', onSeeked);
|
|
1671
|
+
console.log('[Simulator] switchToVideoAndSeek: seeked timeout, current time is', video.currentTime);
|
|
1672
|
+
resolve();
|
|
1673
|
+
}, 1000);
|
|
1674
|
+
});
|
|
1675
|
+
this.lastSetDuration = cumulativeTimestamp;
|
|
1676
|
+
// Double-check: if video is not at target, force it again
|
|
1677
|
+
if (Math.abs(video.currentTime - targetTime) > 0.5) {
|
|
1678
|
+
console.warn('[Simulator] switchToVideoAndSeek: video not at target after seeked, forcing again', {
|
|
1679
|
+
currentTime: video.currentTime,
|
|
1680
|
+
targetTime
|
|
1681
|
+
});
|
|
1682
|
+
video.currentTime = targetTime;
|
|
1683
|
+
await new Promise(resolve => setTimeout(resolve, 200));
|
|
1684
|
+
}
|
|
1685
|
+
// Verify seek completed
|
|
1686
|
+
const actualGlobal = this.getActualGlobalTime();
|
|
1687
|
+
const actualVideoTime = video.currentTime;
|
|
1688
|
+
console.log('[Simulator] switchToVideoAndSeek: seek verification', {
|
|
1689
|
+
actualGlobal,
|
|
1690
|
+
targetGlobal: cumulativeTimestamp,
|
|
1691
|
+
diff: Math.abs(actualGlobal - cumulativeTimestamp),
|
|
1692
|
+
actualVideoTime,
|
|
1693
|
+
targetVideoTime: targetTime,
|
|
1694
|
+
videoDiff: Math.abs(actualVideoTime - targetTime)
|
|
1695
|
+
});
|
|
1696
|
+
// Clear override only if video is actually at the target
|
|
1697
|
+
const tolerance = 500;
|
|
1698
|
+
if (Math.abs(actualGlobal - cumulativeTimestamp) < tolerance) {
|
|
1699
|
+
this.targetGlobalTimeDuringSwitch = null;
|
|
1700
|
+
console.log('[Simulator] switchToVideoAndSeek: cleared override - video at target');
|
|
1701
|
+
}
|
|
1702
|
+
// Update UI with actual current position
|
|
1703
|
+
if (total > 0) {
|
|
1704
|
+
this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / total) * 100));
|
|
1705
|
+
}
|
|
1706
|
+
this.videoTimeUpdate.emit(this.globalCurrentTime);
|
|
1707
|
+
}
|
|
1708
|
+
catch (err) {
|
|
1709
|
+
console.error('[Simulator] switchToVideoAndSeek failed:', err);
|
|
1710
|
+
// Even on error, try to update to target position
|
|
1711
|
+
if (total > 0) {
|
|
1712
|
+
this.progress = Math.min(100, Math.max(0, (cumulativeTimestamp / total) * 100));
|
|
1713
|
+
}
|
|
1714
|
+
this.videoTimeUpdate.emit(cumulativeTimestamp);
|
|
1715
|
+
}
|
|
1716
|
+
finally {
|
|
1717
|
+
this.playerState = 'paused';
|
|
1718
|
+
this.hitMarkers.clear();
|
|
1719
|
+
// Don't clear targetGlobalTimeDuringSwitch here - it's either already cleared
|
|
1720
|
+
// or will be cleared by timeupdate handler when video reaches target
|
|
1721
|
+
}
|
|
1722
|
+
console.log('[Simulator] switchToVideoAndSeek: complete', {
|
|
1723
|
+
currentVideoIndex: this.currentVideoIndex,
|
|
1724
|
+
videoCurrentTime: video.currentTime,
|
|
1725
|
+
globalCurrentTime: this.globalCurrentTime,
|
|
1726
|
+
targetGlobalTimeDuringSwitch: this.targetGlobalTimeDuringSwitch,
|
|
1727
|
+
playerState: this.playerState,
|
|
1728
|
+
progress: this.progress
|
|
1729
|
+
});
|
|
595
1730
|
}
|
|
596
1731
|
}
|
|
597
1732
|
SimulatorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SimulatorComponent, deps: [{ token: i1.DomSanitizer }], target: i0.ɵɵFactoryTarget.Component });
|
|
598
|
-
SimulatorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: SimulatorComponent, selector: "cqa-simulator", inputs: { videoUrl: "videoUrl", videoUrls: "videoUrls", videoCurrentDuration: "videoCurrentDuration", stepMarkers: "stepMarkers", screenShotUrl: "screenShotUrl", traceViewUrl: "traceViewUrl", platformName: "platformName", platformType: "platformType", platform: "platform", deviceName: "deviceName", isLive: "isLive", liveStatus: "liveStatus", liveLoadingLabel: "liveLoadingLabel", isContentVideoLoading: "isContentVideoLoading", failedStatusMessage: "failedStatusMessage", isVNCSessionIntruppted: "isVNCSessionIntruppted", vncSessionIntupptedMessage: "vncSessionIntupptedMessage", selectedView: "selectedView", hideVideoTab: "hideVideoTab", browserViewPort: "browserViewPort" }, outputs: { videoTimeUpdate: "videoTimeUpdate", videoPlay: "videoPlay", videoPause: "videoPause" }, viewQueries: [{ propertyName: "vplayerRef", first: true, predicate: ["vplayer"], descendants: true }, { propertyName: "timelineBar", first: true, predicate: ["timelineBar"], descendants: true }, { propertyName: "speedControlContainer", first: true, predicate: ["speedControlContainer"], 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 cqa-flex-wrap\">\n <div class=\"cqa-flex cqa-items-center\">\n <div *ngIf=\"isLive\" class=\"cqa-h-[21px] cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-mr-2 cqa-px-[9px] cqa-py-[3px] cqa-bg-[#FCD9D9] cqa-rounded-[6px]\" style=\"border: 1px solid #F9BFBF;\">\n <span class=\"cqa-relative cqa-w-2 cqa-h-2 cqa-rounded-full cqa-bg-[#F47F7F]\" style=\"flex-shrink: 0;\">\n <span class=\"cqa-absolute cqa-inset-0 cqa-rounded-full cqa-bg-[#F47F7F] cqa-opacity-75 cqa-animate-ping\"></span>\n </span>\n <span class=\"cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]\">Live</span>\n </div>\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\">\n {{ platformName }}\n <span\n *ngIf=\"platformType === 'browser'\"\n class=\"cqa-ml-1\"\n [matTooltip]=\"'Screen size: ' + effectiveBrowserViewPort.width + 'x' + effectiveBrowserViewPort.height\"\n matTooltipPosition=\"below\"\n >\n \u00B7\n <span class=\"cqa-ml-1\">\n {{ effectiveBrowserViewPort.width }}x{{ effectiveBrowserViewPort.height }}\n </span>\n </span>\n </p>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div *ngIf=\"isLive\" [ngClass]=\"getStatusBadgeClass()\">\n <span [ngClass]=\"getStatusTextClass()\">{{ liveStatus }}</span>\n </div>\n\n <ng-container *ngIf=\"!isLive\">\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 </ng-container>\n </div>\n </div>\n </div>\n <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n <!-- Live Content View -->\n <div *ngIf=\"isLive\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center cqa-relative\">\n <ng-container *ngIf=\"hasDeviceFrame;\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-p-4\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-max-h-full cqa-h-full': platformType !== 'browser' || !isLive, 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-z-20': platformType === 'browser', 'cqa-bg-white': platformType !== 'browser'}\">\n <!-- Loading State -->\n <div *ngIf=\"isContentVideoLoading\" class=\"cqa-p-10 cqa-text-center cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n <div class=\"cqa-mb-4\">\n <mat-progress-spinner mode=\"indeterminate\" diameter=\"40\"></mat-progress-spinner>\n </div>\n <p class=\"cqa-text-gray-400 cqa-text-sm\">{{ liveLoadingLabel }}</p>\n </div>\n\n <!-- Live Content (when not loading) -->\n <div *ngIf=\"!isContentVideoLoading\" class=\"cqa-w-full cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-relative\">\n <div *ngIf=\"liveStatus === 'Failed' && failedStatusMessage\" class=\"cqa-p-6 cqa-text-center cqa-w-full\">\n <div class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-4 cqa-py-3 cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF] cqa-rounded-lg\">\n <mat-icon style=\"width: 18px; height: 18px; color: #C63535; font-size: 18px;\">error</mat-icon>\n <p class=\"cqa-text-[#C63535] cqa-text-sm cqa-font-medium cqa-m-0\">{{ failedStatusMessage }}</p>\n </div>\n </div>\n <ng-content *ngIf=\"liveStatus !== 'Failed' || !failedStatusMessage\"></ng-content>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n\n <!-- Normal Video View (when not live) -->\n <div *ngIf=\"!isLive && currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col\">\n <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-83px)]\" *ngIf=\"currentVideoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device', 'cqa-mt-auto': hasDeviceFrame}\">\n <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <video\n #vplayer\n class=\"cqa-object-cover cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n [src]=\"currentVideoUrl\"\n type=\"video/webm\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n (loadedmetadata)=\"onVideoMetadataLoaded()\"\n (canplay)=\"onVideoCanPlay()\"\n (ended)=\"onVideoEnded()\"\n ></video>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!currentVideoUrl\">\n <ng-container *ngIf=\"isVNCSessionIntruppted && vncSessionIntupptedMessage; else noVideoDefault\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">\n {{ vncSessionIntupptedMessage }}\n </p>\n </ng-container>\n <ng-template #noVideoDefault>\n <span>No video recording found</span>\n </ng-template>\n </div>\n \n <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" [ngClass]=\"{'cqa-mt-auto': hasDeviceFrame}\" *ngIf=\"currentVideoUrl && !isLive\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button \n *ngIf=\"hasMultipleVideos\"\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 [disabled]=\"currentVideoIndex === 0\"\n [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': currentVideoIndex === 0}\"\n (click)=\"prevVideo()\"\n matTooltip=\"Previous video\"\n matTooltipPosition=\"above\">\n <mat-icon class=\"cqa-w-4 cqa-h-4 !cqa-text-[16px] cqa-text-[#374151]\" [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': currentVideoIndex === 0}\">skip_previous</mat-icon>\n </button>\n\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 matTooltip=\"{{ isPlaying ? 'Pause' : 'Play' }}\"\n matTooltipPosition=\"above\">\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 <button \n *ngIf=\"hasMultipleVideos\"\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 [disabled]=\"videoUrls && (currentVideoIndex >= videoUrls.length - 1)\"\n [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': videoUrls && (currentVideoIndex >= videoUrls.length - 1)}\"\n (click)=\"nextVideo()\"\n matTooltip=\"Next video\"\n matTooltipPosition=\"above\">\n <mat-icon class=\"cqa-w-4 cqa-h-4 !cqa-text-[16px] cqa-text-[#374151]\" [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': videoUrls && (currentVideoIndex >= videoUrls.length - 1)}\">skip_next</mat-icon>\n </button>\n\n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n {{ formatTime(vplayer?.nativeElement?.currentTime || 0) }}\n </span>\n\n <div #speedControlContainer class=\"cqa-relative cqa-mr-[8px] cqa-flex cqa-items-center cqa-justify-center\">\n <button\n type=\"button\"\n class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer cqa-text-[#9CA3AF] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-whitespace-nowrap cqa-select-none hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none cqa-px-1\"\n (click)=\"toggleSpeedControl()\"\n [matTooltip]=\"'Playback Speed'\"\n [matTooltipPosition]=\"'below'\">\n {{ currentSpeed }}\n </button>\n \n <div \n *ngIf=\"isSpeedControlOpen\"\n class=\"cqa-absolute cqa-bottom-full cqa-mb-2 cqa-right-0 cqa-bg-[#F0F0F1] cqa-rounded-lg cqa-overflow-hidden cqa-shadow-lg cqa-z-50\"\n style=\"min-width: max-content; left: 50%; bottom: 0%; transform: translate(-50%, -50%);\">\n <cqa-segment-control\n [segments]=\"speedSegments\"\n [value]=\"currentSpeed\"\n [containerBgColor]=\"'#F0F0F1'\"\n (valueChange)=\"onSpeedChange($event)\">\n </cqa-segment-control>\n </div>\n </div>\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 \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-blue-500 cqa-rounded-full\"\n [style.width.%]=\"progress\"\n [style.transition]=\"dragging ? 'none' : 'width 100ms'\"\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-blue-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: 40;\"\n (mousedown)=\"startDrag($event)\">\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"!isLive && 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 <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n [ngClass]=\"{'cqa-max-h-[inherit]': platformType === 'browser', 'cqa-max-h-full': platformType !== 'browser'}\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <img\n [src]=\"screenShotUrl\"\n alt=\"Screenshot\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n class=\"cqa-object-contain cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n />\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!screenShotUrl\">\n No screenshot available\n </div>\n </div>\n\n <div *ngIf=\"!isLive && currentView === 'trace'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-relative\" *ngIf=\"traceViewUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\" style=\"padding-top: 48px; padding-bottom: 0px;\">\n <div class=\"cqa-w-full cqa-h-full cqa-overflow-hidden cqa-relative\">\n <iframe \n [src]=\"safeTraceUrl\" \n title=\"Trace Viewer\"\n class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-[##F2F2F2]\"\n style=\"margin-top: -48px; height: calc(100% + 48px);\"\n frameborder=\"0\"\n allowfullscreen\n width=\"100%\"\n loading=\"lazy\"\n (load)=\"onTraceViewerLoad()\"\n (error)=\"onTraceViewerError()\">\n </iframe>\n </div>\n \n <div *ngIf=\"traceViewerLoading\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Loading trace viewer...\n </div>\n </div>\n \n <div *ngIf=\"traceViewerError\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Failed to load trace viewer\n </div>\n </div>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!traceViewUrl\">\n No trace available\n </div>\n </div> \n </div>\n</div>", components: [{ type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled", "containerBgColor"], outputs: ["valueChange"] }, { type: i4.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }], directives: [{ type: i5.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
1733
|
+
SimulatorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: SimulatorComponent, selector: "cqa-simulator", inputs: { videoUrl: "videoUrl", videoUrls: "videoUrls", videoCurrentDuration: "videoCurrentDuration", stepMarkers: "stepMarkers", screenShotUrl: "screenShotUrl", traceViewUrl: "traceViewUrl", platformName: "platformName", platformType: "platformType", platform: "platform", deviceName: "deviceName", isLive: "isLive", liveStatus: "liveStatus", liveLoadingLabel: "liveLoadingLabel", isContentVideoLoading: "isContentVideoLoading", failedStatusMessage: "failedStatusMessage", isVNCSessionIntruppted: "isVNCSessionIntruppted", vncSessionIntupptedMessage: "vncSessionIntupptedMessage", selectedView: "selectedView", hideVideoTab: "hideVideoTab", browserViewPort: "browserViewPort" }, outputs: { videoTimeUpdate: "videoTimeUpdate", videoPlay: "videoPlay", videoPause: "videoPause", markerHit: "markerHit", isVideoPlayingChange: "isVideoPlayingChange" }, viewQueries: [{ propertyName: "vplayerRef", first: true, predicate: ["vplayer"], descendants: true }, { propertyName: "timelineBarRef", first: true, predicate: ["timelineBar"], descendants: true }, { propertyName: "speedControlContainerRef", first: true, predicate: ["speedControlContainer"], 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 cqa-flex-wrap\">\n <div class=\"cqa-flex cqa-items-center\">\n <div *ngIf=\"isLive\" class=\"cqa-h-[21px] cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-mr-2 cqa-px-[9px] cqa-py-[3px] cqa-bg-[#FCD9D9] cqa-rounded-[6px]\" style=\"border: 1px solid #F9BFBF;\">\n <span class=\"cqa-relative cqa-w-2 cqa-h-2 cqa-rounded-full cqa-bg-[#F47F7F]\" style=\"flex-shrink: 0;\">\n <span class=\"cqa-absolute cqa-inset-0 cqa-rounded-full cqa-bg-[#F47F7F] cqa-opacity-75 cqa-animate-ping\"></span>\n </span>\n <span class=\"cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]\">Live</span>\n </div>\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\">\n {{ platformName }}\n <span\n *ngIf=\"platformType === 'browser'\"\n class=\"cqa-ml-1\"\n [matTooltip]=\"'Screen size: ' + effectiveBrowserViewPort.width + 'x' + effectiveBrowserViewPort.height\"\n matTooltipPosition=\"below\"\n >\n \u00B7\n <span class=\"cqa-ml-1\">\n {{ effectiveBrowserViewPort.width }}x{{ effectiveBrowserViewPort.height }}\n </span>\n </span>\n </p>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div *ngIf=\"isLive\" [ngClass]=\"getStatusBadgeClass()\">\n <span [ngClass]=\"getStatusTextClass()\">{{ liveStatus }}</span>\n </div>\n\n <ng-container *ngIf=\"!isLive\">\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 </ng-container>\n </div>\n </div>\n </div>\n <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n <!-- Live Content View -->\n <div *ngIf=\"isLive\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center cqa-relative\">\n <ng-container *ngIf=\"hasDeviceFrame;\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-p-4\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-max-h-full cqa-h-full': platformType !== 'browser' || !isLive, 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-z-20': platformType === 'browser', 'cqa-bg-white': platformType !== 'browser'}\">\n <!-- Loading State -->\n <div *ngIf=\"isContentVideoLoading\" class=\"cqa-p-10 cqa-text-center cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n <div class=\"cqa-mb-4\">\n <mat-progress-spinner mode=\"indeterminate\" diameter=\"40\"></mat-progress-spinner>\n </div>\n <p class=\"cqa-text-gray-400 cqa-text-sm\">{{ liveLoadingLabel }}</p>\n </div>\n\n <!-- Live Content (when not loading) -->\n <div *ngIf=\"!isContentVideoLoading\" class=\"cqa-w-full cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-relative\">\n <div *ngIf=\"liveStatus === 'Failed' && failedStatusMessage\" class=\"cqa-p-6 cqa-text-center cqa-w-full\">\n <div class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-4 cqa-py-3 cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF] cqa-rounded-lg\">\n <mat-icon style=\"width: 18px; height: 18px; color: #C63535; font-size: 18px;\">error</mat-icon>\n <p class=\"cqa-text-[#C63535] cqa-text-sm cqa-font-medium cqa-m-0\">{{ failedStatusMessage }}</p>\n </div>\n </div>\n <ng-content *ngIf=\"liveStatus !== 'Failed' || !failedStatusMessage\"></ng-content>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n\n <!-- Normal Video View (when not live) -->\n <div *ngIf=\"!isLive && currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col\">\n <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-83px)]\" *ngIf=\"currentVideoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device', 'cqa-mt-auto': hasDeviceFrame}\">\n <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <video\n #vplayer\n class=\"cqa-object-cover cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n [src]=\"currentVideoUrl\"\n type=\"video/webm\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n (loadedmetadata)=\"onVideoMetadataLoaded()\"\n (canplay)=\"onVideoCanPlay()\"\n (ended)=\"onVideoEnded()\"\n ></video>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!currentVideoUrl\">\n <ng-container *ngIf=\"isVNCSessionIntruppted && vncSessionIntupptedMessage; else noVideoDefault\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">\n {{ vncSessionIntupptedMessage }}\n </p>\n </ng-container>\n <ng-template #noVideoDefault>\n <span>No video recording found</span>\n </ng-template>\n </div>\n \n <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" [ngClass]=\"{'cqa-mt-auto': hasDeviceFrame}\" *ngIf=\"currentVideoUrl && !isLive\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-center\" style=\"width: 16px; height: 16px;\">\n <mat-progress-spinner\n *ngIf=\"isPlayerSwitching\"\n mode=\"indeterminate\"\n diameter=\"16\"\n class=\"cqa-inline-block\">\n </mat-progress-spinner>\n <button \n *ngIf=\"!isPlayerSwitching\"\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 matTooltip=\"{{ isPlaying ? 'Pause' : 'Play' }}\"\n matTooltipPosition=\"above\">\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 </div>\n\n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n {{ formatTime(globalCurrentTime / 1000) }}\n </span>\n\n <div #speedControlContainer class=\"cqa-relative cqa-mr-[8px] cqa-flex cqa-items-center cqa-justify-center\">\n <button\n type=\"button\"\n class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer cqa-text-[#9CA3AF] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-whitespace-nowrap cqa-select-none hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none cqa-px-1\"\n (click)=\"toggleSpeedControl()\"\n [matTooltip]=\"'Playback Speed'\"\n [matTooltipPosition]=\"'below'\">\n {{ currentSpeed }}\n </button>\n \n <div \n *ngIf=\"isSpeedControlOpen\"\n class=\"cqa-absolute cqa-bottom-full cqa-mb-2 cqa-right-0 cqa-bg-[#F0F0F1] cqa-rounded-lg cqa-overflow-hidden cqa-shadow-lg cqa-z-50\"\n style=\"min-width: max-content; left: 50%; bottom: 0%; transform: translate(-50%, -50%);\">\n <cqa-segment-control\n [segments]=\"speedSegments\"\n [value]=\"currentSpeed\"\n [containerBgColor]=\"'#F0F0F1'\"\n (valueChange)=\"onSpeedChange($event)\">\n </cqa-segment-control>\n </div>\n </div>\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 \n <div \n *ngFor=\"let marker of globalStepMarkers\" \n class=\"cqa-absolute cqa-rounded-full\"\n [style.left.%]=\"totalDuration > 0 ? (marker.globalTime / totalDuration) * 100 : 0\"\n [style.width]=\"'8px'\"\n [style.height]=\"'8px'\"\n [style.background]=\"getGlobalMarkerColor(marker.level)\"\n [style.border]=\"'2px solid ' + getGlobalMarkerResultColor(marker.result)\"\n [style.box-sizing]=\"'border-box'\"\n [attr.title]=\"marker.title || ''\"\n style=\"pointer-events: auto; z-index: 50; cursor: pointer; transform: translate(-50%, -50%); top: 50%;\"\n (click)=\"onMarkerClick($event, marker)\">\n </div>\n \n <div \n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-h-full cqa-bg-blue-500 cqa-rounded-full\"\n [style.width.%]=\"progress\"\n [style.transition]=\"dragging ? 'none' : 'width 100ms'\"\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-blue-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: 60;\"\n (mousedown)=\"startDrag($event)\">\n </div>\n </div>\n\n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n {{ formatTime(totalDuration / 1000) }}\n </span>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"!isLive && 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 <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n [ngClass]=\"{'cqa-max-h-[inherit]': platformType === 'browser', 'cqa-max-h-full': platformType !== 'browser'}\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <img\n [src]=\"screenShotUrl\"\n alt=\"Screenshot\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n class=\"cqa-object-contain cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n />\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!screenShotUrl\">\n No screenshot available\n </div>\n </div>\n\n <div *ngIf=\"!isLive && currentView === 'trace'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-relative\" *ngIf=\"traceViewUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\" style=\"padding-top: 48px; padding-bottom: 0px;\">\n <div class=\"cqa-w-full cqa-h-full cqa-overflow-hidden cqa-relative\">\n <iframe \n [src]=\"safeTraceUrl\" \n title=\"Trace Viewer\"\n class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-[##F2F2F2]\"\n style=\"margin-top: -48px; height: calc(100% + 48px);\"\n frameborder=\"0\"\n allowfullscreen\n width=\"100%\"\n loading=\"lazy\"\n (load)=\"onTraceViewerLoad()\"\n (error)=\"onTraceViewerError()\">\n </iframe>\n </div>\n \n <div *ngIf=\"traceViewerLoading\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Loading trace viewer...\n </div>\n </div>\n \n <div *ngIf=\"traceViewerError\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Failed to load trace viewer\n </div>\n </div>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!traceViewUrl\">\n No trace available\n </div>\n </div> \n </div>\n</div>", components: [{ type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled", "containerBgColor"], outputs: ["valueChange"] }, { type: i4.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }], directives: [{ type: i5.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
599
1734
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SimulatorComponent, decorators: [{
|
|
600
1735
|
type: Component,
|
|
601
|
-
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 cqa-flex-wrap\">\n <div class=\"cqa-flex cqa-items-center\">\n <div *ngIf=\"isLive\" class=\"cqa-h-[21px] cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-mr-2 cqa-px-[9px] cqa-py-[3px] cqa-bg-[#FCD9D9] cqa-rounded-[6px]\" style=\"border: 1px solid #F9BFBF;\">\n <span class=\"cqa-relative cqa-w-2 cqa-h-2 cqa-rounded-full cqa-bg-[#F47F7F]\" style=\"flex-shrink: 0;\">\n <span class=\"cqa-absolute cqa-inset-0 cqa-rounded-full cqa-bg-[#F47F7F] cqa-opacity-75 cqa-animate-ping\"></span>\n </span>\n <span class=\"cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]\">Live</span>\n </div>\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\">\n {{ platformName }}\n <span\n *ngIf=\"platformType === 'browser'\"\n class=\"cqa-ml-1\"\n [matTooltip]=\"'Screen size: ' + effectiveBrowserViewPort.width + 'x' + effectiveBrowserViewPort.height\"\n matTooltipPosition=\"below\"\n >\n \u00B7\n <span class=\"cqa-ml-1\">\n {{ effectiveBrowserViewPort.width }}x{{ effectiveBrowserViewPort.height }}\n </span>\n </span>\n </p>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div *ngIf=\"isLive\" [ngClass]=\"getStatusBadgeClass()\">\n <span [ngClass]=\"getStatusTextClass()\">{{ liveStatus }}</span>\n </div>\n\n <ng-container *ngIf=\"!isLive\">\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 </ng-container>\n </div>\n </div>\n </div>\n <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n <!-- Live Content View -->\n <div *ngIf=\"isLive\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center cqa-relative\">\n <ng-container *ngIf=\"hasDeviceFrame;\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-p-4\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-max-h-full cqa-h-full': platformType !== 'browser' || !isLive, 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-z-20': platformType === 'browser', 'cqa-bg-white': platformType !== 'browser'}\">\n <!-- Loading State -->\n <div *ngIf=\"isContentVideoLoading\" class=\"cqa-p-10 cqa-text-center cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n <div class=\"cqa-mb-4\">\n <mat-progress-spinner mode=\"indeterminate\" diameter=\"40\"></mat-progress-spinner>\n </div>\n <p class=\"cqa-text-gray-400 cqa-text-sm\">{{ liveLoadingLabel }}</p>\n </div>\n\n <!-- Live Content (when not loading) -->\n <div *ngIf=\"!isContentVideoLoading\" class=\"cqa-w-full cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-relative\">\n <div *ngIf=\"liveStatus === 'Failed' && failedStatusMessage\" class=\"cqa-p-6 cqa-text-center cqa-w-full\">\n <div class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-4 cqa-py-3 cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF] cqa-rounded-lg\">\n <mat-icon style=\"width: 18px; height: 18px; color: #C63535; font-size: 18px;\">error</mat-icon>\n <p class=\"cqa-text-[#C63535] cqa-text-sm cqa-font-medium cqa-m-0\">{{ failedStatusMessage }}</p>\n </div>\n </div>\n <ng-content *ngIf=\"liveStatus !== 'Failed' || !failedStatusMessage\"></ng-content>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n\n <!-- Normal Video View (when not live) -->\n <div *ngIf=\"!isLive && currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col\">\n <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-83px)]\" *ngIf=\"currentVideoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device', 'cqa-mt-auto': hasDeviceFrame}\">\n <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <video\n #vplayer\n class=\"cqa-object-cover cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n [src]=\"currentVideoUrl\"\n type=\"video/webm\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n (loadedmetadata)=\"onVideoMetadataLoaded()\"\n (canplay)=\"onVideoCanPlay()\"\n (ended)=\"onVideoEnded()\"\n ></video>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!currentVideoUrl\">\n <ng-container *ngIf=\"isVNCSessionIntruppted && vncSessionIntupptedMessage; else noVideoDefault\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">\n {{ vncSessionIntupptedMessage }}\n </p>\n </ng-container>\n <ng-template #noVideoDefault>\n <span>No video recording found</span>\n </ng-template>\n </div>\n \n <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" [ngClass]=\"{'cqa-mt-auto': hasDeviceFrame}\" *ngIf=\"currentVideoUrl && !isLive\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button \n *ngIf=\"hasMultipleVideos\"\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 [disabled]=\"currentVideoIndex === 0\"\n [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': currentVideoIndex === 0}\"\n (click)=\"prevVideo()\"\n matTooltip=\"Previous video\"\n matTooltipPosition=\"above\">\n <mat-icon class=\"cqa-w-4 cqa-h-4 !cqa-text-[16px] cqa-text-[#374151]\" [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': currentVideoIndex === 0}\">skip_previous</mat-icon>\n </button>\n\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 matTooltip=\"{{ isPlaying ? 'Pause' : 'Play' }}\"\n matTooltipPosition=\"above\">\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 <button \n *ngIf=\"hasMultipleVideos\"\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 [disabled]=\"videoUrls && (currentVideoIndex >= videoUrls.length - 1)\"\n [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': videoUrls && (currentVideoIndex >= videoUrls.length - 1)}\"\n (click)=\"nextVideo()\"\n matTooltip=\"Next video\"\n matTooltipPosition=\"above\">\n <mat-icon class=\"cqa-w-4 cqa-h-4 !cqa-text-[16px] cqa-text-[#374151]\" [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': videoUrls && (currentVideoIndex >= videoUrls.length - 1)}\">skip_next</mat-icon>\n </button>\n\n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n {{ formatTime(vplayer?.nativeElement?.currentTime || 0) }}\n </span>\n\n <div #speedControlContainer class=\"cqa-relative cqa-mr-[8px] cqa-flex cqa-items-center cqa-justify-center\">\n <button\n type=\"button\"\n class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer cqa-text-[#9CA3AF] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-whitespace-nowrap cqa-select-none hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none cqa-px-1\"\n (click)=\"toggleSpeedControl()\"\n [matTooltip]=\"'Playback Speed'\"\n [matTooltipPosition]=\"'below'\">\n {{ currentSpeed }}\n </button>\n \n <div \n *ngIf=\"isSpeedControlOpen\"\n class=\"cqa-absolute cqa-bottom-full cqa-mb-2 cqa-right-0 cqa-bg-[#F0F0F1] cqa-rounded-lg cqa-overflow-hidden cqa-shadow-lg cqa-z-50\"\n style=\"min-width: max-content; left: 50%; bottom: 0%; transform: translate(-50%, -50%);\">\n <cqa-segment-control\n [segments]=\"speedSegments\"\n [value]=\"currentSpeed\"\n [containerBgColor]=\"'#F0F0F1'\"\n (valueChange)=\"onSpeedChange($event)\">\n </cqa-segment-control>\n </div>\n </div>\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 \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-blue-500 cqa-rounded-full\"\n [style.width.%]=\"progress\"\n [style.transition]=\"dragging ? 'none' : 'width 100ms'\"\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-blue-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: 40;\"\n (mousedown)=\"startDrag($event)\">\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"!isLive && 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 <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n [ngClass]=\"{'cqa-max-h-[inherit]': platformType === 'browser', 'cqa-max-h-full': platformType !== 'browser'}\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <img\n [src]=\"screenShotUrl\"\n alt=\"Screenshot\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n class=\"cqa-object-contain cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n />\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!screenShotUrl\">\n No screenshot available\n </div>\n </div>\n\n <div *ngIf=\"!isLive && currentView === 'trace'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-relative\" *ngIf=\"traceViewUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\" style=\"padding-top: 48px; padding-bottom: 0px;\">\n <div class=\"cqa-w-full cqa-h-full cqa-overflow-hidden cqa-relative\">\n <iframe \n [src]=\"safeTraceUrl\" \n title=\"Trace Viewer\"\n class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-[##F2F2F2]\"\n style=\"margin-top: -48px; height: calc(100% + 48px);\"\n frameborder=\"0\"\n allowfullscreen\n width=\"100%\"\n loading=\"lazy\"\n (load)=\"onTraceViewerLoad()\"\n (error)=\"onTraceViewerError()\">\n </iframe>\n </div>\n \n <div *ngIf=\"traceViewerLoading\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Loading trace viewer...\n </div>\n </div>\n \n <div *ngIf=\"traceViewerError\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Failed to load trace viewer\n </div>\n </div>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!traceViewUrl\">\n No trace available\n </div>\n </div> \n </div>\n</div>", styles: [] }]
|
|
1736
|
+
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 cqa-flex-wrap\">\n <div class=\"cqa-flex cqa-items-center\">\n <div *ngIf=\"isLive\" class=\"cqa-h-[21px] cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-mr-2 cqa-px-[9px] cqa-py-[3px] cqa-bg-[#FCD9D9] cqa-rounded-[6px]\" style=\"border: 1px solid #F9BFBF;\">\n <span class=\"cqa-relative cqa-w-2 cqa-h-2 cqa-rounded-full cqa-bg-[#F47F7F]\" style=\"flex-shrink: 0;\">\n <span class=\"cqa-absolute cqa-inset-0 cqa-rounded-full cqa-bg-[#F47F7F] cqa-opacity-75 cqa-animate-ping\"></span>\n </span>\n <span class=\"cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]\">Live</span>\n </div>\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\">\n {{ platformName }}\n <span\n *ngIf=\"platformType === 'browser'\"\n class=\"cqa-ml-1\"\n [matTooltip]=\"'Screen size: ' + effectiveBrowserViewPort.width + 'x' + effectiveBrowserViewPort.height\"\n matTooltipPosition=\"below\"\n >\n \u00B7\n <span class=\"cqa-ml-1\">\n {{ effectiveBrowserViewPort.width }}x{{ effectiveBrowserViewPort.height }}\n </span>\n </span>\n </p>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div *ngIf=\"isLive\" [ngClass]=\"getStatusBadgeClass()\">\n <span [ngClass]=\"getStatusTextClass()\">{{ liveStatus }}</span>\n </div>\n\n <ng-container *ngIf=\"!isLive\">\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 </ng-container>\n </div>\n </div>\n </div>\n <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n <!-- Live Content View -->\n <div *ngIf=\"isLive\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center cqa-relative\">\n <ng-container *ngIf=\"hasDeviceFrame;\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-p-4\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-max-h-full cqa-h-full': platformType !== 'browser' || !isLive, 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-z-20': platformType === 'browser', 'cqa-bg-white': platformType !== 'browser'}\">\n <!-- Loading State -->\n <div *ngIf=\"isContentVideoLoading\" class=\"cqa-p-10 cqa-text-center cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n <div class=\"cqa-mb-4\">\n <mat-progress-spinner mode=\"indeterminate\" diameter=\"40\"></mat-progress-spinner>\n </div>\n <p class=\"cqa-text-gray-400 cqa-text-sm\">{{ liveLoadingLabel }}</p>\n </div>\n\n <!-- Live Content (when not loading) -->\n <div *ngIf=\"!isContentVideoLoading\" class=\"cqa-w-full cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-relative\">\n <div *ngIf=\"liveStatus === 'Failed' && failedStatusMessage\" class=\"cqa-p-6 cqa-text-center cqa-w-full\">\n <div class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-4 cqa-py-3 cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF] cqa-rounded-lg\">\n <mat-icon style=\"width: 18px; height: 18px; color: #C63535; font-size: 18px;\">error</mat-icon>\n <p class=\"cqa-text-[#C63535] cqa-text-sm cqa-font-medium cqa-m-0\">{{ failedStatusMessage }}</p>\n </div>\n </div>\n <ng-content *ngIf=\"liveStatus !== 'Failed' || !failedStatusMessage\"></ng-content>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n\n <!-- Normal Video View (when not live) -->\n <div *ngIf=\"!isLive && currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col\">\n <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-83px)]\" *ngIf=\"currentVideoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device', 'cqa-mt-auto': hasDeviceFrame}\">\n <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <video\n #vplayer\n class=\"cqa-object-cover cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n [src]=\"currentVideoUrl\"\n type=\"video/webm\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n (loadedmetadata)=\"onVideoMetadataLoaded()\"\n (canplay)=\"onVideoCanPlay()\"\n (ended)=\"onVideoEnded()\"\n ></video>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!currentVideoUrl\">\n <ng-container *ngIf=\"isVNCSessionIntruppted && vncSessionIntupptedMessage; else noVideoDefault\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">\n {{ vncSessionIntupptedMessage }}\n </p>\n </ng-container>\n <ng-template #noVideoDefault>\n <span>No video recording found</span>\n </ng-template>\n </div>\n \n <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" [ngClass]=\"{'cqa-mt-auto': hasDeviceFrame}\" *ngIf=\"currentVideoUrl && !isLive\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-center\" style=\"width: 16px; height: 16px;\">\n <mat-progress-spinner\n *ngIf=\"isPlayerSwitching\"\n mode=\"indeterminate\"\n diameter=\"16\"\n class=\"cqa-inline-block\">\n </mat-progress-spinner>\n <button \n *ngIf=\"!isPlayerSwitching\"\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 matTooltip=\"{{ isPlaying ? 'Pause' : 'Play' }}\"\n matTooltipPosition=\"above\">\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 </div>\n\n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n {{ formatTime(globalCurrentTime / 1000) }}\n </span>\n\n <div #speedControlContainer class=\"cqa-relative cqa-mr-[8px] cqa-flex cqa-items-center cqa-justify-center\">\n <button\n type=\"button\"\n class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer cqa-text-[#9CA3AF] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-whitespace-nowrap cqa-select-none hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none cqa-px-1\"\n (click)=\"toggleSpeedControl()\"\n [matTooltip]=\"'Playback Speed'\"\n [matTooltipPosition]=\"'below'\">\n {{ currentSpeed }}\n </button>\n \n <div \n *ngIf=\"isSpeedControlOpen\"\n class=\"cqa-absolute cqa-bottom-full cqa-mb-2 cqa-right-0 cqa-bg-[#F0F0F1] cqa-rounded-lg cqa-overflow-hidden cqa-shadow-lg cqa-z-50\"\n style=\"min-width: max-content; left: 50%; bottom: 0%; transform: translate(-50%, -50%);\">\n <cqa-segment-control\n [segments]=\"speedSegments\"\n [value]=\"currentSpeed\"\n [containerBgColor]=\"'#F0F0F1'\"\n (valueChange)=\"onSpeedChange($event)\">\n </cqa-segment-control>\n </div>\n </div>\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 \n <div \n *ngFor=\"let marker of globalStepMarkers\" \n class=\"cqa-absolute cqa-rounded-full\"\n [style.left.%]=\"totalDuration > 0 ? (marker.globalTime / totalDuration) * 100 : 0\"\n [style.width]=\"'8px'\"\n [style.height]=\"'8px'\"\n [style.background]=\"getGlobalMarkerColor(marker.level)\"\n [style.border]=\"'2px solid ' + getGlobalMarkerResultColor(marker.result)\"\n [style.box-sizing]=\"'border-box'\"\n [attr.title]=\"marker.title || ''\"\n style=\"pointer-events: auto; z-index: 50; cursor: pointer; transform: translate(-50%, -50%); top: 50%;\"\n (click)=\"onMarkerClick($event, marker)\">\n </div>\n \n <div \n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-h-full cqa-bg-blue-500 cqa-rounded-full\"\n [style.width.%]=\"progress\"\n [style.transition]=\"dragging ? 'none' : 'width 100ms'\"\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-blue-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: 60;\"\n (mousedown)=\"startDrag($event)\">\n </div>\n </div>\n\n <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n {{ formatTime(totalDuration / 1000) }}\n </span>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"!isLive && 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 <ng-container *ngIf=\"hasDeviceFrame\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n <img\n [src]=\"deviceMockupImage\"\n alt=\"Device mockup\"\n class=\"cqa-h-full cqa-w-auto cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n [ngClass]=\"{'cqa-max-h-[inherit]': platformType === 'browser', 'cqa-max-h-full': platformType !== 'browser'}\"\n />\n <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n <img\n [src]=\"screenShotUrl\"\n alt=\"Screenshot\"\n [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n class=\"cqa-object-contain cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n />\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!screenShotUrl\">\n No screenshot available\n </div>\n </div>\n\n <div *ngIf=\"!isLive && currentView === 'trace'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-relative\" *ngIf=\"traceViewUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\" style=\"padding-top: 48px; padding-bottom: 0px;\">\n <div class=\"cqa-w-full cqa-h-full cqa-overflow-hidden cqa-relative\">\n <iframe \n [src]=\"safeTraceUrl\" \n title=\"Trace Viewer\"\n class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-[##F2F2F2]\"\n style=\"margin-top: -48px; height: calc(100% + 48px);\"\n frameborder=\"0\"\n allowfullscreen\n width=\"100%\"\n loading=\"lazy\"\n (load)=\"onTraceViewerLoad()\"\n (error)=\"onTraceViewerError()\">\n </iframe>\n </div>\n \n <div *ngIf=\"traceViewerLoading\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Loading trace viewer...\n </div>\n </div>\n \n <div *ngIf=\"traceViewerError\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n Failed to load trace viewer\n </div>\n </div>\n </div>\n \n <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!traceViewUrl\">\n No trace available\n </div>\n </div> \n </div>\n</div>", styles: [] }]
|
|
602
1737
|
}], ctorParameters: function () { return [{ type: i1.DomSanitizer }]; }, propDecorators: { videoUrl: [{
|
|
603
1738
|
type: Input
|
|
604
1739
|
}], videoUrls: [{
|
|
@@ -645,14 +1780,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
645
1780
|
type: Output
|
|
646
1781
|
}], videoPause: [{
|
|
647
1782
|
type: Output
|
|
1783
|
+
}], markerHit: [{
|
|
1784
|
+
type: Output
|
|
1785
|
+
}], isVideoPlayingChange: [{
|
|
1786
|
+
type: Output
|
|
648
1787
|
}], vplayerRef: [{
|
|
649
1788
|
type: ViewChild,
|
|
650
1789
|
args: ['vplayer']
|
|
651
|
-
}],
|
|
1790
|
+
}], timelineBarRef: [{
|
|
652
1791
|
type: ViewChild,
|
|
653
1792
|
args: ['timelineBar', { static: false }]
|
|
654
|
-
}],
|
|
1793
|
+
}], speedControlContainerRef: [{
|
|
655
1794
|
type: ViewChild,
|
|
656
1795
|
args: ['speedControlContainer', { static: false }]
|
|
657
1796
|
}] } });
|
|
658
|
-
//# 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';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { getEmptyStatePreset } from '../empty-state/empty-state-presets.constants';\n\nexport interface StepMarker {\n  cumulativeDuration: number;\n  result: 'SUCCESS' | 'FAILURE';\n}\n\nexport type SegmentType = 'screenshots' | 'video' | 'trace';\n\ntype SegmentOption = {\n  label: string;\n  value: string;\n  icon?: string;\n};\ninterface DeviceFrameConfig {\n  mockupImage: string;\n  screenInset: {\n    top: string;\n    right: string;\n    bottom: string;\n    left: string;\n  };\n  borderRadius: string;\n}\n\n@Component({\n  selector: 'cqa-simulator',\n  templateUrl: './simulator.component.html',\n  styleUrls: []\n})\n\nexport class SimulatorComponent implements AfterViewInit, OnDestroy, OnChanges {\n\n  @Input() videoUrl: string = '';\n  @Input() videoUrls: string[] = [];\n  @Input() videoCurrentDuration: number = 0;\n  @Input() stepMarkers: StepMarker[] = [];\n  @Input() screenShotUrl: string = '';\n  @Input() traceViewUrl: string = '';\n  @Input() platformName: string = 'Web - Chrome';\n  @Input() platformType: 'browser' | 'device' = 'browser';\n  @Input() platform: string = '';\n  @Input() deviceName: string | null = null;\n  @Input() isLive: boolean = false;\n  @Input() liveStatus: 'In Progress' | 'Paused' | 'Aborted' | 'Failed' | 'Passed' | 'Execution' = 'In Progress';\n  @Input() liveLoadingLabel: string = 'Loading...';\n  @Input() isContentVideoLoading: boolean = false;\n  @Input() failedStatusMessage: string = '';\n  @Input() isVNCSessionIntruppted: boolean = false;\n  @Input() vncSessionIntupptedMessage: string = '';\n  @Input() selectedView: SegmentType = 'video';\n  @Input() hideVideoTab: boolean = false;\n  @Input() browserViewPort: { width: number; height: number } | null = null;\n\n  @Output() videoTimeUpdate = new EventEmitter<number>();\n  @Output() videoPlay = new EventEmitter<void>();\n  @Output() videoPause = new EventEmitter<void>();\n\n  private _vplayer?: ElementRef<HTMLVideoElement>;\n\n  @ViewChild('vplayer')\n  set vplayerRef(ref: ElementRef<HTMLVideoElement> | undefined) {\n    if (!ref) return;\n\n    this._vplayer = ref;\n    this.onVideoElementReady();\n  }\n\n  get vplayer(): ElementRef<HTMLVideoElement> | undefined {\n    return this._vplayer;\n  }\n\n  @ViewChild('timelineBar', { static: false }) timelineBar!: ElementRef<HTMLDivElement>;\n  @ViewChild('speedControlContainer', { static: false }) speedControlContainer!: ElementRef<HTMLDivElement>;\n\n  progress = 0;\n  dragging = false;\n  isPlaying: boolean = false;\n  isFullScreen: boolean = false;\n  currentView: string = 'video';\n  currentVideoIndex: number = 0;\n  currentSpeed: string = '1x';\n  isSpeedControlOpen: boolean = false;\n\n  get hasDeviceFrame(): boolean {\n    return !!this.deviceFrameConfig;\n  }\n\n  segments: SegmentOption[] = [\n    { label: 'Screenshots', value: 'screenshots', icon: 'photo'  },\n    { label: 'Video', value: 'video', icon: 'videocam' },\n  ];\n\n  speedSegments: SegmentOption[] = [\n    { label: '1x', value: '1x' },\n    { label: '2x', value: '2x' },\n    { label: '5x', value: '5x' },\n  ];\n\n  private videoEventListenerCleanup: (() => void) | null = null;\n  private lastSetDuration: number = -1;\n  private dragMouseMoveHandler: ((e: MouseEvent) => void) | null = null;\n  private dragMouseUpHandler: ((e: MouseEvent) => void) | null = null;\n  private speedControlClickOutsideHandler: ((e: MouseEvent) => void) | null = null;\n  \n  safeTraceUrl: SafeResourceUrl;\n  traceViewerLoading: boolean = false;\n  traceViewerError: boolean = false;\n  isCorrectDeviceFrameAvailable: boolean = false;\n  private readonly emptyStateConfig = getEmptyStatePreset('nothingToDisplay');\n  readonly deviceErrorEmptyStateConfig = {\n    ...this.emptyStateConfig,\n    title: 'Device Not Supported',\n    description: 'The device you are trying to use is not supported.',\n    actions: []\n  };\n\n  get effectiveBrowserViewPort(): { width: number; height: number } {\n    const defaultViewport = { width: 1280, height: 720 };\n\n    if (!this.browserViewPort) {\n      return defaultViewport;\n    }\n\n    const width = Number((this.browserViewPort as any).width);\n    const height = Number((this.browserViewPort as any).height);\n\n    if (!Number.isFinite(width) || !Number.isFinite(height) || width <= 0 || height <= 0) {\n      return defaultViewport;\n    }\n\n    return { width, height };\n  }\n\n  private readonly deviceFrameConfigs: Record<string, DeviceFrameConfig> = {\n    // Pixels\n    'Pixel 5': {\n      mockupImage: 'assets/images/mockups/pixel-5.png',\n      screenInset: {\n        top: '2%',\n        right: '4.5%',\n        bottom: '2%',\n        left: '4.5%'\n      },\n      borderRadius: '4%'\n    },\n    'Pixel 8': {\n      mockupImage: 'assets/images/mockups/pixel-8.png',\n      screenInset: {\n        top: '2%',\n        right: '4.5%',\n        bottom: '2.5%',\n        left: '4.5%'\n      },\n      borderRadius: '4%'\n    },\n    'Pixel 9': {\n      mockupImage: 'assets/images/mockups/pixel-9-pro.webp',\n      screenInset: {\n        top: '4%',\n        right: '5%',\n        bottom: '4%',\n        left: '5%'\n      },\n      borderRadius: '8%'\n    },\n    // iPhones\n    'iPhone 11': {\n      mockupImage: 'assets/images/mockups/apple-iphone-11.png',\n      screenInset: {\n        top: '3%',\n        right: '7.5%',\n        bottom: '3%',\n        left: '7.5%'\n      },\n      borderRadius: '4%'\n    },\n    'iPhone 13': {\n      mockupImage: 'assets/images/mockups/apple-iphone-13.png',\n      screenInset: {\n        top: '2%',\n        right: '5%',\n        bottom: '2%',\n        left: '5%'\n      },\n      borderRadius: '4%'\n    },\n    'iPhone 15': {\n      mockupImage: 'assets/images/mockups/apple-iphone-15.png',\n      screenInset: {\n        top: '1.5%',\n        right: '4.5%',\n        bottom: '1.5%',\n        left: '4.5%'\n      },\n      borderRadius: '4%'\n    },\n    'iPhone XR': {\n      mockupImage: 'assets/images/mockups/apple-iphone-xr.png',\n      screenInset: {\n        top: '3.5%',\n        right: '7.5%',\n        bottom: '3.5%',\n        left: '7.5%'\n      },\n      borderRadius: '4%'\n    },\n\n    'Galaxy S20 FE': {\n      mockupImage: 'assets/images/mockups/samsung-galaxy-s20.png',\n      screenInset: {\n        top: '1.5%',\n        right: '2%',\n        bottom: '1.5%',\n        left: '2%'\n      },\n      borderRadius: '4%'\n    },\n\n    browser: {\n      mockupImage: 'assets/images/mockups/browser-mockup.png',\n      screenInset: {\n        top: '6%',\n        right: '0%',\n        bottom: '0%',\n        left: '0%'\n      },\n      borderRadius: '0%'\n    }\n  };\n\n  constructor(private sanitizer: DomSanitizer) {\n    this.safeTraceUrl = this.sanitizer.bypassSecurityTrustResourceUrl('');\n    this.updateSegments();\n  }\n\n  private get deviceFrameConfig(): DeviceFrameConfig | null {\n    if (this.platformType === 'browser') {\n      return this.deviceFrameConfigs['browser'] || null;\n    }\n\n    if (this.platformType === 'device') {\n      // If we have a deviceName and its frame is available, return it\n      if (this.deviceName && this.deviceFrameConfigs[this.deviceName]) {\n        return this.deviceFrameConfigs[this.deviceName];\n      }\n      \n      // If no deviceName or deviceName's frame not available, use defaults based on platform\n      if (this.platform === 'android') {\n        return this.deviceFrameConfigs['Pixel 9'] || null;\n      }\n      if (this.platform === 'ios') {\n        return this.deviceFrameConfigs['iPhone 15'] || null;\n      }\n      \n      return null;\n    }\n\n    return null;\n  }\n\n  get deviceMockupImage(): string {\n    return this.deviceFrameConfig ? this.deviceFrameConfig.mockupImage : '';\n  }\n\n  get deviceScreenStyle(): { [key: string]: string } {\n    const cfg = this.deviceFrameConfig;\n    if (!cfg) {\n      return {};\n    }\n\n    return {\n      position: 'absolute',\n      top: cfg.screenInset.top,\n      right: cfg.screenInset.right,\n      bottom: cfg.screenInset.bottom,\n      left: cfg.screenInset.left,\n      'border-radius': cfg.borderRadius,\n      overflow: 'hidden'\n    };\n  }\n\n  ngAfterViewInit(): void {\n    this.currentView = this.selectedView;\n    this.attachVideoListeners();\n    this.updateSafeTraceUrl();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['videoUrls']) {\n      const arr = changes['videoUrls'].currentValue as string[] | undefined;\n      if (arr && arr.length > 0) {\n        this.currentVideoIndex = 0;\n      }\n    }\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    if (changes['traceViewUrl']) {\n      this.updateSafeTraceUrl();\n      this.updateSegments();\n      if (!this.traceViewUrl && this.currentView === 'trace') {\n        this.currentView = this.hideVideoTab ? 'screenshots' : 'video';\n      }\n    }\n    if (changes['hideVideoTab']) {\n      this.updateSegments();\n    }\n    if (changes['selectedView']) {\n      this.currentView = changes['selectedView'].currentValue;\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.videoEventListenerCleanup) {\n      this.videoEventListenerCleanup();\n    }\n    this.removeDragListeners();\n    this.removeSpeedControlClickOutsideListener();\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    const video = this.vplayer?.nativeElement;\n    if (!video) return;\n\n    // Do not attempt to play if there is no valid source yet\n    if (!this.currentVideoUrl || video.networkState === HTMLMediaElement.NETWORK_NO_SOURCE || video.error) {\n      return;\n    }\n\n    if (this.isPlaying) {\n      video.pause();\n      this.isPlaying = false;\n      this.videoPause.emit();\n    } else {\n      video.play().then(() => {\n        this.isPlaying = true;\n        this.videoPlay.emit();\n      }).catch(err => {\n        console.error('Video play failed:', err);\n      });\n    }\n  }\n\n  get hasMultipleVideos(): boolean {\n    return Array.isArray(this.videoUrls) && this.videoUrls.length > 1;\n  }\n\n  get currentVideoUrl(): string {\n    if (Array.isArray(this.videoUrls) && this.videoUrls.length > 0) {\n      const idx = Math.min(Math.max(this.currentVideoIndex, 0), this.videoUrls.length - 1);\n      return this.videoUrls[idx];\n    }\n    return this.videoUrl;\n  }\n\n  private resetVideoState(): void {\n    this.isPlaying = false;\n    this.progress = 0;\n    if (this.vplayer?.nativeElement) {\n      this.vplayer.nativeElement.pause();\n      this.vplayer.nativeElement.currentTime = 0;\n    }\n    this.lastSetDuration = -1;\n  }\n\n  prevVideo(): void {\n    if (!this.hasMultipleVideos) return;\n    if (this.currentVideoIndex > 0) {\n      this.currentVideoIndex -= 1;\n      this.resetVideoState();\n    }\n  }\n\n  nextVideo(): void {\n    if (!this.hasMultipleVideos) return;\n    if (this.videoUrls && this.currentVideoIndex < this.videoUrls.length - 1) {\n      this.currentVideoIndex += 1;\n      this.resetVideoState();\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 duration = this.vplayer.nativeElement.duration;\n    if (!duration || !isFinite(duration)) {\n      return;\n    }\n\n    const targetTimeSeconds = percent * duration;\n    this.vplayer.nativeElement.currentTime = targetTimeSeconds;\n    this.lastSetDuration = targetTimeSeconds * 1000;\n  }\n\n  startDrag(event: MouseEvent): void {\n    event.preventDefault();\n    this.dragging = true;\n    this.addDragListeners();\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) return;\n\n    this.dragging = false;\n    this.removeDragListeners();\n\n    if (this.vplayer?.nativeElement) {\n      const duration = this.vplayer.nativeElement.duration;\n      if (!duration || !isFinite(duration)) {\n        return;\n      }\n\n      const targetTimeSeconds = (this.progress / 100) * duration;\n      this.vplayer.nativeElement.currentTime = targetTimeSeconds;\n      this.lastSetDuration = targetTimeSeconds * 1000;\n    }\n  }\n\n  private addDragListeners(): void {\n    this.removeDragListeners();\n\n    this.dragMouseMoveHandler = (e: MouseEvent) => {\n      this.onDrag(e);\n    };\n\n    this.dragMouseUpHandler = () => {\n      this.stopDrag();\n    };\n\n    document.addEventListener('mousemove', this.dragMouseMoveHandler);\n    document.addEventListener('mouseup', this.dragMouseUpHandler);\n  }\n\n  private removeDragListeners(): void {\n    if (this.dragMouseMoveHandler) {\n      document.removeEventListener('mousemove', this.dragMouseMoveHandler);\n      this.dragMouseMoveHandler = null;\n    }\n\n    if (this.dragMouseUpHandler) {\n      document.removeEventListener('mouseup', this.dragMouseUpHandler);\n      this.dragMouseUpHandler = null;\n    }\n  }\n\n  onVideoMetadataLoaded(): void {\n    this.attachVideoListeners();\n  }\n\n  onVideoCanPlay(): void {\n    this.attachVideoListeners();\n  }\n\n  onVideoEnded(): void {\n    this.isPlaying = false;\n    this.videoPause.emit();\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    this.isPlaying = false;\n    this.vplayer?.nativeElement?.pause();\n    this.progress = 0;\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  getStatusBadgeClass(): string {\n    const baseClasses = 'cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-px-[9px] cqa-py-[3px] cqa-rounded-[6px] cqa-h-[21px]';\n    switch (this.liveStatus) {\n      case 'In Progress':\n      case 'Execution':\n        return `${baseClasses} cqa-bg-[#E0F2FE] cqa-border cqa-border-[#7DD3FC]`;\n      case 'Paused':\n        return `${baseClasses} cqa-bg-[#FEF3C7] cqa-border cqa-border-[#FCD34D]`;\n      case 'Aborted':\n        return `${baseClasses} cqa-bg-[#FEE2E2] cqa-border cqa-border-[#FCA5A5]`;\n      case 'Failed':\n        return `${baseClasses} cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF]`;\n      case 'Passed':\n        return `${baseClasses} cqa-bg-[#D1FAE5] cqa-border cqa-border-[#6EE7B7]`;\n      default:\n        return `${baseClasses} cqa-bg-gray-200 cqa-border cqa-border-gray-300`;\n    }\n  }\n\n  getStatusTextClass(): string {\n    switch (this.liveStatus) {\n      case 'In Progress':\n      case 'Execution':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#0284C7] cqa-leading-[15px]';\n      case 'Paused':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#D97706] cqa-leading-[15px]';\n      case 'Aborted':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#DC2626] cqa-leading-[15px]';\n      case 'Failed':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]';\n      case 'Passed':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#059669] cqa-leading-[15px]';\n      default:\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-gray-600 cqa-leading-[15px]';\n    }\n  }\n\n  get processedTraceUrl(): string {\n    if (!this.traceViewUrl) return '';\n    \n    if (this.traceViewUrl.endsWith('.zip')) {\n      const urlParams = new URLSearchParams({\n        trace: this.traceViewUrl,\n        hideHeader: 'true',\n        hideBranding: 'true',\n        embed: 'true',\n        minimal: 'true',\n        noHeader: 'true'\n      });\n      \n      return `https://trace.playwright.dev/?${urlParams.toString()}`;\n    }\n    \n    return this.traceViewUrl;\n  }\n\n  private updateSafeTraceUrl(): void {\n    const url = this.processedTraceUrl;\n    if (url) {\n      this.traceViewerLoading = true;\n      this.traceViewerError = false;\n      this.safeTraceUrl = this.sanitizer.bypassSecurityTrustResourceUrl(url);\n    } else {\n      this.traceViewerLoading = false;\n      this.traceViewerError = false;\n      this.safeTraceUrl = this.sanitizer.bypassSecurityTrustResourceUrl('');\n    }\n  }\n\n  onTraceViewerLoad(): void {\n    this.traceViewerLoading = false;\n    this.traceViewerError = false;\n  }\n\n  onTraceViewerError(): void {\n    this.traceViewerLoading = false;\n    this.traceViewerError = true;\n  }\n\n  onSpeedChange(value: string): void {\n    this.currentSpeed = value;\n    const numeric = parseFloat(value.replace('x', ''));\n    const rate = !isNaN(numeric) && numeric > 0 ? numeric : 1;\n    if (this.vplayer?.nativeElement) {\n      this.vplayer.nativeElement.playbackRate = rate;\n    }\n    this.isSpeedControlOpen = false;\n  }\n\n  toggleSpeedControl(): void {\n    this.isSpeedControlOpen = !this.isSpeedControlOpen;\n    if (this.isSpeedControlOpen) {\n      this.addSpeedControlClickOutsideListener();\n    } else {\n      this.removeSpeedControlClickOutsideListener();\n    }\n  }\n\n  private addSpeedControlClickOutsideListener(): void {\n    this.removeSpeedControlClickOutsideListener();\n    \n    setTimeout(() => {\n      this.speedControlClickOutsideHandler = (e: MouseEvent) => {\n        const target = e.target as HTMLElement;\n        if (target && this.speedControlContainer?.nativeElement && \n            !this.speedControlContainer.nativeElement.contains(target)) {\n          this.isSpeedControlOpen = false;\n          this.removeSpeedControlClickOutsideListener();\n        }\n      };\n      \n      document.addEventListener('click', this.speedControlClickOutsideHandler);\n    }, 0);\n  }\n\n  private removeSpeedControlClickOutsideListener(): void {\n    if (this.speedControlClickOutsideHandler) {\n      document.removeEventListener('click', this.speedControlClickOutsideHandler);\n      this.speedControlClickOutsideHandler = null;\n    }\n  }\n\n  private updateSegments(): void {\n    const baseSegments: SegmentOption[] = [\n      { label: 'Screenshots', value: 'screenshots', icon: 'photo'  },\n    ];\n    \n    // Only add Video tab if hideVideoTab is false\n    if (!this.hideVideoTab) {\n      baseSegments.push({ label: 'Video', value: 'video', icon: 'videocam' });\n    }\n    \n    if (this.traceViewUrl) {\n      baseSegments.push({ label: 'Trace', value: 'trace', icon: 'graphic_eq' });\n    }\n    \n    this.segments = baseSegments;\n    \n    // If current view is 'video' and video tab is hidden, switch to screenshots\n    if (this.currentView === 'video' && this.hideVideoTab) {\n      this.currentView = 'screenshots';\n    }\n  }\n\n  private onVideoElementReady(): void {\n    const video = this._vplayer?.nativeElement;\n    if (!video) return;\n\n    const numeric = parseFloat(this.currentSpeed.replace('x', ''));\n    const rate = !isNaN(numeric) && numeric > 0 ? numeric : 1;\n    video.playbackRate = rate;\n\n    this.attachVideoListeners();\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 cqa-flex-wrap\">\n      <div class=\"cqa-flex cqa-items-center\">\n        <div *ngIf=\"isLive\" class=\"cqa-h-[21px] cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-mr-2 cqa-px-[9px] cqa-py-[3px] cqa-bg-[#FCD9D9] cqa-rounded-[6px]\" style=\"border: 1px solid #F9BFBF;\">\n          <span class=\"cqa-relative cqa-w-2 cqa-h-2 cqa-rounded-full cqa-bg-[#F47F7F]\" style=\"flex-shrink: 0;\">\n            <span class=\"cqa-absolute cqa-inset-0 cqa-rounded-full cqa-bg-[#F47F7F] cqa-opacity-75 cqa-animate-ping\"></span>\n          </span>\n          <span class=\"cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]\">Live</span>\n        </div>\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\">\n          {{ platformName }}\n          <span\n            *ngIf=\"platformType === 'browser'\"\n            class=\"cqa-ml-1\"\n            [matTooltip]=\"'Screen size: ' + effectiveBrowserViewPort.width + 'x' + effectiveBrowserViewPort.height\"\n            matTooltipPosition=\"below\"\n          >\n            ·\n            <span class=\"cqa-ml-1\">\n              {{ effectiveBrowserViewPort.width }}x{{ effectiveBrowserViewPort.height }}\n            </span>\n          </span>\n        </p>\n      </div>\n      <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n        <div *ngIf=\"isLive\" [ngClass]=\"getStatusBadgeClass()\">\n          <span [ngClass]=\"getStatusTextClass()\">{{ liveStatus }}</span>\n        </div>\n\n        <ng-container *ngIf=\"!isLive\">\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        </ng-container>\n      </div>\n    </div>\n  </div>\n  <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n  <!-- Live Content View -->\n    <div *ngIf=\"isLive\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center cqa-relative\">\n      <ng-container *ngIf=\"hasDeviceFrame;\">\n        <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-p-4\">\n          <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-max-h-full cqa-h-full': platformType !== 'browser' || !isLive, 'cqa-min-w-max': platformType === 'device'}\">\n            <img\n              [src]=\"deviceMockupImage\"\n              alt=\"Device mockup\"\n              class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n            />\n            <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-z-20': platformType === 'browser', 'cqa-bg-white': platformType !== 'browser'}\">\n              <!-- Loading State -->\n              <div *ngIf=\"isContentVideoLoading\" class=\"cqa-p-10 cqa-text-center cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n                <div class=\"cqa-mb-4\">\n                  <mat-progress-spinner mode=\"indeterminate\" diameter=\"40\"></mat-progress-spinner>\n                </div>\n                <p class=\"cqa-text-gray-400 cqa-text-sm\">{{ liveLoadingLabel }}</p>\n              </div>\n\n              <!-- Live Content (when not loading) -->\n              <div *ngIf=\"!isContentVideoLoading\" class=\"cqa-w-full cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-relative\">\n                <div *ngIf=\"liveStatus === 'Failed' && failedStatusMessage\" class=\"cqa-p-6 cqa-text-center cqa-w-full\">\n                  <div class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-4 cqa-py-3 cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF] cqa-rounded-lg\">\n                    <mat-icon style=\"width: 18px; height: 18px; color: #C63535; font-size: 18px;\">error</mat-icon>\n                    <p class=\"cqa-text-[#C63535] cqa-text-sm cqa-font-medium cqa-m-0\">{{ failedStatusMessage }}</p>\n                  </div>\n                </div>\n                <ng-content *ngIf=\"liveStatus !== 'Failed' || !failedStatusMessage\"></ng-content>\n              </div>\n            </div>\n          </div>\n        </div>\n      </ng-container>\n    </div>\n\n    <!-- Normal Video View (when not live) -->\n    <div *ngIf=\"!isLive && currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col\">\n      <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-83px)]\" *ngIf=\"currentVideoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device', 'cqa-mt-auto': hasDeviceFrame}\">\n        <ng-container *ngIf=\"hasDeviceFrame\">\n          <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n            <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n              <img\n                [src]=\"deviceMockupImage\"\n                alt=\"Device mockup\"\n                class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n              />\n              <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n                <video\n                  #vplayer\n                  class=\"cqa-object-cover cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n                  [src]=\"currentVideoUrl\"\n                  type=\"video/webm\"\n                  [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n                  (loadedmetadata)=\"onVideoMetadataLoaded()\"\n                  (canplay)=\"onVideoCanPlay()\"\n                  (ended)=\"onVideoEnded()\"\n                ></video>\n              </div>\n            </div>\n          </div>\n        </ng-container>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!currentVideoUrl\">\n        <ng-container *ngIf=\"isVNCSessionIntruppted && vncSessionIntupptedMessage; else noVideoDefault\">\n          <p class=\"cqa-text-sm cqa-text-gray-600\">\n            {{ vncSessionIntupptedMessage }}\n          </p>\n        </ng-container>\n        <ng-template #noVideoDefault>\n          <span>No video recording found</span>\n        </ng-template>\n      </div>\n    \n      <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" [ngClass]=\"{'cqa-mt-auto': hasDeviceFrame}\" *ngIf=\"currentVideoUrl && !isLive\">\n        <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n          <button \n            *ngIf=\"hasMultipleVideos\"\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            [disabled]=\"currentVideoIndex === 0\"\n            [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': currentVideoIndex === 0}\"\n            (click)=\"prevVideo()\"\n            matTooltip=\"Previous video\"\n            matTooltipPosition=\"above\">\n            <mat-icon class=\"cqa-w-4 cqa-h-4 !cqa-text-[16px] cqa-text-[#374151]\" [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': currentVideoIndex === 0}\">skip_previous</mat-icon>\n          </button>\n\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            matTooltip=\"{{ isPlaying ? 'Pause' : 'Play' }}\"\n            matTooltipPosition=\"above\">\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          <button \n            *ngIf=\"hasMultipleVideos\"\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            [disabled]=\"videoUrls && (currentVideoIndex >= videoUrls.length - 1)\"\n            [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': videoUrls && (currentVideoIndex >= videoUrls.length - 1)}\"\n            (click)=\"nextVideo()\"\n            matTooltip=\"Next video\"\n            matTooltipPosition=\"above\">\n            <mat-icon class=\"cqa-w-4 cqa-h-4 !cqa-text-[16px] cqa-text-[#374151]\" [ngClass]=\"{'cqa-opacity-50 cqa-cursor-not-allowed': videoUrls && (currentVideoIndex >= videoUrls.length - 1)}\">skip_next</mat-icon>\n          </button>\n\n          <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n            {{ formatTime(vplayer?.nativeElement?.currentTime || 0) }}\n          </span>\n\n          <div #speedControlContainer class=\"cqa-relative cqa-mr-[8px] cqa-flex cqa-items-center cqa-justify-center\">\n            <button\n              type=\"button\"\n              class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer cqa-text-[#9CA3AF] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-whitespace-nowrap cqa-select-none hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none cqa-px-1\"\n              (click)=\"toggleSpeedControl()\"\n              [matTooltip]=\"'Playback Speed'\"\n              [matTooltipPosition]=\"'below'\">\n              {{ currentSpeed }}\n            </button>\n            \n            <div \n              *ngIf=\"isSpeedControlOpen\"\n              class=\"cqa-absolute cqa-bottom-full cqa-mb-2 cqa-right-0 cqa-bg-[#F0F0F1] cqa-rounded-lg cqa-overflow-hidden cqa-shadow-lg cqa-z-50\"\n              style=\"min-width: max-content; left: 50%; bottom: 0%; transform: translate(-50%, -50%);\">\n              <cqa-segment-control\n                [segments]=\"speedSegments\"\n                [value]=\"currentSpeed\"\n                [containerBgColor]=\"'#F0F0F1'\"\n                (valueChange)=\"onSpeedChange($event)\">\n              </cqa-segment-control>\n            </div>\n          </div>\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            \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-blue-500 cqa-rounded-full\"\n              [style.width.%]=\"progress\"\n              [style.transition]=\"dragging ? 'none' : 'width 100ms'\"\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-blue-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: 40;\"\n              (mousedown)=\"startDrag($event)\">\n            </div>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <div *ngIf=\"!isLive && 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        <ng-container *ngIf=\"hasDeviceFrame\">\n          <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n            <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n              <img\n                [src]=\"deviceMockupImage\"\n                alt=\"Device mockup\"\n                class=\"cqa-h-full cqa-w-auto cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n                [ngClass]=\"{'cqa-max-h-[inherit]': platformType === 'browser', 'cqa-max-h-full': platformType !== 'browser'}\"\n              />\n              <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n                <img\n                  [src]=\"screenShotUrl\"\n                  alt=\"Screenshot\"\n                  [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n                  class=\"cqa-object-contain cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n                />\n              </div>\n            </div>\n          </div>\n        </ng-container>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!screenShotUrl\">\n        No screenshot available\n      </div>\n    </div>\n\n    <div *ngIf=\"!isLive && currentView === 'trace'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n      <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-relative\" *ngIf=\"traceViewUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\" style=\"padding-top: 48px; padding-bottom: 0px;\">\n        <div class=\"cqa-w-full cqa-h-full cqa-overflow-hidden cqa-relative\">\n          <iframe \n            [src]=\"safeTraceUrl\" \n            title=\"Trace Viewer\"\n            class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-[##F2F2F2]\"\n            style=\"margin-top: -48px; height: calc(100% + 48px);\"\n            frameborder=\"0\"\n            allowfullscreen\n            width=\"100%\"\n            loading=\"lazy\"\n            (load)=\"onTraceViewerLoad()\"\n            (error)=\"onTraceViewerError()\">\n          </iframe>\n        </div>\n        \n        <div *ngIf=\"traceViewerLoading\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n          <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n            Loading trace viewer...\n          </div>\n        </div>\n        \n        <div *ngIf=\"traceViewerError\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n          <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n            Failed to load trace viewer\n          </div>\n        </div>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!traceViewUrl\">\n        No trace available\n      </div>\n    </div>  \n  </div>\n</div>"]}
|
|
1797
|
+
//# 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';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { getEmptyStatePreset } from '../empty-state/empty-state-presets.constants';\n\nexport interface StepMarker {\n  cumulativeDuration: number;\n  result: 'SUCCESS' | 'FAILURE' | 'ABORTED' | 'SKIPPED';\n  testStepId?: number;\n  level?: number;\n  childSteps?: StepMarker[];\n  title?: string;\n}\n\nexport type SegmentType = 'screenshots' | 'video' | 'trace';\n\ntype SegmentOption = {\n  label: string;\n  value: string;\n  icon?: string;\n};\ninterface DeviceFrameConfig {\n  mockupImage: string;\n  screenInset: {\n    top: string;\n    right: string;\n    bottom: string;\n    left: string;\n  };\n  borderRadius: string;\n}\n\n@Component({\n  selector: 'cqa-simulator',\n  templateUrl: './simulator.component.html',\n  styleUrls: []\n})\n\nexport class SimulatorComponent implements AfterViewInit, OnDestroy, OnChanges {\n\n  @Input() videoUrl: string = '';\n  @Input() videoUrls: string[] = [];\n  @Input() videoCurrentDuration: number = 0;\n  @Input() stepMarkers: StepMarker[] = [];\n  @Input() screenShotUrl: string = '';\n  @Input() traceViewUrl: string = '';\n  @Input() platformName: string = 'Web - Chrome';\n  @Input() platformType: 'browser' | 'device' = 'browser';\n  @Input() platform: string = '';\n  @Input() deviceName: string | null = null;\n  @Input() isLive: boolean = false;\n  @Input() liveStatus: 'In Progress' | 'Paused' | 'Aborted' | 'Failed' | 'Passed' | 'Execution' = 'In Progress';\n  @Input() liveLoadingLabel: string = 'Loading...';\n  @Input() isContentVideoLoading: boolean = false;\n  @Input() failedStatusMessage: string = '';\n  @Input() isVNCSessionIntruppted: boolean = false;\n  @Input() vncSessionIntupptedMessage: string = '';\n  @Input() selectedView: SegmentType = 'video';\n  @Input() hideVideoTab: boolean = false;\n  @Input() browserViewPort: { width: number; height: number } | null = null;\n\n  @Output() videoTimeUpdate = new EventEmitter<number>();\n  @Output() videoPlay = new EventEmitter<void>();\n  @Output() videoPause = new EventEmitter<void>();\n  @Output() markerHit = new EventEmitter<StepMarker>();\n  @Output() isVideoPlayingChange = new EventEmitter<boolean>();\n\n  private _vplayer?: ElementRef<HTMLVideoElement>;\n\n  @ViewChild('vplayer')\n  set vplayerRef(ref: ElementRef<HTMLVideoElement> | undefined) {\n    if (!ref) return;\n\n    this._vplayer = ref;\n    this.onVideoElementReady();\n  }\n\n  get vplayer(): ElementRef<HTMLVideoElement> | undefined {\n    return this._vplayer;\n  }\n\n  private _timelineBar?: ElementRef<HTMLDivElement>;\n\n  @ViewChild('timelineBar', { static: false })\n  set timelineBarRef(ref: ElementRef<HTMLDivElement> | undefined) {\n    if (!ref) return;\n    this._timelineBar = ref;\n  }\n\n  get timelineBar(): ElementRef<HTMLDivElement> | undefined {\n    return this._timelineBar;\n  }\n\n  private _speedControlContainer?: ElementRef<HTMLDivElement>;\n\n  @ViewChild('speedControlContainer', { static: false })\n  set speedControlContainerRef(ref: ElementRef<HTMLDivElement> | undefined) {\n    if (!ref) return;\n    this._speedControlContainer = ref;\n  }\n\n  get speedControlContainer(): ElementRef<HTMLDivElement> | undefined {\n    return this._speedControlContainer;\n  }\n\n  progress = 0;\n  dragging = false;\n  isPlaying: boolean = false;\n  isFullScreen: boolean = false;\n  currentView: string = 'video';\n  currentVideoIndex: number = 0;\n  currentSpeed: string = '1x';\n  isSpeedControlOpen: boolean = false;\n  private detectedVideoDurations: number[] = []; // Actual durations detected from video metadata\n  private wasPlayingBeforeDrag: boolean = false; // Track if video was playing before drag started\n  private playPromise: Promise<void> | null = null; // Track the current play promise\n  private hitMarkers: Set<number> = new Set(); // Track which markers have been hit (by testStepId)\n\n  // State machine for robust media operations\n  private playerState: 'idle' | 'loading' | 'ready' | 'playing' | 'paused' | 'seeking' | 'switching' | 'error' = 'idle';\n  private operationQueue: Promise<void> = Promise.resolve(); // Serialize all media operations\n\n  get hasDeviceFrame(): boolean {\n    return !!this.deviceFrameConfig;\n  }\n\n  get isPlayerSwitching(): boolean {\n    return this.playerState === 'switching';\n  }\n\n  segments: SegmentOption[] = [\n    { label: 'Screenshots', value: 'screenshots', icon: 'photo'  },\n    { label: 'Video', value: 'video', icon: 'videocam' },\n  ];\n\n  speedSegments: SegmentOption[] = [\n    { label: '1x', value: '1x' },\n    { label: '2x', value: '2x' },\n    { label: '5x', value: '5x' },\n  ];\n\n  private videoEventListenerCleanup: (() => void) | null = null;\n  private lastSetDuration: number = -1;\n  private dragMouseMoveHandler: ((e: MouseEvent) => void) | null = null;\n  private dragMouseUpHandler: ((e: MouseEvent) => void) | null = null;\n  private speedControlClickOutsideHandler: ((e: MouseEvent) => void) | null = null;\n  private preloadVideoElement: HTMLVideoElement | null = null;\n  private preloadAllVideoElement: HTMLVideoElement | null = null;\n  private targetGlobalTimeDuringSwitch: number | null = null;\n  \n  safeTraceUrl: SafeResourceUrl;\n  traceViewerLoading: boolean = false;\n  traceViewerError: boolean = false;\n  isCorrectDeviceFrameAvailable: boolean = false;\n  private readonly emptyStateConfig = getEmptyStatePreset('nothingToDisplay');\n  readonly deviceErrorEmptyStateConfig = {\n    ...this.emptyStateConfig,\n    title: 'Device Not Supported',\n    description: 'The device you are trying to use is not supported.',\n    actions: []\n  };\n\n  get effectiveBrowserViewPort(): { width: number; height: number } {\n    const defaultViewport = { width: 1280, height: 720 };\n\n    if (!this.browserViewPort) {\n      return defaultViewport;\n    }\n\n    const width = Number((this.browserViewPort as any).width);\n    const height = Number((this.browserViewPort as any).height);\n\n    if (!Number.isFinite(width) || !Number.isFinite(height) || width <= 0 || height <= 0) {\n      return defaultViewport;\n    }\n\n    return { width, height };\n  }\n\n  private readonly deviceFrameConfigs: Record<string, DeviceFrameConfig> = {\n    // Pixels\n    'Pixel 5': {\n      mockupImage: 'assets/images/mockups/pixel-5.png',\n      screenInset: {\n        top: '2%',\n        right: '4.5%',\n        bottom: '2%',\n        left: '4.5%'\n      },\n      borderRadius: '4%'\n    },\n    'Pixel 8': {\n      mockupImage: 'assets/images/mockups/pixel-8.png',\n      screenInset: {\n        top: '2%',\n        right: '4.5%',\n        bottom: '2.5%',\n        left: '4.5%'\n      },\n      borderRadius: '4%'\n    },\n    'Pixel 9': {\n      mockupImage: 'assets/images/mockups/pixel-9-pro.webp',\n      screenInset: {\n        top: '4%',\n        right: '5%',\n        bottom: '4%',\n        left: '5%'\n      },\n      borderRadius: '8%'\n    },\n    // iPhones\n    'iPhone 11': {\n      mockupImage: 'assets/images/mockups/apple-iphone-11.png',\n      screenInset: {\n        top: '3%',\n        right: '7.5%',\n        bottom: '3%',\n        left: '7.5%'\n      },\n      borderRadius: '4%'\n    },\n    'iPhone 13': {\n      mockupImage: 'assets/images/mockups/apple-iphone-13.png',\n      screenInset: {\n        top: '2%',\n        right: '5%',\n        bottom: '2%',\n        left: '5%'\n      },\n      borderRadius: '4%'\n    },\n    'iPhone 15': {\n      mockupImage: 'assets/images/mockups/apple-iphone-15.png',\n      screenInset: {\n        top: '1.5%',\n        right: '4.5%',\n        bottom: '1.5%',\n        left: '4.5%'\n      },\n      borderRadius: '4%'\n    },\n    'iPhone XR': {\n      mockupImage: 'assets/images/mockups/apple-iphone-xr.png',\n      screenInset: {\n        top: '3.5%',\n        right: '7.5%',\n        bottom: '3.5%',\n        left: '7.5%'\n      },\n      borderRadius: '4%'\n    },\n\n    'Galaxy S20 FE': {\n      mockupImage: 'assets/images/mockups/samsung-galaxy-s20.png',\n      screenInset: {\n        top: '1.5%',\n        right: '2%',\n        bottom: '1.5%',\n        left: '2%'\n      },\n      borderRadius: '4%'\n    },\n\n    browser: {\n      mockupImage: 'assets/images/mockups/browser-mockup.png',\n      screenInset: {\n        top: '6%',\n        right: '0%',\n        bottom: '0%',\n        left: '0%'\n      },\n      borderRadius: '0%'\n    }\n  };\n\n  constructor(private sanitizer: DomSanitizer) {\n    this.safeTraceUrl = this.sanitizer.bypassSecurityTrustResourceUrl('');\n    this.updateSegments();\n  }\n\n  private get deviceFrameConfig(): DeviceFrameConfig | null {\n    if (this.platformType === 'browser') {\n      return this.deviceFrameConfigs['browser'] || null;\n    }\n\n    if (this.platformType === 'device') {\n      // If we have a deviceName and its frame is available, return it\n      if (this.deviceName && this.deviceFrameConfigs[this.deviceName]) {\n        return this.deviceFrameConfigs[this.deviceName];\n      }\n      \n      // If no deviceName or deviceName's frame not available, use defaults based on platform\n      if (this.platform === 'android') {\n        return this.deviceFrameConfigs['Pixel 9'] || null;\n      }\n      if (this.platform === 'ios') {\n        return this.deviceFrameConfigs['iPhone 15'] || null;\n      }\n      \n      return null;\n    }\n\n    return null;\n  }\n\n  get deviceMockupImage(): string {\n    return this.deviceFrameConfig ? this.deviceFrameConfig.mockupImage : '';\n  }\n\n  get deviceScreenStyle(): { [key: string]: string } {\n    const cfg = this.deviceFrameConfig;\n    if (!cfg) {\n      return {};\n    }\n\n    return {\n      position: 'absolute',\n      top: cfg.screenInset.top,\n      right: cfg.screenInset.right,\n      bottom: cfg.screenInset.bottom,\n      left: cfg.screenInset.left,\n      'border-radius': cfg.borderRadius,\n      overflow: 'hidden'\n    };\n  }\n\n  ngAfterViewInit(): void {\n    this.currentView = this.selectedView;\n    this.attachVideoListeners();\n    this.updateSafeTraceUrl();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['videoUrls']) {\n      const arr = changes['videoUrls'].currentValue as string[] | undefined;\n      if (arr && arr.length > 0) {\n        this.currentVideoIndex = 0;\n        this.detectVideoDurations(arr);\n        this.schedulePreloadAllSegments(arr);\n      }\n    }\n    if (changes['videoCurrentDuration'] && !changes['videoCurrentDuration'].firstChange) {\n      const newDuration = changes['videoCurrentDuration'].currentValue;\n      // Ignore sentinel value (-1) used to force change detection\n      if (newDuration === -1) {\n        return;\n      }\n      \n      if (this.isPlaying) {\n        this.pauseVideo();\n        console.log('[Simulator] ngOnChanges videoCurrentDuration: paused video before seeking');\n      }\n      \n      if (this.hasMultipleVideos) {\n        const targetVideoIndex = this.findVideoIndexForTimestamp(newDuration);\n        if (targetVideoIndex !== null) {\n          const boundaries = this.getVideoDurationBoundaries();\n          const targetBoundary = boundaries[targetVideoIndex];\n          const relativeTimestamp = newDuration - targetBoundary.start;\n          \n          const needsSwitch = this.currentVideoIndex !== targetVideoIndex;\n          \n          if (this.vplayer?.nativeElement) {\n            const currentVideoTimeMs = this.vplayer.nativeElement.currentTime * 1000;\n            const timeDifference = Math.abs(relativeTimestamp - currentVideoTimeMs);\n            \n            if (needsSwitch || newDuration !== this.lastSetDuration || timeDifference > 100) {\n              this.switchToVideoAndSeek(newDuration);\n            }\n          } else {\n            if (needsSwitch) {\n              this.currentVideoIndex = targetVideoIndex;\n            }\n          }\n        }\n      } else {\n        if (this.vplayer?.nativeElement) {\n          const currentVideoTimeMs = this.vplayer.nativeElement.currentTime * 1000;\n          const timeDifference = Math.abs(newDuration - currentVideoTimeMs);\n          if (newDuration !== this.lastSetDuration || timeDifference > 100) {\n            this.seekToTime(newDuration);\n          }\n        }\n      }\n    }\n    if (changes['traceViewUrl']) {\n      this.updateSafeTraceUrl();\n      this.updateSegments();\n      if (!this.traceViewUrl && this.currentView === 'trace') {\n        this.currentView = this.hideVideoTab ? 'screenshots' : 'video';\n      }\n    }\n    if (changes['hideVideoTab']) {\n      this.updateSegments();\n    }\n    if (changes['selectedView']) {\n      this.currentView = changes['selectedView'].currentValue;\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.videoEventListenerCleanup) {\n      this.videoEventListenerCleanup();\n    }\n    if (this.preloadVideoElement) {\n      this.preloadVideoElement.src = '';\n      this.preloadVideoElement.remove();\n      this.preloadVideoElement = null;\n    }\n    if (this.preloadAllVideoElement) {\n      this.preloadAllVideoElement.src = '';\n      this.preloadAllVideoElement.remove();\n      this.preloadAllVideoElement = null;\n    }\n    this.removeDragListeners();\n    this.removeSpeedControlClickOutsideListener();\n    console.log('[Simulator] ngOnDestroy: cleaned up listeners');\n  }\n\n  /**\n   * Seek video - public method (enqueues operation)\n   */\n  seekToTime(milliseconds: number): void {\n    if (!this.vplayer?.nativeElement) return;\n    this.enqueueOperation(() => this.seekToTimeInternal(milliseconds));\n  }\n\n  /**\n   * Seek to a global time on the combined timeline (converts to segment + local time)\n   */\n  private async seekToGlobalTime(globalTimeMs: number): Promise<void> {\n    const total = this.totalDuration;\n    if (total <= 0) return;\n\n    const clampedGlobal = Math.max(0, Math.min(globalTimeMs, total));\n    const boundaries = this.segmentBoundaries;\n\n    let segmentIndex = 0;\n    let segmentStart = 0;\n    for (let i = 0; i < boundaries.length; i++) {\n      if (clampedGlobal < boundaries[i]) {\n        segmentIndex = i;\n        segmentStart = i === 0 ? 0 : boundaries[i - 1];\n        break;\n      }\n      segmentIndex = i;\n      segmentStart = i === 0 ? 0 : boundaries[i - 1];\n    }\n    if (boundaries.length > 0 && clampedGlobal >= boundaries[boundaries.length - 1]) {\n      segmentIndex = boundaries.length - 1;\n      segmentStart = segmentIndex === 0 ? 0 : boundaries[segmentIndex - 1];\n    }\n\n    const localTimeMs = clampedGlobal - segmentStart;\n\n    if (this.currentVideoIndex === segmentIndex) {\n      await this.seekToTimeInternal(localTimeMs);\n      // Update progress after seek in same segment\n      if (this.totalDuration > 0) {\n        this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / this.totalDuration) * 100));\n      }\n    } else {\n      const cumulativeTimestamp = segmentStart + localTimeMs;\n      await this.switchToVideoAndSeekInternal(cumulativeTimestamp);\n      // Progress is already updated inside switchToVideoAndSeekInternal\n    }\n  }\n\n  /**\n   * Seek video internal implementation (no enqueue - called from queue)\n   * This prevents pipeline read errors during rapid seek operations\n   */\n  private async seekToTimeInternal(milliseconds: number): Promise<void> {\n    const video = this.vplayer?.nativeElement;\n    if (!video || !video.duration) return;\n\n    const seconds = milliseconds / 1000;\n    const targetTime = Math.max(0, Math.min(seconds, video.duration));\n\n    console.log('[Simulator] seekToTimeInternal: seeking', {\n      milliseconds,\n      targetTime,\n      playerState: this.playerState,\n      currentTimeBefore: video.currentTime\n    });\n    \n    // Set seeking state\n    this.playerState = 'seeking';\n\n    // CRITICAL: Pause before seeking to avoid pipeline errors\n      video.pause();\n    \n    // Update state\n    this.isPlaying = false;\n    this.playPromise = null;\n    this.videoPause.emit();\n    this.isVideoPlayingChange.emit(false);\n\n    // Perform seek\n    video.currentTime = targetTime;\n    this.lastSetDuration = milliseconds;\n    \n    // Wait briefly for seek to complete (prevents rapid seek issues)\n    await new Promise(resolve => setTimeout(resolve, 50));\n\n    const total = this.totalDuration;\n    if (total > 0) {\n      this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / total) * 100));\n    }\n    this.videoTimeUpdate.emit(this.globalCurrentTime);\n    \n    // Set to paused state (user must explicitly play)\n    this.playerState = 'paused';\n\n    console.log('[Simulator] seekToTimeInternal: seek complete', {\n      currentTimeAfter: video.currentTime,\n      playerState: this.playerState\n    });\n  }\n\n  /**\n   * Toggle play/pause (now trivial with state machine)\n   */\n  togglePlay(): void {\n    const video = this.vplayer?.nativeElement;\n    if (!video) {\n      console.error('[Simulator] togglePlay: video element not found', {\n        playerState: this.playerState,\n        isFullScreen: this.isFullScreen\n      });\n      return;\n    }\n\n    // Check for valid source\n    if (!this.currentVideoUrl || video.networkState === HTMLMediaElement.NETWORK_NO_SOURCE || video.error) {\n      console.error('[Simulator] togglePlay: no valid source', {\n        currentVideoUrl: this.currentVideoUrl,\n        networkState: video.networkState,\n        error: video.error\n      });\n      return;\n    }\n\n    // Recover from stuck 'loading' state when video is actually ready (e.g. after jump-to-segment)\n    if (this.playerState === 'loading' && video.readyState >= HTMLMediaElement.HAVE_METADATA) {\n      this.playerState = 'paused';\n    }\n\n    // Don't allow play during error state\n    if (this.playerState === 'error') {\n      console.warn('[Simulator] togglePlay: player in error state', { readyState: video.readyState });\n      this.playerState = 'idle';\n    }\n\n    // If still loading and video not ready, wait and retry\n    if (this.playerState === 'loading') {\n      console.warn('[Simulator] togglePlay: player not ready', {\n        playerState: this.playerState,\n        readyState: video.readyState\n      });\n      setTimeout(() => {\n        if (video.readyState >= HTMLMediaElement.HAVE_METADATA && this.playerState !== 'playing') {\n          this.togglePlay();\n        }\n      }, 50);\n      return;\n    }\n\n    console.log('[Simulator] togglePlay:', {\n      playerState: this.playerState,\n      isPlaying: this.isPlaying,\n      readyState: video.readyState\n    });\n\n    // Simple toggle based on state\n    if (this.playerState === 'playing') {\n      this.pauseVideo();\n    } else {\n      this.playVideo();\n    }\n  }\n\n  /**\n   * Play video - public method (enqueues operation)\n   */\n  private playVideo(): void {\n    this.enqueueOperation(() => this.playVideoInternal());\n  }\n\n  /**\n   * Play video internal implementation (no enqueue - called from queue)\n   * This eliminates FFmpegDemuxer errors and play promise interruptions\n   */\n  private async playVideoInternal(): Promise<void> {\n    const video = this.vplayer?.nativeElement;\n    if (!video) {\n      console.error('[Simulator] playVideoInternal: video element not found');\n      return;\n    }\n\n    // Already playing - skip\n    if (this.playerState === 'playing') {\n      console.log('[Simulator] playVideoInternal: already playing, skipping');\n      return;\n    }\n\n    if (this.hasMultipleVideos && this.videoUrls) {\n      const total = this.totalDuration;\n      const currentGlobalTime = this.globalCurrentTime;\n      const isAtEnd = total > 0 && (this.progress >= 100 || currentGlobalTime >= total);\n      const isLastVideo = this.currentVideoIndex >= this.videoUrls.length - 1;\n      const isLastVideoAtEnd = isLastVideo && video.duration && video.currentTime >= video.duration - 0.1;\n      \n      if (isAtEnd || isLastVideoAtEnd) {\n        console.log('[Simulator] playVideoInternal: all videos completed, resetting to first video', {\n          progress: this.progress,\n          currentGlobalTime,\n          total,\n          currentVideoIndex: this.currentVideoIndex,\n          isAtEnd,\n          isLastVideoAtEnd\n        });\n        \n        // Reset to first video at 0 progress\n        this.currentVideoIndex = 0;\n        this.progress = 0;\n        this.targetGlobalTimeDuringSwitch = 0;\n        this.hitMarkers.clear();\n        \n        // Switch to first video and seek to 0\n        await this.switchToVideoAndSeekInternal(0);\n        \n        // Update video reference after switch\n        const videoAfterSwitch = this.vplayer?.nativeElement;\n        if (!videoAfterSwitch) {\n          console.error('[Simulator] playVideoInternal: video element not found after switch');\n          return;\n        }\n        \n        // Continue with normal play logic below\n        // The video element reference has changed, so we'll use videoAfterSwitch\n        // But we need to wait for metadata again\n        if (videoAfterSwitch.readyState < HTMLMediaElement.HAVE_METADATA) {\n          await new Promise<void>((resolve, reject) => {\n            const onLoadedMetadata = () => {\n              videoAfterSwitch.removeEventListener('loadedmetadata', onLoadedMetadata);\n              videoAfterSwitch.removeEventListener('error', onError);\n              resolve();\n            };\n            const onError = () => {\n              videoAfterSwitch.removeEventListener('loadedmetadata', onLoadedMetadata);\n              videoAfterSwitch.removeEventListener('error', onError);\n              reject(new Error('Failed to load video metadata'));\n            };\n            videoAfterSwitch.addEventListener('loadedmetadata', onLoadedMetadata, { once: true });\n            videoAfterSwitch.addEventListener('error', onError, { once: true });\n            setTimeout(() => {\n              videoAfterSwitch.removeEventListener('loadedmetadata', onLoadedMetadata);\n              videoAfterSwitch.removeEventListener('error', onError);\n              if (videoAfterSwitch.readyState >= HTMLMediaElement.HAVE_METADATA) {\n                resolve();\n              } else {\n                reject(new Error('Timeout waiting for metadata'));\n              }\n            }, 500);\n          });\n        }\n        \n        // Set state to loading\n        this.playerState = 'loading';\n        \n        try {\n          await videoAfterSwitch.play();\n          this.playerState = 'playing';\n          this.isPlaying = true;\n          this.playPromise = null;\n          this.videoPlay.emit();\n          this.isVideoPlayingChange.emit(true);\n          this.targetGlobalTimeDuringSwitch = null;\n          \n          console.log('[Simulator] playVideoInternal: restarted from first video');\n          return;\n        } catch (err: any) {\n          this.playerState = 'error';\n          this.isPlaying = false;\n          this.playPromise = null;\n          this.isVideoPlayingChange.emit(false);\n          if (err.name === 'AbortError') {\n            this.playerState = 'paused';\n          } else {\n            console.error('[Simulator] playVideoInternal: failed after reset', err);\n          }\n          return;\n        }\n      }\n    }\n\n    console.log('[Simulator] playVideoInternal: starting', {\n      playerState: this.playerState,\n      currentTime: video.currentTime,\n      readyState: video.readyState\n    });\n\n    // Wait for video to be ready if needed\n    if (video.readyState < HTMLMediaElement.HAVE_METADATA) {\n      console.log('[Simulator] playVideoInternal: waiting for metadata');\n      await new Promise<void>((resolve, reject) => {\n        const onLoadedMetadata = () => {\n          video.removeEventListener('loadedmetadata', onLoadedMetadata);\n          video.removeEventListener('error', onError);\n          resolve();\n        };\n        const onError = () => {\n          video.removeEventListener('loadedmetadata', onLoadedMetadata);\n          video.removeEventListener('error', onError);\n          reject(new Error('Failed to load video metadata'));\n        };\n        video.addEventListener('loadedmetadata', onLoadedMetadata, { once: true });\n        video.addEventListener('error', onError, { once: true });\n\n        // Timeout fallback\n        setTimeout(() => {\n          video.removeEventListener('loadedmetadata', onLoadedMetadata);\n          video.removeEventListener('error', onError);\n          if (video.readyState >= HTMLMediaElement.HAVE_METADATA) {\n            resolve();\n          } else {\n            reject(new Error('Timeout waiting for metadata'));\n          }\n        }, 500);\n      });\n    }\n\n    // Clear hit markers if starting from beginning\n    const currentTimeMs = video.currentTime * 1000;\n    if (currentTimeMs < 100) {\n      this.hitMarkers.clear();\n    }\n\n    // Set state to loading\n    this.playerState = 'loading';\n\n    try {\n      // Verify video element is still available before playing\n      const currentVideo = this.vplayer?.nativeElement;\n      if (!currentVideo || currentVideo !== video) {\n        console.warn('[Simulator] playVideoInternal: video element changed or unavailable');\n        this.playerState = 'paused';\n        this.isPlaying = false;\n        this.playPromise = null;\n        return;\n      }\n\n      // Await play promise - this is critical\n      await video.play();\n      \n      // Verify video element is still available after play promise resolves\n      const videoAfterPlay = this.vplayer?.nativeElement;\n      if (!videoAfterPlay || videoAfterPlay !== video) {\n        console.warn('[Simulator] playVideoInternal: video element changed during play');\n        this.playerState = 'paused';\n        this.isPlaying = false;\n        this.playPromise = null;\n        return;\n      }\n      \n      // Play succeeded\n      this.playerState = 'playing';\n      this.isPlaying = true;\n      this.playPromise = null;\n      this.videoPlay.emit();\n      this.isVideoPlayingChange.emit(true);\n      \n      console.log('[Simulator] playVideoInternal: playing successfully', {\n        currentTime: video.currentTime,\n        playerState: this.playerState\n      });\n        \n      // Check early markers (non-blocking, doesn't need to be queued)\n      setTimeout(() => {\n        if (this.isPlaying) {\n          this.checkMarkerHit(this.globalCurrentTime);\n        }\n      }, 50);\n    } catch (err: any) {\n      // Play failed\n      // Check if video element is still available\n      const videoAfterError = this.vplayer?.nativeElement;\n      if (!videoAfterError || videoAfterError !== video) {\n        console.warn('[Simulator] playVideoInternal: video element changed during error');\n        this.playerState = 'paused';\n      } else {\n        this.playerState = 'error';\n      }\n      \n      this.isPlaying = false;\n      this.playPromise = null;\n      this.isVideoPlayingChange.emit(false);\n      \n      if (err.name === 'AbortError') {\n        console.log('[Simulator] playVideoInternal: aborted (expected during pause)');\n        // AbortError is expected, reset to paused state instead of error\n        if (videoAfterError && videoAfterError === video) {\n          this.playerState = 'paused';\n        }\n      } else {\n        console.error('[Simulator] playVideoInternal: failed', err);\n      }\n    }\n  }\n\n  /**\n   * Pause video - public method (enqueues operation)\n   */\n  private pauseVideo(): void {\n    this.enqueueOperation(() => this.pauseVideoInternal());\n  }\n\n  /**\n   * Pause video internal implementation (no enqueue - called from queue)\n   */\n  private async pauseVideoInternal(): Promise<void> {\n    const video = this.vplayer?.nativeElement;\n    if (!video) return;\n\n    // Already paused - skip\n    if (this.playerState === 'paused') {\n      console.log('[Simulator] pauseVideoInternal: already paused, skipping');\n      return;\n    }\n\n    console.log('[Simulator] pauseVideoInternal: pausing', {\n      playerState: this.playerState,\n      currentTime: video.currentTime\n    });\n\n      video.pause();\n    \n    this.playerState = 'paused';\n    this.isPlaying = false;\n      this.playPromise = null;\n    this.videoPause.emit();\n    this.isVideoPlayingChange.emit(false);\n    \n    console.log('[Simulator] pauseVideoInternal: paused successfully');\n  }\n\n  /**\n   * Enqueue media operation to prevent race conditions\n   * This is the CRITICAL fix for FFmpegDemuxer errors and play promise interruptions\n   * All media operations (play, pause, seek, switch) go through this queue\n   */\n  private enqueueOperation(operation: () => Promise<void>): void {\n    this.operationQueue = this.operationQueue\n      .then(() => operation())\n      .catch(err => {\n        console.error('[Simulator] Media operation failed:', err);\n        this.playerState = 'error';\n      });\n  }\n\n  get hasMultipleVideos(): boolean {\n    return Array.isArray(this.videoUrls) && this.videoUrls.length > 1;\n  }\n\n  /** Cumulative duration boundaries [ms]. segmentBoundaries[i] = sum of durations [0..i] */\n  get segmentBoundaries(): number[] {\n    const durations = this.videoDurations;\n    if (durations.length === 0) return [];\n    const boundaries: number[] = [];\n    let sum = 0;\n    for (const d of durations) {\n      sum += d;\n      boundaries.push(sum);\n    }\n    return boundaries;\n  }\n\n  /** Total duration across all segments in ms */\n  get totalDuration(): number {\n    const durations = this.videoDurations;\n    if (durations.length === 0) return 0;\n    return durations.reduce((a, b) => a + b, 0);\n  }\n\n  /** Video durations in ms (detected or from single video element) */\n  private get videoDurations(): number[] {\n    if (this.hasMultipleVideos && this.detectedVideoDurations.length > 0) {\n      return this.detectedVideoDurations;\n    }\n    const video = this.vplayer?.nativeElement;\n    if (video?.duration && isFinite(video.duration)) {\n      return [video.duration * 1000];\n    }\n    return [];\n  }\n\n  get globalCurrentTime(): number {\n    if (this.targetGlobalTimeDuringSwitch !== null && !this.isPlaying && this.playerState !== 'playing') {\n      return this.targetGlobalTimeDuringSwitch;\n    }\n    const video = this.vplayer?.nativeElement;\n    if (!video?.duration || !isFinite(video.duration)) {\n      // Fallback to override if video not ready yet\n      if (this.targetGlobalTimeDuringSwitch !== null) {\n        return this.targetGlobalTimeDuringSwitch;\n      }\n      return 0;\n    }\n    const segmentStart = this.currentVideoIndex === 0 ? 0 : (this.segmentBoundaries[this.currentVideoIndex - 1] ?? 0);\n    return segmentStart + video.currentTime * 1000;\n  }\n\n  private getActualGlobalTime(): number {\n    const video = this.vplayer?.nativeElement;\n    if (!video?.duration || !isFinite(video.duration)) return 0;\n    const segmentStart = this.currentVideoIndex === 0 ? 0 : (this.segmentBoundaries[this.currentVideoIndex - 1] ?? 0);\n    return segmentStart + video.currentTime * 1000;\n  }\n\n  /** Global step markers for rendering on the combined timeline */\n  get globalStepMarkers(): Array<{ globalTime: number; result: StepMarker['result']; testStepId?: number; title?: string; level?: number }> {\n    const allMarkers = this.flattenMarkers(this.stepMarkers);\n    return allMarkers.map(m => ({\n      globalTime: m.cumulativeDuration,\n      result: m.result,\n      testStepId: m.testStepId,\n      title: m.title,\n      level: m.level\n    }));\n  }\n\n  get currentVideoUrl(): string {\n    if (Array.isArray(this.videoUrls) && this.videoUrls.length > 0) {\n      const idx = Math.min(Math.max(this.currentVideoIndex, 0), this.videoUrls.length - 1);\n      return this.videoUrls[idx];\n    }\n    return this.videoUrl;\n  }\n\n  /**\n   * Reset video state - public method (enqueues operation)\n   */\n  private resetVideoState(): void {\n    this.enqueueOperation(() => this.resetVideoStateInternal());\n  }\n\n  /**\n   * Reset video state internal implementation (no enqueue - called from queue)\n   */\n  private async resetVideoStateInternal(): Promise<void> {\n    const video = this.vplayer?.nativeElement;\n    if (!video) return;\n\n    console.log('[Simulator] resetVideoStateInternal: resetting');\n\n    video.pause();\n    video.currentTime = 0;\n\n    this.playerState = 'paused';\n    this.isPlaying = false;\n    this.progress = 0;\n    this.playPromise = null;\n    this.hitMarkers.clear();\n    this.lastSetDuration = -1;\n    this.isVideoPlayingChange.emit(false);\n\n    console.log('[Simulator] resetVideoStateInternal: complete');\n  }\n\n  prevVideo(): void {\n    if (!this.hasMultipleVideos) return;\n    if (this.currentVideoIndex > 0) {\n      this.currentVideoIndex -= 1;\n      this.resetVideoState();\n    }\n  }\n\n  nextVideo(): void {\n    if (!this.hasMultipleVideos) return;\n    if (this.videoUrls && this.currentVideoIndex < this.videoUrls.length - 1) {\n      this.currentVideoIndex += 1;\n      this.resetVideoState();\n    }\n  }\n\n  onTimelineClick(event: MouseEvent): void {\n    if (!this.timelineBar?.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 total = this.totalDuration;\n    if (total <= 0) return;\n\n    const globalTimeMs = percent * total;\n    const shouldResumePlaying = this.isPlaying;\n\n    console.log('[Simulator] onTimelineClick', {\n      percent,\n      globalTimeMs,\n      shouldResumePlaying\n    });\n    \n    this.enqueueOperation(async () => {\n      await this.seekToGlobalTime(globalTimeMs);\n      if (shouldResumePlaying) {\n        await new Promise(resolve => setTimeout(resolve, 100));\n        await this.playVideoInternal();\n      }\n    });\n  }\n\n  startDrag(event: MouseEvent): void {\n    event.preventDefault();\n    this.dragging = true;\n    \n    // CRITICAL: Capture playing state BEFORE pausing\n    // Use both isPlaying flag and playerState for reliability\n    this.wasPlayingBeforeDrag = this.isPlaying || this.playerState === 'playing';\n    \n    console.log('[Simulator] startDrag', {\n      clientX: event.clientX,\n      playerState: this.playerState,\n      isPlaying: this.isPlaying,\n      wasPlayingBeforeDrag: this.wasPlayingBeforeDrag\n    });\n    \n    // Pause video during drag to avoid conflicts\n    if (this.isPlaying || this.playerState === 'playing') {\n      this.pauseVideo();\n    }\n    \n    this.addDragListeners();\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; // Global timeline position\n  }\n\n  stopDrag(): void {\n    if (!this.dragging) return;\n\n    this.dragging = false;\n    this.removeDragListeners();\n    \n    // CRITICAL: Capture the playing state BEFORE any async operations\n    const shouldResumePlaying = this.wasPlayingBeforeDrag;\n    \n    console.log('[Simulator] stopDrag: drag ended', {\n      progress: this.progress,\n      wasPlayingBeforeDrag: this.wasPlayingBeforeDrag,\n      shouldResumePlaying: shouldResumePlaying,\n      currentPlayerState: this.playerState\n    });\n\n    const total = this.totalDuration;\n    if (total <= 0) return;\n\n    const globalTimeMs = (this.progress / 100) * total;\n\n    // CRITICAL FIX: Queue both seek and resume play together\n    // This ensures seek completes before attempting to play\n    this.enqueueOperation(async () => {\n      // Perform seek (this will pause the video)\n      await this.seekToGlobalTime(globalTimeMs);\n      \n      // If video was playing before drag, resume it\n      if (shouldResumePlaying) {\n        console.log('[Simulator] stopDrag: resuming playback after seek', {\n          globalTimeMs,\n          playerState: this.playerState\n        });\n        \n        // Wait a bit longer to ensure seek is fully complete\n        await new Promise(resolve => setTimeout(resolve, 150));\n        \n        // Verify state is 'paused' after seek (seekToTimeInternal sets it to 'paused')\n        // If state is still 'seeking', wait a bit more\n        if (this.playerState === 'seeking') {\n          console.log('[Simulator] stopDrag: seek still in progress, waiting more');\n          await new Promise(resolve => setTimeout(resolve, 100));\n        }\n        \n        // Now resume playback (state should be 'paused' at this point)\n        if (this.playerState === 'paused') {\n          await this.playVideoInternal();\n          console.log('[Simulator] stopDrag: playback resumed successfully', {\n            playerState: this.playerState,\n            isPlaying: this.isPlaying\n          });\n        } else {\n          console.warn('[Simulator] stopDrag: unexpected state after seek, attempting to resume anyway', {\n            playerState: this.playerState\n          });\n          // Try to resume anyway - better to try than to leave it stuck\n          await this.playVideoInternal();\n        }\n      } else {\n        console.log('[Simulator] stopDrag: video was paused, not resuming');\n      }\n    });\n  }\n\n  private addDragListeners(): void {\n    this.removeDragListeners();\n\n    this.dragMouseMoveHandler = (e: MouseEvent) => {\n      this.onDrag(e);\n    };\n\n    this.dragMouseUpHandler = () => {\n      this.stopDrag();\n    };\n\n    document.addEventListener('mousemove', this.dragMouseMoveHandler);\n    document.addEventListener('mouseup', this.dragMouseUpHandler);\n  }\n\n  private removeDragListeners(): void {\n    if (this.dragMouseMoveHandler) {\n      document.removeEventListener('mousemove', this.dragMouseMoveHandler);\n      this.dragMouseMoveHandler = null;\n    }\n\n    if (this.dragMouseUpHandler) {\n      document.removeEventListener('mouseup', this.dragMouseUpHandler);\n      this.dragMouseUpHandler = null;\n    }\n  }\n\n  onVideoMetadataLoaded(): void {\n    console.log('[Simulator] onVideoMetadataLoaded');\n    // Ensure playback speed is consistent across all videos\n    this.applyCurrentPlaybackRate();\n    this.attachVideoListeners();\n  }\n  \n  onVideoCanPlay(): void {\n    console.log('[Simulator] onVideoCanPlay');\n    // Ensure playback speed is consistent across all videos\n    this.applyCurrentPlaybackRate();\n    this.attachVideoListeners();\n  }\n\n  onVideoEnded(): void {\n    this.isPlaying = false;\n    this.playerState = 'paused';\n    this.videoPause.emit();\n    this.isVideoPlayingChange.emit(false);\n    \n    // Seamlessly load and play next segment\n    if (this.hasMultipleVideos && this.videoUrls && this.currentVideoIndex < this.videoUrls.length - 1) {\n      const nextIndex = this.currentVideoIndex + 1;\n      const segmentStart = nextIndex === 0 ? 0 : (this.segmentBoundaries[nextIndex - 1] ?? 0);\n      \n      this.enqueueOperation(async () => {\n        await this.switchToVideoAndSeekInternal(segmentStart);\n        if (this.vplayer?.nativeElement && this.currentVideoUrl) {\n          await this.playVideoInternal();\n        }\n      });\n    }\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 && this.playerState !== 'switching') {\n          const total = this.totalDuration;\n          \n          // Clear targetGlobalTimeDuringSwitch if we've reached the target position\n          // When playing, be more aggressive about clearing to allow live updates\n          if (this.targetGlobalTimeDuringSwitch !== null) {\n            const currentGlobal = this.getActualGlobalTime();\n            const tolerance = this.isPlaying ? 1000 : 500; // More tolerance when playing\n            if (Math.abs(currentGlobal - this.targetGlobalTimeDuringSwitch) < tolerance) {\n              this.targetGlobalTimeDuringSwitch = null;\n            }\n          }\n          \n          // globalCurrentTime now handles override logic - use it directly\n          if (total > 0) {\n            this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / total) * 100));\n          }\n          this.videoTimeUpdate.emit(this.globalCurrentTime);\n          \n          // Check if current time hits any marker\n          this.checkMarkerHit(this.globalCurrentTime);\n\n          // Preload next segment when ~5-10 seconds from ending\n          this.maybePreloadNextSegment();\n        }\n      };\n\n      video.addEventListener('timeupdate', handler);\n\n      this.videoEventListenerCleanup = () => {\n        video.removeEventListener('timeupdate', handler);\n      };\n    }, 100);\n  }\n\n  private schedulePreloadAllSegments(videoUrls: string[]): void {\n    if (!videoUrls || videoUrls.length === 0) return;\n    this.cancelPreloadAllSegments();\n    setTimeout(() => this.preloadAllSegments(videoUrls), 500);\n  }\n\n  private cancelPreloadAllSegments(): void {\n    if (this.preloadAllVideoElement) {\n      this.preloadAllVideoElement.src = '';\n      this.preloadAllVideoElement.remove();\n      this.preloadAllVideoElement = null;\n    }\n  }\n\n  /**\n   * Preload all video URLs in sequence (first, then second, ...) so they are cached.\n   * When user jumps to any segment, that video is likely already loaded.\n   */\n  private preloadAllSegments(videoUrls: string[]): void {\n    if (!videoUrls || videoUrls.length === 0) return;\n    this.cancelPreloadAllSegments();\n    const video = document.createElement('video');\n    video.preload = 'auto';\n    video.muted = true;\n    video.playsInline = true;\n    video.style.display = 'none';\n    document.body.appendChild(video);\n    this.preloadAllVideoElement = video;\n\n    let index = 0;\n    const loadNext = (): void => {\n      if (this.preloadAllVideoElement !== video || index >= videoUrls.length) {\n        video.remove();\n        if (this.preloadAllVideoElement === video) this.preloadAllVideoElement = null;\n        return;\n      }\n      const url = videoUrls[index];\n      video.src = url;\n      index += 1;\n      const onDone = (): void => {\n        clearTimeout(timeoutId);\n        video.removeEventListener('loadeddata', onDone);\n        video.removeEventListener('error', onDone);\n        loadNext();\n      };\n      video.addEventListener('loadeddata', onDone, { once: true });\n      video.addEventListener('error', onDone, { once: true });\n      const timeoutId = setTimeout(() => onDone(), 15000);\n    };\n    loadNext();\n  }\n\n  /** Preload next segment when ~5-10 seconds from ending to minimize switching delay */\n  private maybePreloadNextSegment(): void {\n    if (!this.hasMultipleVideos || !this.videoUrls || this.currentVideoIndex >= this.videoUrls.length - 1) {\n      return;\n    }\n    const video = this.vplayer?.nativeElement;\n    if (!video?.duration) return;\n    const timeLeftInSegment = (video.duration - video.currentTime) * 1000; // ms\n    if (timeLeftInSegment > 10000 || timeLeftInSegment < 5000) return; // Preload when 5-10s from end\n    const nextUrl = this.videoUrls[this.currentVideoIndex + 1];\n    if (!nextUrl) return;\n    if (this.preloadVideoElement?.src === nextUrl) return; // Already preloading\n    this.preloadNextSegment(nextUrl);\n  }\n\n  private preloadNextSegment(url: string): void {\n    if (this.preloadVideoElement) {\n      this.preloadVideoElement.src = '';\n      this.preloadVideoElement.remove();\n      this.preloadVideoElement = null;\n    }\n    const video = document.createElement('video');\n    video.preload = 'auto';\n    video.src = url;\n    video.style.display = 'none';\n    document.body.appendChild(video);\n    this.preloadVideoElement = video;\n    video.addEventListener('loadeddata', () => {\n      video.remove();\n      if (this.preloadVideoElement === video) this.preloadVideoElement = null;\n    }, { once: true });\n  }\n\n  /**\n   * Detect actual video durations by loading video metadata\n   * This creates temporary video elements to read duration without playing\n   */\n  private detectVideoDurations(videoUrls: string[]): void {\n    if (!videoUrls || videoUrls.length === 0) {\n      this.detectedVideoDurations = [];\n      return;\n    }\n\n    // Reset detected durations\n    this.detectedVideoDurations = [];\n    let loadedCount = 0;\n\n    videoUrls.forEach((url, index) => {\n      const tempVideo = document.createElement('video');\n      tempVideo.preload = 'metadata';\n      \n      const onLoadedMetadata = () => {\n        const durationMs = tempVideo.duration * 1000;\n        this.detectedVideoDurations[index] = durationMs;\n        loadedCount++;\n        \n        // Clean up\n        tempVideo.removeEventListener('loadedmetadata', onLoadedMetadata);\n        tempVideo.removeEventListener('error', onError);\n        tempVideo.src = '';\n        tempVideo.remove();\n        \n        if (loadedCount === videoUrls.length) {\n          console.log('[Simulator] All video durations detected:', this.detectedVideoDurations);\n        }\n      };\n      \n      const onError = () => {\n        console.warn(`[Simulator] Failed to load metadata for video ${index + 1}`);\n        this.detectedVideoDurations[index] = 0;\n        loadedCount++;\n        tempVideo.removeEventListener('loadedmetadata', onLoadedMetadata);\n        tempVideo.removeEventListener('error', onError);\n        tempVideo.src = '';\n        tempVideo.remove();\n      };\n      \n      tempVideo.addEventListener('loadedmetadata', onLoadedMetadata);\n      tempVideo.addEventListener('error', onError);\n      tempVideo.src = url;\n    });\n  }\n\n  /**\n   * Get cumulative duration boundaries for each video\n   * Returns an array where each element is [startDuration, endDuration] for that video\n   * Uses detected video durations if available, otherwise falls back to videoDurations input\n   */\n  private getVideoDurationBoundaries(): Array<{ start: number; end: number }> {\n    if (!this.hasMultipleVideos) {\n      return [];\n    }\n    \n    // Use detected durations if available, otherwise use provided durations\n    const durations = this.detectedVideoDurations.length > 0 \n      ? this.detectedVideoDurations \n      : [];\n    \n    if (durations.length === 0) {\n      return [];\n    }\n    \n    const boundaries: Array<{ start: number; end: number }> = [];\n    let cumulativeStart = 0;\n    \n    for (const duration of durations) {\n      boundaries.push({\n        start: cumulativeStart,\n        end: cumulativeStart + duration\n      });\n      cumulativeStart += duration;\n    }\n    \n    return boundaries;\n  }\n\n  /**\n   * Flatten all markers recursively including childSteps\n   */\n  private flattenMarkers(markers: StepMarker[]): StepMarker[] {\n    const flattened: StepMarker[] = [];\n    \n    const flattenRecursive = (markers: StepMarker[]) => {\n      markers.forEach(marker => {\n        // Add the marker itself\n        flattened.push(marker);\n        \n        // Recursively add child markers\n        if (marker.childSteps && marker.childSteps.length > 0) {\n          flattenRecursive(marker.childSteps);\n        }\n      });\n    };\n    \n    flattenRecursive(markers);\n    return flattened;\n  }\n\n  private checkMarkerHit(globalTimeMs: number): void {\n    if (!this.isPlaying || this.dragging) return;\n    \n    // Check all markers (including nested ones) for a match\n    const allMarkers = this.flattenMarkers(this.stepMarkers);\n    const tolerance = 200; // 200ms tolerance for matching\n    \n    for (const marker of allMarkers) {\n      // Skip if marker doesn't have testStepId\n      if (!marker.testStepId) continue;\n      \n      // Skip if already hit\n      if (this.hitMarkers.has(marker.testStepId)) continue;\n      \n      // Check if current time matches marker time (within tolerance)\n      const timeDifference = Math.abs(globalTimeMs - marker.cumulativeDuration);\n      if (timeDifference <= tolerance) {\n        // Mark as hit and emit event\n        this.hitMarkers.add(marker.testStepId);\n        this.markerHit.emit(marker);\n        console.log('[Simulator] Marker hit detected', {\n          testStepId: marker.testStepId,\n          cumulativeDuration: marker.cumulativeDuration,\n          currentTime: globalTimeMs,\n          timeDifference,\n          title: marker.title,\n          level: marker.level\n        });\n        break; // Only emit one marker per check\n      }\n    }\n  }\n\n  getGlobalMarkerResultColor(result: StepMarker['result']): string {\n    switch (result) {\n      case 'SUCCESS':\n        return '#12B76A';\n      case 'FAILURE':\n        return '#B91C1C';\n      case 'ABORTED':\n        return '#F79009';\n      case 'SKIPPED':\n        return '#667085';\n      default:\n        return '#6366F1';\n    }\n  }\n\n  getGlobalMarkerColor(level?: number): string {\n    return level === 1 ? '#000000' : '#FFA500';\n  }\n\n  onMarkerClick(event: MouseEvent, marker: { globalTime: number; result: StepMarker['result']; testStepId?: number }): void {\n    event.stopPropagation();\n    event.preventDefault();\n    this.enqueueOperation(async () => {\n      await this.seekToGlobalTime(marker.globalTime);\n      if (marker.testStepId != null) {\n        this.hitMarkers.add(marker.testStepId);\n        const fullMarker = this.flattenMarkers(this.stepMarkers).find(m => m.testStepId === marker.testStepId);\n        if (fullMarker) this.markerHit.emit(fullMarker);\n      }\n    });\n  }\n\n  toggleFullScreen(): void {\n    const wasPlaying = this.isPlaying;\n    const currentTime = this.vplayer?.nativeElement?.currentTime || 0;\n    \n    // Pause video before fullscreen transition to avoid state issues\n    if (wasPlaying) {\n      this.pauseVideo();\n    }\n    \n    // Toggle fullscreen\n    this.isFullScreen = !this.isFullScreen;\n    \n    // Reset player state to allow re-initialization after DOM update\n    // The state might be stuck in 'loading' or 'error' after fullscreen transition\n    if (this.playerState === 'loading' || this.playerState === 'error') {\n      this.playerState = 'idle';\n    }\n    \n    // Wait for Angular to update the DOM, then re-initialize video\n    setTimeout(() => {\n      const video = this.vplayer?.nativeElement;\n      if (video) {\n        // Restore video position if it was set\n        if (currentTime > 0 && video.duration > currentTime) {\n          video.currentTime = currentTime;\n        }\n        \n        // Re-apply playback rate\n        this.applyCurrentPlaybackRate();\n        \n        // Re-attach listeners\n        this.attachVideoListeners();\n        \n        // Reset state to 'paused' (ready to play)\n        this.playerState = 'paused';\n        this.isPlaying = false;\n        \n        console.log('[Simulator] toggleFullScreen: video re-initialized', {\n          isFullScreen: this.isFullScreen,\n          currentTime: video.currentTime,\n          playerState: this.playerState\n        });\n      } else {\n        console.warn('[Simulator] toggleFullScreen: video element not found after transition');\n        // Try again after a bit more time\n        setTimeout(() => {\n          const videoRetry = this.vplayer?.nativeElement;\n          if (videoRetry) {\n            this.applyCurrentPlaybackRate();\n            this.attachVideoListeners();\n            this.playerState = 'paused';\n            this.isPlaying = false;\n          }\n        }, 100);\n      }\n    }, 100);\n  }\n\n  onSegmentChange(value: string): void {\n    this.currentView = value;\n    // Pause video when switching away from video view\n    if (this.currentView !== 'video') {\n      this.pauseVideo();\n      this.progress = 0;\n    } else {\n      // Reset everything when switching back to video view\n      this.enqueueOperation(async () => {\n        await this.resetVideoStateInternal();\n        \n        // If multiple videos, reset to first video\n        if (this.hasMultipleVideos && this.videoUrls) {\n          this.currentVideoIndex = 0;\n          this.targetGlobalTimeDuringSwitch = 0;\n          \n          // Switch to first video and seek to 0\n          await this.switchToVideoAndSeekInternal(0);\n        }\n      });\n    }\n  }\n\n  formatTime(seconds: number): string {\n    if (!seconds || isNaN(seconds)) return '00:00';\n    \n    const totalSeconds = Math.floor(seconds);\n    const hours = Math.floor(totalSeconds / 3600);\n    const mins = Math.floor((totalSeconds % 3600) / 60);\n    const secs = totalSeconds % 60;\n    \n    // If hours > 0, show HH:MM:SS format, otherwise show MM:SS format\n    if (hours > 0) {\n      return `${hours.toString().padStart(2, '0')}:${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;\n    }\n    \n    return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;\n  }\n\n  getStatusBadgeClass(): string {\n    const baseClasses = 'cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-px-[9px] cqa-py-[3px] cqa-rounded-[6px] cqa-h-[21px]';\n    switch (this.liveStatus) {\n      case 'In Progress':\n      case 'Execution':\n        return `${baseClasses} cqa-bg-[#E0F2FE] cqa-border cqa-border-[#7DD3FC]`;\n      case 'Paused':\n        return `${baseClasses} cqa-bg-[#FEF3C7] cqa-border cqa-border-[#FCD34D]`;\n      case 'Aborted':\n        return `${baseClasses} cqa-bg-[#FEE2E2] cqa-border cqa-border-[#FCA5A5]`;\n      case 'Failed':\n        return `${baseClasses} cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF]`;\n      case 'Passed':\n        return `${baseClasses} cqa-bg-[#D1FAE5] cqa-border cqa-border-[#6EE7B7]`;\n      default:\n        return `${baseClasses} cqa-bg-gray-200 cqa-border cqa-border-gray-300`;\n    }\n  }\n\n  getStatusTextClass(): string {\n    switch (this.liveStatus) {\n      case 'In Progress':\n      case 'Execution':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#0284C7] cqa-leading-[15px]';\n      case 'Paused':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#D97706] cqa-leading-[15px]';\n      case 'Aborted':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#DC2626] cqa-leading-[15px]';\n      case 'Failed':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]';\n      case 'Passed':\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-[#059669] cqa-leading-[15px]';\n      default:\n        return 'cqa-text-[10px] cqa-font-medium cqa-text-gray-600 cqa-leading-[15px]';\n    }\n  }\n\n  get processedTraceUrl(): string {\n    if (!this.traceViewUrl) return '';\n    \n    if (this.traceViewUrl.endsWith('.zip')) {\n      const urlParams = new URLSearchParams({\n        trace: this.traceViewUrl,\n        hideHeader: 'true',\n        hideBranding: 'true',\n        embed: 'true',\n        minimal: 'true',\n        noHeader: 'true'\n      });\n      \n      return `https://trace.playwright.dev/?${urlParams.toString()}`;\n    }\n    \n    return this.traceViewUrl;\n  }\n\n  private updateSafeTraceUrl(): void {\n    const url = this.processedTraceUrl;\n    if (url) {\n      this.traceViewerLoading = true;\n      this.traceViewerError = false;\n      this.safeTraceUrl = this.sanitizer.bypassSecurityTrustResourceUrl(url);\n    } else {\n      this.traceViewerLoading = false;\n      this.traceViewerError = false;\n      this.safeTraceUrl = this.sanitizer.bypassSecurityTrustResourceUrl('');\n    }\n  }\n\n  onTraceViewerLoad(): void {\n    this.traceViewerLoading = false;\n    this.traceViewerError = false;\n  }\n\n  onTraceViewerError(): void {\n    this.traceViewerLoading = false;\n    this.traceViewerError = true;\n  }\n\n  onSpeedChange(value: string): void {\n    this.currentSpeed = value;\n    // Whenever speed changes, apply it to the current video\n    this.applyCurrentPlaybackRate();\n    this.isSpeedControlOpen = false;\n  }\n\n  toggleSpeedControl(): void {\n    this.isSpeedControlOpen = !this.isSpeedControlOpen;\n    if (this.isSpeedControlOpen) {\n      this.addSpeedControlClickOutsideListener();\n    } else {\n      this.removeSpeedControlClickOutsideListener();\n    }\n  }\n\n  private addSpeedControlClickOutsideListener(): void {\n    this.removeSpeedControlClickOutsideListener();\n    \n    setTimeout(() => {\n      this.speedControlClickOutsideHandler = (e: MouseEvent) => {\n        const target = e.target as HTMLElement;\n        if (target && this.speedControlContainer?.nativeElement && \n            !this.speedControlContainer.nativeElement.contains(target)) {\n          this.isSpeedControlOpen = false;\n          this.removeSpeedControlClickOutsideListener();\n        }\n      };\n      \n      document.addEventListener('click', this.speedControlClickOutsideHandler);\n    }, 0);\n  }\n\n  private removeSpeedControlClickOutsideListener(): void {\n    if (this.speedControlClickOutsideHandler) {\n      document.removeEventListener('click', this.speedControlClickOutsideHandler);\n      this.speedControlClickOutsideHandler = null;\n    }\n  }\n\n  private updateSegments(): void {\n    const baseSegments: SegmentOption[] = [\n      { label: 'Screenshots', value: 'screenshots', icon: 'photo'  },\n    ];\n    \n    // Only add Video tab if hideVideoTab is false\n    if (!this.hideVideoTab) {\n      baseSegments.push({ label: 'Video', value: 'video', icon: 'videocam' });\n    }\n    \n    if (this.traceViewUrl) {\n      baseSegments.push({ label: 'Trace', value: 'trace', icon: 'graphic_eq' });\n    }\n    \n    this.segments = baseSegments;\n    \n    // If current view is 'video' and video tab is hidden, switch to screenshots\n    if (this.currentView === 'video' && this.hideVideoTab) {\n      this.currentView = 'screenshots';\n    }\n  }\n\n  private onVideoElementReady(): void {\n    const video = this._vplayer?.nativeElement;\n    if (!video) return;\n\n    console.log('[Simulator] onVideoElementReady: video element ready', {\n      playerState: this.playerState,\n      isFullScreen: this.isFullScreen,\n      readyState: video.readyState\n    });\n\n    // Reset error/loading states when video element is recreated (e.g., during fullscreen toggle)\n    if (this.playerState === 'error' || this.playerState === 'loading') {\n      console.log('[Simulator] onVideoElementReady: resetting error/loading state');\n      this.playerState = 'idle';\n      this.isPlaying = false;\n      this.playPromise = null;\n    }\n\n    // Apply current playback speed when the video element is first ready\n    this.applyCurrentPlaybackRate();\n\n    this.attachVideoListeners();\n  }\n\n  private applyCurrentPlaybackRate(): void {\n    const video = this.vplayer?.nativeElement;\n    if (!video) return;\n\n    const numeric = parseFloat(this.currentSpeed.replace('x', ''));\n    const rate = !isNaN(numeric) && numeric > 0 ? numeric : 1;\n    video.playbackRate = rate;\n  }\n\n  private findVideoIndexForTimestamp(cumulativeTimestamp: number): number | null {\n    if (!this.hasMultipleVideos) {\n      return null;\n    }\n\n    const boundaries = this.getVideoDurationBoundaries();\n    if (boundaries.length === 0) {\n      return null;\n    }\n\n    // Find which video contains this timestamp\n    for (let i = 0; i < boundaries.length; i++) {\n      const boundary = boundaries[i];\n      if (cumulativeTimestamp >= boundary.start && cumulativeTimestamp < boundary.end) {\n        return i;\n      }\n    }\n\n    // If timestamp is exactly at the end of the last video, return the last video index\n    if (boundaries.length > 0) {\n      const lastBoundary = boundaries[boundaries.length - 1];\n      if (cumulativeTimestamp === lastBoundary.end) {\n        return boundaries.length - 1;\n      }\n    }\n\n    return null;\n  }\n\n  private switchToVideoAndSeek(cumulativeTimestamp: number): void {\n    this.enqueueOperation(() => this.switchToVideoAndSeekInternal(cumulativeTimestamp));\n  }\n\n  private async switchToVideoAndSeekInternal(cumulativeTimestamp: number): Promise<void> {\n    if (!this.hasMultipleVideos) {\n      await this.seekToTimeInternal(cumulativeTimestamp);\n      return;\n    }\n\n    const targetVideoIndex = this.findVideoIndexForTimestamp(cumulativeTimestamp);\n    if (targetVideoIndex === null) {\n      console.warn('[Simulator] switchToVideoAndSeek: Could not find video for timestamp', cumulativeTimestamp);\n      return;\n    }\n\n    const boundaries = this.getVideoDurationBoundaries();\n    const targetBoundary = boundaries[targetVideoIndex];\n    const relativeTimestamp = cumulativeTimestamp - targetBoundary.start;\n\n    if (this.currentVideoIndex === targetVideoIndex) {\n      await this.seekToTimeInternal(relativeTimestamp);\n      this.lastSetDuration = cumulativeTimestamp;\n      return;\n    }\n\n    const video = this.vplayer?.nativeElement;\n    if (!video) return;\n\n    const total = this.totalDuration;\n    this.targetGlobalTimeDuringSwitch = cumulativeTimestamp;\n    if (total > 0) {\n      this.progress = Math.min(100, Math.max(0, (cumulativeTimestamp / total) * 100));\n    }\n\n    this.playerState = 'switching';\n\n    try {\n      // CRITICAL: Pause before switching source\n      video.pause();\n      this.isPlaying = false;\n      this.playPromise = null;\n      this.videoPause.emit();\n      this.isVideoPlayingChange.emit(false);\n\n      // Switch video source\n      this.currentVideoIndex = targetVideoIndex;\n\n      await new Promise<void>((resolve, reject) => {\n        let resolved = false;\n        const cleanup = () => {\n          if (resolved) return;\n          resolved = true;\n          video.removeEventListener('loadedmetadata', onLoadedMetadata);\n          video.removeEventListener('error', onError);\n          clearTimeout(timeoutId);\n        };\n        \n        const onLoadedMetadata = () => {\n          cleanup();\n          console.log('[Simulator] switchToVideoAndSeek: loadedmetadata', {\n            readyState: video.readyState,\n            duration: video.duration,\n            currentTime: video.currentTime\n          });\n          resolve();\n        };\n        const onError = () => {\n          cleanup();\n          console.error('[Simulator] switchToVideoAndSeek: video error');\n          reject(new Error('Failed to load video'));\n        };\n\n        setTimeout(() => {\n          if (resolved) return;\n          video.addEventListener('loadedmetadata', onLoadedMetadata, { once: true });\n          video.addEventListener('error', onError, { once: true });\n        }, 100);\n\n        const timeoutId = setTimeout(() => {\n          cleanup();\n          if (video.readyState >= HTMLMediaElement.HAVE_METADATA) {\n            console.log('[Simulator] switchToVideoAndSeek: metadata timeout but video ready', {\n              readyState: video.readyState,\n              duration: video.duration\n            });\n            resolve();\n          } else {\n            console.error('[Simulator] switchToVideoAndSeek: metadata timeout and video not ready');\n            reject(new Error('Timeout waiting for video to load'));\n          }\n        }, 2000);\n      });\n\n      // Seek to the target position using 'seeked' event for reliability\n      const seekSeconds = relativeTimestamp / 1000;\n      const targetTime = Math.max(0, Math.min(seekSeconds, video.duration || seekSeconds));\n      \n      console.log('[Simulator] switchToVideoAndSeek: setting currentTime', {\n        targetTime,\n        seekSeconds,\n        relativeTimestamp,\n        videoDuration: video.duration\n      });\n      \n      await new Promise<void>((resolve) => {\n        let resolved = false;\n        const onSeeked = () => {\n          if (resolved) return;\n          resolved = true;\n          video.removeEventListener('seeked', onSeeked);\n          clearTimeout(timeoutId);\n          console.log('[Simulator] switchToVideoAndSeek: seeked event fired', {\n            currentTime: video.currentTime,\n            targetTime,\n            diff: Math.abs(video.currentTime - targetTime)\n          });\n          \n          // Small delay to ensure browser has processed the seek\n          setTimeout(() => resolve(), 100);\n        };\n        \n        video.addEventListener('seeked', onSeeked, { once: true });\n        video.currentTime = targetTime;\n        \n        // Fallback timeout\n        const timeoutId = setTimeout(() => {\n          if (resolved) return;\n          resolved = true;\n          video.removeEventListener('seeked', onSeeked);\n          console.log('[Simulator] switchToVideoAndSeek: seeked timeout, current time is', video.currentTime);\n          resolve();\n        }, 1000);\n      });\n\n      this.lastSetDuration = cumulativeTimestamp;\n      \n      // Double-check: if video is not at target, force it again\n      if (Math.abs(video.currentTime - targetTime) > 0.5) {\n        console.warn('[Simulator] switchToVideoAndSeek: video not at target after seeked, forcing again', {\n          currentTime: video.currentTime,\n          targetTime\n        });\n        video.currentTime = targetTime;\n        await new Promise(resolve => setTimeout(resolve, 200));\n      }\n\n      // Verify seek completed\n      const actualGlobal = this.getActualGlobalTime();\n      const actualVideoTime = video.currentTime;\n      \n      console.log('[Simulator] switchToVideoAndSeek: seek verification', {\n        actualGlobal,\n        targetGlobal: cumulativeTimestamp,\n        diff: Math.abs(actualGlobal - cumulativeTimestamp),\n        actualVideoTime,\n        targetVideoTime: targetTime,\n        videoDiff: Math.abs(actualVideoTime - targetTime)\n      });\n      \n      // Clear override only if video is actually at the target\n      const tolerance = 500;\n      if (Math.abs(actualGlobal - cumulativeTimestamp) < tolerance) {\n        this.targetGlobalTimeDuringSwitch = null;\n        console.log('[Simulator] switchToVideoAndSeek: cleared override - video at target');\n      }\n\n      // Update UI with actual current position\n      if (total > 0) {\n        this.progress = Math.min(100, Math.max(0, (this.globalCurrentTime / total) * 100));\n      }\n      this.videoTimeUpdate.emit(this.globalCurrentTime);\n    } catch (err) {\n      console.error('[Simulator] switchToVideoAndSeek failed:', err);\n      // Even on error, try to update to target position\n      if (total > 0) {\n        this.progress = Math.min(100, Math.max(0, (cumulativeTimestamp / total) * 100));\n      }\n      this.videoTimeUpdate.emit(cumulativeTimestamp);\n    } finally {\n      this.playerState = 'paused';\n      this.hitMarkers.clear();\n      // Don't clear targetGlobalTimeDuringSwitch here - it's either already cleared\n      // or will be cleared by timeupdate handler when video reaches target\n    }\n\n    console.log('[Simulator] switchToVideoAndSeek: complete', {\n      currentVideoIndex: this.currentVideoIndex,\n      videoCurrentTime: video.currentTime,\n      globalCurrentTime: this.globalCurrentTime,\n      targetGlobalTimeDuringSwitch: this.targetGlobalTimeDuringSwitch,\n      playerState: this.playerState,\n      progress: this.progress\n    });\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 cqa-flex-wrap\">\n      <div class=\"cqa-flex cqa-items-center\">\n        <div *ngIf=\"isLive\" class=\"cqa-h-[21px] cqa-inline-flex cqa-items-center cqa-gap-1.5 cqa-mr-2 cqa-px-[9px] cqa-py-[3px] cqa-bg-[#FCD9D9] cqa-rounded-[6px]\" style=\"border: 1px solid #F9BFBF;\">\n          <span class=\"cqa-relative cqa-w-2 cqa-h-2 cqa-rounded-full cqa-bg-[#F47F7F]\" style=\"flex-shrink: 0;\">\n            <span class=\"cqa-absolute cqa-inset-0 cqa-rounded-full cqa-bg-[#F47F7F] cqa-opacity-75 cqa-animate-ping\"></span>\n          </span>\n          <span class=\"cqa-text-[10px] cqa-font-medium cqa-text-[#C63535] cqa-leading-[15px]\">Live</span>\n        </div>\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\">\n          {{ platformName }}\n          <span\n            *ngIf=\"platformType === 'browser'\"\n            class=\"cqa-ml-1\"\n            [matTooltip]=\"'Screen size: ' + effectiveBrowserViewPort.width + 'x' + effectiveBrowserViewPort.height\"\n            matTooltipPosition=\"below\"\n          >\n            ·\n            <span class=\"cqa-ml-1\">\n              {{ effectiveBrowserViewPort.width }}x{{ effectiveBrowserViewPort.height }}\n            </span>\n          </span>\n        </p>\n      </div>\n      <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n        <div *ngIf=\"isLive\" [ngClass]=\"getStatusBadgeClass()\">\n          <span [ngClass]=\"getStatusTextClass()\">{{ liveStatus }}</span>\n        </div>\n\n        <ng-container *ngIf=\"!isLive\">\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        </ng-container>\n      </div>\n    </div>\n  </div>\n  <div class=\"cqa-w-full cqa-bg-[#F3F4F6] cqa-h-[calc(100%-41px)]\">\n  <!-- Live Content View -->\n    <div *ngIf=\"isLive\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center cqa-relative\">\n      <ng-container *ngIf=\"hasDeviceFrame;\">\n        <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center cqa-p-4\">\n          <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-max-h-full cqa-h-full': platformType !== 'browser' || !isLive, 'cqa-min-w-max': platformType === 'device'}\">\n            <img\n              [src]=\"deviceMockupImage\"\n              alt=\"Device mockup\"\n              class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n            />\n            <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-z-20': platformType === 'browser', 'cqa-bg-white': platformType !== 'browser'}\">\n              <!-- Loading State -->\n              <div *ngIf=\"isContentVideoLoading\" class=\"cqa-p-10 cqa-text-center cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center\">\n                <div class=\"cqa-mb-4\">\n                  <mat-progress-spinner mode=\"indeterminate\" diameter=\"40\"></mat-progress-spinner>\n                </div>\n                <p class=\"cqa-text-gray-400 cqa-text-sm\">{{ liveLoadingLabel }}</p>\n              </div>\n\n              <!-- Live Content (when not loading) -->\n              <div *ngIf=\"!isContentVideoLoading\" class=\"cqa-w-full cqa-h-full cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-relative\">\n                <div *ngIf=\"liveStatus === 'Failed' && failedStatusMessage\" class=\"cqa-p-6 cqa-text-center cqa-w-full\">\n                  <div class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-4 cqa-py-3 cqa-bg-[#FCD9D9] cqa-border cqa-border-[#F9BFBF] cqa-rounded-lg\">\n                    <mat-icon style=\"width: 18px; height: 18px; color: #C63535; font-size: 18px;\">error</mat-icon>\n                    <p class=\"cqa-text-[#C63535] cqa-text-sm cqa-font-medium cqa-m-0\">{{ failedStatusMessage }}</p>\n                  </div>\n                </div>\n                <ng-content *ngIf=\"liveStatus !== 'Failed' || !failedStatusMessage\"></ng-content>\n              </div>\n            </div>\n          </div>\n        </div>\n      </ng-container>\n    </div>\n\n    <!-- Normal Video View (when not live) -->\n    <div *ngIf=\"!isLive && currentView === 'video'\" class=\"cqa-h-full cqa-flex cqa-flex-col\">\n      <div class=\"cqa-w-full cqa-py-4 cqa-flex cqa-items-center cqa-max-h-[calc(100%-83px)]\" *ngIf=\"currentVideoUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device', 'cqa-mt-auto': hasDeviceFrame}\">\n        <ng-container *ngIf=\"hasDeviceFrame\">\n          <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n            <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n              <img\n                [src]=\"deviceMockupImage\"\n                alt=\"Device mockup\"\n                class=\"cqa-h-full cqa-w-auto cqa-max-h-full cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n              />\n              <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n                <video\n                  #vplayer\n                  class=\"cqa-object-cover cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n                  [src]=\"currentVideoUrl\"\n                  type=\"video/webm\"\n                  [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n                  (loadedmetadata)=\"onVideoMetadataLoaded()\"\n                  (canplay)=\"onVideoCanPlay()\"\n                  (ended)=\"onVideoEnded()\"\n                ></video>\n              </div>\n            </div>\n          </div>\n        </ng-container>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!currentVideoUrl\">\n        <ng-container *ngIf=\"isVNCSessionIntruppted && vncSessionIntupptedMessage; else noVideoDefault\">\n          <p class=\"cqa-text-sm cqa-text-gray-600\">\n            {{ vncSessionIntupptedMessage }}\n          </p>\n        </ng-container>\n        <ng-template #noVideoDefault>\n          <span>No video recording found</span>\n        </ng-template>\n      </div>\n    \n      <div class=\"cqa-px-2 cqa-py-2 cqa-bg-white\" style=\"border-top: 1px solid #E5E7EB;\" [ngClass]=\"{'cqa-mt-auto': hasDeviceFrame}\" *ngIf=\"currentVideoUrl && !isLive\">\n        <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n          <div class=\"cqa-flex cqa-items-center cqa-justify-center\" style=\"width: 16px; height: 16px;\">\n            <mat-progress-spinner\n              *ngIf=\"isPlayerSwitching\"\n              mode=\"indeterminate\"\n              diameter=\"16\"\n              class=\"cqa-inline-block\">\n            </mat-progress-spinner>\n            <button \n              *ngIf=\"!isPlayerSwitching\"\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              matTooltip=\"{{ isPlaying ? 'Pause' : 'Play' }}\"\n              matTooltipPosition=\"above\">\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          </div>\n\n          <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n            {{ formatTime(globalCurrentTime / 1000) }}\n          </span>\n\n          <div #speedControlContainer class=\"cqa-relative cqa-mr-[8px] cqa-flex cqa-items-center cqa-justify-center\">\n            <button\n              type=\"button\"\n              class=\"cqa-bg-transparent cqa-border-none cqa-cursor-pointer cqa-text-[#9CA3AF] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-whitespace-nowrap cqa-select-none hover:cqa-opacity-70 cqa-transition-opacity cqa-outline-none cqa-px-1\"\n              (click)=\"toggleSpeedControl()\"\n              [matTooltip]=\"'Playback Speed'\"\n              [matTooltipPosition]=\"'below'\">\n              {{ currentSpeed }}\n            </button>\n            \n            <div \n              *ngIf=\"isSpeedControlOpen\"\n              class=\"cqa-absolute cqa-bottom-full cqa-mb-2 cqa-right-0 cqa-bg-[#F0F0F1] cqa-rounded-lg cqa-overflow-hidden cqa-shadow-lg cqa-z-50\"\n              style=\"min-width: max-content; left: 50%; bottom: 0%; transform: translate(-50%, -50%);\">\n              <cqa-segment-control\n                [segments]=\"speedSegments\"\n                [value]=\"currentSpeed\"\n                [containerBgColor]=\"'#F0F0F1'\"\n                (valueChange)=\"onSpeedChange($event)\">\n              </cqa-segment-control>\n            </div>\n          </div>\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            \n            <div \n              *ngFor=\"let marker of globalStepMarkers\" \n              class=\"cqa-absolute cqa-rounded-full\"\n              [style.left.%]=\"totalDuration > 0 ? (marker.globalTime / totalDuration) * 100 : 0\"\n              [style.width]=\"'8px'\"\n              [style.height]=\"'8px'\"\n              [style.background]=\"getGlobalMarkerColor(marker.level)\"\n              [style.border]=\"'2px solid ' + getGlobalMarkerResultColor(marker.result)\"\n              [style.box-sizing]=\"'border-box'\"\n              [attr.title]=\"marker.title || ''\"\n              style=\"pointer-events: auto; z-index: 50; cursor: pointer; transform: translate(-50%, -50%); top: 50%;\"\n              (click)=\"onMarkerClick($event, marker)\">\n            </div>\n            \n            <div \n              class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-h-full cqa-bg-blue-500 cqa-rounded-full\"\n              [style.width.%]=\"progress\"\n              [style.transition]=\"dragging ? 'none' : 'width 100ms'\"\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-blue-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: 60;\"\n              (mousedown)=\"startDrag($event)\">\n            </div>\n          </div>\n\n          <span class=\"cqa-text-[#9CA3AF] cqa-text-[9px] cqa-font-normal cqa-whitespace-nowrap cqa-select-none\">\n            {{ formatTime(totalDuration / 1000) }}\n          </span>\n        </div>\n      </div>\n    </div>\n\n    <div *ngIf=\"!isLive && 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        <ng-container *ngIf=\"hasDeviceFrame\">\n          <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" [ngClass]=\"{'cqa-p-4': platformType === 'browser'}\">\n            <div class=\"cqa-relative cqa-h-full cqa-w-auto cqa-flex cqa-items-center cqa-justify-center cqa-max-h-full\" [ngClass]=\"{'cqa-rounded-md cqa-overflow-hidden': platformType === 'browser', 'cqa-min-w-max': platformType === 'device'}\">\n              <img\n                [src]=\"deviceMockupImage\"\n                alt=\"Device mockup\"\n                class=\"cqa-h-full cqa-w-auto cqa-object-contain cqa-block cqa-pointer-events-none cqa-z-10\"\n                [ngClass]=\"{'cqa-max-h-[inherit]': platformType === 'browser', 'cqa-max-h-full': platformType !== 'browser'}\"\n              />\n              <div class=\"cqa-absolute cqa-flex cqa-flex-col\" [ngStyle]=\"deviceScreenStyle\" [ngClass]=\"{'cqa-bg-white': platformType !== 'browser'}\">\n                <img\n                  [src]=\"screenShotUrl\"\n                  alt=\"Screenshot\"\n                  [ngClass]=\"{'cqa-z-20': platformType === 'browser'}\"\n                  class=\"cqa-object-contain cqa-w-full cqa-h-full cqa-block cqa-bg-[##F2F2F2]\"\n                />\n              </div>\n            </div>\n          </div>\n        </ng-container>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!screenShotUrl\">\n        No screenshot available\n      </div>\n    </div>\n\n    <div *ngIf=\"!isLive && currentView === 'trace'\" class=\"cqa-h-full cqa-flex cqa-flex-col cqa-justify-center\">\n      <div class=\"cqa-w-full cqa-h-full cqa-flex cqa-items-center cqa-relative\" *ngIf=\"traceViewUrl\" [ngClass]=\"{'!cqa-h-full': platformType === 'device'}\" style=\"padding-top: 48px; padding-bottom: 0px;\">\n        <div class=\"cqa-w-full cqa-h-full cqa-overflow-hidden cqa-relative\">\n          <iframe \n            [src]=\"safeTraceUrl\" \n            title=\"Trace Viewer\"\n            class=\"cqa-object-contain cqa-w-full cqa-min-h-[250px] cqa-max-h-full cqa-block cqa-bg-[##F2F2F2]\"\n            style=\"margin-top: -48px; height: calc(100% + 48px);\"\n            frameborder=\"0\"\n            allowfullscreen\n            width=\"100%\"\n            loading=\"lazy\"\n            (load)=\"onTraceViewerLoad()\"\n            (error)=\"onTraceViewerError()\">\n          </iframe>\n        </div>\n        \n        <div *ngIf=\"traceViewerLoading\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n          <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n            Loading trace viewer...\n          </div>\n        </div>\n        \n        <div *ngIf=\"traceViewerError\" class=\"cqa-absolute cqa-inset-0 cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center cqa-z-10\">\n          <div class=\"cqa-text-center cqa-text-gray-400 cqa-text-sm\">\n            Failed to load trace viewer\n          </div>\n        </div>\n      </div>\n    \n      <div class=\"cqa-p-10 cqa-text-center cqa-text-gray-400 cqa-text-sm cqa-h-full cqa-flex cqa-items-center cqa-justify-center\" *ngIf=\"!traceViewUrl\">\n        No trace available\n      </div>\n    </div>  \n  </div>\n</div>"]}
|