angular-debug-recorder 1.0.0

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.
@@ -0,0 +1,166 @@
1
+ import { Component, ViewChild, inject, signal, } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { RrwebRecorderService } from '../services/rrweb-recorder.service';
4
+ import * as i0 from "@angular/core";
5
+ export class SessionReplayComponent {
6
+ constructor() {
7
+ this.rrweb = inject(RrwebRecorderService);
8
+ this.overlayOpen = signal(false);
9
+ this.isPlaying = signal(false);
10
+ }
11
+ async openOverlay() {
12
+ this.overlayOpen.set(true);
13
+ this.isPlaying.set(false);
14
+ // Wait for the DOM to render the overlay
15
+ await new Promise(r => setTimeout(r, 50));
16
+ await this.startPlay();
17
+ }
18
+ closeOverlay() {
19
+ this.rrweb.destroyReplayer();
20
+ this.overlayOpen.set(false);
21
+ this.isPlaying.set(false);
22
+ }
23
+ async pauseResume() {
24
+ if (this.isPlaying()) {
25
+ this.rrweb.pauseReplay();
26
+ this.isPlaying.set(false);
27
+ }
28
+ else {
29
+ this.rrweb.resumeReplay();
30
+ this.isPlaying.set(true);
31
+ }
32
+ }
33
+ async restart() {
34
+ this.rrweb.destroyReplayer();
35
+ await new Promise(r => setTimeout(r, 50));
36
+ await this.startPlay();
37
+ }
38
+ exportSession() {
39
+ this.rrweb.downloadEvents();
40
+ }
41
+ async startPlay() {
42
+ if (!this.replayContainer)
43
+ return;
44
+ const container = this.replayContainer.nativeElement;
45
+ await this.rrweb.startReplay(container);
46
+ // Scale iframe to fill the stage container
47
+ this.scaleReplayer(container);
48
+ this.isPlaying.set(true);
49
+ }
50
+ scaleReplayer(container) {
51
+ // rrweb injects .replayer-wrapper with an iframe sized to the recorded viewport
52
+ setTimeout(() => {
53
+ const wrapper = container.querySelector('.replayer-wrapper');
54
+ if (!wrapper)
55
+ return;
56
+ const iframe = wrapper.querySelector('iframe');
57
+ const wrapW = iframe?.offsetWidth || wrapper.offsetWidth || 1280;
58
+ const wrapH = iframe?.offsetHeight || wrapper.offsetHeight || 720;
59
+ const stageW = container.offsetWidth;
60
+ const stageH = container.offsetHeight;
61
+ if (!stageW || !stageH || !wrapW || !wrapH)
62
+ return;
63
+ const scale = Math.min(stageW / wrapW, stageH / wrapH);
64
+ const offsetX = (stageW - wrapW * scale) / 2;
65
+ const offsetY = (stageH - wrapH * scale) / 2;
66
+ wrapper.style.transform = `translate(${offsetX}px, ${offsetY}px) scale(${scale})`;
67
+ }, 300);
68
+ }
69
+ ngOnDestroy() {
70
+ this.rrweb.destroyReplayer();
71
+ }
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
+ }
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
+ }
117
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SessionReplayComponent, decorators: [{
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
+ }
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
+ }], propDecorators: { replayContainer: [{
163
+ type: ViewChild,
164
+ args: ['replayContainer']
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":"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","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"]}
@@ -0,0 +1,105 @@
1
+ import { Component, Output, EventEmitter, inject } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { FormsModule } from '@angular/forms';
4
+ import { AiGeneratorService } from '../services/ai-generator.service';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@angular/forms";
7
+ export class SettingsDialogComponent {
8
+ constructor() {
9
+ this.close = new EventEmitter();
10
+ this.ai = inject(AiGeneratorService);
11
+ this.localUrl = this.ai.webhookUrl();
12
+ }
13
+ save() {
14
+ this.ai.setWebhookUrl(this.localUrl.trim());
15
+ this.close.emit();
16
+ }
17
+ onOverlayClick(e) {
18
+ if (e.target.classList.contains('overlay')) {
19
+ this.close.emit();
20
+ }
21
+ }
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>
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
+ }
62
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SettingsDialogComponent, decorators: [{
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>
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
+ }], propDecorators: { close: [{
103
+ type: Output
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"]}
@@ -0,0 +1,120 @@
1
+ import { Component, Input, signal, computed } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from "@angular/core";
4
+ export class TestPreviewComponent {
5
+ constructor() {
6
+ this.test = null;
7
+ this.copied = signal(false);
8
+ this.highlightedCode = computed(() => {
9
+ if (!this.test)
10
+ return '';
11
+ return this.syntaxHighlight(this.test.code);
12
+ });
13
+ }
14
+ syntaxHighlight(code) {
15
+ return code
16
+ .replace(/&/g, '&amp;')
17
+ .replace(/</g, '&lt;')
18
+ .replace(/>/g, '&gt;')
19
+ // Comments
20
+ .replace(/(\/\/[^\n]*)/g, '<span class="cm">$1</span>')
21
+ // cy. commands
22
+ .replace(/\b(cy)\b/g, '<span class="cy">$1</span>')
23
+ // Keywords
24
+ .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>')
25
+ // Strings
26
+ .replace(/('[^']*'|"[^"]*"|`[^`]*`)/g, '<span class="str">$1</span>')
27
+ // Numbers
28
+ .replace(/\b(\d+)\b/g, '<span class="num">$1</span>');
29
+ }
30
+ async copyCode() {
31
+ if (!this.test)
32
+ return;
33
+ await navigator.clipboard.writeText(this.test.code);
34
+ this.copied.set(true);
35
+ setTimeout(() => this.copied.set(false), 2000);
36
+ }
37
+ downloadCode() {
38
+ if (!this.test)
39
+ return;
40
+ const blob = new Blob([this.test.code], { type: 'text/typescript' });
41
+ const url = URL.createObjectURL(blob);
42
+ const a = document.createElement('a');
43
+ a.href = url;
44
+ a.download = `cypress-test-${new Date().toISOString().slice(0, 10)}.cy.ts`;
45
+ a.click();
46
+ URL.revokeObjectURL(url);
47
+ }
48
+ formatDate(ts) {
49
+ return new Date(ts).toLocaleString('de-DE', {
50
+ day: '2-digit', month: '2-digit', hour: '2-digit', minute: '2-digit',
51
+ });
52
+ }
53
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TestPreviewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
54
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: TestPreviewComponent, isStandalone: true, selector: "app-test-preview", inputs: { test: "test" }, ngImport: i0, template: `
55
+ <div class="test-preview" data-debug-panel>
56
+ @if (!test) {
57
+ <div class="empty-state">
58
+ <div class="empty-icon">🤖</div>
59
+ <p>Noch kein Test generiert.</p>
60
+ <p class="hint">Zeichne Aktionen auf und klicke „🤖 → Cypress Test".</p>
61
+ </div>
62
+ } @else {
63
+ <div class="test-toolbar" data-debug-panel>
64
+ <div class="test-meta">
65
+ <span class="model-tag">{{ test.model }}</span>
66
+ <span class="gen-time">{{ formatDate(test.generatedAt) }}</span>
67
+ </div>
68
+ <div class="test-actions">
69
+ <button class="action-btn" title="Kopieren" (click)="copyCode()">
70
+ {{ copied() ? '✅ Kopiert!' : '📋 Kopieren' }}
71
+ </button>
72
+ <button class="action-btn" title="Herunterladen" (click)="downloadCode()">
73
+ 💾 Download
74
+ </button>
75
+ </div>
76
+ </div>
77
+
78
+ <div class="code-container">
79
+ <pre class="code-block"><code [innerHTML]="highlightedCode()"></code></pre>
80
+ </div>
81
+ }
82
+ </div>
83
+ `, isInline: true, styles: [".test-preview{height:100%;display:flex;flex-direction:column}.empty-state{text-align:center;padding:32px 20px;color:#64748b}.empty-icon{font-size:40px;margin-bottom:10px}.empty-state p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.test-toolbar{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;background:#1e293b;border-bottom:1px solid #334155;flex-shrink:0;gap:8px}.test-meta{display:flex;align-items:center;gap:8px;flex:1;min-width:0}.model-tag{background:#312e81;color:#a5b4fc;font-size:10px;padding:2px 7px;border-radius:4px;font-weight:600;white-space:nowrap}.gen-time{font-size:10px;color:#64748b;white-space:nowrap}.test-actions{display:flex;gap:6px;flex-shrink:0}.action-btn{background:#334155;border:none;color:#cbd5e1;padding:4px 10px;border-radius:5px;font-size:11px;cursor:pointer;white-space:nowrap;transition:background .15s}.action-btn:hover{background:#475569}.code-container{flex:1;overflow:auto}.code-container::-webkit-scrollbar{width:5px;height:5px}.code-container::-webkit-scrollbar-thumb{background:#334155;border-radius:3px}.code-block{margin:0;padding:14px;font-family:Cascadia Code,Consolas,Fira Code,monospace;font-size:11px;line-height:1.7;color:#e2e8f0;white-space:pre;tab-size:2}:global(.kw){color:#c084fc}:global(.str){color:#86efac}:global(.fn){color:#67e8f9}:global(.cm){color:#64748b;font-style:italic}:global(.num){color:#fb923c}:global(.cy){color:#fbbf24;font-weight:600}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
84
+ }
85
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TestPreviewComponent, decorators: [{
86
+ type: Component,
87
+ args: [{ selector: 'app-test-preview', standalone: true, imports: [CommonModule], template: `
88
+ <div class="test-preview" data-debug-panel>
89
+ @if (!test) {
90
+ <div class="empty-state">
91
+ <div class="empty-icon">🤖</div>
92
+ <p>Noch kein Test generiert.</p>
93
+ <p class="hint">Zeichne Aktionen auf und klicke „🤖 → Cypress Test".</p>
94
+ </div>
95
+ } @else {
96
+ <div class="test-toolbar" data-debug-panel>
97
+ <div class="test-meta">
98
+ <span class="model-tag">{{ test.model }}</span>
99
+ <span class="gen-time">{{ formatDate(test.generatedAt) }}</span>
100
+ </div>
101
+ <div class="test-actions">
102
+ <button class="action-btn" title="Kopieren" (click)="copyCode()">
103
+ {{ copied() ? '✅ Kopiert!' : '📋 Kopieren' }}
104
+ </button>
105
+ <button class="action-btn" title="Herunterladen" (click)="downloadCode()">
106
+ 💾 Download
107
+ </button>
108
+ </div>
109
+ </div>
110
+
111
+ <div class="code-container">
112
+ <pre class="code-block"><code [innerHTML]="highlightedCode()"></code></pre>
113
+ </div>
114
+ }
115
+ </div>
116
+ `, styles: [".test-preview{height:100%;display:flex;flex-direction:column}.empty-state{text-align:center;padding:32px 20px;color:#64748b}.empty-icon{font-size:40px;margin-bottom:10px}.empty-state p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.test-toolbar{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;background:#1e293b;border-bottom:1px solid #334155;flex-shrink:0;gap:8px}.test-meta{display:flex;align-items:center;gap:8px;flex:1;min-width:0}.model-tag{background:#312e81;color:#a5b4fc;font-size:10px;padding:2px 7px;border-radius:4px;font-weight:600;white-space:nowrap}.gen-time{font-size:10px;color:#64748b;white-space:nowrap}.test-actions{display:flex;gap:6px;flex-shrink:0}.action-btn{background:#334155;border:none;color:#cbd5e1;padding:4px 10px;border-radius:5px;font-size:11px;cursor:pointer;white-space:nowrap;transition:background .15s}.action-btn:hover{background:#475569}.code-container{flex:1;overflow:auto}.code-container::-webkit-scrollbar{width:5px;height:5px}.code-container::-webkit-scrollbar-thumb{background:#334155;border-radius:3px}.code-block{margin:0;padding:14px;font-family:Cascadia Code,Consolas,Fira Code,monospace;font-size:11px;line-height:1.7;color:#e2e8f0;white-space:pre;tab-size:2}:global(.kw){color:#c084fc}:global(.str){color:#86efac}:global(.fn){color:#67e8f9}:global(.cm){color:#64748b;font-style:italic}:global(.num){color:#fb923c}:global(.cy){color:#fbbf24;font-weight:600}\n"] }]
117
+ }], propDecorators: { test: [{
118
+ type: Input
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';\nimport { CommonModule } from '@angular/common';\nimport { GeneratedTest } from '../models/recorded-action.model';\n\n@Component({\n  selector: 'app-test-preview',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <div class=\"test-preview\" data-debug-panel>\n      @if (!test) {\n        <div class=\"empty-state\">\n          <div class=\"empty-icon\">🤖</div>\n          <p>Noch kein Test generiert.</p>\n          <p class=\"hint\">Zeichne Aktionen auf und klicke „🤖 → Cypress Test\".</p>\n        </div>\n      } @else {\n        <div class=\"test-toolbar\" data-debug-panel>\n          <div class=\"test-meta\">\n            <span class=\"model-tag\">{{ test.model }}</span>\n            <span class=\"gen-time\">{{ formatDate(test.generatedAt) }}</span>\n          </div>\n          <div class=\"test-actions\">\n            <button class=\"action-btn\" title=\"Kopieren\" (click)=\"copyCode()\">\n              {{ copied() ? '✅ Kopiert!' : '📋 Kopieren' }}\n            </button>\n            <button class=\"action-btn\" title=\"Herunterladen\" (click)=\"downloadCode()\">\n              💾 Download\n            </button>\n          </div>\n        </div>\n\n        <div class=\"code-container\">\n          <pre class=\"code-block\"><code [innerHTML]=\"highlightedCode()\"></code></pre>\n        </div>\n      }\n    </div>\n  `,\n  styles: [`\n    .test-preview { height: 100%; display: flex; flex-direction: column; }\n\n    .empty-state {\n      text-align: center;\n      padding: 32px 20px;\n      color: #64748b;\n    }\n    .empty-icon { font-size: 40px; margin-bottom: 10px; }\n    .empty-state p { margin: 4px 0; font-size: 13px; }\n    .hint { font-size: 11px; color: #475569; }\n\n    .test-toolbar {\n      display: flex;\n      justify-content: space-between;\n      align-items: center;\n      padding: 8px 12px;\n      background: #1e293b;\n      border-bottom: 1px solid #334155;\n      flex-shrink: 0;\n      gap: 8px;\n    }\n    .test-meta { display: flex; align-items: center; gap: 8px; flex: 1; min-width: 0; }\n    .model-tag {\n      background: #312e81;\n      color: #a5b4fc;\n      font-size: 10px;\n      padding: 2px 7px;\n      border-radius: 4px;\n      font-weight: 600;\n      white-space: nowrap;\n    }\n    .gen-time { font-size: 10px; color: #64748b; white-space: nowrap; }\n    .test-actions { display: flex; gap: 6px; flex-shrink: 0; }\n    .action-btn {\n      background: #334155;\n      border: none;\n      color: #cbd5e1;\n      padding: 4px 10px;\n      border-radius: 5px;\n      font-size: 11px;\n      cursor: pointer;\n      white-space: nowrap;\n      transition: background 0.15s;\n    }\n    .action-btn:hover { background: #475569; }\n\n    .code-container {\n      flex: 1;\n      overflow: auto;\n    }\n    .code-container::-webkit-scrollbar { width: 5px; height: 5px; }\n    .code-container::-webkit-scrollbar-thumb { background: #334155; border-radius: 3px; }\n\n    .code-block {\n      margin: 0;\n      padding: 14px;\n      font-family: 'Cascadia Code', 'Consolas', 'Fira Code', monospace;\n      font-size: 11px;\n      line-height: 1.7;\n      color: #e2e8f0;\n      white-space: pre;\n      tab-size: 2;\n    }\n\n    /* Syntax Highlighting */\n    :global(.kw)  { color: #c084fc; }\n    :global(.str) { color: #86efac; }\n    :global(.fn)  { color: #67e8f9; }\n    :global(.cm)  { color: #64748b; font-style: italic; }\n    :global(.num) { color: #fb923c; }\n    :global(.cy)  { color: #fbbf24; font-weight: 600; }\n  `],\n})\nexport class TestPreviewComponent {\n  @Input() test: GeneratedTest | null = null;\n\n  copied = signal(false);\n\n  highlightedCode = computed(() => {\n    if (!this.test) return '';\n    return this.syntaxHighlight(this.test.code);\n  });\n\n  private syntaxHighlight(code: string): string {\n    return code\n      .replace(/&/g, '&amp;')\n      .replace(/</g, '&lt;')\n      .replace(/>/g, '&gt;')\n      // Comments\n      .replace(/(\\/\\/[^\\n]*)/g, '<span class=\"cm\">$1</span>')\n      // cy. commands\n      .replace(/\\b(cy)\\b/g, '<span class=\"cy\">$1</span>')\n      // Keywords\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>')\n      // Strings\n      .replace(/('[^']*'|\"[^\"]*\"|`[^`]*`)/g, '<span class=\"str\">$1</span>')\n      // Numbers\n      .replace(/\\b(\\d+)\\b/g, '<span class=\"num\">$1</span>');\n  }\n\n  async copyCode(): Promise<void> {\n    if (!this.test) return;\n    await navigator.clipboard.writeText(this.test.code);\n    this.copied.set(true);\n    setTimeout(() => this.copied.set(false), 2000);\n  }\n\n  downloadCode(): void {\n    if (!this.test) return;\n    const blob = new Blob([this.test.code], { type: 'text/typescript' });\n    const url = URL.createObjectURL(blob);\n    const a = document.createElement('a');\n    a.href = url;\n    a.download = `cypress-test-${new Date().toISOString().slice(0, 10)}.cy.ts`;\n    a.click();\n    URL.revokeObjectURL(url);\n  }\n\n  formatDate(ts: number): string {\n    return new Date(ts).toLocaleString('de-DE', {\n      day: '2-digit', month: '2-digit', hour: '2-digit', minute: '2-digit',\n    });\n  }\n}\n"]}
@@ -0,0 +1,17 @@
1
+ /*
2
+ * Public API Surface of debug-recorder
3
+ */
4
+ // Main entry component — add this to your Angular app
5
+ export * from './lib/debug-panel/debug-panel.component';
6
+ // Sub-components (if you need them individually)
7
+ export * from './lib/action-list/action-list.component';
8
+ export * from './lib/test-preview/test-preview.component';
9
+ export * from './lib/session-replay/session-replay.component';
10
+ export * from './lib/settings-dialog/settings-dialog.component';
11
+ // Services
12
+ export * from './lib/services/recorder.service';
13
+ export * from './lib/services/ai-generator.service';
14
+ export * from './lib/services/rrweb-recorder.service';
15
+ // Models / interfaces
16
+ export * from './lib/models/recorded-action.model';
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2RlYnVnLXJlY29yZGVyL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsc0RBQXNEO0FBQ3RELGNBQWMseUNBQXlDLENBQUM7QUFFeEQsaURBQWlEO0FBQ2pELGNBQWMseUNBQXlDLENBQUM7QUFDeEQsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLCtDQUErQyxDQUFDO0FBQzlELGNBQWMsaURBQWlELENBQUM7QUFFaEUsV0FBVztBQUNYLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLHVDQUF1QyxDQUFDO0FBRXRELHNCQUFzQjtBQUN0QixjQUFjLG9DQUFvQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcclxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIGRlYnVnLXJlY29yZGVyXHJcbiAqL1xyXG5cclxuLy8gTWFpbiBlbnRyeSBjb21wb25lbnQg4oCUIGFkZCB0aGlzIHRvIHlvdXIgQW5ndWxhciBhcHBcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGVidWctcGFuZWwvZGVidWctcGFuZWwuY29tcG9uZW50JztcclxuXHJcbi8vIFN1Yi1jb21wb25lbnRzIChpZiB5b3UgbmVlZCB0aGVtIGluZGl2aWR1YWxseSlcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvYWN0aW9uLWxpc3QvYWN0aW9uLWxpc3QuY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvdGVzdC1wcmV2aWV3L3Rlc3QtcHJldmlldy5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXNzaW9uLXJlcGxheS9zZXNzaW9uLXJlcGxheS5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXR0aW5ncy1kaWFsb2cvc2V0dGluZ3MtZGlhbG9nLmNvbXBvbmVudCc7XHJcblxyXG4vLyBTZXJ2aWNlc1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9yZWNvcmRlci5zZXJ2aWNlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvc2VydmljZXMvYWktZ2VuZXJhdG9yLnNlcnZpY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9ycndlYi1yZWNvcmRlci5zZXJ2aWNlJztcclxuXHJcbi8vIE1vZGVscyAvIGludGVyZmFjZXNcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvbW9kZWxzL3JlY29yZGVkLWFjdGlvbi5tb2RlbCc7XHJcbiJdfQ==