@cqa-lib/cqa-ui 1.1.177 → 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.
Files changed (34) hide show
  1. package/esm2020/lib/execution-screen/ai-action-step/ai-action-step.component.mjs +23 -7
  2. package/esm2020/lib/execution-screen/ai-agent-step/ai-agent-step.component.mjs +23 -8
  3. package/esm2020/lib/execution-screen/api-step/api-step.component.mjs +24 -7
  4. package/esm2020/lib/execution-screen/base-step.component.mjs +12 -1
  5. package/esm2020/lib/execution-screen/basic-step/basic-step.component.mjs +25 -8
  6. package/esm2020/lib/execution-screen/condition-step/condition-step.component.mjs +25 -8
  7. package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +28 -11
  8. package/esm2020/lib/execution-screen/document-verification-step/document-verification-step.component.mjs +22 -4
  9. package/esm2020/lib/execution-screen/execution-step.models.mjs +1 -1
  10. package/esm2020/lib/execution-screen/file-download-step/file-download-step.component.mjs +23 -5
  11. package/esm2020/lib/execution-screen/loop-step/loop-step.component.mjs +26 -9
  12. package/esm2020/lib/execution-screen/step-group/step-group.component.mjs +22 -5
  13. package/esm2020/lib/execution-screen/step-renderer/step-renderer.component.mjs +6 -2
  14. package/esm2020/lib/simulator/simulator.component.mjs +1202 -63
  15. package/fesm2015/cqa-lib-cqa-ui.mjs +1531 -189
  16. package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
  17. package/fesm2020/cqa-lib-cqa-ui.mjs +1482 -178
  18. package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
  19. package/lib/execution-screen/ai-action-step/ai-action-step.component.d.ts +3 -1
  20. package/lib/execution-screen/ai-agent-step/ai-agent-step.component.d.ts +3 -1
  21. package/lib/execution-screen/api-step/api-step.component.d.ts +3 -1
  22. package/lib/execution-screen/base-step.component.d.ts +1 -0
  23. package/lib/execution-screen/basic-step/basic-step.component.d.ts +3 -1
  24. package/lib/execution-screen/condition-step/condition-step.component.d.ts +3 -1
  25. package/lib/execution-screen/db-verification-step/db-verification-step.component.d.ts +3 -1
  26. package/lib/execution-screen/document-verification-step/document-verification-step.component.d.ts +4 -1
  27. package/lib/execution-screen/execution-step.models.d.ts +4 -0
  28. package/lib/execution-screen/file-download-step/file-download-step.component.d.ts +4 -1
  29. package/lib/execution-screen/loop-step/loop-step.component.d.ts +3 -1
  30. package/lib/execution-screen/step-group/step-group.component.d.ts +3 -1
  31. package/lib/execution-screen/step-renderer/step-renderer.component.d.ts +2 -1
  32. package/lib/simulator/simulator.component.d.ts +122 -6
  33. package/package.json +1 -1
  34. 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
- if (newDuration !== this.lastSetDuration && this.vplayer?.nativeElement) {
242
- this.seekToTime(newDuration);
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 || !this.vplayer.nativeElement.duration)
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
- if (this.isPlaying) {
275
- video.play().catch(err => console.error('Play failed:', err));
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
- // Do not attempt to play if there is no valid source yet
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
- if (this.isPlaying) {
287
- video.pause();
288
- this.isPlaying = false;
289
- this.videoPause.emit();
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
- video.play().then(() => {
293
- this.isPlaying = true;
294
- this.videoPlay.emit();
295
- }).catch(err => {
296
- console.error('Video play failed:', err);
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
- if (this.vplayer?.nativeElement) {
314
- this.vplayer.nativeElement.pause();
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 || !this.vplayer?.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 duration = this.vplayer.nativeElement.duration;
342
- if (!duration || !isFinite(duration)) {
864
+ const total = this.totalDuration;
865
+ if (total <= 0)
343
866
  return;
344
- }
345
- const targetTimeSeconds = percent * duration;
346
- this.vplayer.nativeElement.currentTime = targetTimeSeconds;
347
- this.lastSetDuration = targetTimeSeconds * 1000;
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
- if (this.vplayer?.nativeElement) {
368
- const duration = this.vplayer.nativeElement.duration;
369
- if (!duration || !isFinite(duration)) {
370
- return;
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
- const targetTimeSeconds = (this.progress / 100) * duration;
373
- this.vplayer.nativeElement.currentTime = targetTimeSeconds;
374
- this.lastSetDuration = targetTimeSeconds * 1000;
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 percent = (currentMs / (video.duration * 1000)) * 100;
426
- this.progress = Math.min(100, Math.max(0, percent));
427
- this.videoTimeUpdate.emit(currentMs);
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
- getStepLeftPosition(step) {
437
- if (!this.vplayer?.nativeElement?.duration)
438
- return 0;
439
- return (step.cumulativeDuration / (this.vplayer.nativeElement.duration * 1000)) * 100;
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
- getStepColor(step) {
442
- return step.result === 'SUCCESS' ? '#28a745' : '#dc3545';
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
- this.isPlaying = false;
450
- this.vplayer?.nativeElement?.pause();
451
- this.progress = 0;
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 mins = Math.floor(seconds / 60);
457
- const secs = Math.floor(seconds % 60);
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
- const numeric = parseFloat(value.replace('x', ''));
535
- const rate = !isNaN(numeric) && numeric > 0 ? numeric : 1;
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
- this.attachVideoListeners();
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
- }], timelineBar: [{
1790
+ }], timelineBarRef: [{
652
1791
  type: ViewChild,
653
1792
  args: ['timelineBar', { static: false }]
654
- }], speedControlContainer: [{
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>"]}