angular-debug-recorder 1.0.0 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -123,394 +123,394 @@ export class DebugPanelComponent {
123
123
  e.preventDefault();
124
124
  }
125
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
- }
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.eventCount() > 0) {
249
+ <span class="tab-badge">{{ rrweb.eventCount() }}</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
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
319
  }
320
320
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DebugPanelComponent, decorators: [{
321
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
- }
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.eventCount() > 0) {
445
+ <span class="tab-badge">{{ rrweb.eventCount() }}</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
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
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"]}
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 {\r\n  Component, signal, computed, inject, OnInit, OnDestroy,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { RecorderService } from '../services/recorder.service';\r\nimport { AiGeneratorService } from '../services/ai-generator.service';\r\nimport { RrwebRecorderService } from '../services/rrweb-recorder.service';\r\nimport { RecordingSession } from '../models/recorded-action.model';\r\nimport { ActionListComponent } from '../action-list/action-list.component';\r\nimport { TestPreviewComponent } from '../test-preview/test-preview.component';\r\nimport { SettingsDialogComponent } from '../settings-dialog/settings-dialog.component';\r\nimport { SessionReplayComponent } from '../session-replay/session-replay.component';\r\n\r\ntype PanelTab = 'actions' | 'test' | 'sessions' | 'replay';\r\ntype PanelPosition = 'bottom-right' | 'bottom-left' | 'top-right' | 'top-left';\r\n\r\n@Component({\r\n  selector: 'app-debug-panel',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule, ActionListComponent, TestPreviewComponent, SettingsDialogComponent, SessionReplayComponent],\r\n  template: `\r\n    <!-- Toggle FAB -->\r\n    <button\r\n      data-debug-panel\r\n      class=\"debug-fab\"\r\n      [class.recording]=\"recorder.isRecording()\"\r\n      [class.pulse]=\"recorder.isRecording() && !recorder.isPaused()\"\r\n      (click)=\"togglePanel()\"\r\n      [title]=\"panelOpen() ? 'Debug Panel schließen' : 'Debug Panel öffnen'\"\r\n    >\r\n      <span class=\"fab-icon\">{{ recorder.isRecording() ? '⏺' : '🐛' }}</span>\r\n      @if (recorder.isRecording() && recorder.actionCount() > 0) {\r\n        <span class=\"fab-badge\">{{ recorder.actionCount() }}</span>\r\n      }\r\n    </button>\r\n\r\n    <!-- Main Panel -->\r\n    @if (panelOpen()) {\r\n      <div\r\n        data-debug-panel\r\n        class=\"debug-panel\"\r\n        [class]=\"'pos-' + position()\"\r\n        [style.width.px]=\"panelWidth()\"\r\n      >\r\n        <!-- Header -->\r\n        <div class=\"panel-header\" (mousedown)=\"startDrag($event)\">\r\n          <div class=\"header-left\">\r\n            <span class=\"panel-icon\">🐛</span>\r\n            <span class=\"panel-title\">Debug Recorder</span>\r\n            @if (recorder.isRecording()) {\r\n              <span class=\"rec-indicator\" [class.paused]=\"recorder.isPaused()\">\r\n                {{ recorder.isPaused() ? '⏸ PAUSED' : '⏺ REC' }}\r\n              </span>\r\n            }\r\n          </div>\r\n          <div class=\"header-actions\">\r\n            <button class=\"icon-btn\" title=\"Einstellungen\" (click)=\"showSettings.set(true)\">⚙️</button>\r\n            <button class=\"icon-btn\" title=\"Position wechseln\" (click)=\"cyclePosition()\">📌</button>\r\n            <button class=\"icon-btn\" title=\"Schließen\" (click)=\"togglePanel()\">✕</button>\r\n          </div>\r\n        </div>\r\n\r\n        <!-- Session Name Input (only when not recording) -->\r\n        @if (!recorder.isRecording()) {\r\n          <div class=\"session-setup\">\r\n            <input\r\n              data-debug-panel\r\n              class=\"session-input\"\r\n              type=\"text\"\r\n              [(ngModel)]=\"sessionName\"\r\n              placeholder=\"Session-Name (optional)\"\r\n            />\r\n            <textarea\r\n              data-debug-panel\r\n              class=\"session-desc\"\r\n              [(ngModel)]=\"sessionDesc\"\r\n              placeholder=\"Fehlerbeschreibung...\"\r\n              rows=\"2\"\r\n            ></textarea>\r\n          </div>\r\n        }\r\n\r\n        <!-- Recording Controls -->\r\n        <div class=\"recording-controls\" data-debug-panel>\r\n          @if (!recorder.isRecording()) {\r\n            <button class=\"ctrl-btn start\" (click)=\"startRecording()\">\r\n              ▶ Aufnahme starten\r\n            </button>\r\n          } @else {\r\n            <button class=\"ctrl-btn pause\" (click)=\"recorder.pauseRecording()\">\r\n              {{ recorder.isPaused() ? '▶ Fortsetzen' : '⏸ Pause' }}\r\n            </button>\r\n            <button class=\"ctrl-btn stop\" (click)=\"stopRecording()\">\r\n              ⏹ Stoppen\r\n            </button>\r\n            <button class=\"ctrl-btn clear\" title=\"Aktionen löschen\" (click)=\"recorder.clearCurrentSession()\">\r\n              🗑\r\n            </button>\r\n          }\r\n\r\n          @if (hasActions()) {\r\n            <button\r\n              class=\"ctrl-btn generate\"\r\n              [disabled]=\"aiService.isGenerating() || !aiService.webhookUrl()\"\r\n              [title]=\"!aiService.webhookUrl() ? 'Webhook-URL in Einstellungen eintragen' : 'Cypress Test generieren'\"\r\n              (click)=\"generateTest()\"\r\n            >\r\n              @if (aiService.isGenerating()) {\r\n                <span class=\"spinner\">⟳</span> Generiere...\r\n              } @else {\r\n                🤖 → Cypress Test\r\n              }\r\n            </button>\r\n          }\r\n        </div>\r\n\r\n        <!-- Error Banner -->\r\n        @if (aiService.error()) {\r\n          <div class=\"error-banner\" data-debug-panel>\r\n            ⚠️ {{ aiService.error() }}\r\n          </div>\r\n        }\r\n\r\n        <!-- Tabs -->\r\n        <div class=\"tab-bar\" data-debug-panel>\r\n          <button\r\n            class=\"tab-btn\"\r\n            [class.active]=\"activeTab() === 'actions'\"\r\n            (click)=\"activeTab.set('actions')\"\r\n          >\r\n            Aktionen\r\n            @if (recorder.actionCount() > 0) {\r\n              <span class=\"tab-badge\">{{ recorder.actionCount() }}</span>\r\n            }\r\n          </button>\r\n          <button\r\n            class=\"tab-btn\"\r\n            [class.active]=\"activeTab() === 'replay'\"\r\n            (click)=\"activeTab.set('replay')\"\r\n            title=\"rrweb Session Replay\"\r\n          >\r\n            📽️ Replay\r\n            @if (rrweb.eventCount() > 0) {\r\n              <span class=\"tab-badge\">{{ rrweb.eventCount() }}</span>\r\n            }\r\n          </button>\r\n          <button\r\n            class=\"tab-btn\"\r\n            [class.active]=\"activeTab() === 'test'\"\r\n            [disabled]=\"!aiService.lastTest()\"\r\n            (click)=\"activeTab.set('test')\"\r\n          >\r\n            🤖 Test\r\n            @if (aiService.lastTest()) {\r\n              <span class=\"tab-badge new\">NEU</span>\r\n            }\r\n          </button>\r\n          <button\r\n            class=\"tab-btn\"\r\n            [class.active]=\"activeTab() === 'sessions'\"\r\n            [disabled]=\"recorder.sessions().length === 0\"\r\n            (click)=\"activeTab.set('sessions')\"\r\n          >\r\n            Sessions ({{ recorder.sessions().length }})\r\n          </button>\r\n        </div>\r\n\r\n        <!-- Tab Content -->\r\n        <div class=\"tab-content\">\r\n          @if (activeTab() === 'actions') {\r\n            <app-action-list\r\n              [session]=\"recorder.currentSession()\"\r\n              (removeAction)=\"recorder.removeAction($event)\"\r\n              (addNote)=\"recorder.addNote($event.id, $event.note)\"\r\n            />\r\n          }\r\n\r\n          @if (activeTab() === 'replay') {\r\n            <app-session-replay />\r\n          }\r\n\r\n          @if (activeTab() === 'test') {\r\n            <app-test-preview [test]=\"aiService.lastTest()\" />\r\n          }\r\n\r\n          @if (activeTab() === 'sessions') {\r\n            <div class=\"sessions-list\">\r\n              @for (session of recorder.sessions(); track session.id) {\r\n                <div class=\"session-card\">\r\n                  <div class=\"session-card-header\">\r\n                    <span class=\"session-name\">{{ session.name }}</span>\r\n                    <span class=\"session-meta\">{{ session.actions.length }} Aktionen</span>\r\n                  </div>\r\n                  <div class=\"session-card-actions\">\r\n                    <button class=\"sm-btn\" (click)=\"loadAndGenerate(session)\">🤖 Neu generieren</button>\r\n                    <button class=\"sm-btn danger\" (click)=\"recorder.deleteSession(session.id)\">🗑</button>\r\n                  </div>\r\n                </div>\r\n              }\r\n            </div>\r\n          }\r\n        </div>\r\n\r\n        <!-- Resize Handle -->\r\n        <div class=\"resize-handle\" (mousedown)=\"startResize($event)\">⠿</div>\r\n      </div>\r\n    }\r\n\r\n    <!-- Settings Dialog -->\r\n    @if (showSettings()) {\r\n      <app-settings-dialog (close)=\"showSettings.set(false)\" />\r\n    }\r\n  `,\r\n  styles: [`\r\n    .debug-fab {\r\n      position: fixed;\r\n      bottom: 24px;\r\n      right: 24px;\r\n      z-index: 99999;\r\n      width: 52px;\r\n      height: 52px;\r\n      border-radius: 50%;\r\n      border: none;\r\n      background: #1e293b;\r\n      color: white;\r\n      font-size: 22px;\r\n      cursor: pointer;\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      box-shadow: 0 4px 20px rgba(0,0,0,0.4);\r\n      transition: transform 0.2s, background 0.2s;\r\n      position: fixed;\r\n    }\r\n    .debug-fab:hover { transform: scale(1.1); background: #334155; }\r\n    .debug-fab.recording { background: #dc2626; }\r\n    .debug-fab.pulse { animation: fabPulse 1.5s ease-in-out infinite; }\r\n    @keyframes fabPulse {\r\n      0%, 100% { box-shadow: 0 4px 20px rgba(220,38,38,0.4); }\r\n      50% { box-shadow: 0 4px 30px rgba(220,38,38,0.9), 0 0 0 8px rgba(220,38,38,0.15); }\r\n    }\r\n    .fab-badge {\r\n      position: absolute;\r\n      top: -4px;\r\n      right: -4px;\r\n      background: #f59e0b;\r\n      color: #000;\r\n      font-size: 10px;\r\n      font-weight: 700;\r\n      min-width: 18px;\r\n      height: 18px;\r\n      border-radius: 9px;\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      padding: 0 4px;\r\n    }\r\n\r\n    .debug-panel {\r\n      position: fixed;\r\n      z-index: 99998;\r\n      width: 420px;\r\n      max-height: 80vh;\r\n      background: #0f172a;\r\n      color: #e2e8f0;\r\n      border-radius: 12px;\r\n      border: 1px solid #1e3a5f;\r\n      box-shadow: 0 25px 60px rgba(0,0,0,0.6);\r\n      display: flex;\r\n      flex-direction: column;\r\n      font-family: 'Segoe UI', system-ui, sans-serif;\r\n      font-size: 13px;\r\n      overflow: hidden;\r\n    }\r\n    .pos-bottom-right { bottom: 88px; right: 24px; }\r\n    .pos-bottom-left  { bottom: 88px; left: 24px; }\r\n    .pos-top-right    { top: 24px; right: 24px; }\r\n    .pos-top-left     { top: 24px; left: 24px; }\r\n\r\n    .panel-header {\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: space-between;\r\n      padding: 10px 14px;\r\n      background: #1e293b;\r\n      border-bottom: 1px solid #334155;\r\n      cursor: move;\r\n      user-select: none;\r\n      flex-shrink: 0;\r\n    }\r\n    .header-left { display: flex; align-items: center; gap: 8px; }\r\n    .panel-icon { font-size: 16px; }\r\n    .panel-title { font-weight: 600; font-size: 14px; color: #f1f5f9; }\r\n    .rec-indicator {\r\n      font-size: 10px;\r\n      font-weight: 700;\r\n      color: #ef4444;\r\n      background: rgba(239,68,68,0.15);\r\n      padding: 2px 7px;\r\n      border-radius: 4px;\r\n      animation: blink 1s step-end infinite;\r\n    }\r\n    .rec-indicator.paused { color: #f59e0b; background: rgba(245,158,11,0.15); animation: none; }\r\n    @keyframes blink { 50% { opacity: 0.4; } }\r\n    .header-actions { display: flex; gap: 4px; }\r\n    .icon-btn {\r\n      background: none;\r\n      border: none;\r\n      color: #94a3b8;\r\n      cursor: pointer;\r\n      padding: 4px;\r\n      border-radius: 4px;\r\n      font-size: 14px;\r\n      line-height: 1;\r\n      transition: color 0.15s, background 0.15s;\r\n    }\r\n    .icon-btn:hover { color: #f1f5f9; background: #334155; }\r\n\r\n    .session-setup {\r\n      padding: 10px 14px;\r\n      border-bottom: 1px solid #1e293b;\r\n      display: flex;\r\n      flex-direction: column;\r\n      gap: 6px;\r\n      flex-shrink: 0;\r\n    }\r\n    .session-input, .session-desc {\r\n      background: #1e293b;\r\n      border: 1px solid #334155;\r\n      color: #e2e8f0;\r\n      border-radius: 6px;\r\n      padding: 6px 10px;\r\n      font-size: 12px;\r\n      width: 100%;\r\n      box-sizing: border-box;\r\n      resize: vertical;\r\n    }\r\n    .session-input:focus, .session-desc:focus {\r\n      outline: none;\r\n      border-color: #3b82f6;\r\n    }\r\n\r\n    .recording-controls {\r\n      display: flex;\r\n      gap: 6px;\r\n      padding: 10px 14px;\r\n      border-bottom: 1px solid #1e293b;\r\n      flex-wrap: wrap;\r\n      flex-shrink: 0;\r\n    }\r\n    .ctrl-btn {\r\n      border: none;\r\n      border-radius: 6px;\r\n      padding: 6px 14px;\r\n      font-size: 12px;\r\n      font-weight: 600;\r\n      cursor: pointer;\r\n      transition: filter 0.15s, transform 0.1s;\r\n      display: flex;\r\n      align-items: center;\r\n      gap: 5px;\r\n    }\r\n    .ctrl-btn:hover:not(:disabled) { filter: brightness(1.15); transform: translateY(-1px); }\r\n    .ctrl-btn:active:not(:disabled) { transform: translateY(0); }\r\n    .ctrl-btn:disabled { opacity: 0.5; cursor: not-allowed; }\r\n    .ctrl-btn.start    { background: #16a34a; color: #fff; }\r\n    .ctrl-btn.stop     { background: #dc2626; color: #fff; }\r\n    .ctrl-btn.pause    { background: #d97706; color: #fff; }\r\n    .ctrl-btn.generate { background: #7c3aed; color: #fff; flex: 1; justify-content: center; }\r\n    .ctrl-btn.clear    { background: #374151; color: #9ca3af; padding: 6px 10px; }\r\n    .spinner { display: inline-block; animation: spin 0.8s linear infinite; }\r\n    @keyframes spin { to { transform: rotate(360deg); } }\r\n\r\n    .error-banner {\r\n      background: rgba(220,38,38,0.15);\r\n      border-left: 3px solid #dc2626;\r\n      color: #fca5a5;\r\n      padding: 8px 14px;\r\n      font-size: 12px;\r\n      flex-shrink: 0;\r\n    }\r\n\r\n    .tab-bar {\r\n      display: flex;\r\n      border-bottom: 1px solid #1e293b;\r\n      flex-shrink: 0;\r\n    }\r\n    .tab-btn {\r\n      flex: 1;\r\n      background: none;\r\n      border: none;\r\n      color: #64748b;\r\n      padding: 8px 4px;\r\n      font-size: 12px;\r\n      cursor: pointer;\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      gap: 5px;\r\n      border-bottom: 2px solid transparent;\r\n      transition: color 0.15s;\r\n    }\r\n    .tab-btn:hover:not(:disabled) { color: #cbd5e1; }\r\n    .tab-btn.active { color: #60a5fa; border-bottom-color: #3b82f6; }\r\n    .tab-btn:disabled { opacity: 0.4; cursor: not-allowed; }\r\n    .tab-badge {\r\n      background: #334155;\r\n      color: #94a3b8;\r\n      font-size: 10px;\r\n      padding: 1px 5px;\r\n      border-radius: 3px;\r\n    }\r\n    .tab-badge.new { background: #7c3aed; color: #e9d5ff; }\r\n\r\n    .tab-content {\r\n      overflow-y: auto;\r\n      flex: 1;\r\n      min-height: 0;\r\n    }\r\n    .tab-content::-webkit-scrollbar { width: 5px; }\r\n    .tab-content::-webkit-scrollbar-track { background: transparent; }\r\n    .tab-content::-webkit-scrollbar-thumb { background: #334155; border-radius: 3px; }\r\n\r\n    .sessions-list { padding: 10px 14px; display: flex; flex-direction: column; gap: 8px; }\r\n    .session-card {\r\n      background: #1e293b;\r\n      border: 1px solid #334155;\r\n      border-radius: 8px;\r\n      padding: 10px;\r\n    }\r\n    .session-card-header {\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      margin-bottom: 8px;\r\n    }\r\n    .session-name { font-weight: 600; color: #f1f5f9; font-size: 13px; }\r\n    .session-meta { font-size: 11px; color: #64748b; }\r\n    .session-card-actions { display: flex; gap: 6px; }\r\n    .sm-btn {\r\n      background: #334155;\r\n      border: none;\r\n      color: #cbd5e1;\r\n      padding: 4px 10px;\r\n      border-radius: 5px;\r\n      font-size: 11px;\r\n      cursor: pointer;\r\n      transition: background 0.15s;\r\n    }\r\n    .sm-btn:hover { background: #475569; }\r\n    .sm-btn.danger { color: #fca5a5; }\r\n    .sm-btn.danger:hover { background: rgba(220,38,38,0.2); }\r\n\r\n    .resize-handle {\r\n      text-align: center;\r\n      color: #334155;\r\n      cursor: ns-resize;\r\n      font-size: 16px;\r\n      padding: 2px;\r\n      flex-shrink: 0;\r\n      background: #0f172a;\r\n      user-select: none;\r\n    }\r\n    .resize-handle:hover { color: #64748b; }\r\n  `],\r\n})\r\nexport class DebugPanelComponent implements OnInit, OnDestroy {\r\n  recorder = inject(RecorderService);\r\n  aiService = inject(AiGeneratorService);\r\n  rrweb = inject(RrwebRecorderService);\r\n\r\n  panelOpen = signal(false);\r\n  activeTab = signal<PanelTab>('actions');\r\n  position = signal<PanelPosition>('bottom-right');\r\n  panelWidth = signal(420);\r\n  showSettings = signal(false);\r\n  sessionName = '';\r\n  sessionDesc = '';\r\n\r\n  hasActions = computed(\r\n    () => (this.recorder.currentSession()?.actions.length ?? 0) > 0 ||\r\n          (this.recorder.sessions().length > 0)\r\n  );\r\n\r\n  private resizing = false;\r\n  private resizeStartY = 0;\r\n  private resizeStartH = 0;\r\n\r\n  ngOnInit(): void {\r\n    // Keyboard shortcut: Ctrl+Shift+D\r\n    document.addEventListener('keydown', this.handleHotkey);\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    document.removeEventListener('keydown', this.handleHotkey);\r\n  }\r\n\r\n  private handleHotkey = (e: KeyboardEvent) => {\r\n    if (e.ctrlKey && e.shiftKey && e.key === 'D') {\r\n      e.preventDefault();\r\n      this.togglePanel();\r\n    }\r\n    if (e.ctrlKey && e.shiftKey && e.key === 'R') {\r\n      e.preventDefault();\r\n      if (this.recorder.isRecording()) {\r\n        this.stopRecording();\r\n      } else {\r\n        this.startRecording();\r\n      }\r\n    }\r\n  };\r\n\r\n  togglePanel(): void {\r\n    this.panelOpen.update(v => !v);\r\n    if (this.panelOpen() && !this.recorder.currentSession()) {\r\n      this.activeTab.set('actions');\r\n    }\r\n  }\r\n\r\n  cyclePosition(): void {\r\n    const positions: PanelPosition[] = ['bottom-right', 'bottom-left', 'top-right', 'top-left'];\r\n    const idx = positions.indexOf(this.position());\r\n    this.position.set(positions[(idx + 1) % positions.length]);\r\n  }\r\n\r\n  startRecording(): void {\r\n    this.recorder.startRecording(\r\n      this.sessionName || undefined,\r\n      this.sessionDesc || undefined\r\n    );\r\n    // Start rrweb in parallel for visual replay\r\n    this.rrweb.startRecording();\r\n    this.activeTab.set('actions');\r\n  }\r\n\r\n  stopRecording(): void {\r\n    const session = this.recorder.stopRecording();\r\n    this.rrweb.stopRecording();\r\n    if (session) {\r\n      this.activeTab.set('actions');\r\n    }\r\n  }\r\n\r\n  async generateTest(): Promise<void> {\r\n    const session = this.recorder.currentSession() ?? this.recorder.sessions().at(-1);\r\n    if (!session) return;\r\n    await this.aiService.generateCypressTest(session);\r\n    this.activeTab.set('test');\r\n  }\r\n\r\n  async loadAndGenerate(session: RecordingSession): Promise<void> {\r\n    await this.aiService.generateCypressTest(session);\r\n    this.activeTab.set('test');\r\n  }\r\n\r\n  // Drag to reposition\r\n  startDrag(e: MouseEvent): void {\r\n    if ((e.target as HTMLElement).closest('button')) return;\r\n    const panel = (e.currentTarget as HTMLElement).parentElement!;\r\n    const startX = e.clientX - panel.getBoundingClientRect().left;\r\n    const startY = e.clientY - panel.getBoundingClientRect().top;\r\n\r\n    const onMove = (ev: MouseEvent) => {\r\n      panel.style.left = `${ev.clientX - startX}px`;\r\n      panel.style.top  = `${ev.clientY - startY}px`;\r\n      panel.style.right = 'auto';\r\n      panel.style.bottom = 'auto';\r\n    };\r\n    const onUp = () => {\r\n      document.removeEventListener('mousemove', onMove);\r\n      document.removeEventListener('mouseup', onUp);\r\n    };\r\n    document.addEventListener('mousemove', onMove);\r\n    document.addEventListener('mouseup', onUp);\r\n  }\r\n\r\n  // Resize height\r\n  startResize(e: MouseEvent): void {\r\n    const panel = (e.currentTarget as HTMLElement).parentElement!;\r\n    this.resizeStartY = e.clientY;\r\n    this.resizeStartH = panel.getBoundingClientRect().height;\r\n\r\n    const onMove = (ev: MouseEvent) => {\r\n      const newH = Math.max(250, this.resizeStartH + (ev.clientY - this.resizeStartY));\r\n      panel.style.maxHeight = `${newH}px`;\r\n    };\r\n    const onUp = () => {\r\n      document.removeEventListener('mousemove', onMove);\r\n      document.removeEventListener('mouseup', onUp);\r\n    };\r\n    document.addEventListener('mousemove', onMove);\r\n    document.addEventListener('mouseup', onUp);\r\n    e.preventDefault();\r\n  }\r\n}\r\n"]}