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 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 &nbsp;
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">&lt;{{ action.element?.tagName }}&gt;</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 &nbsp;
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">&lt;{{ action.element?.tagName }}&gt;</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"]}