angular-debug-recorder 1.0.0 → 1.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -3,24 +3,39 @@ import * as i0 from "@angular/core";
3
3
  export class RrwebRecorderService {
4
4
  constructor(zone) {
5
5
  this.zone = zone;
6
- this._events = signal([]);
6
+ // Plain array — cheap push, no O(n) spread on every event
7
+ this._eventsArray = [];
8
+ // Signal only for reactive count display in templates
9
+ this._eventCount = signal(0);
7
10
  this._isRecording = signal(false);
8
- this.events = this._events.asReadonly();
11
+ this.eventCount = this._eventCount.asReadonly();
9
12
  this.isRecording = this._isRecording.asReadonly();
10
13
  }
14
+ hasEvents() {
15
+ return this._eventsArray.length > 0;
16
+ }
17
+ getEvents() {
18
+ return this._eventsArray;
19
+ }
11
20
  async startRecording() {
12
- // Dynamically import rrweb to avoid SSR issues and reduce initial bundle
13
21
  const { record } = await import('rrweb');
14
- this._events.set([]);
22
+ this._eventsArray = [];
23
+ this._eventCount.set(0);
15
24
  this._isRecording.set(true);
16
25
  this.stopFn = record({
17
26
  emit: (event) => {
18
- this.zone.run(() => {
19
- this._events.update(ev => [...ev, event]);
27
+ // Push outside Angular zone no change detection per event
28
+ this.zone.runOutsideAngular(() => {
29
+ this._eventsArray.push(event);
20
30
  });
31
+ // Update count signal only every 10 events to keep UI responsive
32
+ const len = this._eventsArray.length;
33
+ if (len === 1 || len % 10 === 0) {
34
+ this.zone.run(() => this._eventCount.set(len));
35
+ }
21
36
  },
22
- // Note: blockSelector is omitted — rrweb 2.0.0-alpha.4 calls node.matches()
23
- // on TextNodes/CommentNodes which don't have that method, crashing the recorder.
37
+ // Note: blockSelector omitted — rrweb 2.0.0-alpha.4 calls node.matches()
38
+ // on TextNodes/CommentNodes which crash the recorder.
24
39
  maskTextSelector: 'input[type="password"]',
25
40
  checkoutEveryNth: 200,
26
41
  });
@@ -31,24 +46,19 @@ export class RrwebRecorderService {
31
46
  this.stopFn = undefined;
32
47
  }
33
48
  this._isRecording.set(false);
34
- return this._events();
35
- }
36
- getEvents() {
37
- return this._events();
49
+ this._eventCount.set(this._eventsArray.length);
50
+ return this._eventsArray;
38
51
  }
39
52
  clearEvents() {
40
- this._events.set([]);
41
- }
42
- hasEvents() {
43
- return this._events().length > 0;
53
+ this._eventsArray = [];
54
+ this._eventCount.set(0);
44
55
  }
45
56
  // ─── Replay ──────────────────────────────────────────────────────────────
46
57
  async startReplay(container, events) {
47
58
  const { Replayer } = await import('rrweb');
48
- const eventsToReplay = events ?? this._events();
59
+ const eventsToReplay = events ?? this._eventsArray;
49
60
  if (eventsToReplay.length === 0)
50
61
  return;
51
- // Destroy previous replayer
52
62
  this.destroyReplayer();
53
63
  this.replayer = new Replayer(eventsToReplay, {
54
64
  root: container,
@@ -76,7 +86,7 @@ export class RrwebRecorderService {
76
86
  }
77
87
  // ─── Export ───────────────────────────────────────────────────────────────
78
88
  exportEvents() {
79
- return JSON.stringify(this._events(), null, 2);
89
+ return JSON.stringify(this._eventsArray, null, 2);
80
90
  }
81
91
  downloadEvents(filename = 'rrweb-session.json') {
82
92
  const blob = new Blob([this.exportEvents()], { type: 'application/json' });
@@ -90,7 +100,8 @@ export class RrwebRecorderService {
90
100
  importEvents(json) {
91
101
  try {
92
102
  const events = JSON.parse(json);
93
- this._events.set(events);
103
+ this._eventsArray = events;
104
+ this._eventCount.set(events.length);
94
105
  return events;
95
106
  }
96
107
  catch {
@@ -105,4 +116,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
105
116
  type: Injectable,
106
117
  args: [{ providedIn: 'root' }]
107
118
  }], ctorParameters: () => [{ type: i0.NgZone }] });
108
- //# 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  private _events = signal<eventWithTime[]>([]);\r\n  private _isRecording = signal(false);\r\n  private stopFn?: () => void;\r\n  private replayer?: any;\r\n\r\n  events = this._events.asReadonly();\r\n  isRecording = this._isRecording.asReadonly();\r\n\r\n  constructor(private zone: NgZone) {}\r\n\r\n  async startRecording(): Promise<void> {\r\n    // Dynamically import rrweb to avoid SSR issues and reduce initial bundle\r\n    const { record } = await import('rrweb');\r\n    this._events.set([]);\r\n    this._isRecording.set(true);\r\n\r\n    this.stopFn = record({\r\n      emit: (event: eventWithTime) => {\r\n        this.zone.run(() => {\r\n          this._events.update(ev => [...ev, event]);\r\n        });\r\n      },\r\n      // Note: blockSelector is omitted — rrweb 2.0.0-alpha.4 calls node.matches()\r\n      // on TextNodes/CommentNodes which don't have that method, crashing 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    return this._events();\r\n  }\r\n\r\n  getEvents(): eventWithTime[] {\r\n    return this._events();\r\n  }\r\n\r\n  clearEvents(): void {\r\n    this._events.set([]);\r\n  }\r\n\r\n  hasEvents(): boolean {\r\n    return this._events().length > 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._events();\r\n\r\n    if (eventsToReplay.length === 0) return;\r\n\r\n    // Destroy previous replayer\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._events(), 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._events.set(events);\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":["../../../../../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"]}
@@ -70,97 +70,97 @@ export class SessionReplayComponent {
70
70
  this.rrweb.destroyReplayer();
71
71
  }
72
72
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SessionReplayComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
73
- 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
- <div class="replay-panel" data-debug-panel>
75
- @if (!rrweb.hasEvents()) {
76
- <div class="replay-empty">
77
- <div class="replay-icon">📽️</div>
78
- <p>Kein Replay verfügbar.</p>
79
- <p class="hint">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>
80
- </div>
81
- } @else {
82
- <div class="replay-info">
83
- <span class="event-count">{{ rrweb.events().length }} Events aufgezeichnet</span>
84
- </div>
85
- <div class="replay-actions">
86
- <button class="replay-btn primary" (click)="openOverlay()">
87
- ▶ Replay abspielen
88
- </button>
89
- <button class="replay-btn" (click)="exportSession()">
90
- 💾 JSON exportieren
91
- </button>
92
- </div>
93
- <p class="replay-hint">
94
- Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.
95
- </p>
96
- }
97
- </div>
98
-
99
- <!-- Fullscreen Replay Overlay -->
100
- @if (overlayOpen()) {
101
- <div class="replay-overlay" data-debug-panel>
102
- <div class="overlay-header" data-debug-panel>
103
- <span class="overlay-title">📽️ Session Replay</span>
104
- <div class="overlay-controls" data-debug-panel>
105
- <button class="ovl-btn" (click)="pauseResume()">
106
- {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}
107
- </button>
108
- <button class="ovl-btn" (click)="restart()">⟳ Neustart</button>
109
- <button class="ovl-btn close-ovl" (click)="closeOverlay()">✕ Schließen</button>
110
- </div>
111
- </div>
112
- <div #replayContainer class="overlay-stage" data-debug-panel></div>
113
- </div>
114
- }
73
+ 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
+ <div class="replay-panel" data-debug-panel>
75
+ @if (!rrweb.hasEvents()) {
76
+ <div class="replay-empty">
77
+ <div class="replay-icon">📽️</div>
78
+ <p>Kein Replay verfügbar.</p>
79
+ <p class="hint">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>
80
+ </div>
81
+ } @else {
82
+ <div class="replay-info">
83
+ <span class="event-count">{{ rrweb.eventCount() }} Events aufgezeichnet</span>
84
+ </div>
85
+ <div class="replay-actions">
86
+ <button class="replay-btn primary" (click)="openOverlay()">
87
+ ▶ Replay abspielen
88
+ </button>
89
+ <button class="replay-btn" (click)="exportSession()">
90
+ 💾 JSON exportieren
91
+ </button>
92
+ </div>
93
+ <p class="replay-hint">
94
+ Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.
95
+ </p>
96
+ }
97
+ </div>
98
+
99
+ <!-- Fullscreen Replay Overlay -->
100
+ @if (overlayOpen()) {
101
+ <div class="replay-overlay" data-debug-panel>
102
+ <div class="overlay-header" data-debug-panel>
103
+ <span class="overlay-title">📽️ Session Replay</span>
104
+ <div class="overlay-controls" data-debug-panel>
105
+ <button class="ovl-btn" (click)="pauseResume()">
106
+ {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}
107
+ </button>
108
+ <button class="ovl-btn" (click)="restart()">⟳ Neustart</button>
109
+ <button class="ovl-btn close-ovl" (click)="closeOverlay()">✕ Schließen</button>
110
+ </div>
111
+ </div>
112
+ <div #replayContainer class="overlay-stage" data-debug-panel></div>
113
+ </div>
114
+ }
115
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 }] }); }
116
116
  }
117
117
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SessionReplayComponent, decorators: [{
118
118
  type: Component,
119
- args: [{ selector: 'app-session-replay', standalone: true, imports: [CommonModule], template: `
120
- <div class="replay-panel" data-debug-panel>
121
- @if (!rrweb.hasEvents()) {
122
- <div class="replay-empty">
123
- <div class="replay-icon">📽️</div>
124
- <p>Kein Replay verfügbar.</p>
125
- <p class="hint">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>
126
- </div>
127
- } @else {
128
- <div class="replay-info">
129
- <span class="event-count">{{ rrweb.events().length }} Events aufgezeichnet</span>
130
- </div>
131
- <div class="replay-actions">
132
- <button class="replay-btn primary" (click)="openOverlay()">
133
- ▶ Replay abspielen
134
- </button>
135
- <button class="replay-btn" (click)="exportSession()">
136
- 💾 JSON exportieren
137
- </button>
138
- </div>
139
- <p class="replay-hint">
140
- Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.
141
- </p>
142
- }
143
- </div>
144
-
145
- <!-- Fullscreen Replay Overlay -->
146
- @if (overlayOpen()) {
147
- <div class="replay-overlay" data-debug-panel>
148
- <div class="overlay-header" data-debug-panel>
149
- <span class="overlay-title">📽️ Session Replay</span>
150
- <div class="overlay-controls" data-debug-panel>
151
- <button class="ovl-btn" (click)="pauseResume()">
152
- {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}
153
- </button>
154
- <button class="ovl-btn" (click)="restart()">⟳ Neustart</button>
155
- <button class="ovl-btn close-ovl" (click)="closeOverlay()">✕ Schließen</button>
156
- </div>
157
- </div>
158
- <div #replayContainer class="overlay-stage" data-debug-panel></div>
159
- </div>
160
- }
119
+ args: [{ selector: 'app-session-replay', standalone: true, imports: [CommonModule], template: `
120
+ <div class="replay-panel" data-debug-panel>
121
+ @if (!rrweb.hasEvents()) {
122
+ <div class="replay-empty">
123
+ <div class="replay-icon">📽️</div>
124
+ <p>Kein Replay verfügbar.</p>
125
+ <p class="hint">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>
126
+ </div>
127
+ } @else {
128
+ <div class="replay-info">
129
+ <span class="event-count">{{ rrweb.eventCount() }} Events aufgezeichnet</span>
130
+ </div>
131
+ <div class="replay-actions">
132
+ <button class="replay-btn primary" (click)="openOverlay()">
133
+ ▶ Replay abspielen
134
+ </button>
135
+ <button class="replay-btn" (click)="exportSession()">
136
+ 💾 JSON exportieren
137
+ </button>
138
+ </div>
139
+ <p class="replay-hint">
140
+ Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.
141
+ </p>
142
+ }
143
+ </div>
144
+
145
+ <!-- Fullscreen Replay Overlay -->
146
+ @if (overlayOpen()) {
147
+ <div class="replay-overlay" data-debug-panel>
148
+ <div class="overlay-header" data-debug-panel>
149
+ <span class="overlay-title">📽️ Session Replay</span>
150
+ <div class="overlay-controls" data-debug-panel>
151
+ <button class="ovl-btn" (click)="pauseResume()">
152
+ {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}
153
+ </button>
154
+ <button class="ovl-btn" (click)="restart()">⟳ Neustart</button>
155
+ <button class="ovl-btn close-ovl" (click)="closeOverlay()">✕ Schließen</button>
156
+ </div>
157
+ </div>
158
+ <div #replayContainer class="overlay-stage" data-debug-panel></div>
159
+ </div>
160
+ }
161
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"] }]
162
162
  }], propDecorators: { replayContainer: [{
163
163
  type: ViewChild,
164
164
  args: ['replayContainer']
165
165
  }] } });
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":"AAAA,OAAO,EACL,SAAS,EAAc,SAAS,EAAa,MAAM,EAAE,MAAM,GAC5D,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,oBAAoB,EAAE,MAAM,oCAAoC,CAAC;;AAiJ1E,MAAM,OAAO,sBAAsB;IA/InC;QAkJE,UAAK,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QACrC,gBAAW,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC5B,cAAS,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;KAwE3B;IAtEC,KAAK,CAAC,WAAW;QACf,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC3B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC1B,yCAAyC;QACzC,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;QAC1C,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;IACzB,CAAC;IAED,YAAY;QACV,IAAI,CAAC,KAAK,CAAC,eAAe,EAAE,CAAC;QAC7B,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC5B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,WAAW;QACf,IAAI,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC;YACrB,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;YACzB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;YAC1B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAED,KAAK,CAAC,OAAO;QACX,IAAI,CAAC,KAAK,CAAC,eAAe,EAAE,CAAC;QAC7B,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;QAC1C,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;IACzB,CAAC;IAED,aAAa;QACX,IAAI,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC;IAC9B,CAAC;IAEO,KAAK,CAAC,SAAS;QACrB,IAAI,CAAC,IAAI,CAAC,eAAe;YAAE,OAAO;QAClC,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC;QAErD,MAAM,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QAExC,2CAA2C;QAC3C,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;QAC9B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC3B,CAAC;IAEO,aAAa,CAAC,SAAsB;QAC1C,gFAAgF;QAChF,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,OAAO,GAAG,SAAS,CAAC,aAAa,CAAC,mBAAmB,CAAgB,CAAC;YAC5E,IAAI,CAAC,OAAO;gBAAE,OAAO;YAErB,MAAM,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC,QAAQ,CAAsB,CAAC;YACpE,MAAM,KAAK,GAAG,MAAM,EAAE,WAAW,IAAK,OAAO,CAAC,WAAW,IAAK,IAAI,CAAC;YACnE,MAAM,KAAK,GAAG,MAAM,EAAE,YAAY,IAAI,OAAO,CAAC,YAAY,IAAI,GAAG,CAAC;YAClE,MAAM,MAAM,GAAG,SAAS,CAAC,WAAW,CAAC;YACrC,MAAM,MAAM,GAAG,SAAS,CAAC,YAAY,CAAC;YAEtC,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,IAAI,CAAC,KAAK,IAAI,CAAC,KAAK;gBAAE,OAAO;YAEnD,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,GAAG,KAAK,EAAE,MAAM,GAAG,KAAK,CAAC,CAAC;YACvD,MAAM,OAAO,GAAG,CAAC,MAAM,GAAG,KAAK,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;YAC7C,MAAM,OAAO,GAAG,CAAC,MAAM,GAAG,KAAK,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;YAE7C,OAAO,CAAC,KAAK,CAAC,SAAS,GAAG,aAAa,OAAO,OAAO,OAAO,aAAa,KAAK,GAAG,CAAC;QACpF,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;IAED,WAAW;QACT,IAAI,CAAC,KAAK,CAAC,eAAe,EAAE,CAAC;IAC/B,CAAC;+GA5EU,sBAAsB;mGAAtB,sBAAsB,kMA3IvB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0CT,stDA3CS,YAAY;;4FA4IX,sBAAsB;kBA/IlC,SAAS;+BACE,oBAAoB,cAClB,IAAI,WACP,CAAC,YAAY,CAAC,YACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0CT;8BAkG6B,eAAe;sBAA5C,SAAS;uBAAC,iBAAiB","sourcesContent":["import {\n  Component, ElementRef, ViewChild, OnDestroy, inject, signal,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { RrwebRecorderService } from '../services/rrweb-recorder.service';\n\n@Component({\n  selector: 'app-session-replay',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <div class=\"replay-panel\" data-debug-panel>\n      @if (!rrweb.hasEvents()) {\n        <div class=\"replay-empty\">\n          <div class=\"replay-icon\">📽️</div>\n          <p>Kein Replay verfügbar.</p>\n          <p class=\"hint\">Starte eine Aufnahme — rrweb zeichnet den DOM parallel mit.</p>\n        </div>\n      } @else {\n        <div class=\"replay-info\">\n          <span class=\"event-count\">{{ rrweb.events().length }} Events aufgezeichnet</span>\n        </div>\n        <div class=\"replay-actions\">\n          <button class=\"replay-btn primary\" (click)=\"openOverlay()\">\n            ▶ Replay abspielen\n          </button>\n          <button class=\"replay-btn\" (click)=\"exportSession()\">\n            💾 JSON exportieren\n          </button>\n        </div>\n        <p class=\"replay-hint\">\n          Der Replay öffnet sich als Vollbild-Overlay über der aktuellen Seite.\n        </p>\n      }\n    </div>\n\n    <!-- Fullscreen Replay Overlay -->\n    @if (overlayOpen()) {\n      <div class=\"replay-overlay\" data-debug-panel>\n        <div class=\"overlay-header\" data-debug-panel>\n          <span class=\"overlay-title\">📽️ Session Replay</span>\n          <div class=\"overlay-controls\" data-debug-panel>\n            <button class=\"ovl-btn\" (click)=\"pauseResume()\">\n              {{ isPlaying() ? '⏸ Pause' : '▶ Play' }}\n            </button>\n            <button class=\"ovl-btn\" (click)=\"restart()\">⟳ Neustart</button>\n            <button class=\"ovl-btn close-ovl\" (click)=\"closeOverlay()\">✕ Schließen</button>\n          </div>\n        </div>\n        <div #replayContainer class=\"overlay-stage\" data-debug-panel></div>\n      </div>\n    }\n  `,\n  styles: [`\n    .replay-panel {\n      padding: 20px 16px;\n      display: flex;\n      flex-direction: column;\n      gap: 14px;\n    }\n\n    .replay-empty {\n      text-align: center;\n      padding: 20px 0;\n      color: #64748b;\n    }\n    .replay-icon { font-size: 36px; margin-bottom: 8px; }\n    .replay-empty p { margin: 4px 0; font-size: 13px; }\n    .hint { font-size: 11px; color: #475569; }\n\n    .replay-info {\n      background: #1e293b;\n      border-radius: 6px;\n      padding: 8px 12px;\n    }\n    .event-count { font-size: 12px; color: #6ee7b7; font-weight: 600; }\n\n    .replay-actions { display: flex; gap: 8px; }\n    .replay-btn {\n      background: #334155;\n      border: none;\n      color: #cbd5e1;\n      padding: 8px 14px;\n      border-radius: 6px;\n      font-size: 12px;\n      font-weight: 600;\n      cursor: pointer;\n      transition: background 0.15s;\n    }\n    .replay-btn:hover { background: #475569; }\n    .replay-btn.primary { background: #1d4ed8; color: #fff; }\n    .replay-btn.primary:hover { background: #2563eb; }\n\n    .replay-hint { font-size: 11px; color: #475569; margin: 0; }\n\n    /* ── Fullscreen Overlay ── */\n    .replay-overlay {\n      position: fixed;\n      inset: 0;\n      z-index: 99997;\n      background: #000;\n      display: flex;\n      flex-direction: column;\n    }\n    .overlay-header {\n      display: flex;\n      align-items: center;\n      justify-content: space-between;\n      padding: 10px 16px;\n      background: #0f172a;\n      border-bottom: 1px solid #1e293b;\n      flex-shrink: 0;\n    }\n    .overlay-title { font-size: 14px; font-weight: 600; color: #f1f5f9; }\n    .overlay-controls { display: flex; gap: 8px; }\n    .ovl-btn {\n      background: #1e293b;\n      border: 1px solid #334155;\n      color: #cbd5e1;\n      padding: 5px 12px;\n      border-radius: 5px;\n      font-size: 12px;\n      font-weight: 600;\n      cursor: pointer;\n      transition: background 0.15s;\n    }\n    .ovl-btn:hover { background: #334155; }\n    .close-ovl { color: #fca5a5; }\n    .close-ovl:hover { background: rgba(220,38,38,0.2); }\n\n    .overlay-stage {\n      flex: 1;\n      overflow: hidden;\n      position: relative;\n      background: #f8fafc;\n    }\n\n    /* Scale the rrweb iframe to fill available space */\n    :host ::ng-deep .replayer-wrapper {\n      position: absolute !important;\n      top: 0 !important;\n      left: 0 !important;\n      transform-origin: top left !important;\n    }\n    :host ::ng-deep .replayer-wrapper iframe {\n      pointer-events: none;\n    }\n  `],\n})\nexport class SessionReplayComponent implements OnDestroy {\n  @ViewChild('replayContainer') replayContainer?: ElementRef<HTMLDivElement>;\n\n  rrweb = inject(RrwebRecorderService);\n  overlayOpen = signal(false);\n  isPlaying = signal(false);\n\n  async openOverlay(): Promise<void> {\n    this.overlayOpen.set(true);\n    this.isPlaying.set(false);\n    // Wait for the DOM to render the overlay\n    await new Promise(r => setTimeout(r, 50));\n    await this.startPlay();\n  }\n\n  closeOverlay(): void {\n    this.rrweb.destroyReplayer();\n    this.overlayOpen.set(false);\n    this.isPlaying.set(false);\n  }\n\n  async pauseResume(): Promise<void> {\n    if (this.isPlaying()) {\n      this.rrweb.pauseReplay();\n      this.isPlaying.set(false);\n    } else {\n      this.rrweb.resumeReplay();\n      this.isPlaying.set(true);\n    }\n  }\n\n  async restart(): Promise<void> {\n    this.rrweb.destroyReplayer();\n    await new Promise(r => setTimeout(r, 50));\n    await this.startPlay();\n  }\n\n  exportSession(): void {\n    this.rrweb.downloadEvents();\n  }\n\n  private async startPlay(): Promise<void> {\n    if (!this.replayContainer) return;\n    const container = this.replayContainer.nativeElement;\n\n    await this.rrweb.startReplay(container);\n\n    // Scale iframe to fill the stage container\n    this.scaleReplayer(container);\n    this.isPlaying.set(true);\n  }\n\n  private scaleReplayer(container: HTMLElement): void {\n    // rrweb injects .replayer-wrapper with an iframe sized to the recorded viewport\n    setTimeout(() => {\n      const wrapper = container.querySelector('.replayer-wrapper') as HTMLElement;\n      if (!wrapper) return;\n\n      const iframe = wrapper.querySelector('iframe') as HTMLIFrameElement;\n      const wrapW = iframe?.offsetWidth  || wrapper.offsetWidth  || 1280;\n      const wrapH = iframe?.offsetHeight || wrapper.offsetHeight || 720;\n      const stageW = container.offsetWidth;\n      const stageH = container.offsetHeight;\n\n      if (!stageW || !stageH || !wrapW || !wrapH) return;\n\n      const scale = Math.min(stageW / wrapW, stageH / wrapH);\n      const offsetX = (stageW - wrapW * scale) / 2;\n      const offsetY = (stageH - wrapH * scale) / 2;\n\n      wrapper.style.transform = `translate(${offsetX}px, ${offsetY}px) scale(${scale})`;\n    }, 300);\n  }\n\n  ngOnDestroy(): void {\n    this.rrweb.destroyReplayer();\n  }\n}\n"]}
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"]}
@@ -20,86 +20,86 @@ export class SettingsDialogComponent {
20
20
  }
21
21
  }
22
22
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SettingsDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
23
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SettingsDialogComponent, isStandalone: true, selector: "app-settings-dialog", outputs: { close: "close" }, ngImport: i0, template: `
24
- <div class="overlay" data-debug-panel (click)="onOverlayClick($event)">
25
- <div class="dialog" data-debug-panel>
26
- <div class="dialog-header">
27
- <h2>⚙️ Einstellungen</h2>
28
- <button class="close-btn" (click)="close.emit()">✕</button>
29
- </div>
30
-
31
- <div class="dialog-body">
32
- <div class="field-group">
33
- <label>KI Webhook-URL</label>
34
- <input
35
- data-debug-panel
36
- class="field-input"
37
- type="url"
38
- [(ngModel)]="localUrl"
39
- placeholder="https://deine-ki-api.de/generate"
40
- />
41
- <p class="field-hint">
42
- Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.
43
- Die Antwort wird als Cypress-Test angezeigt.
44
- </p>
45
- </div>
46
-
47
- <div class="info-box">
48
- <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>
49
- <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>
50
- <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>
51
- </div>
52
- </div>
53
-
54
- <div class="dialog-footer">
55
- <button class="btn-cancel" (click)="close.emit()">Abbrechen</button>
56
- <button class="btn-save" (click)="save()">💾 Speichern</button>
57
- </div>
58
- </div>
59
- </div>
23
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SettingsDialogComponent, isStandalone: true, selector: "app-settings-dialog", outputs: { close: "close" }, ngImport: i0, template: `
24
+ <div class="overlay" data-debug-panel (click)="onOverlayClick($event)">
25
+ <div class="dialog" data-debug-panel>
26
+ <div class="dialog-header">
27
+ <h2>⚙️ Einstellungen</h2>
28
+ <button class="close-btn" (click)="close.emit()">✕</button>
29
+ </div>
30
+
31
+ <div class="dialog-body">
32
+ <div class="field-group">
33
+ <label>KI Webhook-URL</label>
34
+ <input
35
+ data-debug-panel
36
+ class="field-input"
37
+ type="url"
38
+ [(ngModel)]="localUrl"
39
+ placeholder="https://deine-ki-api.de/generate"
40
+ />
41
+ <p class="field-hint">
42
+ Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.
43
+ Die Antwort wird als Cypress-Test angezeigt.
44
+ </p>
45
+ </div>
46
+
47
+ <div class="info-box">
48
+ <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>
49
+ <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>
50
+ <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>
51
+ </div>
52
+ </div>
53
+
54
+ <div class="dialog-footer">
55
+ <button class="btn-cancel" (click)="close.emit()">Abbrechen</button>
56
+ <button class="btn-save" (click)="save()">💾 Speichern</button>
57
+ </div>
58
+ </div>
59
+ </div>
60
60
  `, isInline: true, styles: [".overlay{position:fixed;inset:0;background:#000000b3;z-index:100000;display:flex;align-items:center;justify-content:center;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px)}.dialog{background:#0f172a;border:1px solid #1e3a5f;border-radius:12px;width:440px;max-width:95vw;display:flex;flex-direction:column;box-shadow:0 25px 60px #0009;overflow:hidden}.dialog-header{display:flex;align-items:center;justify-content:space-between;padding:14px 18px;background:#1e293b;border-bottom:1px solid #334155}.dialog-header h2{margin:0;font-size:15px;color:#f1f5f9;font-weight:600}.close-btn{background:none;border:none;color:#94a3b8;cursor:pointer;font-size:16px;padding:4px;border-radius:4px}.close-btn:hover{color:#f1f5f9;background:#334155}.dialog-body{padding:18px;display:flex;flex-direction:column;gap:16px}.field-group{display:flex;flex-direction:column;gap:6px}label{font-size:12px;font-weight:600;color:#94a3b8;text-transform:uppercase;letter-spacing:.05em}.field-input{background:#1e293b;border:1px solid #334155;color:#e2e8f0;border-radius:6px;padding:10px 12px;font-size:13px;width:100%;box-sizing:border-box}.field-input:focus{outline:none;border-color:#3b82f6}.field-hint{font-size:11px;color:#64748b;margin:0;line-height:1.5}.info-box{background:#1e293b80;border:1px solid #1e3a5f;border-radius:8px;padding:12px;display:flex;flex-direction:column;gap:6px}.info-box p{margin:0;font-size:11px;color:#64748b}.info-box code{background:#1e293b;color:#34d399;padding:1px 4px;border-radius:3px;font-family:monospace}kbd{background:#1e293b;border:1px solid #334155;color:#94a3b8;padding:1px 5px;border-radius:3px;font-size:10px}.dialog-footer{display:flex;gap:8px;justify-content:flex-end;padding:14px 18px;background:#1e293b;border-top:1px solid #334155}.btn-cancel,.btn-save{padding:8px 20px;border-radius:6px;font-size:13px;font-weight:600;cursor:pointer;border:none;transition:filter .15s}.btn-cancel{background:#334155;color:#94a3b8}.btn-cancel:hover{filter:brightness(1.2)}.btn-save{background:#2563eb;color:#fff}.btn-save:hover{filter:brightness(1.1)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] }); }
61
61
  }
62
62
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SettingsDialogComponent, decorators: [{
63
63
  type: Component,
64
- args: [{ selector: 'app-settings-dialog', standalone: true, imports: [CommonModule, FormsModule], template: `
65
- <div class="overlay" data-debug-panel (click)="onOverlayClick($event)">
66
- <div class="dialog" data-debug-panel>
67
- <div class="dialog-header">
68
- <h2>⚙️ Einstellungen</h2>
69
- <button class="close-btn" (click)="close.emit()">✕</button>
70
- </div>
71
-
72
- <div class="dialog-body">
73
- <div class="field-group">
74
- <label>KI Webhook-URL</label>
75
- <input
76
- data-debug-panel
77
- class="field-input"
78
- type="url"
79
- [(ngModel)]="localUrl"
80
- placeholder="https://deine-ki-api.de/generate"
81
- />
82
- <p class="field-hint">
83
- Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.
84
- Die Antwort wird als Cypress-Test angezeigt.
85
- </p>
86
- </div>
87
-
88
- <div class="info-box">
89
- <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>
90
- <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>
91
- <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>
92
- </div>
93
- </div>
94
-
95
- <div class="dialog-footer">
96
- <button class="btn-cancel" (click)="close.emit()">Abbrechen</button>
97
- <button class="btn-save" (click)="save()">💾 Speichern</button>
98
- </div>
99
- </div>
100
- </div>
64
+ args: [{ selector: 'app-settings-dialog', standalone: true, imports: [CommonModule, FormsModule], template: `
65
+ <div class="overlay" data-debug-panel (click)="onOverlayClick($event)">
66
+ <div class="dialog" data-debug-panel>
67
+ <div class="dialog-header">
68
+ <h2>⚙️ Einstellungen</h2>
69
+ <button class="close-btn" (click)="close.emit()">✕</button>
70
+ </div>
71
+
72
+ <div class="dialog-body">
73
+ <div class="field-group">
74
+ <label>KI Webhook-URL</label>
75
+ <input
76
+ data-debug-panel
77
+ class="field-input"
78
+ type="url"
79
+ [(ngModel)]="localUrl"
80
+ placeholder="https://deine-ki-api.de/generate"
81
+ />
82
+ <p class="field-hint">
83
+ Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.
84
+ Die Antwort wird als Cypress-Test angezeigt.
85
+ </p>
86
+ </div>
87
+
88
+ <div class="info-box">
89
+ <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>
90
+ <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>
91
+ <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>
92
+ </div>
93
+ </div>
94
+
95
+ <div class="dialog-footer">
96
+ <button class="btn-cancel" (click)="close.emit()">Abbrechen</button>
97
+ <button class="btn-save" (click)="save()">💾 Speichern</button>
98
+ </div>
99
+ </div>
100
+ </div>
101
101
  `, styles: [".overlay{position:fixed;inset:0;background:#000000b3;z-index:100000;display:flex;align-items:center;justify-content:center;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px)}.dialog{background:#0f172a;border:1px solid #1e3a5f;border-radius:12px;width:440px;max-width:95vw;display:flex;flex-direction:column;box-shadow:0 25px 60px #0009;overflow:hidden}.dialog-header{display:flex;align-items:center;justify-content:space-between;padding:14px 18px;background:#1e293b;border-bottom:1px solid #334155}.dialog-header h2{margin:0;font-size:15px;color:#f1f5f9;font-weight:600}.close-btn{background:none;border:none;color:#94a3b8;cursor:pointer;font-size:16px;padding:4px;border-radius:4px}.close-btn:hover{color:#f1f5f9;background:#334155}.dialog-body{padding:18px;display:flex;flex-direction:column;gap:16px}.field-group{display:flex;flex-direction:column;gap:6px}label{font-size:12px;font-weight:600;color:#94a3b8;text-transform:uppercase;letter-spacing:.05em}.field-input{background:#1e293b;border:1px solid #334155;color:#e2e8f0;border-radius:6px;padding:10px 12px;font-size:13px;width:100%;box-sizing:border-box}.field-input:focus{outline:none;border-color:#3b82f6}.field-hint{font-size:11px;color:#64748b;margin:0;line-height:1.5}.info-box{background:#1e293b80;border:1px solid #1e3a5f;border-radius:8px;padding:12px;display:flex;flex-direction:column;gap:6px}.info-box p{margin:0;font-size:11px;color:#64748b}.info-box code{background:#1e293b;color:#34d399;padding:1px 4px;border-radius:3px;font-family:monospace}kbd{background:#1e293b;border:1px solid #334155;color:#94a3b8;padding:1px 5px;border-radius:3px;font-size:10px}.dialog-footer{display:flex;gap:8px;justify-content:flex-end;padding:14px 18px;background:#1e293b;border-top:1px solid #334155}.btn-cancel,.btn-save{padding:8px 20px;border-radius:6px;font-size:13px;font-weight:600;cursor:pointer;border:none;transition:filter .15s}.btn-cancel{background:#334155;color:#94a3b8}.btn-cancel:hover{filter:brightness(1.2)}.btn-save{background:#2563eb;color:#fff}.btn-save:hover{filter:brightness(1.1)}\n"] }]
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';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { AiGeneratorService } from '../services/ai-generator.service';\n\n@Component({\n  selector: 'app-settings-dialog',\n  standalone: true,\n  imports: [CommonModule, FormsModule],\n  template: `\n    <div class=\"overlay\" data-debug-panel (click)=\"onOverlayClick($event)\">\n      <div class=\"dialog\" data-debug-panel>\n        <div class=\"dialog-header\">\n          <h2>⚙️ Einstellungen</h2>\n          <button class=\"close-btn\" (click)=\"close.emit()\">✕</button>\n        </div>\n\n        <div class=\"dialog-body\">\n          <div class=\"field-group\">\n            <label>KI Webhook-URL</label>\n            <input\n              data-debug-panel\n              class=\"field-input\"\n              type=\"url\"\n              [(ngModel)]=\"localUrl\"\n              placeholder=\"https://deine-ki-api.de/generate\"\n            />\n            <p class=\"field-hint\">\n              Die aufgezeichnete Session wird als JSON per POST an diese URL gesendet.\n              Die Antwort wird als Cypress-Test angezeigt.\n            </p>\n          </div>\n\n          <div class=\"info-box\">\n            <p>📦 POST-Body: Die Session als JSON (<code>actions</code>, <code>startUrl</code>, ...)</p>\n            <p>📄 Erwartete Antwort: Cypress-Test-Code als Plain-Text</p>\n            <p>⌨️ Shortcuts: <kbd>Ctrl+Shift+D</kbd> Panel &nbsp; <kbd>Ctrl+Shift+R</kbd> Aufnahme</p>\n          </div>\n        </div>\n\n        <div class=\"dialog-footer\">\n          <button class=\"btn-cancel\" (click)=\"close.emit()\">Abbrechen</button>\n          <button class=\"btn-save\" (click)=\"save()\">💾 Speichern</button>\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [`\n    .overlay {\n      position: fixed; inset: 0; background: rgba(0,0,0,0.7);\n      z-index: 100000; display: flex; align-items: center; justify-content: center;\n      backdrop-filter: blur(4px);\n    }\n    .dialog {\n      background: #0f172a; border: 1px solid #1e3a5f; border-radius: 12px;\n      width: 440px; max-width: 95vw; display: flex; flex-direction: column;\n      box-shadow: 0 25px 60px rgba(0,0,0,0.6); overflow: hidden;\n    }\n    .dialog-header {\n      display: flex; align-items: center; justify-content: space-between;\n      padding: 14px 18px; background: #1e293b; border-bottom: 1px solid #334155;\n    }\n    .dialog-header h2 { margin: 0; font-size: 15px; color: #f1f5f9; font-weight: 600; }\n    .close-btn {\n      background: none; border: none; color: #94a3b8;\n      cursor: pointer; font-size: 16px; padding: 4px; border-radius: 4px;\n    }\n    .close-btn:hover { color: #f1f5f9; background: #334155; }\n    .dialog-body { padding: 18px; display: flex; flex-direction: column; gap: 16px; }\n    .field-group { display: flex; flex-direction: column; gap: 6px; }\n    label {\n      font-size: 12px; font-weight: 600; color: #94a3b8;\n      text-transform: uppercase; letter-spacing: 0.05em;\n    }\n    .field-input {\n      background: #1e293b; border: 1px solid #334155; color: #e2e8f0;\n      border-radius: 6px; padding: 10px 12px; font-size: 13px;\n      width: 100%; box-sizing: border-box;\n    }\n    .field-input:focus { outline: none; border-color: #3b82f6; }\n    .field-hint { font-size: 11px; color: #64748b; margin: 0; line-height: 1.5; }\n    .info-box {\n      background: rgba(30,41,59,0.5); border: 1px solid #1e3a5f;\n      border-radius: 8px; padding: 12px; display: flex; flex-direction: column; gap: 6px;\n    }\n    .info-box p { margin: 0; font-size: 11px; color: #64748b; }\n    .info-box code {\n      background: #1e293b; color: #34d399; padding: 1px 4px;\n      border-radius: 3px; font-family: monospace;\n    }\n    kbd {\n      background: #1e293b; border: 1px solid #334155; color: #94a3b8;\n      padding: 1px 5px; border-radius: 3px; font-size: 10px;\n    }\n    .dialog-footer {\n      display: flex; gap: 8px; justify-content: flex-end;\n      padding: 14px 18px; background: #1e293b; border-top: 1px solid #334155;\n    }\n    .btn-cancel, .btn-save {\n      padding: 8px 20px; border-radius: 6px; font-size: 13px;\n      font-weight: 600; cursor: pointer; border: none; transition: filter 0.15s;\n    }\n    .btn-cancel { background: #334155; color: #94a3b8; }\n    .btn-cancel:hover { filter: brightness(1.2); }\n    .btn-save { background: #2563eb; color: #fff; }\n    .btn-save:hover { filter: brightness(1.1); }\n  `],\n})\nexport class SettingsDialogComponent {\n  @Output() close = new EventEmitter<void>();\n  private ai = inject(AiGeneratorService);\n  localUrl = this.ai.webhookUrl();\n\n  save(): void {\n    this.ai.setWebhookUrl(this.localUrl.trim());\n    this.close.emit();\n  }\n\n  onOverlayClick(e: MouseEvent): void {\n    if ((e.target as HTMLElement).classList.contains('overlay')) {\n      this.close.emit();\n    }\n  }\n}\n"]}
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"]}