angular-debug-recorder 1.0.3 → 1.1.1

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.
@@ -116,4 +116,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
116
116
  type: Injectable,
117
117
  args: [{ providedIn: 'root' }]
118
118
  }], ctorParameters: () => [{ type: i0.NgZone }] });
119
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rrweb-recorder.service.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/services/rrweb-recorder.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, signal } from '@angular/core';\r\nimport type { eventWithTime } from '@rrweb/types';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class RrwebRecorderService {\r\n  // Plain array — cheap push, no O(n) spread on every event\r\n  private _eventsArray: eventWithTime[] = [];\r\n  // Signal only for reactive count display in templates\r\n  private _eventCount = signal(0);\r\n  private _isRecording = signal(false);\r\n  private stopFn?: () => void;\r\n  private replayer?: any;\r\n\r\n  eventCount = this._eventCount.asReadonly();\r\n  isRecording = this._isRecording.asReadonly();\r\n\r\n  constructor(private zone: NgZone) {}\r\n\r\n  hasEvents(): boolean {\r\n    return this._eventsArray.length > 0;\r\n  }\r\n\r\n  getEvents(): eventWithTime[] {\r\n    return this._eventsArray;\r\n  }\r\n\r\n  async startRecording(): Promise<void> {\r\n    const { record } = await import('rrweb');\r\n    this._eventsArray = [];\r\n    this._eventCount.set(0);\r\n    this._isRecording.set(true);\r\n\r\n    this.stopFn = record({\r\n      emit: (event: eventWithTime) => {\r\n        // Push outside Angular zone — no change detection per event\r\n        this.zone.runOutsideAngular(() => {\r\n          this._eventsArray.push(event);\r\n        });\r\n        // Update count signal only every 10 events to keep UI responsive\r\n        const len = this._eventsArray.length;\r\n        if (len === 1 || len % 10 === 0) {\r\n          this.zone.run(() => this._eventCount.set(len));\r\n        }\r\n      },\r\n      // Note: blockSelector omitted — rrweb 2.0.0-alpha.4 calls node.matches()\r\n      // on TextNodes/CommentNodes which crash the recorder.\r\n      maskTextSelector: 'input[type=\"password\"]',\r\n      checkoutEveryNth: 200,\r\n    });\r\n  }\r\n\r\n  stopRecording(): eventWithTime[] {\r\n    if (this.stopFn) {\r\n      this.stopFn();\r\n      this.stopFn = undefined;\r\n    }\r\n    this._isRecording.set(false);\r\n    this._eventCount.set(this._eventsArray.length);\r\n    return this._eventsArray;\r\n  }\r\n\r\n  clearEvents(): void {\r\n    this._eventsArray = [];\r\n    this._eventCount.set(0);\r\n  }\r\n\r\n  // ─── Replay ──────────────────────────────────────────────────────────────\r\n\r\n  async startReplay(container: HTMLElement, events?: eventWithTime[]): Promise<void> {\r\n    const { Replayer } = await import('rrweb');\r\n    const eventsToReplay = events ?? this._eventsArray;\r\n\r\n    if (eventsToReplay.length === 0) return;\r\n\r\n    this.destroyReplayer();\r\n\r\n    this.replayer = new Replayer(eventsToReplay, {\r\n      root: container,\r\n      skipInactive: true,\r\n      showWarning: false,\r\n      showDebug: false,\r\n      blockClass: 'debug-panel',\r\n    });\r\n\r\n    this.replayer.play();\r\n  }\r\n\r\n  pauseReplay(): void {\r\n    this.replayer?.pause();\r\n  }\r\n\r\n  resumeReplay(): void {\r\n    this.replayer?.play();\r\n  }\r\n\r\n  destroyReplayer(): void {\r\n    if (this.replayer) {\r\n      try { this.replayer.pause(); } catch {}\r\n      this.replayer = undefined;\r\n    }\r\n  }\r\n\r\n  // ─── Export ───────────────────────────────────────────────────────────────\r\n\r\n  exportEvents(): string {\r\n    return JSON.stringify(this._eventsArray, null, 2);\r\n  }\r\n\r\n  downloadEvents(filename = 'rrweb-session.json'): void {\r\n    const blob = new Blob([this.exportEvents()], { type: 'application/json' });\r\n    const url = URL.createObjectURL(blob);\r\n    const a = document.createElement('a');\r\n    a.href = url;\r\n    a.download = filename;\r\n    a.click();\r\n    URL.revokeObjectURL(url);\r\n  }\r\n\r\n  importEvents(json: string): eventWithTime[] {\r\n    try {\r\n      const events = JSON.parse(json) as eventWithTime[];\r\n      this._eventsArray = events;\r\n      this._eventCount.set(events.length);\r\n      return events;\r\n    } catch {\r\n      console.error('Invalid rrweb events JSON');\r\n      return [];\r\n    }\r\n  }\r\n}\r\n"]}
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rrweb-recorder.service.js","sourceRoot":"","sources":["../../../../../src/lib/services/rrweb-recorder.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, signal } from '@angular/core';\r\nimport type { eventWithTime } from '@rrweb/types';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class RrwebRecorderService {\r\n  // Plain array — cheap push, no O(n) spread on every event\r\n  private _eventsArray: eventWithTime[] = [];\r\n  // Signal only for reactive count display in templates\r\n  private _eventCount = signal(0);\r\n  private _isRecording = signal(false);\r\n  private stopFn?: () => void;\r\n  private replayer?: any;\r\n\r\n  eventCount = this._eventCount.asReadonly();\r\n  isRecording = this._isRecording.asReadonly();\r\n\r\n  constructor(private zone: NgZone) {}\r\n\r\n  hasEvents(): boolean {\r\n    return this._eventsArray.length > 0;\r\n  }\r\n\r\n  getEvents(): eventWithTime[] {\r\n    return this._eventsArray;\r\n  }\r\n\r\n  async startRecording(): Promise<void> {\r\n    const { record } = await import('rrweb');\r\n    this._eventsArray = [];\r\n    this._eventCount.set(0);\r\n    this._isRecording.set(true);\r\n\r\n    this.stopFn = record({\r\n      emit: (event: eventWithTime) => {\r\n        // Push outside Angular zone — no change detection per event\r\n        this.zone.runOutsideAngular(() => {\r\n          this._eventsArray.push(event);\r\n        });\r\n        // Update count signal only every 10 events to keep UI responsive\r\n        const len = this._eventsArray.length;\r\n        if (len === 1 || len % 10 === 0) {\r\n          this.zone.run(() => this._eventCount.set(len));\r\n        }\r\n      },\r\n      // Note: blockSelector omitted — rrweb 2.0.0-alpha.4 calls node.matches()\r\n      // on TextNodes/CommentNodes which crash the recorder.\r\n      maskTextSelector: 'input[type=\"password\"]',\r\n      checkoutEveryNth: 200,\r\n    });\r\n  }\r\n\r\n  stopRecording(): eventWithTime[] {\r\n    if (this.stopFn) {\r\n      this.stopFn();\r\n      this.stopFn = undefined;\r\n    }\r\n    this._isRecording.set(false);\r\n    this._eventCount.set(this._eventsArray.length);\r\n    return this._eventsArray;\r\n  }\r\n\r\n  clearEvents(): void {\r\n    this._eventsArray = [];\r\n    this._eventCount.set(0);\r\n  }\r\n\r\n  // ─── Replay ──────────────────────────────────────────────────────────────\r\n\r\n  async startReplay(container: HTMLElement, events?: eventWithTime[]): Promise<void> {\r\n    const { Replayer } = await import('rrweb');\r\n    const eventsToReplay = events ?? this._eventsArray;\r\n\r\n    if (eventsToReplay.length === 0) return;\r\n\r\n    this.destroyReplayer();\r\n\r\n    this.replayer = new Replayer(eventsToReplay, {\r\n      root: container,\r\n      skipInactive: true,\r\n      showWarning: false,\r\n      showDebug: false,\r\n      blockClass: 'debug-panel',\r\n    });\r\n\r\n    this.replayer.play();\r\n  }\r\n\r\n  pauseReplay(): void {\r\n    this.replayer?.pause();\r\n  }\r\n\r\n  resumeReplay(): void {\r\n    this.replayer?.play();\r\n  }\r\n\r\n  destroyReplayer(): void {\r\n    if (this.replayer) {\r\n      try { this.replayer.pause(); } catch {}\r\n      this.replayer = undefined;\r\n    }\r\n  }\r\n\r\n  // ─── Export ───────────────────────────────────────────────────────────────\r\n\r\n  exportEvents(): string {\r\n    return JSON.stringify(this._eventsArray, null, 2);\r\n  }\r\n\r\n  downloadEvents(filename = 'rrweb-session.json'): void {\r\n    const blob = new Blob([this.exportEvents()], { type: 'application/json' });\r\n    const url = URL.createObjectURL(blob);\r\n    const a = document.createElement('a');\r\n    a.href = url;\r\n    a.download = filename;\r\n    a.click();\r\n    URL.revokeObjectURL(url);\r\n  }\r\n\r\n  importEvents(json: string): eventWithTime[] {\r\n    try {\r\n      const events = JSON.parse(json) as eventWithTime[];\r\n      this._eventsArray = events;\r\n      this._eventCount.set(events.length);\r\n      return events;\r\n    } catch {\r\n      console.error('Invalid rrweb events JSON');\r\n      return [];\r\n    }\r\n  }\r\n}\r\n"]}
@@ -7,6 +7,8 @@ export class SessionReplayComponent {
7
7
  this.rrweb = inject(RrwebRecorderService);
8
8
  this.overlayOpen = signal(false);
9
9
  this.isPlaying = signal(false);
10
+ this.importError = signal(null);
11
+ this.importedFilename = signal(null);
10
12
  }
11
13
  async openOverlay() {
12
14
  this.overlayOpen.set(true);
@@ -38,6 +40,34 @@ export class SessionReplayComponent {
38
40
  exportSession() {
39
41
  this.rrweb.downloadEvents();
40
42
  }
43
+ onFileSelected(event) {
44
+ const input = event.target;
45
+ const file = input.files?.[0];
46
+ if (!file)
47
+ return;
48
+ // Reset input so the same file can be re-selected after clearing
49
+ input.value = '';
50
+ this.importError.set(null);
51
+ const reader = new FileReader();
52
+ reader.onload = () => {
53
+ try {
54
+ const events = this.rrweb.importEvents(reader.result);
55
+ if (events.length === 0) {
56
+ this.importError.set('Keine Events in der Datei gefunden.');
57
+ this.importedFilename.set(null);
58
+ }
59
+ else {
60
+ this.importedFilename.set(file.name);
61
+ }
62
+ }
63
+ catch {
64
+ this.importError.set('Ungültiges JSON-Format.');
65
+ this.importedFilename.set(null);
66
+ }
67
+ };
68
+ reader.onerror = () => this.importError.set('Datei konnte nicht gelesen werden.');
69
+ reader.readAsText(file);
70
+ }
41
71
  async startPlay() {
42
72
  if (!this.replayContainer)
43
73
  return;
@@ -72,15 +102,34 @@ export class SessionReplayComponent {
72
102
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SessionReplayComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
73
103
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: SessionReplayComponent, isStandalone: true, selector: "app-session-replay", viewQueries: [{ propertyName: "replayContainer", first: true, predicate: ["replayContainer"], descendants: true }], ngImport: i0, template: `
74
104
  <div class="replay-panel" data-debug-panel>
105
+ <!-- Hidden file input for import -->
106
+ <input
107
+ #fileInput
108
+ type="file"
109
+ accept=".json"
110
+ style="display:none"
111
+ data-debug-panel
112
+ (change)="onFileSelected($event)"
113
+ />
114
+
75
115
  @if (!rrweb.hasEvents()) {
76
116
  <div class="replay-empty">
77
117
  <div class="replay-icon">📽️</div>
78
118
  <p>Kein Replay verfügbar.</p>
79
119
  <p class="hint">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>
120
+ <button class="replay-btn import" (click)="fileInput.click()">
121
+ 📂 JSON importieren
122
+ </button>
123
+ @if (importError()) {
124
+ <p class="import-error">{{ importError() }}</p>
125
+ }
80
126
  </div>
81
127
  } @else {
82
128
  <div class="replay-info">
83
129
  <span class="event-count">{{ rrweb.eventCount() }} Events aufgezeichnet</span>
130
+ @if (importedFilename()) {
131
+ <span class="imported-badge">📂 {{ importedFilename() }}</span>
132
+ }
84
133
  </div>
85
134
  <div class="replay-actions">
86
135
  <button class="replay-btn primary" (click)="openOverlay()">
@@ -89,10 +138,16 @@ export class SessionReplayComponent {
89
138
  <button class="replay-btn" (click)="exportSession()">
90
139
  💾 JSON exportieren
91
140
  </button>
141
+ <button class="replay-btn import" (click)="fileInput.click()">
142
+ 📂 Importieren
143
+ </button>
92
144
  </div>
93
145
  <p class="replay-hint">
94
146
  Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.
95
147
  </p>
148
+ @if (importError()) {
149
+ <p class="import-error">{{ importError() }}</p>
150
+ }
96
151
  }
97
152
  </div>
98
153
 
@@ -112,21 +167,40 @@ export class SessionReplayComponent {
112
167
  <div #replayContainer class="overlay-stage" data-debug-panel></div>
113
168
  </div>
114
169
  }
115
- `, isInline: true, styles: [".replay-panel{padding:20px 16px;display:flex;flex-direction:column;gap:14px}.replay-empty{text-align:center;padding:20px 0;color:#64748b}.replay-icon{font-size:36px;margin-bottom:8px}.replay-empty p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.replay-info{background:#1e293b;border-radius:6px;padding:8px 12px}.event-count{font-size:12px;color:#6ee7b7;font-weight:600}.replay-actions{display:flex;gap:8px}.replay-btn{background:#334155;border:none;color:#cbd5e1;padding:8px 14px;border-radius:6px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.replay-btn:hover{background:#475569}.replay-btn.primary{background:#1d4ed8;color:#fff}.replay-btn.primary:hover{background:#2563eb}.replay-hint{font-size:11px;color:#475569;margin:0}.replay-overlay{position:fixed;inset:0;z-index:99997;background:#000;display:flex;flex-direction:column}.overlay-header{display:flex;align-items:center;justify-content:space-between;padding:10px 16px;background:#0f172a;border-bottom:1px solid #1e293b;flex-shrink:0}.overlay-title{font-size:14px;font-weight:600;color:#f1f5f9}.overlay-controls{display:flex;gap:8px}.ovl-btn{background:#1e293b;border:1px solid #334155;color:#cbd5e1;padding:5px 12px;border-radius:5px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.ovl-btn:hover{background:#334155}.close-ovl{color:#fca5a5}.close-ovl:hover{background:#dc262633}.overlay-stage{flex:1;overflow:hidden;position:relative;background:#f8fafc}:host ::ng-deep .replayer-wrapper{position:absolute!important;top:0!important;left:0!important;transform-origin:top left!important}:host ::ng-deep .replayer-wrapper iframe{pointer-events:none}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
170
+ `, isInline: true, styles: [".replay-panel{padding:20px 16px;display:flex;flex-direction:column;gap:14px}.replay-empty{text-align:center;padding:20px 0;color:#64748b}.replay-icon{font-size:36px;margin-bottom:8px}.replay-empty p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.replay-info{background:#1e293b;border-radius:6px;padding:8px 12px}.event-count{font-size:12px;color:#6ee7b7;font-weight:600}.replay-actions{display:flex;gap:8px}.replay-btn{background:#334155;border:none;color:#cbd5e1;padding:8px 14px;border-radius:6px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.replay-btn:hover{background:#475569}.replay-btn.primary{background:#1d4ed8;color:#fff}.replay-btn.primary:hover{background:#2563eb}.replay-hint{font-size:11px;color:#475569;margin:0}.replay-btn.import{background:#1e3a5f;color:#93c5fd}.replay-btn.import:hover{background:#1e4070}.imported-badge{font-size:11px;color:#93c5fd;margin-left:8px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:150px;display:inline-block;vertical-align:middle}.import-error{font-size:11px;color:#fca5a5;margin:4px 0 0}.replay-overlay{position:fixed;inset:0;z-index:99997;background:#000;display:flex;flex-direction:column}.overlay-header{display:flex;align-items:center;justify-content:space-between;padding:10px 16px;background:#0f172a;border-bottom:1px solid #1e293b;flex-shrink:0}.overlay-title{font-size:14px;font-weight:600;color:#f1f5f9}.overlay-controls{display:flex;gap:8px}.ovl-btn{background:#1e293b;border:1px solid #334155;color:#cbd5e1;padding:5px 12px;border-radius:5px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.ovl-btn:hover{background:#334155}.close-ovl{color:#fca5a5}.close-ovl:hover{background:#dc262633}.overlay-stage{flex:1;overflow:hidden;position:relative;background:#f8fafc}:host ::ng-deep .replayer-wrapper{position:absolute!important;top:0!important;left:0!important;transform-origin:top left!important}:host ::ng-deep .replayer-wrapper iframe{pointer-events:none;display:block}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
116
171
  }
117
172
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SessionReplayComponent, decorators: [{
118
173
  type: Component,
119
174
  args: [{ selector: 'app-session-replay', standalone: true, imports: [CommonModule], template: `
120
175
  <div class="replay-panel" data-debug-panel>
176
+ <!-- Hidden file input for import -->
177
+ <input
178
+ #fileInput
179
+ type="file"
180
+ accept=".json"
181
+ style="display:none"
182
+ data-debug-panel
183
+ (change)="onFileSelected($event)"
184
+ />
185
+
121
186
  @if (!rrweb.hasEvents()) {
122
187
  <div class="replay-empty">
123
188
  <div class="replay-icon">📽️</div>
124
189
  <p>Kein Replay verfügbar.</p>
125
190
  <p class="hint">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>
191
+ <button class="replay-btn import" (click)="fileInput.click()">
192
+ 📂 JSON importieren
193
+ </button>
194
+ @if (importError()) {
195
+ <p class="import-error">{{ importError() }}</p>
196
+ }
126
197
  </div>
127
198
  } @else {
128
199
  <div class="replay-info">
129
200
  <span class="event-count">{{ rrweb.eventCount() }} Events aufgezeichnet</span>
201
+ @if (importedFilename()) {
202
+ <span class="imported-badge">📂 {{ importedFilename() }}</span>
203
+ }
130
204
  </div>
131
205
  <div class="replay-actions">
132
206
  <button class="replay-btn primary" (click)="openOverlay()">
@@ -135,10 +209,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
135
209
  <button class="replay-btn" (click)="exportSession()">
136
210
  💾 JSON exportieren
137
211
  </button>
212
+ <button class="replay-btn import" (click)="fileInput.click()">
213
+ 📂 Importieren
214
+ </button>
138
215
  </div>
139
216
  <p class="replay-hint">
140
217
  Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.
141
218
  </p>
219
+ @if (importError()) {
220
+ <p class="import-error">{{ importError() }}</p>
221
+ }
142
222
  }
143
223
  </div>
144
224
 
@@ -158,9 +238,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
158
238
  <div #replayContainer class="overlay-stage" data-debug-panel></div>
159
239
  </div>
160
240
  }
161
- `, styles: [".replay-panel{padding:20px 16px;display:flex;flex-direction:column;gap:14px}.replay-empty{text-align:center;padding:20px 0;color:#64748b}.replay-icon{font-size:36px;margin-bottom:8px}.replay-empty p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.replay-info{background:#1e293b;border-radius:6px;padding:8px 12px}.event-count{font-size:12px;color:#6ee7b7;font-weight:600}.replay-actions{display:flex;gap:8px}.replay-btn{background:#334155;border:none;color:#cbd5e1;padding:8px 14px;border-radius:6px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.replay-btn:hover{background:#475569}.replay-btn.primary{background:#1d4ed8;color:#fff}.replay-btn.primary:hover{background:#2563eb}.replay-hint{font-size:11px;color:#475569;margin:0}.replay-overlay{position:fixed;inset:0;z-index:99997;background:#000;display:flex;flex-direction:column}.overlay-header{display:flex;align-items:center;justify-content:space-between;padding:10px 16px;background:#0f172a;border-bottom:1px solid #1e293b;flex-shrink:0}.overlay-title{font-size:14px;font-weight:600;color:#f1f5f9}.overlay-controls{display:flex;gap:8px}.ovl-btn{background:#1e293b;border:1px solid #334155;color:#cbd5e1;padding:5px 12px;border-radius:5px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.ovl-btn:hover{background:#334155}.close-ovl{color:#fca5a5}.close-ovl:hover{background:#dc262633}.overlay-stage{flex:1;overflow:hidden;position:relative;background:#f8fafc}:host ::ng-deep .replayer-wrapper{position:absolute!important;top:0!important;left:0!important;transform-origin:top left!important}:host ::ng-deep .replayer-wrapper iframe{pointer-events:none}\n"] }]
241
+ `, styles: [".replay-panel{padding:20px 16px;display:flex;flex-direction:column;gap:14px}.replay-empty{text-align:center;padding:20px 0;color:#64748b}.replay-icon{font-size:36px;margin-bottom:8px}.replay-empty p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.replay-info{background:#1e293b;border-radius:6px;padding:8px 12px}.event-count{font-size:12px;color:#6ee7b7;font-weight:600}.replay-actions{display:flex;gap:8px}.replay-btn{background:#334155;border:none;color:#cbd5e1;padding:8px 14px;border-radius:6px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.replay-btn:hover{background:#475569}.replay-btn.primary{background:#1d4ed8;color:#fff}.replay-btn.primary:hover{background:#2563eb}.replay-hint{font-size:11px;color:#475569;margin:0}.replay-btn.import{background:#1e3a5f;color:#93c5fd}.replay-btn.import:hover{background:#1e4070}.imported-badge{font-size:11px;color:#93c5fd;margin-left:8px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:150px;display:inline-block;vertical-align:middle}.import-error{font-size:11px;color:#fca5a5;margin:4px 0 0}.replay-overlay{position:fixed;inset:0;z-index:99997;background:#000;display:flex;flex-direction:column}.overlay-header{display:flex;align-items:center;justify-content:space-between;padding:10px 16px;background:#0f172a;border-bottom:1px solid #1e293b;flex-shrink:0}.overlay-title{font-size:14px;font-weight:600;color:#f1f5f9}.overlay-controls{display:flex;gap:8px}.ovl-btn{background:#1e293b;border:1px solid #334155;color:#cbd5e1;padding:5px 12px;border-radius:5px;font-size:12px;font-weight:600;cursor:pointer;transition:background .15s}.ovl-btn:hover{background:#334155}.close-ovl{color:#fca5a5}.close-ovl:hover{background:#dc262633}.overlay-stage{flex:1;overflow:hidden;position:relative;background:#f8fafc}:host ::ng-deep .replayer-wrapper{position:absolute!important;top:0!important;left:0!important;transform-origin:top left!important}:host ::ng-deep .replayer-wrapper iframe{pointer-events:none;display:block}\n"] }]
162
242
  }], propDecorators: { replayContainer: [{
163
243
  type: ViewChild,
164
244
  args: ['replayContainer']
165
245
  }] } });
166
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"session-replay.component.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/session-replay/session-replay.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component, ElementRef, ViewChild, OnDestroy, inject, signal,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { RrwebRecorderService } from '../services/rrweb-recorder.service';\r\n\r\n@Component({\r\n  selector: 'app-session-replay',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  template: `\r\n    <div class=\"replay-panel\" data-debug-panel>\r\n      @if (!rrweb.hasEvents()) {\r\n        <div class=\"replay-empty\">\r\n          <div class=\"replay-icon\">📽️</div>\r\n          <p>Kein Replay verfügbar.</p>\r\n          <p class=\"hint\">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>\r\n        </div>\r\n      } @else {\r\n        <div class=\"replay-info\">\r\n          <span class=\"event-count\">{{ rrweb.eventCount() }} Events aufgezeichnet</span>\r\n        </div>\r\n        <div class=\"replay-actions\">\r\n          <button class=\"replay-btn primary\" (click)=\"openOverlay()\">\r\n            ▶ Replay abspielen\r\n          </button>\r\n          <button class=\"replay-btn\" (click)=\"exportSession()\">\r\n            💾 JSON exportieren\r\n          </button>\r\n        </div>\r\n        <p class=\"replay-hint\">\r\n          Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.\r\n        </p>\r\n      }\r\n    </div>\r\n\r\n    <!-- Fullscreen Replay Overlay -->\r\n    @if (overlayOpen()) {\r\n      <div class=\"replay-overlay\" data-debug-panel>\r\n        <div class=\"overlay-header\" data-debug-panel>\r\n          <span class=\"overlay-title\">📽️ Session Replay</span>\r\n          <div class=\"overlay-controls\" data-debug-panel>\r\n            <button class=\"ovl-btn\" (click)=\"pauseResume()\">\r\n              {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}\r\n            </button>\r\n            <button class=\"ovl-btn\" (click)=\"restart()\">⟳ Neustart</button>\r\n            <button class=\"ovl-btn close-ovl\" (click)=\"closeOverlay()\">✕ Schließen</button>\r\n          </div>\r\n        </div>\r\n        <div #replayContainer class=\"overlay-stage\" data-debug-panel></div>\r\n      </div>\r\n    }\r\n  `,\r\n  styles: [`\r\n    .replay-panel {\r\n      padding: 20px 16px;\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: 14px;\r\n    }\r\n\r\n    .replay-empty {\r\n      text-align: center;\r\n      padding: 20px 0;\r\n      color: #64748b;\r\n    }\r\n    .replay-icon { font-size: 36px; margin-bottom: 8px; }\r\n    .replay-empty p { margin: 4px 0; font-size: 13px; }\r\n    .hint { font-size: 11px; color: #475569; }\r\n\r\n    .replay-info {\r\n      background: #1e293b;\r\n      border-radius: 6px;\r\n      padding: 8px 12px;\r\n    }\r\n    .event-count { font-size: 12px; color: #6ee7b7; font-weight: 600; }\r\n\r\n    .replay-actions { display: flex; gap: 8px; }\r\n    .replay-btn {\r\n      background: #334155;\r\n      border: none;\r\n      color: #cbd5e1;\r\n      padding: 8px 14px;\r\n      border-radius: 6px;\r\n      font-size: 12px;\r\n      font-weight: 600;\r\n      cursor: pointer;\r\n      transition: background 0.15s;\r\n    }\r\n    .replay-btn:hover { background: #475569; }\r\n    .replay-btn.primary { background: #1d4ed8; color: #fff; }\r\n    .replay-btn.primary:hover { background: #2563eb; }\r\n\r\n    .replay-hint { font-size: 11px; color: #475569; margin: 0; }\r\n\r\n    /* ── Fullscreen Overlay ── */\r\n    .replay-overlay {\r\n      position: fixed;\r\n      inset: 0;\r\n      z-index: 99997;\r\n      background: #000;\r\n      display: flex;\r\n      flex-direction: column;\r\n    }\r\n    .overlay-header {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: space-between;\r\n      padding: 10px 16px;\r\n      background: #0f172a;\r\n      border-bottom: 1px solid #1e293b;\r\n      flex-shrink: 0;\r\n    }\r\n    .overlay-title { font-size: 14px; font-weight: 600; color: #f1f5f9; }\r\n    .overlay-controls { display: flex; gap: 8px; }\r\n    .ovl-btn {\r\n      background: #1e293b;\r\n      border: 1px solid #334155;\r\n      color: #cbd5e1;\r\n      padding: 5px 12px;\r\n      border-radius: 5px;\r\n      font-size: 12px;\r\n      font-weight: 600;\r\n      cursor: pointer;\r\n      transition: background 0.15s;\r\n    }\r\n    .ovl-btn:hover { background: #334155; }\r\n    .close-ovl { color: #fca5a5; }\r\n    .close-ovl:hover { background: rgba(220,38,38,0.2); }\r\n\r\n    .overlay-stage {\r\n      flex: 1;\r\n      overflow: hidden;\r\n      position: relative;\r\n      background: #f8fafc;\r\n    }\r\n\r\n    /* Scale the rrweb iframe to fill available space */\r\n    :host ::ng-deep .replayer-wrapper {\r\n      position: absolute !important;\r\n      top: 0 !important;\r\n      left: 0 !important;\r\n      transform-origin: top left !important;\r\n    }\r\n    :host ::ng-deep .replayer-wrapper iframe {\r\n      pointer-events: none;\r\n    }\r\n  `],\r\n})\r\nexport class SessionReplayComponent implements OnDestroy {\r\n  @ViewChild('replayContainer') replayContainer?: ElementRef<HTMLDivElement>;\r\n\r\n  rrweb = inject(RrwebRecorderService);\r\n  overlayOpen = signal(false);\r\n  isPlaying = signal(false);\r\n\r\n  async openOverlay(): Promise<void> {\r\n    this.overlayOpen.set(true);\r\n    this.isPlaying.set(false);\r\n    // Wait for the DOM to render the overlay\r\n    await new Promise(r => setTimeout(r, 50));\r\n    await this.startPlay();\r\n  }\r\n\r\n  closeOverlay(): void {\r\n    this.rrweb.destroyReplayer();\r\n    this.overlayOpen.set(false);\r\n    this.isPlaying.set(false);\r\n  }\r\n\r\n  async pauseResume(): Promise<void> {\r\n    if (this.isPlaying()) {\r\n      this.rrweb.pauseReplay();\r\n      this.isPlaying.set(false);\r\n    } else {\r\n      this.rrweb.resumeReplay();\r\n      this.isPlaying.set(true);\r\n    }\r\n  }\r\n\r\n  async restart(): Promise<void> {\r\n    this.rrweb.destroyReplayer();\r\n    await new Promise(r => setTimeout(r, 50));\r\n    await this.startPlay();\r\n  }\r\n\r\n  exportSession(): void {\r\n    this.rrweb.downloadEvents();\r\n  }\r\n\r\n  private async startPlay(): Promise<void> {\r\n    if (!this.replayContainer) return;\r\n    const container = this.replayContainer.nativeElement;\r\n\r\n    await this.rrweb.startReplay(container);\r\n\r\n    // Scale iframe to fill the stage container\r\n    this.scaleReplayer(container);\r\n    this.isPlaying.set(true);\r\n  }\r\n\r\n  private scaleReplayer(container: HTMLElement): void {\r\n    // rrweb injects .replayer-wrapper with an iframe sized to the recorded viewport\r\n    setTimeout(() => {\r\n      const wrapper = container.querySelector('.replayer-wrapper') as HTMLElement;\r\n      if (!wrapper) return;\r\n\r\n      const iframe = wrapper.querySelector('iframe') as HTMLIFrameElement;\r\n      const wrapW = iframe?.offsetWidth  || wrapper.offsetWidth  || 1280;\r\n      const wrapH = iframe?.offsetHeight || wrapper.offsetHeight || 720;\r\n      const stageW = container.offsetWidth;\r\n      const stageH = container.offsetHeight;\r\n\r\n      if (!stageW || !stageH || !wrapW || !wrapH) return;\r\n\r\n      const scale = Math.min(stageW / wrapW, stageH / wrapH);\r\n      const offsetX = (stageW - wrapW * scale) / 2;\r\n      const offsetY = (stageH - wrapH * scale) / 2;\r\n\r\n      wrapper.style.transform = `translate(${offsetX}px, ${offsetY}px) scale(${scale})`;\r\n    }, 300);\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.rrweb.destroyReplayer();\r\n  }\r\n}\r\n"]}
246
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"session-replay.component.js","sourceRoot":"","sources":["../../../../../src/lib/session-replay/session-replay.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component, ElementRef, ViewChild, OnDestroy, inject, signal,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { RrwebRecorderService } from '../services/rrweb-recorder.service';\r\n\r\n@Component({\r\n  selector: 'app-session-replay',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  template: `\r\n    <div class=\"replay-panel\" data-debug-panel>\r\n      <!-- Hidden file input for import -->\r\n      <input\r\n        #fileInput\r\n        type=\"file\"\r\n        accept=\".json\"\r\n        style=\"display:none\"\r\n        data-debug-panel\r\n        (change)=\"onFileSelected($event)\"\r\n      />\r\n\r\n      @if (!rrweb.hasEvents()) {\r\n        <div class=\"replay-empty\">\r\n          <div class=\"replay-icon\">📽️</div>\r\n          <p>Kein Replay verfügbar.</p>\r\n          <p class=\"hint\">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>\r\n          <button class=\"replay-btn import\" (click)=\"fileInput.click()\">\r\n            📂 JSON importieren\r\n          </button>\r\n          @if (importError()) {\r\n            <p class=\"import-error\">{{ importError() }}</p>\r\n          }\r\n        </div>\r\n      } @else {\r\n        <div class=\"replay-info\">\r\n          <span class=\"event-count\">{{ rrweb.eventCount() }} Events aufgezeichnet</span>\r\n          @if (importedFilename()) {\r\n            <span class=\"imported-badge\">📂 {{ importedFilename() }}</span>\r\n          }\r\n        </div>\r\n        <div class=\"replay-actions\">\r\n          <button class=\"replay-btn primary\" (click)=\"openOverlay()\">\r\n            ▶ Replay abspielen\r\n          </button>\r\n          <button class=\"replay-btn\" (click)=\"exportSession()\">\r\n            💾 JSON exportieren\r\n          </button>\r\n          <button class=\"replay-btn import\" (click)=\"fileInput.click()\">\r\n            📂 Importieren\r\n          </button>\r\n        </div>\r\n        <p class=\"replay-hint\">\r\n          Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.\r\n        </p>\r\n        @if (importError()) {\r\n          <p class=\"import-error\">{{ importError() }}</p>\r\n        }\r\n      }\r\n    </div>\r\n\r\n    <!-- Fullscreen Replay Overlay -->\r\n    @if (overlayOpen()) {\r\n      <div class=\"replay-overlay\" data-debug-panel>\r\n        <div class=\"overlay-header\" data-debug-panel>\r\n          <span class=\"overlay-title\">📽️ Session Replay</span>\r\n          <div class=\"overlay-controls\" data-debug-panel>\r\n            <button class=\"ovl-btn\" (click)=\"pauseResume()\">\r\n              {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}\r\n            </button>\r\n            <button class=\"ovl-btn\" (click)=\"restart()\">⟳ Neustart</button>\r\n            <button class=\"ovl-btn close-ovl\" (click)=\"closeOverlay()\">✕ Schließen</button>\r\n          </div>\r\n        </div>\r\n        <div #replayContainer class=\"overlay-stage\" data-debug-panel></div>\r\n      </div>\r\n    }\r\n  `,\r\n  styles: [`\r\n    .replay-panel {\r\n      padding: 20px 16px;\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: 14px;\r\n    }\r\n\r\n    .replay-empty {\r\n      text-align: center;\r\n      padding: 20px 0;\r\n      color: #64748b;\r\n    }\r\n    .replay-icon { font-size: 36px; margin-bottom: 8px; }\r\n    .replay-empty p { margin: 4px 0; font-size: 13px; }\r\n    .hint { font-size: 11px; color: #475569; }\r\n\r\n    .replay-info {\r\n      background: #1e293b;\r\n      border-radius: 6px;\r\n      padding: 8px 12px;\r\n    }\r\n    .event-count { font-size: 12px; color: #6ee7b7; font-weight: 600; }\r\n\r\n    .replay-actions { display: flex; gap: 8px; }\r\n    .replay-btn {\r\n      background: #334155;\r\n      border: none;\r\n      color: #cbd5e1;\r\n      padding: 8px 14px;\r\n      border-radius: 6px;\r\n      font-size: 12px;\r\n      font-weight: 600;\r\n      cursor: pointer;\r\n      transition: background 0.15s;\r\n    }\r\n    .replay-btn:hover { background: #475569; }\r\n    .replay-btn.primary { background: #1d4ed8; color: #fff; }\r\n    .replay-btn.primary:hover { background: #2563eb; }\r\n\r\n    .replay-hint { font-size: 11px; color: #475569; margin: 0; }\r\n\r\n    .replay-btn.import { background: #1e3a5f; color: #93c5fd; }\r\n    .replay-btn.import:hover { background: #1e4070; }\r\n\r\n    .imported-badge {\r\n      font-size: 11px;\r\n      color: #93c5fd;\r\n      margin-left: 8px;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n      white-space: nowrap;\r\n      max-width: 150px;\r\n      display: inline-block;\r\n      vertical-align: middle;\r\n    }\r\n\r\n    .import-error {\r\n      font-size: 11px;\r\n      color: #fca5a5;\r\n      margin: 4px 0 0;\r\n    }\r\n\r\n    /* ── Fullscreen Overlay ── */\r\n    .replay-overlay {\r\n      position: fixed;\r\n      inset: 0;\r\n      z-index: 99997;\r\n      background: #000;\r\n      display: flex;\r\n      flex-direction: column;\r\n    }\r\n    .overlay-header {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: space-between;\r\n      padding: 10px 16px;\r\n      background: #0f172a;\r\n      border-bottom: 1px solid #1e293b;\r\n      flex-shrink: 0;\r\n    }\r\n    .overlay-title { font-size: 14px; font-weight: 600; color: #f1f5f9; }\r\n    .overlay-controls { display: flex; gap: 8px; }\r\n    .ovl-btn {\r\n      background: #1e293b;\r\n      border: 1px solid #334155;\r\n      color: #cbd5e1;\r\n      padding: 5px 12px;\r\n      border-radius: 5px;\r\n      font-size: 12px;\r\n      font-weight: 600;\r\n      cursor: pointer;\r\n      transition: background 0.15s;\r\n    }\r\n    .ovl-btn:hover { background: #334155; }\r\n    .close-ovl { color: #fca5a5; }\r\n    .close-ovl:hover { background: rgba(220,38,38,0.2); }\r\n\r\n    .overlay-stage {\r\n      flex: 1;\r\n      overflow: hidden;\r\n      position: relative;\r\n      background: #f8fafc;\r\n    }\r\n\r\n    /* Scale the rrweb iframe to fill available space */\r\n    :host ::ng-deep .replayer-wrapper {\r\n      position: absolute !important;\r\n      top: 0 !important;\r\n      left: 0 !important;\r\n      transform-origin: top left !important;\r\n    }\r\n    :host ::ng-deep .replayer-wrapper iframe {\r\n      pointer-events: none;\r\n      display: block;\r\n    }\r\n  `],\r\n})\r\nexport class SessionReplayComponent implements OnDestroy {\r\n  @ViewChild('replayContainer') replayContainer?: ElementRef<HTMLDivElement>;\r\n\r\n  rrweb = inject(RrwebRecorderService);\r\n  overlayOpen = signal(false);\r\n  isPlaying = signal(false);\r\n  importError = signal<string | null>(null);\r\n  importedFilename = signal<string | null>(null);\r\n\r\n  async openOverlay(): Promise<void> {\r\n    this.overlayOpen.set(true);\r\n    this.isPlaying.set(false);\r\n    // Wait for the DOM to render the overlay\r\n    await new Promise(r => setTimeout(r, 50));\r\n    await this.startPlay();\r\n  }\r\n\r\n  closeOverlay(): void {\r\n    this.rrweb.destroyReplayer();\r\n    this.overlayOpen.set(false);\r\n    this.isPlaying.set(false);\r\n  }\r\n\r\n  async pauseResume(): Promise<void> {\r\n    if (this.isPlaying()) {\r\n      this.rrweb.pauseReplay();\r\n      this.isPlaying.set(false);\r\n    } else {\r\n      this.rrweb.resumeReplay();\r\n      this.isPlaying.set(true);\r\n    }\r\n  }\r\n\r\n  async restart(): Promise<void> {\r\n    this.rrweb.destroyReplayer();\r\n    await new Promise(r => setTimeout(r, 50));\r\n    await this.startPlay();\r\n  }\r\n\r\n  exportSession(): void {\r\n    this.rrweb.downloadEvents();\r\n  }\r\n\r\n  onFileSelected(event: Event): void {\r\n    const input = event.target as HTMLInputElement;\r\n    const file = input.files?.[0];\r\n    if (!file) return;\r\n\r\n    // Reset input so the same file can be re-selected after clearing\r\n    input.value = '';\r\n\r\n    this.importError.set(null);\r\n\r\n    const reader = new FileReader();\r\n    reader.onload = () => {\r\n      try {\r\n        const events = this.rrweb.importEvents(reader.result as string);\r\n        if (events.length === 0) {\r\n          this.importError.set('Keine Events in der Datei gefunden.');\r\n          this.importedFilename.set(null);\r\n        } else {\r\n          this.importedFilename.set(file.name);\r\n        }\r\n      } catch {\r\n        this.importError.set('Ungültiges JSON-Format.');\r\n        this.importedFilename.set(null);\r\n      }\r\n    };\r\n    reader.onerror = () => this.importError.set('Datei konnte nicht gelesen werden.');\r\n    reader.readAsText(file);\r\n  }\r\n\r\n  private async startPlay(): Promise<void> {\r\n    if (!this.replayContainer) return;\r\n    const container = this.replayContainer.nativeElement;\r\n\r\n    await this.rrweb.startReplay(container);\r\n\r\n    // Scale iframe to fill the stage container\r\n    this.scaleReplayer(container);\r\n    this.isPlaying.set(true);\r\n  }\r\n\r\n  private scaleReplayer(container: HTMLElement): void {\r\n    // rrweb injects .replayer-wrapper with an iframe sized to the recorded viewport\r\n    setTimeout(() => {\r\n      const wrapper = container.querySelector('.replayer-wrapper') as HTMLElement;\r\n      if (!wrapper) return;\r\n\r\n      const iframe = wrapper.querySelector('iframe') as HTMLIFrameElement;\r\n      const wrapW = iframe?.offsetWidth  || wrapper.offsetWidth  || 1280;\r\n      const wrapH = iframe?.offsetHeight || wrapper.offsetHeight || 720;\r\n      const stageW = container.offsetWidth;\r\n      const stageH = container.offsetHeight;\r\n\r\n      if (!stageW || !stageH || !wrapW || !wrapH) return;\r\n\r\n      const scale = Math.min(stageW / wrapW, stageH / wrapH);\r\n      const offsetX = (stageW - wrapW * scale) / 2;\r\n      const offsetY = (stageH - wrapH * scale) / 2;\r\n\r\n      wrapper.style.transform = `translate(${offsetX}px, ${offsetY}px) scale(${scale})`;\r\n    }, 300);\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.rrweb.destroyReplayer();\r\n  }\r\n}\r\n"]}
@@ -102,4 +102,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
102
102
  }], propDecorators: { close: [{
103
103
  type: Output
104
104
  }] } });
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"settings-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/settings-dialog/settings-dialog.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACxE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,kBAAkB,EAAE,MAAM,kCAAkC,CAAC;;;AAyGtE,MAAM,OAAO,uBAAuB;IAvGpC;QAwGY,UAAK,GAAG,IAAI,YAAY,EAAQ,CAAC;QACnC,OAAE,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAC;QACxC,aAAQ,GAAG,IAAI,CAAC,EAAE,CAAC,UAAU,EAAE,CAAC;KAYjC;IAVC,IAAI;QACF,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;QAC5C,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;IACpB,CAAC;IAED,cAAc,CAAC,CAAa;QAC1B,IAAK,CAAC,CAAC,MAAsB,CAAC,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YAC5D,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QACpB,CAAC;IACH,CAAC;+GAdU,uBAAuB;mGAAvB,uBAAuB,4GAnGxB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCT,2lEAtCS,YAAY,8BAAE,WAAW;;4FAoGxB,uBAAuB;kBAvGnC,SAAS;+BACE,qBAAqB,cACnB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,CAAC,YAC1B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCT;8BA+DS,KAAK;sBAAd,MAAM","sourcesContent":["import { Component, Output, EventEmitter, inject } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { AiGeneratorService } from '../services/ai-generator.service';\r\n\r\n@Component({\r\n  selector: 'app-settings-dialog',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule],\r\n  template: `\r\n    <div class=\"overlay\" data-debug-panel (click)=\"onOverlayClick($event)\">\r\n      <div class=\"dialog\" data-debug-panel>\r\n        <div class=\"dialog-header\">\r\n          <h2>⚙️ Einstellungen</h2>\r\n          <button class=\"close-btn\" (click)=\"close.emit()\">✕</button>\r\n        </div>\r\n\r\n        <div class=\"dialog-body\">\r\n          <div class=\"field-group\">\r\n            <label>KI Webhook-URL</label>\r\n            <input\r\n              data-debug-panel\r\n              class=\"field-input\"\r\n              type=\"url\"\r\n              [(ngModel)]=\"localUrl\"\r\n              placeholder=\"https://deine-ki-api.de/generate\"\r\n            />\r\n            <p class=\"field-hint\">\r\n              Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.\r\n              Die Antwort wird als Cypress-Test angezeigt.\r\n            </p>\r\n          </div>\r\n\r\n          <div class=\"info-box\">\r\n            <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>\r\n            <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>\r\n            <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>\r\n          </div>\r\n        </div>\r\n\r\n        <div class=\"dialog-footer\">\r\n          <button class=\"btn-cancel\" (click)=\"close.emit()\">Abbrechen</button>\r\n          <button class=\"btn-save\" (click)=\"save()\">💾 Speichern</button>\r\n        </div>\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .overlay {\r\n      position: fixed; inset: 0; background: rgba(0,0,0,0.7);\r\n      z-index: 100000; display: flex; align-items: center; justify-content: center;\r\n      backdrop-filter: blur(4px);\r\n    }\r\n    .dialog {\r\n      background: #0f172a; border: 1px solid #1e3a5f; border-radius: 12px;\r\n      width: 440px; max-width: 95vw; display: flex; flex-direction: column;\r\n      box-shadow: 0 25px 60px rgba(0,0,0,0.6); overflow: hidden;\r\n    }\r\n    .dialog-header {\r\n      display: flex; align-items: center; justify-content: space-between;\r\n      padding: 14px 18px; background: #1e293b; border-bottom: 1px solid #334155;\r\n    }\r\n    .dialog-header h2 { margin: 0; font-size: 15px; color: #f1f5f9; font-weight: 600; }\r\n    .close-btn {\r\n      background: none; border: none; color: #94a3b8;\r\n      cursor: pointer; font-size: 16px; padding: 4px; border-radius: 4px;\r\n    }\r\n    .close-btn:hover { color: #f1f5f9; background: #334155; }\r\n    .dialog-body { padding: 18px; display: flex; flex-direction: column; gap: 16px; }\r\n    .field-group { display: flex; flex-direction: column; gap: 6px; }\r\n    label {\r\n      font-size: 12px; font-weight: 600; color: #94a3b8;\r\n      text-transform: uppercase; letter-spacing: 0.05em;\r\n    }\r\n    .field-input {\r\n      background: #1e293b; border: 1px solid #334155; color: #e2e8f0;\r\n      border-radius: 6px; padding: 10px 12px; font-size: 13px;\r\n      width: 100%; box-sizing: border-box;\r\n    }\r\n    .field-input:focus { outline: none; border-color: #3b82f6; }\r\n    .field-hint { font-size: 11px; color: #64748b; margin: 0; line-height: 1.5; }\r\n    .info-box {\r\n      background: rgba(30,41,59,0.5); border: 1px solid #1e3a5f;\r\n      border-radius: 8px; padding: 12px; display: flex; flex-direction: column; gap: 6px;\r\n    }\r\n    .info-box p { margin: 0; font-size: 11px; color: #64748b; }\r\n    .info-box code {\r\n      background: #1e293b; color: #34d399; padding: 1px 4px;\r\n      border-radius: 3px; font-family: monospace;\r\n    }\r\n    kbd {\r\n      background: #1e293b; border: 1px solid #334155; color: #94a3b8;\r\n      padding: 1px 5px; border-radius: 3px; font-size: 10px;\r\n    }\r\n    .dialog-footer {\r\n      display: flex; gap: 8px; justify-content: flex-end;\r\n      padding: 14px 18px; background: #1e293b; border-top: 1px solid #334155;\r\n    }\r\n    .btn-cancel, .btn-save {\r\n      padding: 8px 20px; border-radius: 6px; font-size: 13px;\r\n      font-weight: 600; cursor: pointer; border: none; transition: filter 0.15s;\r\n    }\r\n    .btn-cancel { background: #334155; color: #94a3b8; }\r\n    .btn-cancel:hover { filter: brightness(1.2); }\r\n    .btn-save { background: #2563eb; color: #fff; }\r\n    .btn-save:hover { filter: brightness(1.1); }\r\n  `],\r\n})\r\nexport class SettingsDialogComponent {\r\n  @Output() close = new EventEmitter<void>();\r\n  private ai = inject(AiGeneratorService);\r\n  localUrl = this.ai.webhookUrl();\r\n\r\n  save(): void {\r\n    this.ai.setWebhookUrl(this.localUrl.trim());\r\n    this.close.emit();\r\n  }\r\n\r\n  onOverlayClick(e: MouseEvent): void {\r\n    if ((e.target as HTMLElement).classList.contains('overlay')) {\r\n      this.close.emit();\r\n    }\r\n  }\r\n}\r\n"]}
105
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"settings-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/settings-dialog/settings-dialog.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACxE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,kBAAkB,EAAE,MAAM,kCAAkC,CAAC;;;AAyGtE,MAAM,OAAO,uBAAuB;IAvGpC;QAwGY,UAAK,GAAG,IAAI,YAAY,EAAQ,CAAC;QACnC,OAAE,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAC;QACxC,aAAQ,GAAG,IAAI,CAAC,EAAE,CAAC,UAAU,EAAE,CAAC;KAYjC;IAVC,IAAI;QACF,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;QAC5C,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;IACpB,CAAC;IAED,cAAc,CAAC,CAAa;QAC1B,IAAK,CAAC,CAAC,MAAsB,CAAC,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YAC5D,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QACpB,CAAC;IACH,CAAC;+GAdU,uBAAuB;mGAAvB,uBAAuB,4GAnGxB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCT,2lEAtCS,YAAY,8BAAE,WAAW;;4FAoGxB,uBAAuB;kBAvGnC,SAAS;+BACE,qBAAqB,cACnB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,CAAC,YAC1B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCT;8BA+DS,KAAK;sBAAd,MAAM","sourcesContent":["import { Component, Output, EventEmitter, inject } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { AiGeneratorService } from '../services/ai-generator.service';\r\n\r\n@Component({\r\n  selector: 'app-settings-dialog',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule],\r\n  template: `\r\n    <div class=\"overlay\" data-debug-panel (click)=\"onOverlayClick($event)\">\r\n      <div class=\"dialog\" data-debug-panel>\r\n        <div class=\"dialog-header\">\r\n          <h2>⚙️ Einstellungen</h2>\r\n          <button class=\"close-btn\" (click)=\"close.emit()\">✕</button>\r\n        </div>\r\n\r\n        <div class=\"dialog-body\">\r\n          <div class=\"field-group\">\r\n            <label>KI Webhook-URL</label>\r\n            <input\r\n              data-debug-panel\r\n              class=\"field-input\"\r\n              type=\"url\"\r\n              [(ngModel)]=\"localUrl\"\r\n              placeholder=\"https://deine-ki-api.de/generate\"\r\n            />\r\n            <p class=\"field-hint\">\r\n              Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.\r\n              Die Antwort wird als Cypress-Test angezeigt.\r\n            </p>\r\n          </div>\r\n\r\n          <div class=\"info-box\">\r\n            <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>\r\n            <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>\r\n            <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>\r\n          </div>\r\n        </div>\r\n\r\n        <div class=\"dialog-footer\">\r\n          <button class=\"btn-cancel\" (click)=\"close.emit()\">Abbrechen</button>\r\n          <button class=\"btn-save\" (click)=\"save()\">💾 Speichern</button>\r\n        </div>\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .overlay {\r\n      position: fixed; inset: 0; background: rgba(0,0,0,0.7);\r\n      z-index: 100000; display: flex; align-items: center; justify-content: center;\r\n      backdrop-filter: blur(4px);\r\n    }\r\n    .dialog {\r\n      background: #0f172a; border: 1px solid #1e3a5f; border-radius: 12px;\r\n      width: 440px; max-width: 95vw; display: flex; flex-direction: column;\r\n      box-shadow: 0 25px 60px rgba(0,0,0,0.6); overflow: hidden;\r\n    }\r\n    .dialog-header {\r\n      display: flex; align-items: center; justify-content: space-between;\r\n      padding: 14px 18px; background: #1e293b; border-bottom: 1px solid #334155;\r\n    }\r\n    .dialog-header h2 { margin: 0; font-size: 15px; color: #f1f5f9; font-weight: 600; }\r\n    .close-btn {\r\n      background: none; border: none; color: #94a3b8;\r\n      cursor: pointer; font-size: 16px; padding: 4px; border-radius: 4px;\r\n    }\r\n    .close-btn:hover { color: #f1f5f9; background: #334155; }\r\n    .dialog-body { padding: 18px; display: flex; flex-direction: column; gap: 16px; }\r\n    .field-group { display: flex; flex-direction: column; gap: 6px; }\r\n    label {\r\n      font-size: 12px; font-weight: 600; color: #94a3b8;\r\n      text-transform: uppercase; letter-spacing: 0.05em;\r\n    }\r\n    .field-input {\r\n      background: #1e293b; border: 1px solid #334155; color: #e2e8f0;\r\n      border-radius: 6px; padding: 10px 12px; font-size: 13px;\r\n      width: 100%; box-sizing: border-box;\r\n    }\r\n    .field-input:focus { outline: none; border-color: #3b82f6; }\r\n    .field-hint { font-size: 11px; color: #64748b; margin: 0; line-height: 1.5; }\r\n    .info-box {\r\n      background: rgba(30,41,59,0.5); border: 1px solid #1e3a5f;\r\n      border-radius: 8px; padding: 12px; display: flex; flex-direction: column; gap: 6px;\r\n    }\r\n    .info-box p { margin: 0; font-size: 11px; color: #64748b; }\r\n    .info-box code {\r\n      background: #1e293b; color: #34d399; padding: 1px 4px;\r\n      border-radius: 3px; font-family: monospace;\r\n    }\r\n    kbd {\r\n      background: #1e293b; border: 1px solid #334155; color: #94a3b8;\r\n      padding: 1px 5px; border-radius: 3px; font-size: 10px;\r\n    }\r\n    .dialog-footer {\r\n      display: flex; gap: 8px; justify-content: flex-end;\r\n      padding: 14px 18px; background: #1e293b; border-top: 1px solid #334155;\r\n    }\r\n    .btn-cancel, .btn-save {\r\n      padding: 8px 20px; border-radius: 6px; font-size: 13px;\r\n      font-weight: 600; cursor: pointer; border: none; transition: filter 0.15s;\r\n    }\r\n    .btn-cancel { background: #334155; color: #94a3b8; }\r\n    .btn-cancel:hover { filter: brightness(1.2); }\r\n    .btn-save { background: #2563eb; color: #fff; }\r\n    .btn-save:hover { filter: brightness(1.1); }\r\n  `],\r\n})\r\nexport class SettingsDialogComponent {\r\n  @Output() close = new EventEmitter<void>();\r\n  private ai = inject(AiGeneratorService);\r\n  localUrl = this.ai.webhookUrl();\r\n\r\n  save(): void {\r\n    this.ai.setWebhookUrl(this.localUrl.trim());\r\n    this.close.emit();\r\n  }\r\n\r\n  onOverlayClick(e: MouseEvent): void {\r\n    if ((e.target as HTMLElement).classList.contains('overlay')) {\r\n      this.close.emit();\r\n    }\r\n  }\r\n}\r\n"]}
@@ -117,4 +117,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
117
117
  }], propDecorators: { test: [{
118
118
  type: Input
119
119
  }] } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-preview.component.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/test-preview/test-preview.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;;AA+G/C,MAAM,OAAO,oBAAoB;IA5GjC;QA6GW,SAAI,GAAyB,IAAI,CAAC;QAE3C,WAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAEvB,oBAAe,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC9B,IAAI,CAAC,IAAI,CAAC,IAAI;gBAAE,OAAO,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9C,CAAC,CAAC,CAAC;KA0CJ;IAxCS,eAAe,CAAC,IAAY;QAClC,OAAO,IAAI;aACR,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC;aACtB,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC;aACrB,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC;YACtB,WAAW;aACV,OAAO,CAAC,eAAe,EAAE,4BAA4B,CAAC;YACvD,eAAe;aACd,OAAO,CAAC,WAAW,EAAE,4BAA4B,CAAC;YACnD,WAAW;aACV,OAAO,CAAC,uIAAuI,EAAE,4BAA4B,CAAC;YAC/K,UAAU;aACT,OAAO,CAAC,4BAA4B,EAAE,6BAA6B,CAAC;YACrE,UAAU;aACT,OAAO,CAAC,YAAY,EAAE,6BAA6B,CAAC,CAAC;IAC1D,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QACvB,MAAM,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACtB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,IAAI,CAAC,CAAC;IACjD,CAAC;IAED,YAAY;QACV,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QACvB,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,EAAE,iBAAiB,EAAE,CAAC,CAAC;QACrE,MAAM,GAAG,GAAG,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,CAAC,GAAG,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;QACtC,CAAC,CAAC,IAAI,GAAG,GAAG,CAAC;QACb,CAAC,CAAC,QAAQ,GAAG,gBAAgB,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC;QAC3E,CAAC,CAAC,KAAK,EAAE,CAAC;QACV,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;IAC3B,CAAC;IAED,UAAU,CAAC,EAAU;QACnB,OAAO,IAAI,IAAI,CAAC,EAAE,CAAC,CAAC,cAAc,CAAC,OAAO,EAAE;YAC1C,GAAG,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS;SACrE,CAAC,CAAC;IACL,CAAC;+GAjDU,oBAAoB;mGAApB,oBAAoB,sGAxGrB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BT,y/CA9BS,YAAY;;4FAyGX,oBAAoB;kBA5GhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,CAAC,YACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BT;8BA4EQ,IAAI;sBAAZ,KAAK","sourcesContent":["import { Component, Input, signal, computed } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { GeneratedTest } from '../models/recorded-action.model';\r\n\r\n@Component({\r\n  selector: 'app-test-preview',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  template: `\r\n    <div class=\"test-preview\" data-debug-panel>\r\n      @if (!test) {\r\n        <div class=\"empty-state\">\r\n          <div class=\"empty-icon\">🤖</div>\r\n          <p>Noch kein Test generiert.</p>\r\n          <p class=\"hint\">Zeichne Aktionen auf und klicke „🤖 → Cypress Test\".</p>\r\n        </div>\r\n      } @else {\r\n        <div class=\"test-toolbar\" data-debug-panel>\r\n          <div class=\"test-meta\">\r\n            <span class=\"model-tag\">{{ test.model }}</span>\r\n            <span class=\"gen-time\">{{ formatDate(test.generatedAt) }}</span>\r\n          </div>\r\n          <div class=\"test-actions\">\r\n            <button class=\"action-btn\" title=\"Kopieren\" (click)=\"copyCode()\">\r\n              {{ copied() ? '✅ Kopiert!' : '📋 Kopieren' }}\r\n            </button>\r\n            <button class=\"action-btn\" title=\"Herunterladen\" (click)=\"downloadCode()\">\r\n              💾 Download\r\n            </button>\r\n          </div>\r\n        </div>\r\n\r\n        <div class=\"code-container\">\r\n          <pre class=\"code-block\"><code [innerHTML]=\"highlightedCode()\"></code></pre>\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .test-preview { height: 100%; display: flex; flex-direction: column; }\r\n\r\n    .empty-state {\r\n      text-align: center;\r\n      padding: 32px 20px;\r\n      color: #64748b;\r\n    }\r\n    .empty-icon { font-size: 40px; margin-bottom: 10px; }\r\n    .empty-state p { margin: 4px 0; font-size: 13px; }\r\n    .hint { font-size: 11px; color: #475569; }\r\n\r\n    .test-toolbar {\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      padding: 8px 12px;\r\n      background: #1e293b;\r\n      border-bottom: 1px solid #334155;\r\n      flex-shrink: 0;\r\n      gap: 8px;\r\n    }\r\n    .test-meta { display: flex; align-items: center; gap: 8px; flex: 1; min-width: 0; }\r\n    .model-tag {\r\n      background: #312e81;\r\n      color: #a5b4fc;\r\n      font-size: 10px;\r\n      padding: 2px 7px;\r\n      border-radius: 4px;\r\n      font-weight: 600;\r\n      white-space: nowrap;\r\n    }\r\n    .gen-time { font-size: 10px; color: #64748b; white-space: nowrap; }\r\n    .test-actions { display: flex; gap: 6px; flex-shrink: 0; }\r\n    .action-btn {\r\n      background: #334155;\r\n      border: none;\r\n      color: #cbd5e1;\r\n      padding: 4px 10px;\r\n      border-radius: 5px;\r\n      font-size: 11px;\r\n      cursor: pointer;\r\n      white-space: nowrap;\r\n      transition: background 0.15s;\r\n    }\r\n    .action-btn:hover { background: #475569; }\r\n\r\n    .code-container {\r\n      flex: 1;\r\n      overflow: auto;\r\n    }\r\n    .code-container::-webkit-scrollbar { width: 5px; height: 5px; }\r\n    .code-container::-webkit-scrollbar-thumb { background: #334155; border-radius: 3px; }\r\n\r\n    .code-block {\r\n      margin: 0;\r\n      padding: 14px;\r\n      font-family: 'Cascadia Code', 'Consolas', 'Fira Code', monospace;\r\n      font-size: 11px;\r\n      line-height: 1.7;\r\n      color: #e2e8f0;\r\n      white-space: pre;\r\n      tab-size: 2;\r\n    }\r\n\r\n    /* Syntax Highlighting */\r\n    :global(.kw)  { color: #c084fc; }\r\n    :global(.str) { color: #86efac; }\r\n    :global(.fn)  { color: #67e8f9; }\r\n    :global(.cm)  { color: #64748b; font-style: italic; }\r\n    :global(.num) { color: #fb923c; }\r\n    :global(.cy)  { color: #fbbf24; font-weight: 600; }\r\n  `],\r\n})\r\nexport class TestPreviewComponent {\r\n  @Input() test: GeneratedTest | null = null;\r\n\r\n  copied = signal(false);\r\n\r\n  highlightedCode = computed(() => {\r\n    if (!this.test) return '';\r\n    return this.syntaxHighlight(this.test.code);\r\n  });\r\n\r\n  private syntaxHighlight(code: string): string {\r\n    return code\r\n      .replace(/&/g, '&amp;')\r\n      .replace(/</g, '&lt;')\r\n      .replace(/>/g, '&gt;')\r\n      // Comments\r\n      .replace(/(\\/\\/[^\\n]*)/g, '<span class=\"cm\">$1</span>')\r\n      // cy. commands\r\n      .replace(/\\b(cy)\\b/g, '<span class=\"cy\">$1</span>')\r\n      // Keywords\r\n      .replace(/\\b(describe|it|beforeEach|afterEach|before|after|context|specify|const|let|var|function|return|import|export|from|async|await|new)\\b/g, '<span class=\"kw\">$1</span>')\r\n      // Strings\r\n      .replace(/('[^']*'|\"[^\"]*\"|`[^`]*`)/g, '<span class=\"str\">$1</span>')\r\n      // Numbers\r\n      .replace(/\\b(\\d+)\\b/g, '<span class=\"num\">$1</span>');\r\n  }\r\n\r\n  async copyCode(): Promise<void> {\r\n    if (!this.test) return;\r\n    await navigator.clipboard.writeText(this.test.code);\r\n    this.copied.set(true);\r\n    setTimeout(() => this.copied.set(false), 2000);\r\n  }\r\n\r\n  downloadCode(): void {\r\n    if (!this.test) return;\r\n    const blob = new Blob([this.test.code], { type: 'text/typescript' });\r\n    const url = URL.createObjectURL(blob);\r\n    const a = document.createElement('a');\r\n    a.href = url;\r\n    a.download = `cypress-test-${new Date().toISOString().slice(0, 10)}.cy.ts`;\r\n    a.click();\r\n    URL.revokeObjectURL(url);\r\n  }\r\n\r\n  formatDate(ts: number): string {\r\n    return new Date(ts).toLocaleString('de-DE', {\r\n      day: '2-digit', month: '2-digit', hour: '2-digit', minute: '2-digit',\r\n    });\r\n  }\r\n}\r\n"]}
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-preview.component.js","sourceRoot":"","sources":["../../../../../src/lib/test-preview/test-preview.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;;AA+G/C,MAAM,OAAO,oBAAoB;IA5GjC;QA6GW,SAAI,GAAyB,IAAI,CAAC;QAE3C,WAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAEvB,oBAAe,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC9B,IAAI,CAAC,IAAI,CAAC,IAAI;gBAAE,OAAO,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9C,CAAC,CAAC,CAAC;KA0CJ;IAxCS,eAAe,CAAC,IAAY;QAClC,OAAO,IAAI;aACR,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC;aACtB,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC;aACrB,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC;YACtB,WAAW;aACV,OAAO,CAAC,eAAe,EAAE,4BAA4B,CAAC;YACvD,eAAe;aACd,OAAO,CAAC,WAAW,EAAE,4BAA4B,CAAC;YACnD,WAAW;aACV,OAAO,CAAC,uIAAuI,EAAE,4BAA4B,CAAC;YAC/K,UAAU;aACT,OAAO,CAAC,4BAA4B,EAAE,6BAA6B,CAAC;YACrE,UAAU;aACT,OAAO,CAAC,YAAY,EAAE,6BAA6B,CAAC,CAAC;IAC1D,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QACvB,MAAM,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACtB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,IAAI,CAAC,CAAC;IACjD,CAAC;IAED,YAAY;QACV,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QACvB,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,EAAE,iBAAiB,EAAE,CAAC,CAAC;QACrE,MAAM,GAAG,GAAG,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,CAAC,GAAG,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;QACtC,CAAC,CAAC,IAAI,GAAG,GAAG,CAAC;QACb,CAAC,CAAC,QAAQ,GAAG,gBAAgB,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC;QAC3E,CAAC,CAAC,KAAK,EAAE,CAAC;QACV,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;IAC3B,CAAC;IAED,UAAU,CAAC,EAAU;QACnB,OAAO,IAAI,IAAI,CAAC,EAAE,CAAC,CAAC,cAAc,CAAC,OAAO,EAAE;YAC1C,GAAG,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS;SACrE,CAAC,CAAC;IACL,CAAC;+GAjDU,oBAAoB;mGAApB,oBAAoB,sGAxGrB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BT,y/CA9BS,YAAY;;4FAyGX,oBAAoB;kBA5GhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,CAAC,YACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BT;8BA4EQ,IAAI;sBAAZ,KAAK","sourcesContent":["import { Component, Input, signal, computed } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { GeneratedTest } from '../models/recorded-action.model';\r\n\r\n@Component({\r\n  selector: 'app-test-preview',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  template: `\r\n    <div class=\"test-preview\" data-debug-panel>\r\n      @if (!test) {\r\n        <div class=\"empty-state\">\r\n          <div class=\"empty-icon\">🤖</div>\r\n          <p>Noch kein Test generiert.</p>\r\n          <p class=\"hint\">Zeichne Aktionen auf und klicke „🤖 → Cypress Test\".</p>\r\n        </div>\r\n      } @else {\r\n        <div class=\"test-toolbar\" data-debug-panel>\r\n          <div class=\"test-meta\">\r\n            <span class=\"model-tag\">{{ test.model }}</span>\r\n            <span class=\"gen-time\">{{ formatDate(test.generatedAt) }}</span>\r\n          </div>\r\n          <div class=\"test-actions\">\r\n            <button class=\"action-btn\" title=\"Kopieren\" (click)=\"copyCode()\">\r\n              {{ copied() ? '✅ Kopiert!' : '📋 Kopieren' }}\r\n            </button>\r\n            <button class=\"action-btn\" title=\"Herunterladen\" (click)=\"downloadCode()\">\r\n              💾 Download\r\n            </button>\r\n          </div>\r\n        </div>\r\n\r\n        <div class=\"code-container\">\r\n          <pre class=\"code-block\"><code [innerHTML]=\"highlightedCode()\"></code></pre>\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .test-preview { height: 100%; display: flex; flex-direction: column; }\r\n\r\n    .empty-state {\r\n      text-align: center;\r\n      padding: 32px 20px;\r\n      color: #64748b;\r\n    }\r\n    .empty-icon { font-size: 40px; margin-bottom: 10px; }\r\n    .empty-state p { margin: 4px 0; font-size: 13px; }\r\n    .hint { font-size: 11px; color: #475569; }\r\n\r\n    .test-toolbar {\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      padding: 8px 12px;\r\n      background: #1e293b;\r\n      border-bottom: 1px solid #334155;\r\n      flex-shrink: 0;\r\n      gap: 8px;\r\n    }\r\n    .test-meta { display: flex; align-items: center; gap: 8px; flex: 1; min-width: 0; }\r\n    .model-tag {\r\n      background: #312e81;\r\n      color: #a5b4fc;\r\n      font-size: 10px;\r\n      padding: 2px 7px;\r\n      border-radius: 4px;\r\n      font-weight: 600;\r\n      white-space: nowrap;\r\n    }\r\n    .gen-time { font-size: 10px; color: #64748b; white-space: nowrap; }\r\n    .test-actions { display: flex; gap: 6px; flex-shrink: 0; }\r\n    .action-btn {\r\n      background: #334155;\r\n      border: none;\r\n      color: #cbd5e1;\r\n      padding: 4px 10px;\r\n      border-radius: 5px;\r\n      font-size: 11px;\r\n      cursor: pointer;\r\n      white-space: nowrap;\r\n      transition: background 0.15s;\r\n    }\r\n    .action-btn:hover { background: #475569; }\r\n\r\n    .code-container {\r\n      flex: 1;\r\n      overflow: auto;\r\n    }\r\n    .code-container::-webkit-scrollbar { width: 5px; height: 5px; }\r\n    .code-container::-webkit-scrollbar-thumb { background: #334155; border-radius: 3px; }\r\n\r\n    .code-block {\r\n      margin: 0;\r\n      padding: 14px;\r\n      font-family: 'Cascadia Code', 'Consolas', 'Fira Code', monospace;\r\n      font-size: 11px;\r\n      line-height: 1.7;\r\n      color: #e2e8f0;\r\n      white-space: pre;\r\n      tab-size: 2;\r\n    }\r\n\r\n    /* Syntax Highlighting */\r\n    :global(.kw)  { color: #c084fc; }\r\n    :global(.str) { color: #86efac; }\r\n    :global(.fn)  { color: #67e8f9; }\r\n    :global(.cm)  { color: #64748b; font-style: italic; }\r\n    :global(.num) { color: #fb923c; }\r\n    :global(.cy)  { color: #fbbf24; font-weight: 600; }\r\n  `],\r\n})\r\nexport class TestPreviewComponent {\r\n  @Input() test: GeneratedTest | null = null;\r\n\r\n  copied = signal(false);\r\n\r\n  highlightedCode = computed(() => {\r\n    if (!this.test) return '';\r\n    return this.syntaxHighlight(this.test.code);\r\n  });\r\n\r\n  private syntaxHighlight(code: string): string {\r\n    return code\r\n      .replace(/&/g, '&amp;')\r\n      .replace(/</g, '&lt;')\r\n      .replace(/>/g, '&gt;')\r\n      // Comments\r\n      .replace(/(\\/\\/[^\\n]*)/g, '<span class=\"cm\">$1</span>')\r\n      // cy. commands\r\n      .replace(/\\b(cy)\\b/g, '<span class=\"cy\">$1</span>')\r\n      // Keywords\r\n      .replace(/\\b(describe|it|beforeEach|afterEach|before|after|context|specify|const|let|var|function|return|import|export|from|async|await|new)\\b/g, '<span class=\"kw\">$1</span>')\r\n      // Strings\r\n      .replace(/('[^']*'|\"[^\"]*\"|`[^`]*`)/g, '<span class=\"str\">$1</span>')\r\n      // Numbers\r\n      .replace(/\\b(\\d+)\\b/g, '<span class=\"num\">$1</span>');\r\n  }\r\n\r\n  async copyCode(): Promise<void> {\r\n    if (!this.test) return;\r\n    await navigator.clipboard.writeText(this.test.code);\r\n    this.copied.set(true);\r\n    setTimeout(() => this.copied.set(false), 2000);\r\n  }\r\n\r\n  downloadCode(): void {\r\n    if (!this.test) return;\r\n    const blob = new Blob([this.test.code], { type: 'text/typescript' });\r\n    const url = URL.createObjectURL(blob);\r\n    const a = document.createElement('a');\r\n    a.href = url;\r\n    a.download = `cypress-test-${new Date().toISOString().slice(0, 10)}.cy.ts`;\r\n    a.click();\r\n    URL.revokeObjectURL(url);\r\n  }\r\n\r\n  formatDate(ts: number): string {\r\n    return new Date(ts).toLocaleString('de-DE', {\r\n      day: '2-digit', month: '2-digit', hour: '2-digit', minute: '2-digit',\r\n    });\r\n  }\r\n}\r\n"]}
@@ -14,4 +14,4 @@ export * from './lib/services/ai-generator.service';
14
14
  export * from './lib/services/rrweb-recorder.service';
15
15
  // Models / interfaces
16
16
  export * from './lib/models/recorded-action.model';
17
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2RlYnVnLXJlY29yZGVyL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsc0RBQXNEO0FBQ3RELGNBQWMseUNBQXlDLENBQUM7QUFFeEQsaURBQWlEO0FBQ2pELGNBQWMseUNBQXlDLENBQUM7QUFDeEQsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLCtDQUErQyxDQUFDO0FBQzlELGNBQWMsaURBQWlELENBQUM7QUFFaEUsV0FBVztBQUNYLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLHVDQUF1QyxDQUFDO0FBRXRELHNCQUFzQjtBQUN0QixjQUFjLG9DQUFvQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcclxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIGRlYnVnLXJlY29yZGVyXHJcbiAqL1xyXG5cclxuLy8gTWFpbiBlbnRyeSBjb21wb25lbnQg4oCUIGFkZCB0aGlzIHRvIHlvdXIgQW5ndWxhciBhcHBcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGVidWctcGFuZWwvZGVidWctcGFuZWwuY29tcG9uZW50JztcclxuXHJcbi8vIFN1Yi1jb21wb25lbnRzIChpZiB5b3UgbmVlZCB0aGVtIGluZGl2aWR1YWxseSlcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvYWN0aW9uLWxpc3QvYWN0aW9uLWxpc3QuY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvdGVzdC1wcmV2aWV3L3Rlc3QtcHJldmlldy5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXNzaW9uLXJlcGxheS9zZXNzaW9uLXJlcGxheS5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXR0aW5ncy1kaWFsb2cvc2V0dGluZ3MtZGlhbG9nLmNvbXBvbmVudCc7XHJcblxyXG4vLyBTZXJ2aWNlc1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9yZWNvcmRlci5zZXJ2aWNlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvc2VydmljZXMvYWktZ2VuZXJhdG9yLnNlcnZpY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9ycndlYi1yZWNvcmRlci5zZXJ2aWNlJztcclxuXHJcbi8vIE1vZGVscyAvIGludGVyZmFjZXNcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvbW9kZWxzL3JlY29yZGVkLWFjdGlvbi5tb2RlbCc7XHJcbiJdfQ==
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsc0RBQXNEO0FBQ3RELGNBQWMseUNBQXlDLENBQUM7QUFFeEQsaURBQWlEO0FBQ2pELGNBQWMseUNBQXlDLENBQUM7QUFDeEQsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLCtDQUErQyxDQUFDO0FBQzlELGNBQWMsaURBQWlELENBQUM7QUFFaEUsV0FBVztBQUNYLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLHVDQUF1QyxDQUFDO0FBRXRELHNCQUFzQjtBQUN0QixjQUFjLG9DQUFvQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcclxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIGRlYnVnLXJlY29yZGVyXHJcbiAqL1xyXG5cclxuLy8gTWFpbiBlbnRyeSBjb21wb25lbnQg4oCUIGFkZCB0aGlzIHRvIHlvdXIgQW5ndWxhciBhcHBcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGVidWctcGFuZWwvZGVidWctcGFuZWwuY29tcG9uZW50JztcclxuXHJcbi8vIFN1Yi1jb21wb25lbnRzIChpZiB5b3UgbmVlZCB0aGVtIGluZGl2aWR1YWxseSlcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvYWN0aW9uLWxpc3QvYWN0aW9uLWxpc3QuY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvdGVzdC1wcmV2aWV3L3Rlc3QtcHJldmlldy5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXNzaW9uLXJlcGxheS9zZXNzaW9uLXJlcGxheS5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXR0aW5ncy1kaWFsb2cvc2V0dGluZ3MtZGlhbG9nLmNvbXBvbmVudCc7XHJcblxyXG4vLyBTZXJ2aWNlc1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9yZWNvcmRlci5zZXJ2aWNlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvc2VydmljZXMvYWktZ2VuZXJhdG9yLnNlcnZpY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9ycndlYi1yZWNvcmRlci5zZXJ2aWNlJztcclxuXHJcbi8vIE1vZGVscyAvIGludGVyZmFjZXNcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvbW9kZWxzL3JlY29yZGVkLWFjdGlvbi5tb2RlbCc7XHJcbiJdfQ==