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.
- package/esm2022/lib/action-list/action-list.component.mjs +161 -161
- package/esm2022/lib/debug-panel/debug-panel.component.mjs +385 -385
- package/esm2022/lib/services/rrweb-recorder.service.mjs +32 -21
- package/esm2022/lib/session-replay/session-replay.component.mjs +85 -85
- package/esm2022/lib/settings-dialog/settings-dialog.component.mjs +75 -75
- package/esm2022/lib/test-preview/test-preview.component.mjs +59 -59
- package/fesm2022/angular-debug-recorder.mjs +791 -780
- package/fesm2022/angular-debug-recorder.mjs.map +1 -1
- package/lib/services/rrweb-recorder.service.d.ts +5 -4
- package/package.json +1 -1
|
@@ -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.
|
|
249
|
-
<span class="tab-badge">{{ rrweb.
|
|
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.
|
|
445
|
-
<span class="tab-badge">{{ rrweb.
|
|
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":"AAAA,OAAO,EACL,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GACpC,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAC/D,OAAO,EAAE,kBAAkB,EAAE,MAAM,kCAAkC,CAAC;AACtE,OAAO,EAAE,oBAAoB,EAAE,MAAM,oCAAoC,CAAC;AAE1E,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAC3E,OAAO,EAAE,oBAAoB,EAAE,MAAM,wCAAwC,CAAC;AAC9E,OAAO,EAAE,uBAAuB,EAAE,MAAM,8CAA8C,CAAC;AACvF,OAAO,EAAE,sBAAsB,EAAE,MAAM,4CAA4C,CAAC;;;AAucpF,MAAM,OAAO,mBAAmB;IAlchC;QAmcE,aAAQ,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QACnC,cAAS,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAC;QACvC,UAAK,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAErC,cAAS,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC1B,cAAS,GAAG,MAAM,CAAW,SAAS,CAAC,CAAC;QACxC,aAAQ,GAAG,MAAM,CAAgB,cAAc,CAAC,CAAC;QACjD,eAAU,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;QACzB,iBAAY,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC7B,gBAAW,GAAG,EAAE,CAAC;QACjB,gBAAW,GAAG,EAAE,CAAC;QAEjB,eAAU,GAAG,QAAQ,CACnB,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE,EAAE,OAAO,CAAC,MAAM,IAAI,CAAC,CAAC,GAAG,CAAC;YACzD,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,CAC5C,CAAC;QAEM,aAAQ,GAAG,KAAK,CAAC;QACjB,iBAAY,GAAG,CAAC,CAAC;QACjB,iBAAY,GAAG,CAAC,CAAC;QAWjB,iBAAY,GAAG,CAAC,CAAgB,EAAE,EAAE;YAC1C,IAAI,CAAC,CAAC,OAAO,IAAI,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC,GAAG,KAAK,GAAG,EAAE,CAAC;gBAC7C,CAAC,CAAC,cAAc,EAAE,CAAC;gBACnB,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,CAAC;YACD,IAAI,CAAC,CAAC,OAAO,IAAI,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC,GAAG,KAAK,GAAG,EAAE,CAAC;gBAC7C,CAAC,CAAC,cAAc,EAAE,CAAC;gBACnB,IAAI,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,EAAE,CAAC;oBAChC,IAAI,CAAC,aAAa,EAAE,CAAC;gBACvB,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,cAAc,EAAE,CAAC;gBACxB,CAAC;YACH,CAAC;QACH,CAAC,CAAC;KAoFH;IA1GC,QAAQ;QACN,kCAAkC;QAClC,QAAQ,CAAC,gBAAgB,CAAC,SAAS,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;IAC1D,CAAC;IAED,WAAW;QACT,QAAQ,CAAC,mBAAmB,CAAC,SAAS,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;IAC7D,CAAC;IAiBD,WAAW;QACT,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE,EAAE,CAAC;YACxD,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAChC,CAAC;IACH,CAAC;IAED,aAAa;QACX,MAAM,SAAS,GAAoB,CAAC,cAAc,EAAE,aAAa,EAAE,WAAW,EAAE,UAAU,CAAC,CAAC;QAC5F,MAAM,GAAG,GAAG,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC/C,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;IAC7D,CAAC;IAED,cAAc;QACZ,IAAI,CAAC,QAAQ,CAAC,cAAc,CAC1B,IAAI,CAAC,WAAW,IAAI,SAAS,EAC7B,IAAI,CAAC,WAAW,IAAI,SAAS,CAC9B,CAAC;QACF,4CAA4C;QAC5C,IAAI,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC;QAC5B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAChC,CAAC;IAED,aAAa;QACX,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE,CAAC;QAC9C,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;QAC3B,IAAI,OAAO,EAAE,CAAC;YACZ,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAChC,CAAC;IACH,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAClF,IAAI,CAAC,OAAO;YAAE,OAAO;QACrB,MAAM,IAAI,CAAC,SAAS,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAClD,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAC7B,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,OAAyB;QAC7C,MAAM,IAAI,CAAC,SAAS,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAClD,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAC7B,CAAC;IAED,qBAAqB;IACrB,SAAS,CAAC,CAAa;QACrB,IAAK,CAAC,CAAC,MAAsB,CAAC,OAAO,CAAC,QAAQ,CAAC;YAAE,OAAO;QACxD,MAAM,KAAK,GAAI,CAAC,CAAC,aAA6B,CAAC,aAAc,CAAC;QAC9D,MAAM,MAAM,GAAG,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,qBAAqB,EAAE,CAAC,IAAI,CAAC;QAC9D,MAAM,MAAM,GAAG,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,qBAAqB,EAAE,CAAC,GAAG,CAAC;QAE7D,MAAM,MAAM,GAAG,CAAC,EAAc,EAAE,EAAE;YAChC,KAAK,CAAC,KAAK,CAAC,IAAI,GAAG,GAAG,EAAE,CAAC,OAAO,GAAG,MAAM,IAAI,CAAC;YAC9C,KAAK,CAAC,KAAK,CAAC,GAAG,GAAI,GAAG,EAAE,CAAC,OAAO,GAAG,MAAM,IAAI,CAAC;YAC9C,KAAK,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YAC3B,KAAK,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC;QAC9B,CAAC,CAAC;QACF,MAAM,IAAI,GAAG,GAAG,EAAE;YAChB,QAAQ,CAAC,mBAAmB,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;YAClD,QAAQ,CAAC,mBAAmB,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAChD,CAAC,CAAC;QACF,QAAQ,CAAC,gBAAgB,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;QAC/C,QAAQ,CAAC,gBAAgB,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;IAC7C,CAAC;IAED,gBAAgB;IAChB,WAAW,CAAC,CAAa;QACvB,MAAM,KAAK,GAAI,CAAC,CAAC,aAA6B,CAAC,aAAc,CAAC;QAC9D,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC,OAAO,CAAC;QAC9B,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,qBAAqB,EAAE,CAAC,MAAM,CAAC;QAEzD,MAAM,MAAM,GAAG,CAAC,EAAc,EAAE,EAAE;YAChC,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,YAAY,GAAG,CAAC,EAAE,CAAC,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;YACjF,KAAK,CAAC,KAAK,CAAC,SAAS,GAAG,GAAG,IAAI,IAAI,CAAC;QACtC,CAAC,CAAC;QACF,MAAM,IAAI,GAAG,GAAG,EAAE;YAChB,QAAQ,CAAC,mBAAmB,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;YAClD,QAAQ,CAAC,mBAAmB,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAChD,CAAC,CAAC;QACF,QAAQ,CAAC,gBAAgB,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;QAC/C,QAAQ,CAAC,gBAAgB,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAC3C,CAAC,CAAC,cAAc,EAAE,CAAC;IACrB,CAAC;+GA/HU,mBAAmB;mGAAnB,mBAAmB,2EA9bpB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgMT,qmKAjMS,YAAY,8BAAE,WAAW,+mBAAE,mBAAmB,uHAAE,oBAAoB,+EAAE,uBAAuB,oFAAE,sBAAsB;;4FA+bpH,mBAAmB;kBAlc/B,SAAS;+BACE,iBAAiB,cACf,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,mBAAmB,EAAE,oBAAoB,EAAE,uBAAuB,EAAE,sBAAsB,CAAC,YACtH;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgMT","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"]}
|