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,516 @@
1
+ import { Component, signal, computed, inject, } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { FormsModule } from '@angular/forms';
4
+ import { RecorderService } from '../services/recorder.service';
5
+ import { AiGeneratorService } from '../services/ai-generator.service';
6
+ import { RrwebRecorderService } from '../services/rrweb-recorder.service';
7
+ import { ActionListComponent } from '../action-list/action-list.component';
8
+ import { TestPreviewComponent } from '../test-preview/test-preview.component';
9
+ import { SettingsDialogComponent } from '../settings-dialog/settings-dialog.component';
10
+ import { SessionReplayComponent } from '../session-replay/session-replay.component';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "@angular/forms";
13
+ export class DebugPanelComponent {
14
+ constructor() {
15
+ this.recorder = inject(RecorderService);
16
+ this.aiService = inject(AiGeneratorService);
17
+ this.rrweb = inject(RrwebRecorderService);
18
+ this.panelOpen = signal(false);
19
+ this.activeTab = signal('actions');
20
+ this.position = signal('bottom-right');
21
+ this.panelWidth = signal(420);
22
+ this.showSettings = signal(false);
23
+ this.sessionName = '';
24
+ this.sessionDesc = '';
25
+ this.hasActions = computed(() => (this.recorder.currentSession()?.actions.length ?? 0) > 0 ||
26
+ (this.recorder.sessions().length > 0));
27
+ this.resizing = false;
28
+ this.resizeStartY = 0;
29
+ this.resizeStartH = 0;
30
+ this.handleHotkey = (e) => {
31
+ if (e.ctrlKey && e.shiftKey && e.key === 'D') {
32
+ e.preventDefault();
33
+ this.togglePanel();
34
+ }
35
+ if (e.ctrlKey && e.shiftKey && e.key === 'R') {
36
+ e.preventDefault();
37
+ if (this.recorder.isRecording()) {
38
+ this.stopRecording();
39
+ }
40
+ else {
41
+ this.startRecording();
42
+ }
43
+ }
44
+ };
45
+ }
46
+ ngOnInit() {
47
+ // Keyboard shortcut: Ctrl+Shift+D
48
+ document.addEventListener('keydown', this.handleHotkey);
49
+ }
50
+ ngOnDestroy() {
51
+ document.removeEventListener('keydown', this.handleHotkey);
52
+ }
53
+ togglePanel() {
54
+ this.panelOpen.update(v => !v);
55
+ if (this.panelOpen() && !this.recorder.currentSession()) {
56
+ this.activeTab.set('actions');
57
+ }
58
+ }
59
+ cyclePosition() {
60
+ const positions = ['bottom-right', 'bottom-left', 'top-right', 'top-left'];
61
+ const idx = positions.indexOf(this.position());
62
+ this.position.set(positions[(idx + 1) % positions.length]);
63
+ }
64
+ startRecording() {
65
+ this.recorder.startRecording(this.sessionName || undefined, this.sessionDesc || undefined);
66
+ // Start rrweb in parallel for visual replay
67
+ this.rrweb.startRecording();
68
+ this.activeTab.set('actions');
69
+ }
70
+ stopRecording() {
71
+ const session = this.recorder.stopRecording();
72
+ this.rrweb.stopRecording();
73
+ if (session) {
74
+ this.activeTab.set('actions');
75
+ }
76
+ }
77
+ async generateTest() {
78
+ const session = this.recorder.currentSession() ?? this.recorder.sessions().at(-1);
79
+ if (!session)
80
+ return;
81
+ await this.aiService.generateCypressTest(session);
82
+ this.activeTab.set('test');
83
+ }
84
+ async loadAndGenerate(session) {
85
+ await this.aiService.generateCypressTest(session);
86
+ this.activeTab.set('test');
87
+ }
88
+ // Drag to reposition
89
+ startDrag(e) {
90
+ if (e.target.closest('button'))
91
+ return;
92
+ const panel = e.currentTarget.parentElement;
93
+ const startX = e.clientX - panel.getBoundingClientRect().left;
94
+ const startY = e.clientY - panel.getBoundingClientRect().top;
95
+ const onMove = (ev) => {
96
+ panel.style.left = `${ev.clientX - startX}px`;
97
+ panel.style.top = `${ev.clientY - startY}px`;
98
+ panel.style.right = 'auto';
99
+ panel.style.bottom = 'auto';
100
+ };
101
+ const onUp = () => {
102
+ document.removeEventListener('mousemove', onMove);
103
+ document.removeEventListener('mouseup', onUp);
104
+ };
105
+ document.addEventListener('mousemove', onMove);
106
+ document.addEventListener('mouseup', onUp);
107
+ }
108
+ // Resize height
109
+ startResize(e) {
110
+ const panel = e.currentTarget.parentElement;
111
+ this.resizeStartY = e.clientY;
112
+ this.resizeStartH = panel.getBoundingClientRect().height;
113
+ const onMove = (ev) => {
114
+ const newH = Math.max(250, this.resizeStartH + (ev.clientY - this.resizeStartY));
115
+ panel.style.maxHeight = `${newH}px`;
116
+ };
117
+ const onUp = () => {
118
+ document.removeEventListener('mousemove', onMove);
119
+ document.removeEventListener('mouseup', onUp);
120
+ };
121
+ document.addEventListener('mousemove', onMove);
122
+ document.addEventListener('mouseup', onUp);
123
+ e.preventDefault();
124
+ }
125
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DebugPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
126
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: DebugPanelComponent, isStandalone: true, selector: "app-debug-panel", ngImport: i0, template: `
127
+ <!-- Toggle FAB -->
128
+ <button
129
+ data-debug-panel
130
+ class="debug-fab"
131
+ [class.recording]="recorder.isRecording()"
132
+ [class.pulse]="recorder.isRecording() && !recorder.isPaused()"
133
+ (click)="togglePanel()"
134
+ [title]="panelOpen() ? 'Debug Panel schließen' : 'Debug Panel öffnen'"
135
+ >
136
+ <span class="fab-icon">{{ recorder.isRecording() ? '⏺' : '🐛' }}</span>
137
+ @if (recorder.isRecording() && recorder.actionCount() > 0) {
138
+ <span class="fab-badge">{{ recorder.actionCount() }}</span>
139
+ }
140
+ </button>
141
+
142
+ <!-- Main Panel -->
143
+ @if (panelOpen()) {
144
+ <div
145
+ data-debug-panel
146
+ class="debug-panel"
147
+ [class]="'pos-' + position()"
148
+ [style.width.px]="panelWidth()"
149
+ >
150
+ <!-- Header -->
151
+ <div class="panel-header" (mousedown)="startDrag($event)">
152
+ <div class="header-left">
153
+ <span class="panel-icon">🐛</span>
154
+ <span class="panel-title">Debug Recorder</span>
155
+ @if (recorder.isRecording()) {
156
+ <span class="rec-indicator" [class.paused]="recorder.isPaused()">
157
+ {{ recorder.isPaused() ? '⏸ PAUSED' : '⏺ REC' }}
158
+ </span>
159
+ }
160
+ </div>
161
+ <div class="header-actions">
162
+ <button class="icon-btn" title="Einstellungen" (click)="showSettings.set(true)">⚙️</button>
163
+ <button class="icon-btn" title="Position wechseln" (click)="cyclePosition()">📌</button>
164
+ <button class="icon-btn" title="Schließen" (click)="togglePanel()">✕</button>
165
+ </div>
166
+ </div>
167
+
168
+ <!-- Session Name Input (only when not recording) -->
169
+ @if (!recorder.isRecording()) {
170
+ <div class="session-setup">
171
+ <input
172
+ data-debug-panel
173
+ class="session-input"
174
+ type="text"
175
+ [(ngModel)]="sessionName"
176
+ placeholder="Session-Name (optional)"
177
+ />
178
+ <textarea
179
+ data-debug-panel
180
+ class="session-desc"
181
+ [(ngModel)]="sessionDesc"
182
+ placeholder="Fehlerbeschreibung..."
183
+ rows="2"
184
+ ></textarea>
185
+ </div>
186
+ }
187
+
188
+ <!-- Recording Controls -->
189
+ <div class="recording-controls" data-debug-panel>
190
+ @if (!recorder.isRecording()) {
191
+ <button class="ctrl-btn start" (click)="startRecording()">
192
+ ▶ Aufnahme starten
193
+ </button>
194
+ } @else {
195
+ <button class="ctrl-btn pause" (click)="recorder.pauseRecording()">
196
+ {{ recorder.isPaused() ? '▶ Fortsetzen' : '⏸ Pause' }}
197
+ </button>
198
+ <button class="ctrl-btn stop" (click)="stopRecording()">
199
+ ⏹ Stoppen
200
+ </button>
201
+ <button class="ctrl-btn clear" title="Aktionen löschen" (click)="recorder.clearCurrentSession()">
202
+ 🗑
203
+ </button>
204
+ }
205
+
206
+ @if (hasActions()) {
207
+ <button
208
+ class="ctrl-btn generate"
209
+ [disabled]="aiService.isGenerating() || !aiService.webhookUrl()"
210
+ [title]="!aiService.webhookUrl() ? 'Webhook-URL in Einstellungen eintragen' : 'Cypress Test generieren'"
211
+ (click)="generateTest()"
212
+ >
213
+ @if (aiService.isGenerating()) {
214
+ <span class="spinner">⟳</span> Generiere...
215
+ } @else {
216
+ 🤖 → Cypress Test
217
+ }
218
+ </button>
219
+ }
220
+ </div>
221
+
222
+ <!-- Error Banner -->
223
+ @if (aiService.error()) {
224
+ <div class="error-banner" data-debug-panel>
225
+ ⚠️ {{ aiService.error() }}
226
+ </div>
227
+ }
228
+
229
+ <!-- Tabs -->
230
+ <div class="tab-bar" data-debug-panel>
231
+ <button
232
+ class="tab-btn"
233
+ [class.active]="activeTab() === 'actions'"
234
+ (click)="activeTab.set('actions')"
235
+ >
236
+ Aktionen
237
+ @if (recorder.actionCount() > 0) {
238
+ <span class="tab-badge">{{ recorder.actionCount() }}</span>
239
+ }
240
+ </button>
241
+ <button
242
+ class="tab-btn"
243
+ [class.active]="activeTab() === 'replay'"
244
+ (click)="activeTab.set('replay')"
245
+ title="rrweb Session Replay"
246
+ >
247
+ 📽️ Replay
248
+ @if (rrweb.events().length > 0) {
249
+ <span class="tab-badge">{{ rrweb.events().length }}</span>
250
+ }
251
+ </button>
252
+ <button
253
+ class="tab-btn"
254
+ [class.active]="activeTab() === 'test'"
255
+ [disabled]="!aiService.lastTest()"
256
+ (click)="activeTab.set('test')"
257
+ >
258
+ 🤖 Test
259
+ @if (aiService.lastTest()) {
260
+ <span class="tab-badge new">NEU</span>
261
+ }
262
+ </button>
263
+ <button
264
+ class="tab-btn"
265
+ [class.active]="activeTab() === 'sessions'"
266
+ [disabled]="recorder.sessions().length === 0"
267
+ (click)="activeTab.set('sessions')"
268
+ >
269
+ Sessions ({{ recorder.sessions().length }})
270
+ </button>
271
+ </div>
272
+
273
+ <!-- Tab Content -->
274
+ <div class="tab-content">
275
+ @if (activeTab() === 'actions') {
276
+ <app-action-list
277
+ [session]="recorder.currentSession()"
278
+ (removeAction)="recorder.removeAction($event)"
279
+ (addNote)="recorder.addNote($event.id, $event.note)"
280
+ />
281
+ }
282
+
283
+ @if (activeTab() === 'replay') {
284
+ <app-session-replay />
285
+ }
286
+
287
+ @if (activeTab() === 'test') {
288
+ <app-test-preview [test]="aiService.lastTest()" />
289
+ }
290
+
291
+ @if (activeTab() === 'sessions') {
292
+ <div class="sessions-list">
293
+ @for (session of recorder.sessions(); track session.id) {
294
+ <div class="session-card">
295
+ <div class="session-card-header">
296
+ <span class="session-name">{{ session.name }}</span>
297
+ <span class="session-meta">{{ session.actions.length }} Aktionen</span>
298
+ </div>
299
+ <div class="session-card-actions">
300
+ <button class="sm-btn" (click)="loadAndGenerate(session)">🤖 Neu generieren</button>
301
+ <button class="sm-btn danger" (click)="recorder.deleteSession(session.id)">🗑</button>
302
+ </div>
303
+ </div>
304
+ }
305
+ </div>
306
+ }
307
+ </div>
308
+
309
+ <!-- Resize Handle -->
310
+ <div class="resize-handle" (mousedown)="startResize($event)">⠿</div>
311
+ </div>
312
+ }
313
+
314
+ <!-- Settings Dialog -->
315
+ @if (showSettings()) {
316
+ <app-settings-dialog (close)="showSettings.set(false)" />
317
+ }
318
+ `, isInline: true, styles: [".debug-fab{bottom:24px;right:24px;z-index:99999;width:52px;height:52px;border-radius:50%;border:none;background:#1e293b;color:#fff;font-size:22px;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:0 4px 20px #0006;transition:transform .2s,background .2s;position:fixed}.debug-fab:hover{transform:scale(1.1);background:#334155}.debug-fab.recording{background:#dc2626}.debug-fab.pulse{animation:fabPulse 1.5s ease-in-out infinite}@keyframes fabPulse{0%,to{box-shadow:0 4px 20px #dc262666}50%{box-shadow:0 4px 30px #dc2626e6,0 0 0 8px #dc262626}}.fab-badge{position:absolute;top:-4px;right:-4px;background:#f59e0b;color:#000;font-size:10px;font-weight:700;min-width:18px;height:18px;border-radius:9px;display:flex;align-items:center;justify-content:center;padding:0 4px}.debug-panel{position:fixed;z-index:99998;width:420px;max-height:80vh;background:#0f172a;color:#e2e8f0;border-radius:12px;border:1px solid #1e3a5f;box-shadow:0 25px 60px #0009;display:flex;flex-direction:column;font-family:Segoe UI,system-ui,sans-serif;font-size:13px;overflow:hidden}.pos-bottom-right{bottom:88px;right:24px}.pos-bottom-left{bottom:88px;left:24px}.pos-top-right{top:24px;right:24px}.pos-top-left{top:24px;left:24px}.panel-header{display:flex;align-items:center;justify-content:space-between;padding:10px 14px;background:#1e293b;border-bottom:1px solid #334155;cursor:move;-webkit-user-select:none;user-select:none;flex-shrink:0}.header-left{display:flex;align-items:center;gap:8px}.panel-icon{font-size:16px}.panel-title{font-weight:600;font-size:14px;color:#f1f5f9}.rec-indicator{font-size:10px;font-weight:700;color:#ef4444;background:#ef444426;padding:2px 7px;border-radius:4px;animation:blink 1s step-end infinite}.rec-indicator.paused{color:#f59e0b;background:#f59e0b26;animation:none}@keyframes blink{50%{opacity:.4}}.header-actions{display:flex;gap:4px}.icon-btn{background:none;border:none;color:#94a3b8;cursor:pointer;padding:4px;border-radius:4px;font-size:14px;line-height:1;transition:color .15s,background .15s}.icon-btn:hover{color:#f1f5f9;background:#334155}.session-setup{padding:10px 14px;border-bottom:1px solid #1e293b;display:flex;flex-direction:column;gap:6px;flex-shrink:0}.session-input,.session-desc{background:#1e293b;border:1px solid #334155;color:#e2e8f0;border-radius:6px;padding:6px 10px;font-size:12px;width:100%;box-sizing:border-box;resize:vertical}.session-input:focus,.session-desc:focus{outline:none;border-color:#3b82f6}.recording-controls{display:flex;gap:6px;padding:10px 14px;border-bottom:1px solid #1e293b;flex-wrap:wrap;flex-shrink:0}.ctrl-btn{border:none;border-radius:6px;padding:6px 14px;font-size:12px;font-weight:600;cursor:pointer;transition:filter .15s,transform .1s;display:flex;align-items:center;gap:5px}.ctrl-btn:hover:not(:disabled){filter:brightness(1.15);transform:translateY(-1px)}.ctrl-btn:active:not(:disabled){transform:translateY(0)}.ctrl-btn:disabled{opacity:.5;cursor:not-allowed}.ctrl-btn.start{background:#16a34a;color:#fff}.ctrl-btn.stop{background:#dc2626;color:#fff}.ctrl-btn.pause{background:#d97706;color:#fff}.ctrl-btn.generate{background:#7c3aed;color:#fff;flex:1;justify-content:center}.ctrl-btn.clear{background:#374151;color:#9ca3af;padding:6px 10px}.spinner{display:inline-block;animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.error-banner{background:#dc262626;border-left:3px solid #dc2626;color:#fca5a5;padding:8px 14px;font-size:12px;flex-shrink:0}.tab-bar{display:flex;border-bottom:1px solid #1e293b;flex-shrink:0}.tab-btn{flex:1;background:none;border:none;color:#64748b;padding:8px 4px;font-size:12px;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:5px;border-bottom:2px solid transparent;transition:color .15s}.tab-btn:hover:not(:disabled){color:#cbd5e1}.tab-btn.active{color:#60a5fa;border-bottom-color:#3b82f6}.tab-btn:disabled{opacity:.4;cursor:not-allowed}.tab-badge{background:#334155;color:#94a3b8;font-size:10px;padding:1px 5px;border-radius:3px}.tab-badge.new{background:#7c3aed;color:#e9d5ff}.tab-content{overflow-y:auto;flex:1;min-height:0}.tab-content::-webkit-scrollbar{width:5px}.tab-content::-webkit-scrollbar-track{background:transparent}.tab-content::-webkit-scrollbar-thumb{background:#334155;border-radius:3px}.sessions-list{padding:10px 14px;display:flex;flex-direction:column;gap:8px}.session-card{background:#1e293b;border:1px solid #334155;border-radius:8px;padding:10px}.session-card-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:8px}.session-name{font-weight:600;color:#f1f5f9;font-size:13px}.session-meta{font-size:11px;color:#64748b}.session-card-actions{display:flex;gap:6px}.sm-btn{background:#334155;border:none;color:#cbd5e1;padding:4px 10px;border-radius:5px;font-size:11px;cursor:pointer;transition:background .15s}.sm-btn:hover{background:#475569}.sm-btn.danger{color:#fca5a5}.sm-btn.danger:hover{background:#dc262633}.resize-handle{text-align:center;color:#334155;cursor:ns-resize;font-size:16px;padding:2px;flex-shrink:0;background:#0f172a;-webkit-user-select:none;user-select:none}.resize-handle:hover{color:#64748b}\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"] }, { kind: "component", type: ActionListComponent, selector: "app-action-list", inputs: ["session"], outputs: ["removeAction", "addNote"] }, { kind: "component", type: TestPreviewComponent, selector: "app-test-preview", inputs: ["test"] }, { kind: "component", type: SettingsDialogComponent, selector: "app-settings-dialog", outputs: ["close"] }, { kind: "component", type: SessionReplayComponent, selector: "app-session-replay" }] }); }
319
+ }
320
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DebugPanelComponent, decorators: [{
321
+ type: Component,
322
+ args: [{ selector: 'app-debug-panel', standalone: true, imports: [CommonModule, FormsModule, ActionListComponent, TestPreviewComponent, SettingsDialogComponent, SessionReplayComponent], template: `
323
+ <!-- Toggle FAB -->
324
+ <button
325
+ data-debug-panel
326
+ class="debug-fab"
327
+ [class.recording]="recorder.isRecording()"
328
+ [class.pulse]="recorder.isRecording() && !recorder.isPaused()"
329
+ (click)="togglePanel()"
330
+ [title]="panelOpen() ? 'Debug Panel schließen' : 'Debug Panel öffnen'"
331
+ >
332
+ <span class="fab-icon">{{ recorder.isRecording() ? '⏺' : '🐛' }}</span>
333
+ @if (recorder.isRecording() && recorder.actionCount() > 0) {
334
+ <span class="fab-badge">{{ recorder.actionCount() }}</span>
335
+ }
336
+ </button>
337
+
338
+ <!-- Main Panel -->
339
+ @if (panelOpen()) {
340
+ <div
341
+ data-debug-panel
342
+ class="debug-panel"
343
+ [class]="'pos-' + position()"
344
+ [style.width.px]="panelWidth()"
345
+ >
346
+ <!-- Header -->
347
+ <div class="panel-header" (mousedown)="startDrag($event)">
348
+ <div class="header-left">
349
+ <span class="panel-icon">🐛</span>
350
+ <span class="panel-title">Debug Recorder</span>
351
+ @if (recorder.isRecording()) {
352
+ <span class="rec-indicator" [class.paused]="recorder.isPaused()">
353
+ {{ recorder.isPaused() ? '⏸ PAUSED' : '⏺ REC' }}
354
+ </span>
355
+ }
356
+ </div>
357
+ <div class="header-actions">
358
+ <button class="icon-btn" title="Einstellungen" (click)="showSettings.set(true)">⚙️</button>
359
+ <button class="icon-btn" title="Position wechseln" (click)="cyclePosition()">📌</button>
360
+ <button class="icon-btn" title="Schließen" (click)="togglePanel()">✕</button>
361
+ </div>
362
+ </div>
363
+
364
+ <!-- Session Name Input (only when not recording) -->
365
+ @if (!recorder.isRecording()) {
366
+ <div class="session-setup">
367
+ <input
368
+ data-debug-panel
369
+ class="session-input"
370
+ type="text"
371
+ [(ngModel)]="sessionName"
372
+ placeholder="Session-Name (optional)"
373
+ />
374
+ <textarea
375
+ data-debug-panel
376
+ class="session-desc"
377
+ [(ngModel)]="sessionDesc"
378
+ placeholder="Fehlerbeschreibung..."
379
+ rows="2"
380
+ ></textarea>
381
+ </div>
382
+ }
383
+
384
+ <!-- Recording Controls -->
385
+ <div class="recording-controls" data-debug-panel>
386
+ @if (!recorder.isRecording()) {
387
+ <button class="ctrl-btn start" (click)="startRecording()">
388
+ ▶ Aufnahme starten
389
+ </button>
390
+ } @else {
391
+ <button class="ctrl-btn pause" (click)="recorder.pauseRecording()">
392
+ {{ recorder.isPaused() ? '▶ Fortsetzen' : '⏸ Pause' }}
393
+ </button>
394
+ <button class="ctrl-btn stop" (click)="stopRecording()">
395
+ ⏹ Stoppen
396
+ </button>
397
+ <button class="ctrl-btn clear" title="Aktionen löschen" (click)="recorder.clearCurrentSession()">
398
+ 🗑
399
+ </button>
400
+ }
401
+
402
+ @if (hasActions()) {
403
+ <button
404
+ class="ctrl-btn generate"
405
+ [disabled]="aiService.isGenerating() || !aiService.webhookUrl()"
406
+ [title]="!aiService.webhookUrl() ? 'Webhook-URL in Einstellungen eintragen' : 'Cypress Test generieren'"
407
+ (click)="generateTest()"
408
+ >
409
+ @if (aiService.isGenerating()) {
410
+ <span class="spinner">⟳</span> Generiere...
411
+ } @else {
412
+ 🤖 → Cypress Test
413
+ }
414
+ </button>
415
+ }
416
+ </div>
417
+
418
+ <!-- Error Banner -->
419
+ @if (aiService.error()) {
420
+ <div class="error-banner" data-debug-panel>
421
+ ⚠️ {{ aiService.error() }}
422
+ </div>
423
+ }
424
+
425
+ <!-- Tabs -->
426
+ <div class="tab-bar" data-debug-panel>
427
+ <button
428
+ class="tab-btn"
429
+ [class.active]="activeTab() === 'actions'"
430
+ (click)="activeTab.set('actions')"
431
+ >
432
+ Aktionen
433
+ @if (recorder.actionCount() > 0) {
434
+ <span class="tab-badge">{{ recorder.actionCount() }}</span>
435
+ }
436
+ </button>
437
+ <button
438
+ class="tab-btn"
439
+ [class.active]="activeTab() === 'replay'"
440
+ (click)="activeTab.set('replay')"
441
+ title="rrweb Session Replay"
442
+ >
443
+ 📽️ Replay
444
+ @if (rrweb.events().length > 0) {
445
+ <span class="tab-badge">{{ rrweb.events().length }}</span>
446
+ }
447
+ </button>
448
+ <button
449
+ class="tab-btn"
450
+ [class.active]="activeTab() === 'test'"
451
+ [disabled]="!aiService.lastTest()"
452
+ (click)="activeTab.set('test')"
453
+ >
454
+ 🤖 Test
455
+ @if (aiService.lastTest()) {
456
+ <span class="tab-badge new">NEU</span>
457
+ }
458
+ </button>
459
+ <button
460
+ class="tab-btn"
461
+ [class.active]="activeTab() === 'sessions'"
462
+ [disabled]="recorder.sessions().length === 0"
463
+ (click)="activeTab.set('sessions')"
464
+ >
465
+ Sessions ({{ recorder.sessions().length }})
466
+ </button>
467
+ </div>
468
+
469
+ <!-- Tab Content -->
470
+ <div class="tab-content">
471
+ @if (activeTab() === 'actions') {
472
+ <app-action-list
473
+ [session]="recorder.currentSession()"
474
+ (removeAction)="recorder.removeAction($event)"
475
+ (addNote)="recorder.addNote($event.id, $event.note)"
476
+ />
477
+ }
478
+
479
+ @if (activeTab() === 'replay') {
480
+ <app-session-replay />
481
+ }
482
+
483
+ @if (activeTab() === 'test') {
484
+ <app-test-preview [test]="aiService.lastTest()" />
485
+ }
486
+
487
+ @if (activeTab() === 'sessions') {
488
+ <div class="sessions-list">
489
+ @for (session of recorder.sessions(); track session.id) {
490
+ <div class="session-card">
491
+ <div class="session-card-header">
492
+ <span class="session-name">{{ session.name }}</span>
493
+ <span class="session-meta">{{ session.actions.length }} Aktionen</span>
494
+ </div>
495
+ <div class="session-card-actions">
496
+ <button class="sm-btn" (click)="loadAndGenerate(session)">🤖 Neu generieren</button>
497
+ <button class="sm-btn danger" (click)="recorder.deleteSession(session.id)">🗑</button>
498
+ </div>
499
+ </div>
500
+ }
501
+ </div>
502
+ }
503
+ </div>
504
+
505
+ <!-- Resize Handle -->
506
+ <div class="resize-handle" (mousedown)="startResize($event)">⠿</div>
507
+ </div>
508
+ }
509
+
510
+ <!-- Settings Dialog -->
511
+ @if (showSettings()) {
512
+ <app-settings-dialog (close)="showSettings.set(false)" />
513
+ }
514
+ `, styles: [".debug-fab{bottom:24px;right:24px;z-index:99999;width:52px;height:52px;border-radius:50%;border:none;background:#1e293b;color:#fff;font-size:22px;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:0 4px 20px #0006;transition:transform .2s,background .2s;position:fixed}.debug-fab:hover{transform:scale(1.1);background:#334155}.debug-fab.recording{background:#dc2626}.debug-fab.pulse{animation:fabPulse 1.5s ease-in-out infinite}@keyframes fabPulse{0%,to{box-shadow:0 4px 20px #dc262666}50%{box-shadow:0 4px 30px #dc2626e6,0 0 0 8px #dc262626}}.fab-badge{position:absolute;top:-4px;right:-4px;background:#f59e0b;color:#000;font-size:10px;font-weight:700;min-width:18px;height:18px;border-radius:9px;display:flex;align-items:center;justify-content:center;padding:0 4px}.debug-panel{position:fixed;z-index:99998;width:420px;max-height:80vh;background:#0f172a;color:#e2e8f0;border-radius:12px;border:1px solid #1e3a5f;box-shadow:0 25px 60px #0009;display:flex;flex-direction:column;font-family:Segoe UI,system-ui,sans-serif;font-size:13px;overflow:hidden}.pos-bottom-right{bottom:88px;right:24px}.pos-bottom-left{bottom:88px;left:24px}.pos-top-right{top:24px;right:24px}.pos-top-left{top:24px;left:24px}.panel-header{display:flex;align-items:center;justify-content:space-between;padding:10px 14px;background:#1e293b;border-bottom:1px solid #334155;cursor:move;-webkit-user-select:none;user-select:none;flex-shrink:0}.header-left{display:flex;align-items:center;gap:8px}.panel-icon{font-size:16px}.panel-title{font-weight:600;font-size:14px;color:#f1f5f9}.rec-indicator{font-size:10px;font-weight:700;color:#ef4444;background:#ef444426;padding:2px 7px;border-radius:4px;animation:blink 1s step-end infinite}.rec-indicator.paused{color:#f59e0b;background:#f59e0b26;animation:none}@keyframes blink{50%{opacity:.4}}.header-actions{display:flex;gap:4px}.icon-btn{background:none;border:none;color:#94a3b8;cursor:pointer;padding:4px;border-radius:4px;font-size:14px;line-height:1;transition:color .15s,background .15s}.icon-btn:hover{color:#f1f5f9;background:#334155}.session-setup{padding:10px 14px;border-bottom:1px solid #1e293b;display:flex;flex-direction:column;gap:6px;flex-shrink:0}.session-input,.session-desc{background:#1e293b;border:1px solid #334155;color:#e2e8f0;border-radius:6px;padding:6px 10px;font-size:12px;width:100%;box-sizing:border-box;resize:vertical}.session-input:focus,.session-desc:focus{outline:none;border-color:#3b82f6}.recording-controls{display:flex;gap:6px;padding:10px 14px;border-bottom:1px solid #1e293b;flex-wrap:wrap;flex-shrink:0}.ctrl-btn{border:none;border-radius:6px;padding:6px 14px;font-size:12px;font-weight:600;cursor:pointer;transition:filter .15s,transform .1s;display:flex;align-items:center;gap:5px}.ctrl-btn:hover:not(:disabled){filter:brightness(1.15);transform:translateY(-1px)}.ctrl-btn:active:not(:disabled){transform:translateY(0)}.ctrl-btn:disabled{opacity:.5;cursor:not-allowed}.ctrl-btn.start{background:#16a34a;color:#fff}.ctrl-btn.stop{background:#dc2626;color:#fff}.ctrl-btn.pause{background:#d97706;color:#fff}.ctrl-btn.generate{background:#7c3aed;color:#fff;flex:1;justify-content:center}.ctrl-btn.clear{background:#374151;color:#9ca3af;padding:6px 10px}.spinner{display:inline-block;animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.error-banner{background:#dc262626;border-left:3px solid #dc2626;color:#fca5a5;padding:8px 14px;font-size:12px;flex-shrink:0}.tab-bar{display:flex;border-bottom:1px solid #1e293b;flex-shrink:0}.tab-btn{flex:1;background:none;border:none;color:#64748b;padding:8px 4px;font-size:12px;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:5px;border-bottom:2px solid transparent;transition:color .15s}.tab-btn:hover:not(:disabled){color:#cbd5e1}.tab-btn.active{color:#60a5fa;border-bottom-color:#3b82f6}.tab-btn:disabled{opacity:.4;cursor:not-allowed}.tab-badge{background:#334155;color:#94a3b8;font-size:10px;padding:1px 5px;border-radius:3px}.tab-badge.new{background:#7c3aed;color:#e9d5ff}.tab-content{overflow-y:auto;flex:1;min-height:0}.tab-content::-webkit-scrollbar{width:5px}.tab-content::-webkit-scrollbar-track{background:transparent}.tab-content::-webkit-scrollbar-thumb{background:#334155;border-radius:3px}.sessions-list{padding:10px 14px;display:flex;flex-direction:column;gap:8px}.session-card{background:#1e293b;border:1px solid #334155;border-radius:8px;padding:10px}.session-card-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:8px}.session-name{font-weight:600;color:#f1f5f9;font-size:13px}.session-meta{font-size:11px;color:#64748b}.session-card-actions{display:flex;gap:6px}.sm-btn{background:#334155;border:none;color:#cbd5e1;padding:4px 10px;border-radius:5px;font-size:11px;cursor:pointer;transition:background .15s}.sm-btn:hover{background:#475569}.sm-btn.danger{color:#fca5a5}.sm-btn.danger:hover{background:#dc262633}.resize-handle{text-align:center;color:#334155;cursor:ns-resize;font-size:16px;padding:2px;flex-shrink:0;background:#0f172a;-webkit-user-select:none;user-select:none}.resize-handle:hover{color:#64748b}\n"] }]
515
+ }] });
516
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"debug-panel.component.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/debug-panel/debug-panel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, signal, computed, inject, OnInit, OnDestroy,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { RecorderService } from '../services/recorder.service';\nimport { AiGeneratorService } from '../services/ai-generator.service';\nimport { RrwebRecorderService } from '../services/rrweb-recorder.service';\nimport { RecordingSession } from '../models/recorded-action.model';\nimport { ActionListComponent } from '../action-list/action-list.component';\nimport { TestPreviewComponent } from '../test-preview/test-preview.component';\nimport { SettingsDialogComponent } from '../settings-dialog/settings-dialog.component';\nimport { SessionReplayComponent } from '../session-replay/session-replay.component';\n\ntype PanelTab = 'actions' | 'test' | 'sessions' | 'replay';\ntype PanelPosition = 'bottom-right' | 'bottom-left' | 'top-right' | 'top-left';\n\n@Component({\n  selector: 'app-debug-panel',\n  standalone: true,\n  imports: [CommonModule, FormsModule, ActionListComponent, TestPreviewComponent, SettingsDialogComponent, SessionReplayComponent],\n  template: `\n    <!-- Toggle FAB -->\n    <button\n      data-debug-panel\n      class=\"debug-fab\"\n      [class.recording]=\"recorder.isRecording()\"\n      [class.pulse]=\"recorder.isRecording() && !recorder.isPaused()\"\n      (click)=\"togglePanel()\"\n      [title]=\"panelOpen() ? 'Debug Panel schließen' : 'Debug Panel öffnen'\"\n    >\n      <span class=\"fab-icon\">{{ recorder.isRecording() ? '⏺' : '🐛' }}</span>\n      @if (recorder.isRecording() && recorder.actionCount() > 0) {\n        <span class=\"fab-badge\">{{ recorder.actionCount() }}</span>\n      }\n    </button>\n\n    <!-- Main Panel -->\n    @if (panelOpen()) {\n      <div\n        data-debug-panel\n        class=\"debug-panel\"\n        [class]=\"'pos-' + position()\"\n        [style.width.px]=\"panelWidth()\"\n      >\n        <!-- Header -->\n        <div class=\"panel-header\" (mousedown)=\"startDrag($event)\">\n          <div class=\"header-left\">\n            <span class=\"panel-icon\">🐛</span>\n            <span class=\"panel-title\">Debug Recorder</span>\n            @if (recorder.isRecording()) {\n              <span class=\"rec-indicator\" [class.paused]=\"recorder.isPaused()\">\n                {{ recorder.isPaused() ? '⏸ PAUSED' : '⏺ REC' }}\n              </span>\n            }\n          </div>\n          <div class=\"header-actions\">\n            <button class=\"icon-btn\" title=\"Einstellungen\" (click)=\"showSettings.set(true)\">⚙️</button>\n            <button class=\"icon-btn\" title=\"Position wechseln\" (click)=\"cyclePosition()\">📌</button>\n            <button class=\"icon-btn\" title=\"Schließen\" (click)=\"togglePanel()\">✕</button>\n          </div>\n        </div>\n\n        <!-- Session Name Input (only when not recording) -->\n        @if (!recorder.isRecording()) {\n          <div class=\"session-setup\">\n            <input\n              data-debug-panel\n              class=\"session-input\"\n              type=\"text\"\n              [(ngModel)]=\"sessionName\"\n              placeholder=\"Session-Name (optional)\"\n            />\n            <textarea\n              data-debug-panel\n              class=\"session-desc\"\n              [(ngModel)]=\"sessionDesc\"\n              placeholder=\"Fehlerbeschreibung...\"\n              rows=\"2\"\n            ></textarea>\n          </div>\n        }\n\n        <!-- Recording Controls -->\n        <div class=\"recording-controls\" data-debug-panel>\n          @if (!recorder.isRecording()) {\n            <button class=\"ctrl-btn start\" (click)=\"startRecording()\">\n              ▶ Aufnahme starten\n            </button>\n          } @else {\n            <button class=\"ctrl-btn pause\" (click)=\"recorder.pauseRecording()\">\n              {{ recorder.isPaused() ? '▶ Fortsetzen' : '⏸ Pause' }}\n            </button>\n            <button class=\"ctrl-btn stop\" (click)=\"stopRecording()\">\n              ⏹ Stoppen\n            </button>\n            <button class=\"ctrl-btn clear\" title=\"Aktionen löschen\" (click)=\"recorder.clearCurrentSession()\">\n              🗑\n            </button>\n          }\n\n          @if (hasActions()) {\n            <button\n              class=\"ctrl-btn generate\"\n              [disabled]=\"aiService.isGenerating() || !aiService.webhookUrl()\"\n              [title]=\"!aiService.webhookUrl() ? 'Webhook-URL in Einstellungen eintragen' : 'Cypress Test generieren'\"\n              (click)=\"generateTest()\"\n            >\n              @if (aiService.isGenerating()) {\n                <span class=\"spinner\">⟳</span> Generiere...\n              } @else {\n                🤖 → Cypress Test\n              }\n            </button>\n          }\n        </div>\n\n        <!-- Error Banner -->\n        @if (aiService.error()) {\n          <div class=\"error-banner\" data-debug-panel>\n            ⚠️ {{ aiService.error() }}\n          </div>\n        }\n\n        <!-- Tabs -->\n        <div class=\"tab-bar\" data-debug-panel>\n          <button\n            class=\"tab-btn\"\n            [class.active]=\"activeTab() === 'actions'\"\n            (click)=\"activeTab.set('actions')\"\n          >\n            Aktionen\n            @if (recorder.actionCount() > 0) {\n              <span class=\"tab-badge\">{{ recorder.actionCount() }}</span>\n            }\n          </button>\n          <button\n            class=\"tab-btn\"\n            [class.active]=\"activeTab() === 'replay'\"\n            (click)=\"activeTab.set('replay')\"\n            title=\"rrweb Session Replay\"\n          >\n            📽️ Replay\n            @if (rrweb.events().length > 0) {\n              <span class=\"tab-badge\">{{ rrweb.events().length }}</span>\n            }\n          </button>\n          <button\n            class=\"tab-btn\"\n            [class.active]=\"activeTab() === 'test'\"\n            [disabled]=\"!aiService.lastTest()\"\n            (click)=\"activeTab.set('test')\"\n          >\n            🤖 Test\n            @if (aiService.lastTest()) {\n              <span class=\"tab-badge new\">NEU</span>\n            }\n          </button>\n          <button\n            class=\"tab-btn\"\n            [class.active]=\"activeTab() === 'sessions'\"\n            [disabled]=\"recorder.sessions().length === 0\"\n            (click)=\"activeTab.set('sessions')\"\n          >\n            Sessions ({{ recorder.sessions().length }})\n          </button>\n        </div>\n\n        <!-- Tab Content -->\n        <div class=\"tab-content\">\n          @if (activeTab() === 'actions') {\n            <app-action-list\n              [session]=\"recorder.currentSession()\"\n              (removeAction)=\"recorder.removeAction($event)\"\n              (addNote)=\"recorder.addNote($event.id, $event.note)\"\n            />\n          }\n\n          @if (activeTab() === 'replay') {\n            <app-session-replay />\n          }\n\n          @if (activeTab() === 'test') {\n            <app-test-preview [test]=\"aiService.lastTest()\" />\n          }\n\n          @if (activeTab() === 'sessions') {\n            <div class=\"sessions-list\">\n              @for (session of recorder.sessions(); track session.id) {\n                <div class=\"session-card\">\n                  <div class=\"session-card-header\">\n                    <span class=\"session-name\">{{ session.name }}</span>\n                    <span class=\"session-meta\">{{ session.actions.length }} Aktionen</span>\n                  </div>\n                  <div class=\"session-card-actions\">\n                    <button class=\"sm-btn\" (click)=\"loadAndGenerate(session)\">🤖 Neu generieren</button>\n                    <button class=\"sm-btn danger\" (click)=\"recorder.deleteSession(session.id)\">🗑</button>\n                  </div>\n                </div>\n              }\n            </div>\n          }\n        </div>\n\n        <!-- Resize Handle -->\n        <div class=\"resize-handle\" (mousedown)=\"startResize($event)\">⠿</div>\n      </div>\n    }\n\n    <!-- Settings Dialog -->\n    @if (showSettings()) {\n      <app-settings-dialog (close)=\"showSettings.set(false)\" />\n    }\n  `,\n  styles: [`\n    .debug-fab {\n      position: fixed;\n      bottom: 24px;\n      right: 24px;\n      z-index: 99999;\n      width: 52px;\n      height: 52px;\n      border-radius: 50%;\n      border: none;\n      background: #1e293b;\n      color: white;\n      font-size: 22px;\n      cursor: pointer;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      box-shadow: 0 4px 20px rgba(0,0,0,0.4);\n      transition: transform 0.2s, background 0.2s;\n      position: fixed;\n    }\n    .debug-fab:hover { transform: scale(1.1); background: #334155; }\n    .debug-fab.recording { background: #dc2626; }\n    .debug-fab.pulse { animation: fabPulse 1.5s ease-in-out infinite; }\n    @keyframes fabPulse {\n      0%, 100% { box-shadow: 0 4px 20px rgba(220,38,38,0.4); }\n      50% { box-shadow: 0 4px 30px rgba(220,38,38,0.9), 0 0 0 8px rgba(220,38,38,0.15); }\n    }\n    .fab-badge {\n      position: absolute;\n      top: -4px;\n      right: -4px;\n      background: #f59e0b;\n      color: #000;\n      font-size: 10px;\n      font-weight: 700;\n      min-width: 18px;\n      height: 18px;\n      border-radius: 9px;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      padding: 0 4px;\n    }\n\n    .debug-panel {\n      position: fixed;\n      z-index: 99998;\n      width: 420px;\n      max-height: 80vh;\n      background: #0f172a;\n      color: #e2e8f0;\n      border-radius: 12px;\n      border: 1px solid #1e3a5f;\n      box-shadow: 0 25px 60px rgba(0,0,0,0.6);\n      display: flex;\n      flex-direction: column;\n      font-family: 'Segoe UI', system-ui, sans-serif;\n      font-size: 13px;\n      overflow: hidden;\n    }\n    .pos-bottom-right { bottom: 88px; right: 24px; }\n    .pos-bottom-left  { bottom: 88px; left: 24px; }\n    .pos-top-right    { top: 24px; right: 24px; }\n    .pos-top-left     { top: 24px; left: 24px; }\n\n    .panel-header {\n      display: flex;\n      align-items: center;\n      justify-content: space-between;\n      padding: 10px 14px;\n      background: #1e293b;\n      border-bottom: 1px solid #334155;\n      cursor: move;\n      user-select: none;\n      flex-shrink: 0;\n    }\n    .header-left { display: flex; align-items: center; gap: 8px; }\n    .panel-icon { font-size: 16px; }\n    .panel-title { font-weight: 600; font-size: 14px; color: #f1f5f9; }\n    .rec-indicator {\n      font-size: 10px;\n      font-weight: 700;\n      color: #ef4444;\n      background: rgba(239,68,68,0.15);\n      padding: 2px 7px;\n      border-radius: 4px;\n      animation: blink 1s step-end infinite;\n    }\n    .rec-indicator.paused { color: #f59e0b; background: rgba(245,158,11,0.15); animation: none; }\n    @keyframes blink { 50% { opacity: 0.4; } }\n    .header-actions { display: flex; gap: 4px; }\n    .icon-btn {\n      background: none;\n      border: none;\n      color: #94a3b8;\n      cursor: pointer;\n      padding: 4px;\n      border-radius: 4px;\n      font-size: 14px;\n      line-height: 1;\n      transition: color 0.15s, background 0.15s;\n    }\n    .icon-btn:hover { color: #f1f5f9; background: #334155; }\n\n    .session-setup {\n      padding: 10px 14px;\n      border-bottom: 1px solid #1e293b;\n      display: flex;\n      flex-direction: column;\n      gap: 6px;\n      flex-shrink: 0;\n    }\n    .session-input, .session-desc {\n      background: #1e293b;\n      border: 1px solid #334155;\n      color: #e2e8f0;\n      border-radius: 6px;\n      padding: 6px 10px;\n      font-size: 12px;\n      width: 100%;\n      box-sizing: border-box;\n      resize: vertical;\n    }\n    .session-input:focus, .session-desc:focus {\n      outline: none;\n      border-color: #3b82f6;\n    }\n\n    .recording-controls {\n      display: flex;\n      gap: 6px;\n      padding: 10px 14px;\n      border-bottom: 1px solid #1e293b;\n      flex-wrap: wrap;\n      flex-shrink: 0;\n    }\n    .ctrl-btn {\n      border: none;\n      border-radius: 6px;\n      padding: 6px 14px;\n      font-size: 12px;\n      font-weight: 600;\n      cursor: pointer;\n      transition: filter 0.15s, transform 0.1s;\n      display: flex;\n      align-items: center;\n      gap: 5px;\n    }\n    .ctrl-btn:hover:not(:disabled) { filter: brightness(1.15); transform: translateY(-1px); }\n    .ctrl-btn:active:not(:disabled) { transform: translateY(0); }\n    .ctrl-btn:disabled { opacity: 0.5; cursor: not-allowed; }\n    .ctrl-btn.start    { background: #16a34a; color: #fff; }\n    .ctrl-btn.stop     { background: #dc2626; color: #fff; }\n    .ctrl-btn.pause    { background: #d97706; color: #fff; }\n    .ctrl-btn.generate { background: #7c3aed; color: #fff; flex: 1; justify-content: center; }\n    .ctrl-btn.clear    { background: #374151; color: #9ca3af; padding: 6px 10px; }\n    .spinner { display: inline-block; animation: spin 0.8s linear infinite; }\n    @keyframes spin { to { transform: rotate(360deg); } }\n\n    .error-banner {\n      background: rgba(220,38,38,0.15);\n      border-left: 3px solid #dc2626;\n      color: #fca5a5;\n      padding: 8px 14px;\n      font-size: 12px;\n      flex-shrink: 0;\n    }\n\n    .tab-bar {\n      display: flex;\n      border-bottom: 1px solid #1e293b;\n      flex-shrink: 0;\n    }\n    .tab-btn {\n      flex: 1;\n      background: none;\n      border: none;\n      color: #64748b;\n      padding: 8px 4px;\n      font-size: 12px;\n      cursor: pointer;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      gap: 5px;\n      border-bottom: 2px solid transparent;\n      transition: color 0.15s;\n    }\n    .tab-btn:hover:not(:disabled) { color: #cbd5e1; }\n    .tab-btn.active { color: #60a5fa; border-bottom-color: #3b82f6; }\n    .tab-btn:disabled { opacity: 0.4; cursor: not-allowed; }\n    .tab-badge {\n      background: #334155;\n      color: #94a3b8;\n      font-size: 10px;\n      padding: 1px 5px;\n      border-radius: 3px;\n    }\n    .tab-badge.new { background: #7c3aed; color: #e9d5ff; }\n\n    .tab-content {\n      overflow-y: auto;\n      flex: 1;\n      min-height: 0;\n    }\n    .tab-content::-webkit-scrollbar { width: 5px; }\n    .tab-content::-webkit-scrollbar-track { background: transparent; }\n    .tab-content::-webkit-scrollbar-thumb { background: #334155; border-radius: 3px; }\n\n    .sessions-list { padding: 10px 14px; display: flex; flex-direction: column; gap: 8px; }\n    .session-card {\n      background: #1e293b;\n      border: 1px solid #334155;\n      border-radius: 8px;\n      padding: 10px;\n    }\n    .session-card-header {\n      display: flex;\n      justify-content: space-between;\n      align-items: center;\n      margin-bottom: 8px;\n    }\n    .session-name { font-weight: 600; color: #f1f5f9; font-size: 13px; }\n    .session-meta { font-size: 11px; color: #64748b; }\n    .session-card-actions { display: flex; gap: 6px; }\n    .sm-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      transition: background 0.15s;\n    }\n    .sm-btn:hover { background: #475569; }\n    .sm-btn.danger { color: #fca5a5; }\n    .sm-btn.danger:hover { background: rgba(220,38,38,0.2); }\n\n    .resize-handle {\n      text-align: center;\n      color: #334155;\n      cursor: ns-resize;\n      font-size: 16px;\n      padding: 2px;\n      flex-shrink: 0;\n      background: #0f172a;\n      user-select: none;\n    }\n    .resize-handle:hover { color: #64748b; }\n  `],\n})\nexport class DebugPanelComponent implements OnInit, OnDestroy {\n  recorder = inject(RecorderService);\n  aiService = inject(AiGeneratorService);\n  rrweb = inject(RrwebRecorderService);\n\n  panelOpen = signal(false);\n  activeTab = signal<PanelTab>('actions');\n  position = signal<PanelPosition>('bottom-right');\n  panelWidth = signal(420);\n  showSettings = signal(false);\n  sessionName = '';\n  sessionDesc = '';\n\n  hasActions = computed(\n    () => (this.recorder.currentSession()?.actions.length ?? 0) > 0 ||\n          (this.recorder.sessions().length > 0)\n  );\n\n  private resizing = false;\n  private resizeStartY = 0;\n  private resizeStartH = 0;\n\n  ngOnInit(): void {\n    // Keyboard shortcut: Ctrl+Shift+D\n    document.addEventListener('keydown', this.handleHotkey);\n  }\n\n  ngOnDestroy(): void {\n    document.removeEventListener('keydown', this.handleHotkey);\n  }\n\n  private handleHotkey = (e: KeyboardEvent) => {\n    if (e.ctrlKey && e.shiftKey && e.key === 'D') {\n      e.preventDefault();\n      this.togglePanel();\n    }\n    if (e.ctrlKey && e.shiftKey && e.key === 'R') {\n      e.preventDefault();\n      if (this.recorder.isRecording()) {\n        this.stopRecording();\n      } else {\n        this.startRecording();\n      }\n    }\n  };\n\n  togglePanel(): void {\n    this.panelOpen.update(v => !v);\n    if (this.panelOpen() && !this.recorder.currentSession()) {\n      this.activeTab.set('actions');\n    }\n  }\n\n  cyclePosition(): void {\n    const positions: PanelPosition[] = ['bottom-right', 'bottom-left', 'top-right', 'top-left'];\n    const idx = positions.indexOf(this.position());\n    this.position.set(positions[(idx + 1) % positions.length]);\n  }\n\n  startRecording(): void {\n    this.recorder.startRecording(\n      this.sessionName || undefined,\n      this.sessionDesc || undefined\n    );\n    // Start rrweb in parallel for visual replay\n    this.rrweb.startRecording();\n    this.activeTab.set('actions');\n  }\n\n  stopRecording(): void {\n    const session = this.recorder.stopRecording();\n    this.rrweb.stopRecording();\n    if (session) {\n      this.activeTab.set('actions');\n    }\n  }\n\n  async generateTest(): Promise<void> {\n    const session = this.recorder.currentSession() ?? this.recorder.sessions().at(-1);\n    if (!session) return;\n    await this.aiService.generateCypressTest(session);\n    this.activeTab.set('test');\n  }\n\n  async loadAndGenerate(session: RecordingSession): Promise<void> {\n    await this.aiService.generateCypressTest(session);\n    this.activeTab.set('test');\n  }\n\n  // Drag to reposition\n  startDrag(e: MouseEvent): void {\n    if ((e.target as HTMLElement).closest('button')) return;\n    const panel = (e.currentTarget as HTMLElement).parentElement!;\n    const startX = e.clientX - panel.getBoundingClientRect().left;\n    const startY = e.clientY - panel.getBoundingClientRect().top;\n\n    const onMove = (ev: MouseEvent) => {\n      panel.style.left = `${ev.clientX - startX}px`;\n      panel.style.top  = `${ev.clientY - startY}px`;\n      panel.style.right = 'auto';\n      panel.style.bottom = 'auto';\n    };\n    const onUp = () => {\n      document.removeEventListener('mousemove', onMove);\n      document.removeEventListener('mouseup', onUp);\n    };\n    document.addEventListener('mousemove', onMove);\n    document.addEventListener('mouseup', onUp);\n  }\n\n  // Resize height\n  startResize(e: MouseEvent): void {\n    const panel = (e.currentTarget as HTMLElement).parentElement!;\n    this.resizeStartY = e.clientY;\n    this.resizeStartH = panel.getBoundingClientRect().height;\n\n    const onMove = (ev: MouseEvent) => {\n      const newH = Math.max(250, this.resizeStartH + (ev.clientY - this.resizeStartY));\n      panel.style.maxHeight = `${newH}px`;\n    };\n    const onUp = () => {\n      document.removeEventListener('mousemove', onMove);\n      document.removeEventListener('mouseup', onUp);\n    };\n    document.addEventListener('mousemove', onMove);\n    document.addEventListener('mouseup', onUp);\n    e.preventDefault();\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb3JkZWQtYWN0aW9uLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZGVidWctcmVjb3JkZXIvc3JjL2xpYi9tb2RlbHMvcmVjb3JkZWQtYWN0aW9uLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgdHlwZSBBY3Rpb25UeXBlID1cclxuICB8ICdjbGljaydcclxuICB8ICdkYmxjbGljaydcclxuICB8ICdpbnB1dCdcclxuICB8ICdzZWxlY3QnXHJcbiAgfCAnc3VibWl0J1xyXG4gIHwgJ25hdmlnYXRpb24nXHJcbiAgfCAna2V5cHJlc3MnXHJcbiAgfCAnc2Nyb2xsJ1xyXG4gIHwgJ2hvdmVyJ1xyXG4gIHwgJ2Fzc2VydGlvbidcclxuICB8ICdzY3JlZW5zaG90JztcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgRWxlbWVudEluZm8ge1xyXG4gIHRhZ05hbWU6IHN0cmluZztcclxuICBpZD86IHN0cmluZztcclxuICBjbGFzc2VzOiBzdHJpbmdbXTtcclxuICBkYXRhVGVzdElkPzogc3RyaW5nO1xyXG4gIGRhdGFDeT86IHN0cmluZztcclxuICBuYW1lPzogc3RyaW5nO1xyXG4gIHR5cGU/OiBzdHJpbmc7XHJcbiAgcGxhY2Vob2xkZXI/OiBzdHJpbmc7XHJcbiAgdGV4dD86IHN0cmluZztcclxuICBocmVmPzogc3RyaW5nO1xyXG4gIGFyaWFMYWJlbD86IHN0cmluZztcclxufVxyXG5cclxuZXhwb3J0IGludGVyZmFjZSBSZWNvcmRlZEFjdGlvbiB7XHJcbiAgaWQ6IHN0cmluZztcclxuICB0eXBlOiBBY3Rpb25UeXBlO1xyXG4gIHRpbWVzdGFtcDogbnVtYmVyO1xyXG4gIHNlbGVjdG9yOiBzdHJpbmc7XHJcbiAgc2VsZWN0b3JTdHJhdGVneTogJ2RhdGEtdGVzdGlkJyB8ICdkYXRhLWN5JyB8ICdpZCcgfCAnbmFtZScgfCAnY2xhc3MnIHwgJ3RleHQnIHwgJ250aCcgfCAnY29tYmluZWQnO1xyXG4gIHZhbHVlPzogc3RyaW5nO1xyXG4gIHVybDogc3RyaW5nO1xyXG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XHJcbiAgZWxlbWVudD86IEVsZW1lbnRJbmZvO1xyXG4gIHNjcmVlbnNob3REYXRhVXJsPzogc3RyaW5nO1xyXG4gIG5vdGU/OiBzdHJpbmc7XHJcbn1cclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgUmVjb3JkaW5nU2Vzc2lvbiB7XHJcbiAgaWQ6IHN0cmluZztcclxuICBuYW1lOiBzdHJpbmc7XHJcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XHJcbiAgc3RhcnRUaW1lOiBudW1iZXI7XHJcbiAgZW5kVGltZT86IG51bWJlcjtcclxuICBzdGFydFVybDogc3RyaW5nO1xyXG4gIGFjdGlvbnM6IFJlY29yZGVkQWN0aW9uW107XHJcbiAgdGFncz86IHN0cmluZ1tdO1xyXG59XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIEdlbmVyYXRlZFRlc3Qge1xyXG4gIGNvZGU6IHN0cmluZztcclxuICBnZW5lcmF0ZWRBdDogbnVtYmVyO1xyXG4gIG1vZGVsOiBzdHJpbmc7XHJcbiAgc2Vzc2lvbklkOiBzdHJpbmc7XHJcbiAgcHJvbXB0VG9rZW5zPzogbnVtYmVyO1xyXG4gIGNvbXBsZXRpb25Ub2tlbnM/OiBudW1iZXI7XHJcbn1cclxuIl19