angular-debug-recorder 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +24 -0
- package/esm2022/angular-debug-recorder.mjs +5 -0
- package/esm2022/lib/action-list/action-list.component.mjs +220 -0
- package/esm2022/lib/debug-panel/debug-panel.component.mjs +516 -0
- package/esm2022/lib/models/recorded-action.model.mjs +2 -0
- package/esm2022/lib/services/ai-generator.service.mjs +61 -0
- package/esm2022/lib/services/recorder.service.mjs +354 -0
- package/esm2022/lib/services/rrweb-recorder.service.mjs +108 -0
- package/esm2022/lib/session-replay/session-replay.component.mjs +166 -0
- package/esm2022/lib/settings-dialog/settings-dialog.component.mjs +105 -0
- package/esm2022/lib/test-preview/test-preview.component.mjs +120 -0
- package/esm2022/public-api.mjs +17 -0
- package/fesm2022/angular-debug-recorder.mjs +1631 -0
- package/fesm2022/angular-debug-recorder.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/action-list/action-list.component.d.ts +21 -0
- package/lib/debug-panel/debug-panel.component.d.ts +38 -0
- package/lib/models/recorded-action.model.d.ts +45 -0
- package/lib/services/ai-generator.service.d.ts +20 -0
- package/lib/services/recorder.service.d.ts +46 -0
- package/lib/services/rrweb-recorder.service.d.ts +27 -0
- package/lib/session-replay/session-replay.component.d.ts +19 -0
- package/lib/settings-dialog/settings-dialog.component.d.ts +11 -0
- package/lib/test-preview/test-preview.component.d.ts +13 -0
- package/package.json +37 -0
- package/public-api.d.ts +9 -0
package/README.md
ADDED
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
# DebugRecorder
|
|
2
|
+
|
|
3
|
+
This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 17.3.0.
|
|
4
|
+
|
|
5
|
+
## Code scaffolding
|
|
6
|
+
|
|
7
|
+
Run `ng generate component component-name --project debug-recorder` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module --project debug-recorder`.
|
|
8
|
+
> Note: Don't forget to add `--project debug-recorder` or else it will be added to the default project in your `angular.json` file.
|
|
9
|
+
|
|
10
|
+
## Build
|
|
11
|
+
|
|
12
|
+
Run `ng build debug-recorder` to build the project. The build artifacts will be stored in the `dist/` directory.
|
|
13
|
+
|
|
14
|
+
## Publishing
|
|
15
|
+
|
|
16
|
+
After building your library with `ng build debug-recorder`, go to the dist folder `cd dist/debug-recorder` and run `npm publish`.
|
|
17
|
+
|
|
18
|
+
## Running unit tests
|
|
19
|
+
|
|
20
|
+
Run `ng test debug-recorder` to execute the unit tests via [Karma](https://karma-runner.github.io).
|
|
21
|
+
|
|
22
|
+
## Further help
|
|
23
|
+
|
|
24
|
+
To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.io/cli) page.
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './public-api';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5ndWxhci1kZWJ1Zy1yZWNvcmRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2RlYnVnLXJlY29yZGVyL3NyYy9hbmd1bGFyLWRlYnVnLXJlY29yZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vcHVibGljLWFwaSc7XG4iXX0=
|
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, signal } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/forms";
|
|
6
|
+
export class ActionListComponent {
|
|
7
|
+
constructor() {
|
|
8
|
+
this.session = null;
|
|
9
|
+
this.removeAction = new EventEmitter();
|
|
10
|
+
this.addNote = new EventEmitter();
|
|
11
|
+
this.expandedId = signal(null);
|
|
12
|
+
this.noteMap = {};
|
|
13
|
+
}
|
|
14
|
+
toggleExpand(id) {
|
|
15
|
+
this.expandedId.update(v => v === id ? null : id);
|
|
16
|
+
}
|
|
17
|
+
onRemove(id, e) {
|
|
18
|
+
e.stopPropagation();
|
|
19
|
+
this.removeAction.emit(id);
|
|
20
|
+
}
|
|
21
|
+
onAddNote(id) {
|
|
22
|
+
this.addNote.emit({ id, note: this.noteMap[id] ?? '' });
|
|
23
|
+
}
|
|
24
|
+
getActionIcon(type) {
|
|
25
|
+
const icons = {
|
|
26
|
+
click: '👆',
|
|
27
|
+
dblclick: '👆👆',
|
|
28
|
+
input: '⌨️',
|
|
29
|
+
select: '📋',
|
|
30
|
+
submit: '📤',
|
|
31
|
+
navigation: '🔗',
|
|
32
|
+
keypress: '⌨️',
|
|
33
|
+
scroll: '↕️',
|
|
34
|
+
hover: '🖱️',
|
|
35
|
+
assertion: '✅',
|
|
36
|
+
screenshot: '📸',
|
|
37
|
+
};
|
|
38
|
+
return icons[type] ?? '•';
|
|
39
|
+
}
|
|
40
|
+
formatTime(ts) {
|
|
41
|
+
return new Date(ts).toLocaleTimeString('de-DE', { hour: '2-digit', minute: '2-digit', second: '2-digit' });
|
|
42
|
+
}
|
|
43
|
+
formatDuration(start, end) {
|
|
44
|
+
const s = Math.round((end - start) / 1000);
|
|
45
|
+
return s < 60 ? `${s}s` : `${Math.floor(s / 60)}m ${s % 60}s`;
|
|
46
|
+
}
|
|
47
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ActionListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
48
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ActionListComponent, isStandalone: true, selector: "app-action-list", inputs: { session: "session" }, outputs: { removeAction: "removeAction", addNote: "addNote" }, ngImport: i0, template: `
|
|
49
|
+
<div class="action-list" data-debug-panel>
|
|
50
|
+
@if (!session || session.actions.length === 0) {
|
|
51
|
+
<div class="empty-state">
|
|
52
|
+
<div class="empty-icon">🎬</div>
|
|
53
|
+
<p>Noch keine Aktionen aufgezeichnet.</p>
|
|
54
|
+
<p class="hint">Starte die Aufnahme und interagiere mit der App.</p>
|
|
55
|
+
<div class="shortcuts-hint">
|
|
56
|
+
<kbd>Ctrl+Shift+D</kbd> Panel
|
|
57
|
+
<kbd>Ctrl+Shift+R</kbd> Record
|
|
58
|
+
</div>
|
|
59
|
+
</div>
|
|
60
|
+
} @else {
|
|
61
|
+
<div class="list-header">
|
|
62
|
+
<span class="list-count">{{ session.actions.length }} Aktionen</span>
|
|
63
|
+
<span class="list-duration">
|
|
64
|
+
@if (session.endTime) {
|
|
65
|
+
{{ formatDuration(session.startTime, session.endTime) }}
|
|
66
|
+
} @else {
|
|
67
|
+
Live
|
|
68
|
+
}
|
|
69
|
+
</span>
|
|
70
|
+
</div>
|
|
71
|
+
|
|
72
|
+
@for (action of session.actions; track action.id; let i = $index) {
|
|
73
|
+
<div class="action-item" [class.expanded]="expandedId() === action.id">
|
|
74
|
+
<div class="action-row" (click)="toggleExpand(action.id)">
|
|
75
|
+
<span class="action-index">{{ i + 1 }}</span>
|
|
76
|
+
<span class="action-type-badge" [class]="'type-' + action.type">
|
|
77
|
+
{{ getActionIcon(action.type) }}
|
|
78
|
+
</span>
|
|
79
|
+
<div class="action-info">
|
|
80
|
+
<span class="action-desc">{{ action.description }}</span>
|
|
81
|
+
<span class="action-selector">{{ action.selector }}</span>
|
|
82
|
+
</div>
|
|
83
|
+
<span class="action-time">{{ formatTime(action.timestamp) }}</span>
|
|
84
|
+
<button
|
|
85
|
+
class="remove-btn"
|
|
86
|
+
data-debug-panel
|
|
87
|
+
title="Aktion entfernen"
|
|
88
|
+
(click)="onRemove(action.id, $event)"
|
|
89
|
+
>✕</button>
|
|
90
|
+
</div>
|
|
91
|
+
|
|
92
|
+
@if (expandedId() === action.id) {
|
|
93
|
+
<div class="action-detail" data-debug-panel>
|
|
94
|
+
<div class="detail-grid">
|
|
95
|
+
<span class="detail-label">Selector</span>
|
|
96
|
+
<code class="detail-value">{{ action.selector }}</code>
|
|
97
|
+
@if (action.value) {
|
|
98
|
+
<span class="detail-label">Wert</span>
|
|
99
|
+
<code class="detail-value">{{ action.value }}</code>
|
|
100
|
+
}
|
|
101
|
+
@if (action.element?.tagName) {
|
|
102
|
+
<span class="detail-label">Element</span>
|
|
103
|
+
<code class="detail-value"><{{ action.element?.tagName }}></code>
|
|
104
|
+
}
|
|
105
|
+
<span class="detail-label">Strategie</span>
|
|
106
|
+
<span class="detail-value strategy-badge" [class]="'strat-' + action.selectorStrategy">
|
|
107
|
+
{{ action.selectorStrategy }}
|
|
108
|
+
</span>
|
|
109
|
+
<span class="detail-label">URL</span>
|
|
110
|
+
<code class="detail-value url-val">{{ action.url }}</code>
|
|
111
|
+
</div>
|
|
112
|
+
<div class="note-area">
|
|
113
|
+
<textarea
|
|
114
|
+
data-debug-panel
|
|
115
|
+
class="note-input"
|
|
116
|
+
[(ngModel)]="noteMap[action.id]"
|
|
117
|
+
placeholder="Notiz zu dieser Aktion..."
|
|
118
|
+
rows="2"
|
|
119
|
+
(blur)="onAddNote(action.id)"
|
|
120
|
+
></textarea>
|
|
121
|
+
</div>
|
|
122
|
+
</div>
|
|
123
|
+
}
|
|
124
|
+
</div>
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
</div>
|
|
128
|
+
`, isInline: true, styles: [".action-list{padding:0}.empty-state{text-align:center;padding:32px 20px;color:#64748b}.empty-icon{font-size:40px;margin-bottom:10px}.empty-state p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.shortcuts-hint{margin-top:12px;font-size:11px;color:#475569}kbd{background:#1e293b;border:1px solid #334155;color:#94a3b8;padding:2px 6px;border-radius:4px;font-size:10px}.list-header{display:flex;justify-content:space-between;padding:8px 14px;font-size:11px;color:#64748b;background:#0f172a;border-bottom:1px solid #1e293b;position:sticky;top:0}.action-item{border-bottom:1px solid #1e293b;transition:background .1s}.action-item:hover{background:#1e293b80}.action-item.expanded{background:#1e293b}.action-row{display:flex;align-items:center;padding:8px 10px;gap:8px;cursor:pointer}.action-index{color:#475569;font-size:10px;min-width:18px;text-align:right}.action-type-badge{font-size:14px;min-width:20px;text-align:center}.action-info{flex:1;min-width:0;display:flex;flex-direction:column;gap:1px}.action-desc{font-size:12px;color:#cbd5e1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.action-selector{font-size:10px;color:#64748b;font-family:Cascadia Code,Consolas,monospace;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.action-time{font-size:10px;color:#475569;white-space:nowrap}.remove-btn{background:none;border:none;color:#475569;cursor:pointer;font-size:12px;padding:2px 5px;border-radius:3px;opacity:0;transition:opacity .15s,color .15s}.action-row:hover .remove-btn{opacity:1}.remove-btn:hover{color:#f87171}.action-detail{padding:10px 14px;background:#0f172ab3;border-top:1px solid #1e293b}.detail-grid{display:grid;grid-template-columns:auto 1fr;gap:4px 10px;margin-bottom:8px;align-items:start}.detail-label{font-size:10px;color:#64748b;padding-top:2px;white-space:nowrap}.detail-value{font-size:11px;color:#93c5fd;font-family:Cascadia Code,Consolas,monospace;word-break:break-all}.url-val{color:#6ee7b7}.strategy-badge{font-size:10px;padding:1px 6px;border-radius:3px;font-family:monospace}.strat-data-testid,.strat-data-cy{background:#064e3b;color:#34d399}.strat-id{background:#1e3a8a;color:#93c5fd}.strat-name{background:#44337a;color:#c4b5fd}.strat-class{background:#374151;color:#9ca3af}.strat-combined{background:#292524;color:#d6d3d1}.note-area{margin-top:6px}.note-input{width:100%;box-sizing:border-box;background:#0f172a;border:1px solid #334155;color:#e2e8f0;border-radius:5px;padding:6px 8px;font-size:11px;resize:vertical}.note-input:focus{outline:none;border-color:#3b82f6}\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"] }] }); }
|
|
129
|
+
}
|
|
130
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ActionListComponent, decorators: [{
|
|
131
|
+
type: Component,
|
|
132
|
+
args: [{ selector: 'app-action-list', standalone: true, imports: [CommonModule, FormsModule], template: `
|
|
133
|
+
<div class="action-list" data-debug-panel>
|
|
134
|
+
@if (!session || session.actions.length === 0) {
|
|
135
|
+
<div class="empty-state">
|
|
136
|
+
<div class="empty-icon">🎬</div>
|
|
137
|
+
<p>Noch keine Aktionen aufgezeichnet.</p>
|
|
138
|
+
<p class="hint">Starte die Aufnahme und interagiere mit der App.</p>
|
|
139
|
+
<div class="shortcuts-hint">
|
|
140
|
+
<kbd>Ctrl+Shift+D</kbd> Panel
|
|
141
|
+
<kbd>Ctrl+Shift+R</kbd> Record
|
|
142
|
+
</div>
|
|
143
|
+
</div>
|
|
144
|
+
} @else {
|
|
145
|
+
<div class="list-header">
|
|
146
|
+
<span class="list-count">{{ session.actions.length }} Aktionen</span>
|
|
147
|
+
<span class="list-duration">
|
|
148
|
+
@if (session.endTime) {
|
|
149
|
+
{{ formatDuration(session.startTime, session.endTime) }}
|
|
150
|
+
} @else {
|
|
151
|
+
Live
|
|
152
|
+
}
|
|
153
|
+
</span>
|
|
154
|
+
</div>
|
|
155
|
+
|
|
156
|
+
@for (action of session.actions; track action.id; let i = $index) {
|
|
157
|
+
<div class="action-item" [class.expanded]="expandedId() === action.id">
|
|
158
|
+
<div class="action-row" (click)="toggleExpand(action.id)">
|
|
159
|
+
<span class="action-index">{{ i + 1 }}</span>
|
|
160
|
+
<span class="action-type-badge" [class]="'type-' + action.type">
|
|
161
|
+
{{ getActionIcon(action.type) }}
|
|
162
|
+
</span>
|
|
163
|
+
<div class="action-info">
|
|
164
|
+
<span class="action-desc">{{ action.description }}</span>
|
|
165
|
+
<span class="action-selector">{{ action.selector }}</span>
|
|
166
|
+
</div>
|
|
167
|
+
<span class="action-time">{{ formatTime(action.timestamp) }}</span>
|
|
168
|
+
<button
|
|
169
|
+
class="remove-btn"
|
|
170
|
+
data-debug-panel
|
|
171
|
+
title="Aktion entfernen"
|
|
172
|
+
(click)="onRemove(action.id, $event)"
|
|
173
|
+
>✕</button>
|
|
174
|
+
</div>
|
|
175
|
+
|
|
176
|
+
@if (expandedId() === action.id) {
|
|
177
|
+
<div class="action-detail" data-debug-panel>
|
|
178
|
+
<div class="detail-grid">
|
|
179
|
+
<span class="detail-label">Selector</span>
|
|
180
|
+
<code class="detail-value">{{ action.selector }}</code>
|
|
181
|
+
@if (action.value) {
|
|
182
|
+
<span class="detail-label">Wert</span>
|
|
183
|
+
<code class="detail-value">{{ action.value }}</code>
|
|
184
|
+
}
|
|
185
|
+
@if (action.element?.tagName) {
|
|
186
|
+
<span class="detail-label">Element</span>
|
|
187
|
+
<code class="detail-value"><{{ action.element?.tagName }}></code>
|
|
188
|
+
}
|
|
189
|
+
<span class="detail-label">Strategie</span>
|
|
190
|
+
<span class="detail-value strategy-badge" [class]="'strat-' + action.selectorStrategy">
|
|
191
|
+
{{ action.selectorStrategy }}
|
|
192
|
+
</span>
|
|
193
|
+
<span class="detail-label">URL</span>
|
|
194
|
+
<code class="detail-value url-val">{{ action.url }}</code>
|
|
195
|
+
</div>
|
|
196
|
+
<div class="note-area">
|
|
197
|
+
<textarea
|
|
198
|
+
data-debug-panel
|
|
199
|
+
class="note-input"
|
|
200
|
+
[(ngModel)]="noteMap[action.id]"
|
|
201
|
+
placeholder="Notiz zu dieser Aktion..."
|
|
202
|
+
rows="2"
|
|
203
|
+
(blur)="onAddNote(action.id)"
|
|
204
|
+
></textarea>
|
|
205
|
+
</div>
|
|
206
|
+
</div>
|
|
207
|
+
}
|
|
208
|
+
</div>
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
</div>
|
|
212
|
+
`, styles: [".action-list{padding:0}.empty-state{text-align:center;padding:32px 20px;color:#64748b}.empty-icon{font-size:40px;margin-bottom:10px}.empty-state p{margin:4px 0;font-size:13px}.hint{font-size:11px;color:#475569}.shortcuts-hint{margin-top:12px;font-size:11px;color:#475569}kbd{background:#1e293b;border:1px solid #334155;color:#94a3b8;padding:2px 6px;border-radius:4px;font-size:10px}.list-header{display:flex;justify-content:space-between;padding:8px 14px;font-size:11px;color:#64748b;background:#0f172a;border-bottom:1px solid #1e293b;position:sticky;top:0}.action-item{border-bottom:1px solid #1e293b;transition:background .1s}.action-item:hover{background:#1e293b80}.action-item.expanded{background:#1e293b}.action-row{display:flex;align-items:center;padding:8px 10px;gap:8px;cursor:pointer}.action-index{color:#475569;font-size:10px;min-width:18px;text-align:right}.action-type-badge{font-size:14px;min-width:20px;text-align:center}.action-info{flex:1;min-width:0;display:flex;flex-direction:column;gap:1px}.action-desc{font-size:12px;color:#cbd5e1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.action-selector{font-size:10px;color:#64748b;font-family:Cascadia Code,Consolas,monospace;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.action-time{font-size:10px;color:#475569;white-space:nowrap}.remove-btn{background:none;border:none;color:#475569;cursor:pointer;font-size:12px;padding:2px 5px;border-radius:3px;opacity:0;transition:opacity .15s,color .15s}.action-row:hover .remove-btn{opacity:1}.remove-btn:hover{color:#f87171}.action-detail{padding:10px 14px;background:#0f172ab3;border-top:1px solid #1e293b}.detail-grid{display:grid;grid-template-columns:auto 1fr;gap:4px 10px;margin-bottom:8px;align-items:start}.detail-label{font-size:10px;color:#64748b;padding-top:2px;white-space:nowrap}.detail-value{font-size:11px;color:#93c5fd;font-family:Cascadia Code,Consolas,monospace;word-break:break-all}.url-val{color:#6ee7b7}.strategy-badge{font-size:10px;padding:1px 6px;border-radius:3px;font-family:monospace}.strat-data-testid,.strat-data-cy{background:#064e3b;color:#34d399}.strat-id{background:#1e3a8a;color:#93c5fd}.strat-name{background:#44337a;color:#c4b5fd}.strat-class{background:#374151;color:#9ca3af}.strat-combined{background:#292524;color:#d6d3d1}.note-area{margin-top:6px}.note-input{width:100%;box-sizing:border-box;background:#0f172a;border:1px solid #334155;color:#e2e8f0;border-radius:5px;padding:6px 8px;font-size:11px;resize:vertical}.note-input:focus{outline:none;border-color:#3b82f6}\n"] }]
|
|
213
|
+
}], propDecorators: { session: [{
|
|
214
|
+
type: Input
|
|
215
|
+
}], removeAction: [{
|
|
216
|
+
type: Output
|
|
217
|
+
}], addNote: [{
|
|
218
|
+
type: Output
|
|
219
|
+
}] } });
|
|
220
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"action-list.component.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/action-list/action-list.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;;;AA+O7C,MAAM,OAAO,mBAAmB;IA5OhC;QA6OW,YAAO,GAA4B,IAAI,CAAC;QACvC,iBAAY,GAAG,IAAI,YAAY,EAAU,CAAC;QAC1C,YAAO,GAAG,IAAI,YAAY,EAAgC,CAAC;QAErE,eAAU,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;QACzC,YAAO,GAA2B,EAAE,CAAC;KAwCtC;IAtCC,YAAY,CAAC,EAAU;QACrB,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IACpD,CAAC;IAED,QAAQ,CAAC,EAAU,EAAE,CAAQ;QAC3B,CAAC,CAAC,eAAe,EAAE,CAAC;QACpB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC7B,CAAC;IAED,SAAS,CAAC,EAAU;QAClB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;IAC1D,CAAC;IAED,aAAa,CAAC,IAAY;QACxB,MAAM,KAAK,GAA2B;YACpC,KAAK,EAAQ,IAAI;YACjB,QAAQ,EAAK,MAAM;YACnB,KAAK,EAAQ,IAAI;YACjB,MAAM,EAAO,IAAI;YACjB,MAAM,EAAO,IAAI;YACjB,UAAU,EAAG,IAAI;YACjB,QAAQ,EAAK,IAAI;YACjB,MAAM,EAAO,IAAI;YACjB,KAAK,EAAQ,KAAK;YAClB,SAAS,EAAI,GAAG;YAChB,UAAU,EAAG,IAAI;SAClB,CAAC;QACF,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC;IAC5B,CAAC;IAED,UAAU,CAAC,EAAU;QACnB,OAAO,IAAI,IAAI,CAAC,EAAE,CAAC,CAAC,kBAAkB,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC;IAC7G,CAAC;IAED,cAAc,CAAC,KAAa,EAAE,GAAW;QACvC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,KAAK,CAAC,GAAG,IAAI,CAAC,CAAC;QAC3C,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,GAAC,EAAE,CAAC,KAAK,CAAC,GAAC,EAAE,GAAG,CAAC;IAC5D,CAAC;+GA7CU,mBAAmB;mGAAnB,mBAAmB,0KAxOpB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgFT,ojFAjFS,YAAY,8BAAE,WAAW;;4FAyOxB,mBAAmB;kBA5O/B,SAAS;+BACE,iBAAiB,cACf,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,CAAC,YAC1B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgFT;8BAyJQ,OAAO;sBAAf,KAAK;gBACI,YAAY;sBAArB,MAAM;gBACG,OAAO;sBAAhB,MAAM","sourcesContent":["import { Component, Input, Output, EventEmitter, signal } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { RecordedAction, RecordingSession } from '../models/recorded-action.model';\n\n@Component({\n  selector: 'app-action-list',\n  standalone: true,\n  imports: [CommonModule, FormsModule],\n  template: `\n    <div class=\"action-list\" data-debug-panel>\n      @if (!session || session.actions.length === 0) {\n        <div class=\"empty-state\">\n          <div class=\"empty-icon\">🎬</div>\n          <p>Noch keine Aktionen aufgezeichnet.</p>\n          <p class=\"hint\">Starte die Aufnahme und interagiere mit der App.</p>\n          <div class=\"shortcuts-hint\">\n            <kbd>Ctrl+Shift+D</kbd> Panel &nbsp;\n            <kbd>Ctrl+Shift+R</kbd> Record\n          </div>\n        </div>\n      } @else {\n        <div class=\"list-header\">\n          <span class=\"list-count\">{{ session.actions.length }} Aktionen</span>\n          <span class=\"list-duration\">\n            @if (session.endTime) {\n              {{ formatDuration(session.startTime, session.endTime) }}\n            } @else {\n              Live\n            }\n          </span>\n        </div>\n\n        @for (action of session.actions; track action.id; let i = $index) {\n          <div class=\"action-item\" [class.expanded]=\"expandedId() === action.id\">\n            <div class=\"action-row\" (click)=\"toggleExpand(action.id)\">\n              <span class=\"action-index\">{{ i + 1 }}</span>\n              <span class=\"action-type-badge\" [class]=\"'type-' + action.type\">\n                {{ getActionIcon(action.type) }}\n              </span>\n              <div class=\"action-info\">\n                <span class=\"action-desc\">{{ action.description }}</span>\n                <span class=\"action-selector\">{{ action.selector }}</span>\n              </div>\n              <span class=\"action-time\">{{ formatTime(action.timestamp) }}</span>\n              <button\n                class=\"remove-btn\"\n                data-debug-panel\n                title=\"Aktion entfernen\"\n                (click)=\"onRemove(action.id, $event)\"\n              >✕</button>\n            </div>\n\n            @if (expandedId() === action.id) {\n              <div class=\"action-detail\" data-debug-panel>\n                <div class=\"detail-grid\">\n                  <span class=\"detail-label\">Selector</span>\n                  <code class=\"detail-value\">{{ action.selector }}</code>\n                  @if (action.value) {\n                    <span class=\"detail-label\">Wert</span>\n                    <code class=\"detail-value\">{{ action.value }}</code>\n                  }\n                  @if (action.element?.tagName) {\n                    <span class=\"detail-label\">Element</span>\n                    <code class=\"detail-value\">&lt;{{ action.element?.tagName }}&gt;</code>\n                  }\n                  <span class=\"detail-label\">Strategie</span>\n                  <span class=\"detail-value strategy-badge\" [class]=\"'strat-' + action.selectorStrategy\">\n                    {{ action.selectorStrategy }}\n                  </span>\n                  <span class=\"detail-label\">URL</span>\n                  <code class=\"detail-value url-val\">{{ action.url }}</code>\n                </div>\n                <div class=\"note-area\">\n                  <textarea\n                    data-debug-panel\n                    class=\"note-input\"\n                    [(ngModel)]=\"noteMap[action.id]\"\n                    placeholder=\"Notiz zu dieser Aktion...\"\n                    rows=\"2\"\n                    (blur)=\"onAddNote(action.id)\"\n                  ></textarea>\n                </div>\n              </div>\n            }\n          </div>\n        }\n      }\n    </div>\n  `,\n  styles: [`\n    .action-list { padding: 0; }\n\n    .empty-state {\n      text-align: center;\n      padding: 32px 20px;\n      color: #64748b;\n    }\n    .empty-icon { font-size: 40px; margin-bottom: 10px; }\n    .empty-state p { margin: 4px 0; font-size: 13px; }\n    .hint { font-size: 11px; color: #475569; }\n    .shortcuts-hint {\n      margin-top: 12px;\n      font-size: 11px;\n      color: #475569;\n    }\n    kbd {\n      background: #1e293b;\n      border: 1px solid #334155;\n      color: #94a3b8;\n      padding: 2px 6px;\n      border-radius: 4px;\n      font-size: 10px;\n    }\n\n    .list-header {\n      display: flex;\n      justify-content: space-between;\n      padding: 8px 14px;\n      font-size: 11px;\n      color: #64748b;\n      background: #0f172a;\n      border-bottom: 1px solid #1e293b;\n      position: sticky;\n      top: 0;\n    }\n\n    .action-item {\n      border-bottom: 1px solid #1e293b;\n      transition: background 0.1s;\n    }\n    .action-item:hover { background: rgba(30,41,59,0.5); }\n    .action-item.expanded { background: #1e293b; }\n\n    .action-row {\n      display: flex;\n      align-items: center;\n      padding: 8px 10px;\n      gap: 8px;\n      cursor: pointer;\n    }\n    .action-index {\n      color: #475569;\n      font-size: 10px;\n      min-width: 18px;\n      text-align: right;\n    }\n    .action-type-badge {\n      font-size: 14px;\n      min-width: 20px;\n      text-align: center;\n    }\n    .action-info {\n      flex: 1;\n      min-width: 0;\n      display: flex;\n      flex-direction: column;\n      gap: 1px;\n    }\n    .action-desc {\n      font-size: 12px;\n      color: #cbd5e1;\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n    .action-selector {\n      font-size: 10px;\n      color: #64748b;\n      font-family: 'Cascadia Code', 'Consolas', monospace;\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n    .action-time {\n      font-size: 10px;\n      color: #475569;\n      white-space: nowrap;\n    }\n    .remove-btn {\n      background: none;\n      border: none;\n      color: #475569;\n      cursor: pointer;\n      font-size: 12px;\n      padding: 2px 5px;\n      border-radius: 3px;\n      opacity: 0;\n      transition: opacity 0.15s, color 0.15s;\n    }\n    .action-row:hover .remove-btn { opacity: 1; }\n    .remove-btn:hover { color: #f87171; }\n\n    .action-detail {\n      padding: 10px 14px;\n      background: rgba(15,23,42,0.7);\n      border-top: 1px solid #1e293b;\n    }\n    .detail-grid {\n      display: grid;\n      grid-template-columns: auto 1fr;\n      gap: 4px 10px;\n      margin-bottom: 8px;\n      align-items: start;\n    }\n    .detail-label { font-size: 10px; color: #64748b; padding-top: 2px; white-space: nowrap; }\n    .detail-value {\n      font-size: 11px;\n      color: #93c5fd;\n      font-family: 'Cascadia Code', 'Consolas', monospace;\n      word-break: break-all;\n    }\n    .url-val { color: #6ee7b7; }\n    .strategy-badge {\n      font-size: 10px;\n      padding: 1px 6px;\n      border-radius: 3px;\n      font-family: monospace;\n    }\n    .strat-data-testid { background: #064e3b; color: #34d399; }\n    .strat-data-cy     { background: #064e3b; color: #34d399; }\n    .strat-id          { background: #1e3a8a; color: #93c5fd; }\n    .strat-name        { background: #44337a; color: #c4b5fd; }\n    .strat-class       { background: #374151; color: #9ca3af; }\n    .strat-combined    { background: #292524; color: #d6d3d1; }\n\n    .note-area { margin-top: 6px; }\n    .note-input {\n      width: 100%;\n      box-sizing: border-box;\n      background: #0f172a;\n      border: 1px solid #334155;\n      color: #e2e8f0;\n      border-radius: 5px;\n      padding: 6px 8px;\n      font-size: 11px;\n      resize: vertical;\n    }\n    .note-input:focus { outline: none; border-color: #3b82f6; }\n  `],\n})\nexport class ActionListComponent {\n  @Input() session: RecordingSession | null = null;\n  @Output() removeAction = new EventEmitter<string>();\n  @Output() addNote = new EventEmitter<{ id: string; note: string }>();\n\n  expandedId = signal<string | null>(null);\n  noteMap: Record<string, string> = {};\n\n  toggleExpand(id: string): void {\n    this.expandedId.update(v => v === id ? null : id);\n  }\n\n  onRemove(id: string, e: Event): void {\n    e.stopPropagation();\n    this.removeAction.emit(id);\n  }\n\n  onAddNote(id: string): void {\n    this.addNote.emit({ id, note: this.noteMap[id] ?? '' });\n  }\n\n  getActionIcon(type: string): string {\n    const icons: Record<string, string> = {\n      click:       '👆',\n      dblclick:    '👆👆',\n      input:       '⌨️',\n      select:      '📋',\n      submit:      '📤',\n      navigation:  '🔗',\n      keypress:    '⌨️',\n      scroll:      '↕️',\n      hover:       '🖱️',\n      assertion:   '✅',\n      screenshot:  '📸',\n    };\n    return icons[type] ?? '•';\n  }\n\n  formatTime(ts: number): string {\n    return new Date(ts).toLocaleTimeString('de-DE', { hour: '2-digit', minute: '2-digit', second: '2-digit' });\n  }\n\n  formatDuration(start: number, end: number): string {\n    const s = Math.round((end - start) / 1000);\n    return s < 60 ? `${s}s` : `${Math.floor(s/60)}m ${s%60}s`;\n  }\n}\n"]}
|