angular-debug-recorder 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,61 @@
1
+ import { Injectable, signal } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common/http";
4
+ export class AiGeneratorService {
5
+ constructor(http) {
6
+ this.http = http;
7
+ this._webhookUrl = signal(localStorage.getItem('debugRecorder_webhookUrl') ?? '');
8
+ this._isGenerating = signal(false);
9
+ this._error = signal(null);
10
+ this._lastTest = signal(null);
11
+ this.webhookUrl = this._webhookUrl.asReadonly();
12
+ this.isGenerating = this._isGenerating.asReadonly();
13
+ this.error = this._error.asReadonly();
14
+ this.lastTest = this._lastTest.asReadonly();
15
+ }
16
+ setWebhookUrl(url) {
17
+ this._webhookUrl.set(url);
18
+ localStorage.setItem('debugRecorder_webhookUrl', url);
19
+ }
20
+ async generateCypressTest(session) {
21
+ const url = this._webhookUrl();
22
+ if (!url)
23
+ throw new Error('Keine Webhook-URL konfiguriert');
24
+ this._isGenerating.set(true);
25
+ this._error.set(null);
26
+ try {
27
+ const code = await this.postSession(url, session);
28
+ const test = {
29
+ code,
30
+ generatedAt: Date.now(),
31
+ model: url,
32
+ sessionId: session.id,
33
+ };
34
+ this._lastTest.set(test);
35
+ return test;
36
+ }
37
+ catch (err) {
38
+ const msg = err?.error?.message || err?.message || 'Fehler beim Senden';
39
+ this._error.set(msg);
40
+ throw err;
41
+ }
42
+ finally {
43
+ this._isGenerating.set(false);
44
+ }
45
+ }
46
+ postSession(url, session) {
47
+ return new Promise((resolve, reject) => {
48
+ this.http.post(url, session, { responseType: 'text' }).subscribe({
49
+ next: (res) => resolve(res),
50
+ error: reject,
51
+ });
52
+ });
53
+ }
54
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AiGeneratorService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable }); }
55
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AiGeneratorService, providedIn: 'root' }); }
56
+ }
57
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AiGeneratorService, decorators: [{
58
+ type: Injectable,
59
+ args: [{ providedIn: 'root' }]
60
+ }], ctorParameters: () => [{ type: i1.HttpClient }] });
61
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWktZ2VuZXJhdG9yLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9kZWJ1Zy1yZWNvcmRlci9zcmMvbGliL3NlcnZpY2VzL2FpLWdlbmVyYXRvci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFLbkQsTUFBTSxPQUFPLGtCQUFrQjtJQVc3QixZQUFvQixJQUFnQjtRQUFoQixTQUFJLEdBQUosSUFBSSxDQUFZO1FBVjVCLGdCQUFXLEdBQUcsTUFBTSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsMEJBQTBCLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUM3RSxrQkFBYSxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM5QixXQUFNLEdBQUcsTUFBTSxDQUFnQixJQUFJLENBQUMsQ0FBQztRQUNyQyxjQUFTLEdBQUcsTUFBTSxDQUF1QixJQUFJLENBQUMsQ0FBQztRQUV2RCxlQUFVLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxVQUFVLEVBQUUsQ0FBQztRQUMzQyxpQkFBWSxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsVUFBVSxFQUFFLENBQUM7UUFDL0MsVUFBSyxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxFQUFFLENBQUM7UUFDakMsYUFBUSxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsVUFBVSxFQUFFLENBQUM7SUFFQSxDQUFDO0lBRXhDLGFBQWEsQ0FBQyxHQUFXO1FBQ3ZCLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQzFCLFlBQVksQ0FBQyxPQUFPLENBQUMsMEJBQTBCLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDeEQsQ0FBQztJQUVELEtBQUssQ0FBQyxtQkFBbUIsQ0FBQyxPQUF5QjtRQUNqRCxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDL0IsSUFBSSxDQUFDLEdBQUc7WUFBRSxNQUFNLElBQUksS0FBSyxDQUFDLGdDQUFnQyxDQUFDLENBQUM7UUFFNUQsSUFBSSxDQUFDLGFBQWEsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDN0IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7UUFFdEIsSUFBSSxDQUFDO1lBQ0gsTUFBTSxJQUFJLEdBQUcsTUFBTSxJQUFJLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQztZQUNsRCxNQUFNLElBQUksR0FBa0I7Z0JBQzFCLElBQUk7Z0JBQ0osV0FBVyxFQUFFLElBQUksQ0FBQyxHQUFHLEVBQUU7Z0JBQ3ZCLEtBQUssRUFBRSxHQUFHO2dCQUNWLFNBQVMsRUFBRSxPQUFPLENBQUMsRUFBRTthQUN0QixDQUFDO1lBQ0YsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDekIsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO1FBQUMsT0FBTyxHQUFRLEVBQUUsQ0FBQztZQUNsQixNQUFNLEdBQUcsR0FBRyxHQUFHLEVBQUUsS0FBSyxFQUFFLE9BQU8sSUFBSSxHQUFHLEVBQUUsT0FBTyxJQUFJLG9CQUFvQixDQUFDO1lBQ3hFLElBQUksQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ3JCLE1BQU0sR0FBRyxDQUFDO1FBQ1osQ0FBQztnQkFBUyxDQUFDO1lBQ1QsSUFBSSxDQUFDLGFBQWEsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDaEMsQ0FBQztJQUNILENBQUM7SUFFTyxXQUFXLENBQUMsR0FBVyxFQUFFLE9BQXlCO1FBQ3hELE9BQU8sSUFBSSxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEVBQUU7WUFDckMsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLE9BQU8sRUFBRSxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsQ0FBQyxDQUFDLFNBQVMsQ0FBQztnQkFDL0QsSUFBSSxFQUFFLENBQUMsR0FBRyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDO2dCQUMzQixLQUFLLEVBQUUsTUFBTTthQUNkLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzsrR0FuRFUsa0JBQWtCO21IQUFsQixrQkFBa0IsY0FETCxNQUFNOzs0RkFDbkIsa0JBQWtCO2tCQUQ5QixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUsIHNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBIdHRwQ2xpZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xyXG5pbXBvcnQgeyBHZW5lcmF0ZWRUZXN0LCBSZWNvcmRpbmdTZXNzaW9uIH0gZnJvbSAnLi4vbW9kZWxzL3JlY29yZGVkLWFjdGlvbi5tb2RlbCc7XHJcblxyXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46ICdyb290JyB9KVxyXG5leHBvcnQgY2xhc3MgQWlHZW5lcmF0b3JTZXJ2aWNlIHtcclxuICBwcml2YXRlIF93ZWJob29rVXJsID0gc2lnbmFsKGxvY2FsU3RvcmFnZS5nZXRJdGVtKCdkZWJ1Z1JlY29yZGVyX3dlYmhvb2tVcmwnKSA/PyAnJyk7XHJcbiAgcHJpdmF0ZSBfaXNHZW5lcmF0aW5nID0gc2lnbmFsKGZhbHNlKTtcclxuICBwcml2YXRlIF9lcnJvciA9IHNpZ25hbDxzdHJpbmcgfCBudWxsPihudWxsKTtcclxuICBwcml2YXRlIF9sYXN0VGVzdCA9IHNpZ25hbDxHZW5lcmF0ZWRUZXN0IHwgbnVsbD4obnVsbCk7XHJcblxyXG4gIHdlYmhvb2tVcmwgPSB0aGlzLl93ZWJob29rVXJsLmFzUmVhZG9ubHkoKTtcclxuICBpc0dlbmVyYXRpbmcgPSB0aGlzLl9pc0dlbmVyYXRpbmcuYXNSZWFkb25seSgpO1xyXG4gIGVycm9yID0gdGhpcy5fZXJyb3IuYXNSZWFkb25seSgpO1xyXG4gIGxhc3RUZXN0ID0gdGhpcy5fbGFzdFRlc3QuYXNSZWFkb25seSgpO1xyXG5cclxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIGh0dHA6IEh0dHBDbGllbnQpIHt9XHJcblxyXG4gIHNldFdlYmhvb2tVcmwodXJsOiBzdHJpbmcpOiB2b2lkIHtcclxuICAgIHRoaXMuX3dlYmhvb2tVcmwuc2V0KHVybCk7XHJcbiAgICBsb2NhbFN0b3JhZ2Uuc2V0SXRlbSgnZGVidWdSZWNvcmRlcl93ZWJob29rVXJsJywgdXJsKTtcclxuICB9XHJcblxyXG4gIGFzeW5jIGdlbmVyYXRlQ3lwcmVzc1Rlc3Qoc2Vzc2lvbjogUmVjb3JkaW5nU2Vzc2lvbik6IFByb21pc2U8R2VuZXJhdGVkVGVzdD4ge1xyXG4gICAgY29uc3QgdXJsID0gdGhpcy5fd2ViaG9va1VybCgpO1xyXG4gICAgaWYgKCF1cmwpIHRocm93IG5ldyBFcnJvcignS2VpbmUgV2ViaG9vay1VUkwga29uZmlndXJpZXJ0Jyk7XHJcblxyXG4gICAgdGhpcy5faXNHZW5lcmF0aW5nLnNldCh0cnVlKTtcclxuICAgIHRoaXMuX2Vycm9yLnNldChudWxsKTtcclxuXHJcbiAgICB0cnkge1xyXG4gICAgICBjb25zdCBjb2RlID0gYXdhaXQgdGhpcy5wb3N0U2Vzc2lvbih1cmwsIHNlc3Npb24pO1xyXG4gICAgICBjb25zdCB0ZXN0OiBHZW5lcmF0ZWRUZXN0ID0ge1xyXG4gICAgICAgIGNvZGUsXHJcbiAgICAgICAgZ2VuZXJhdGVkQXQ6IERhdGUubm93KCksXHJcbiAgICAgICAgbW9kZWw6IHVybCxcclxuICAgICAgICBzZXNzaW9uSWQ6IHNlc3Npb24uaWQsXHJcbiAgICAgIH07XHJcbiAgICAgIHRoaXMuX2xhc3RUZXN0LnNldCh0ZXN0KTtcclxuICAgICAgcmV0dXJuIHRlc3Q7XHJcbiAgICB9IGNhdGNoIChlcnI6IGFueSkge1xyXG4gICAgICBjb25zdCBtc2cgPSBlcnI/LmVycm9yPy5tZXNzYWdlIHx8IGVycj8ubWVzc2FnZSB8fCAnRmVobGVyIGJlaW0gU2VuZGVuJztcclxuICAgICAgdGhpcy5fZXJyb3Iuc2V0KG1zZyk7XHJcbiAgICAgIHRocm93IGVycjtcclxuICAgIH0gZmluYWxseSB7XHJcbiAgICAgIHRoaXMuX2lzR2VuZXJhdGluZy5zZXQoZmFsc2UpO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBwb3N0U2Vzc2lvbih1cmw6IHN0cmluZywgc2Vzc2lvbjogUmVjb3JkaW5nU2Vzc2lvbik6IFByb21pc2U8c3RyaW5nPiB7XHJcbiAgICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xyXG4gICAgICB0aGlzLmh0dHAucG9zdCh1cmwsIHNlc3Npb24sIHsgcmVzcG9uc2VUeXBlOiAndGV4dCcgfSkuc3Vic2NyaWJlKHtcclxuICAgICAgICBuZXh0OiAocmVzKSA9PiByZXNvbHZlKHJlcyksXHJcbiAgICAgICAgZXJyb3I6IHJlamVjdCxcclxuICAgICAgfSk7XHJcbiAgICB9KTtcclxuICB9XHJcbn1cclxuIl19
@@ -0,0 +1,354 @@
1
+ import { Injectable, signal, computed } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class RecorderService {
4
+ constructor(zone) {
5
+ this.zone = zone;
6
+ this._isRecording = signal(false);
7
+ this._currentSession = signal(null);
8
+ this._sessions = signal([]);
9
+ this._isPaused = signal(false);
10
+ this.isRecording = computed(() => this._isRecording());
11
+ this.isPaused = computed(() => this._isPaused());
12
+ this.currentSession = computed(() => this._currentSession());
13
+ this.sessions = computed(() => this._sessions());
14
+ this.actionCount = computed(() => this._currentSession()?.actions.length ?? 0);
15
+ this.listeners = [];
16
+ this.lastScrollTime = 0;
17
+ }
18
+ startRecording(name, description) {
19
+ const session = {
20
+ id: this.generateId(),
21
+ name: name || `Session ${new Date().toLocaleTimeString()}`,
22
+ description,
23
+ startTime: Date.now(),
24
+ startUrl: window.location.href,
25
+ actions: [],
26
+ tags: [],
27
+ };
28
+ this._currentSession.set(session);
29
+ this._isRecording.set(true);
30
+ this._isPaused.set(false);
31
+ this.attachListeners();
32
+ this.recordNavigation('navigation', window.location.href);
33
+ }
34
+ stopRecording() {
35
+ const session = this._currentSession();
36
+ if (!session)
37
+ return null;
38
+ const completed = { ...session, endTime: Date.now() };
39
+ this._sessions.update(s => [...s, completed]);
40
+ this._currentSession.set(null);
41
+ this._isRecording.set(false);
42
+ this._isPaused.set(false);
43
+ this.detachListeners();
44
+ return completed;
45
+ }
46
+ pauseRecording() {
47
+ if (this._isRecording()) {
48
+ this._isPaused.set(!this._isPaused());
49
+ }
50
+ }
51
+ clearCurrentSession() {
52
+ this._currentSession.update(s => s ? { ...s, actions: [] } : null);
53
+ }
54
+ removeAction(actionId) {
55
+ this._currentSession.update(s => s ? { ...s, actions: s.actions.filter(a => a.id !== actionId) } : null);
56
+ }
57
+ addNote(actionId, note) {
58
+ this._currentSession.update(s => s ? {
59
+ ...s,
60
+ actions: s.actions.map(a => a.id === actionId ? { ...a, note } : a)
61
+ } : null);
62
+ }
63
+ deleteSession(sessionId) {
64
+ this._sessions.update(s => s.filter(x => x.id !== sessionId));
65
+ }
66
+ loadSession(session) {
67
+ this._currentSession.set(session);
68
+ }
69
+ // ─── Event Listeners ──────────────────────────────────────────────────────
70
+ attachListeners() {
71
+ const opts = { capture: true, passive: true };
72
+ const onClick = (e) => this.zone.run(() => this.handleClick(e));
73
+ const onDblClick = (e) => this.zone.run(() => this.handleDblClick(e));
74
+ const onInput = (e) => this.zone.run(() => this.handleInput(e));
75
+ const onChange = (e) => this.zone.run(() => this.handleChange(e));
76
+ const onSubmit = (e) => this.zone.run(() => this.handleSubmit(e));
77
+ const onKeydown = (e) => this.zone.run(() => this.handleKeydown(e));
78
+ const onScroll = (e) => this.zone.run(() => this.handleScroll(e));
79
+ document.addEventListener('click', onClick, opts);
80
+ document.addEventListener('dblclick', onDblClick, opts);
81
+ document.addEventListener('input', onInput, opts);
82
+ document.addEventListener('change', onChange, opts);
83
+ document.addEventListener('submit', onSubmit, opts);
84
+ document.addEventListener('keydown', onKeydown, opts);
85
+ document.addEventListener('scroll', onScroll, { capture: true, passive: true });
86
+ this.listeners = [
87
+ { type: 'click', fn: onClick },
88
+ { type: 'dblclick', fn: onDblClick },
89
+ { type: 'input', fn: onInput },
90
+ { type: 'change', fn: onChange },
91
+ { type: 'submit', fn: onSubmit },
92
+ { type: 'keydown', fn: onKeydown },
93
+ { type: 'scroll', fn: onScroll },
94
+ ];
95
+ }
96
+ detachListeners() {
97
+ this.listeners.forEach(({ type, fn }) => document.removeEventListener(type, fn, true));
98
+ this.listeners = [];
99
+ this.mutationObserver?.disconnect();
100
+ }
101
+ // ─── Handlers ─────────────────────────────────────────────────────────────
102
+ handleClick(e) {
103
+ if (!this.shouldRecord(e.target))
104
+ return;
105
+ const el = e.target;
106
+ const info = this.getElementInfo(el);
107
+ const selector = this.buildSelector(el);
108
+ this.addAction({
109
+ type: 'click',
110
+ selector,
111
+ selectorStrategy: this.getSelectorStrategy(el),
112
+ element: info,
113
+ description: `Click on ${this.describeElement(info)}`,
114
+ });
115
+ }
116
+ handleDblClick(e) {
117
+ if (!this.shouldRecord(e.target))
118
+ return;
119
+ const el = e.target;
120
+ const info = this.getElementInfo(el);
121
+ const selector = this.buildSelector(el);
122
+ this.addAction({
123
+ type: 'dblclick',
124
+ selector,
125
+ selectorStrategy: this.getSelectorStrategy(el),
126
+ element: info,
127
+ description: `Double-click on ${this.describeElement(info)}`,
128
+ });
129
+ }
130
+ handleInput(e) {
131
+ if (!this.shouldRecord(e.target))
132
+ return;
133
+ const el = e.target;
134
+ if (['checkbox', 'radio'].includes(el.type))
135
+ return; // handled by change
136
+ const info = this.getElementInfo(el);
137
+ const selector = this.buildSelector(el);
138
+ this.addAction({
139
+ type: 'input',
140
+ selector,
141
+ selectorStrategy: this.getSelectorStrategy(el),
142
+ element: info,
143
+ value: el.value,
144
+ description: `Type "${el.value}" in ${this.describeElement(info)}`,
145
+ });
146
+ }
147
+ handleChange(e) {
148
+ if (!this.shouldRecord(e.target))
149
+ return;
150
+ const el = e.target;
151
+ const info = this.getElementInfo(el);
152
+ const selector = this.buildSelector(el);
153
+ if (el.tagName === 'SELECT') {
154
+ this.addAction({
155
+ type: 'select',
156
+ selector,
157
+ selectorStrategy: this.getSelectorStrategy(el),
158
+ element: info,
159
+ value: el.value,
160
+ description: `Select "${el.value}" in ${this.describeElement(info)}`,
161
+ });
162
+ }
163
+ else if (el.type === 'checkbox') {
164
+ this.addAction({
165
+ type: 'click',
166
+ selector,
167
+ selectorStrategy: this.getSelectorStrategy(el),
168
+ element: info,
169
+ value: String(el.checked),
170
+ description: `${el.checked ? 'Check' : 'Uncheck'} ${this.describeElement(info)}`,
171
+ });
172
+ }
173
+ }
174
+ handleSubmit(e) {
175
+ if (!this.shouldRecord(e.target))
176
+ return;
177
+ const form = e.target;
178
+ const info = this.getElementInfo(form);
179
+ const selector = this.buildSelector(form);
180
+ this.addAction({
181
+ type: 'submit',
182
+ selector,
183
+ selectorStrategy: this.getSelectorStrategy(form),
184
+ element: info,
185
+ description: `Submit form ${info.id ? '#' + info.id : info.name ? info.name : ''}`.trim(),
186
+ });
187
+ }
188
+ handleScroll(e) {
189
+ if (!this.shouldRecord(e.target))
190
+ return;
191
+ const now = Date.now();
192
+ if (now - this.lastScrollTime < 1000)
193
+ return; // debounce 1s
194
+ this.lastScrollTime = now;
195
+ const el = e.target;
196
+ const isDoc = !el.tagName || el.tagName === 'HTML';
197
+ const scrollY = isDoc ? window.scrollY : el.scrollTop;
198
+ const scrollX = isDoc ? window.scrollX : el.scrollLeft;
199
+ const selector = el.tagName === 'HTML' || el.tagName === 'BODY'
200
+ ? 'window'
201
+ : this.buildSelector(el);
202
+ this.addAction({
203
+ type: 'scroll',
204
+ selector,
205
+ selectorStrategy: 'combined',
206
+ value: `${scrollX},${scrollY}`,
207
+ description: `Scroll to (${scrollX}, ${scrollY})`,
208
+ });
209
+ }
210
+ handleKeydown(e) {
211
+ const specialKeys = ['Enter', 'Escape', 'Tab', 'F5', 'F12'];
212
+ if (!specialKeys.includes(e.key))
213
+ return;
214
+ if (!this.shouldRecord(e.target))
215
+ return;
216
+ const el = e.target;
217
+ const selector = this.buildSelector(el);
218
+ this.addAction({
219
+ type: 'keypress',
220
+ selector,
221
+ selectorStrategy: this.getSelectorStrategy(el),
222
+ value: e.key,
223
+ description: `Press "${e.key}" on ${el.tagName.toLowerCase()}`,
224
+ });
225
+ }
226
+ recordNavigation(type, navUrl) {
227
+ const action = {
228
+ id: this.generateId(),
229
+ timestamp: Date.now(),
230
+ url: navUrl,
231
+ type,
232
+ selector: 'window',
233
+ selectorStrategy: 'combined',
234
+ description: `Navigate to ${navUrl}`,
235
+ };
236
+ this._currentSession.update(s => s ? { ...s, actions: [...s.actions, action] } : s);
237
+ }
238
+ // ─── Selector Building ────────────────────────────────────────────────────
239
+ buildSelector(el) {
240
+ // Priority: data-testid > data-cy > id > name > aria-label > combined
241
+ const testId = el.getAttribute('data-testid');
242
+ if (testId)
243
+ return `[data-testid="${testId}"]`;
244
+ const cy = el.getAttribute('data-cy');
245
+ if (cy)
246
+ return `[data-cy="${cy}"]`;
247
+ if (el.id && !el.id.includes(':'))
248
+ return `#${el.id}`;
249
+ const name = el.getAttribute('name');
250
+ if (name)
251
+ return `${el.tagName.toLowerCase()}[name="${name}"]`;
252
+ const ariaLabel = el.getAttribute('aria-label');
253
+ if (ariaLabel)
254
+ return `[aria-label="${ariaLabel}"]`;
255
+ // Class-based fallback
256
+ const relevantClasses = Array.from(el.classList)
257
+ .filter(c => !c.startsWith('ng-') && !c.startsWith('cdk-') && c.length > 0)
258
+ .slice(0, 3);
259
+ if (relevantClasses.length > 0) {
260
+ return `${el.tagName.toLowerCase()}.${relevantClasses.join('.')}`;
261
+ }
262
+ // Text content for buttons/links
263
+ if (['BUTTON', 'A'].includes(el.tagName)) {
264
+ const text = el.textContent?.trim().slice(0, 30);
265
+ if (text)
266
+ return `${el.tagName.toLowerCase()}:contains("${text}")`;
267
+ }
268
+ return el.tagName.toLowerCase();
269
+ }
270
+ getSelectorStrategy(el) {
271
+ if (el.getAttribute('data-testid'))
272
+ return 'data-testid';
273
+ if (el.getAttribute('data-cy'))
274
+ return 'data-cy';
275
+ if (el.id)
276
+ return 'id';
277
+ if (el.getAttribute('name'))
278
+ return 'name';
279
+ return 'class';
280
+ }
281
+ getElementInfo(el) {
282
+ return {
283
+ tagName: el.tagName.toLowerCase(),
284
+ id: el.id || undefined,
285
+ classes: Array.from(el.classList).filter(c => !c.startsWith('ng-')),
286
+ dataTestId: el.getAttribute('data-testid') || undefined,
287
+ dataCy: el.getAttribute('data-cy') || undefined,
288
+ name: el.getAttribute('name') || undefined,
289
+ type: el.getAttribute('type') || undefined,
290
+ placeholder: el.getAttribute('placeholder') || undefined,
291
+ text: el.textContent?.trim().slice(0, 50) || undefined,
292
+ href: el.href || undefined,
293
+ ariaLabel: el.getAttribute('aria-label') || undefined,
294
+ };
295
+ }
296
+ describeElement(info) {
297
+ if (info.dataTestId)
298
+ return `[data-testid="${info.dataTestId}"]`;
299
+ if (info.dataCy)
300
+ return `[data-cy="${info.dataCy}"]`;
301
+ if (info.id)
302
+ return `#${info.id}`;
303
+ if (info.ariaLabel)
304
+ return `"${info.ariaLabel}"`;
305
+ if (info.placeholder)
306
+ return `"${info.placeholder}" input`;
307
+ if (info.text)
308
+ return `"${info.text}"`;
309
+ return info.tagName;
310
+ }
311
+ // ─── Helpers ──────────────────────────────────────────────────────────────
312
+ shouldRecord(target) {
313
+ if (!this._isRecording() || this._isPaused())
314
+ return false;
315
+ if (!target)
316
+ return false;
317
+ // Ignore the debug panel itself
318
+ if (target.closest('[data-debug-panel]'))
319
+ return false;
320
+ return true;
321
+ }
322
+ addAction(partial) {
323
+ const action = {
324
+ id: this.generateId(),
325
+ timestamp: Date.now(),
326
+ url: window.location.href,
327
+ ...partial,
328
+ };
329
+ // Deduplicate consecutive identical inputs (keep only latest value)
330
+ if (action.type === 'input') {
331
+ this._currentSession.update(s => {
332
+ if (!s)
333
+ return s;
334
+ const last = s.actions[s.actions.length - 1];
335
+ if (last?.type === 'input' && last.selector === action.selector) {
336
+ return { ...s, actions: [...s.actions.slice(0, -1), action] };
337
+ }
338
+ return { ...s, actions: [...s.actions, action] };
339
+ });
340
+ return;
341
+ }
342
+ this._currentSession.update(s => s ? { ...s, actions: [...s.actions, action] } : s);
343
+ }
344
+ generateId() {
345
+ return Math.random().toString(36).slice(2, 11);
346
+ }
347
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecorderService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable }); }
348
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecorderService, providedIn: 'root' }); }
349
+ }
350
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecorderService, decorators: [{
351
+ type: Injectable,
352
+ args: [{ providedIn: 'root' }]
353
+ }], ctorParameters: () => [{ type: i0.NgZone }] });
354
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recorder.service.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/services/recorder.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, signal, computed } from '@angular/core';\r\nimport { ActionType, ElementInfo, RecordedAction, RecordingSession } from '../models/recorded-action.model';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class RecorderService {\r\n  private _isRecording = signal(false);\r\n  private _currentSession = signal<RecordingSession | null>(null);\r\n  private _sessions = signal<RecordingSession[]>([]);\r\n  private _isPaused = signal(false);\r\n\r\n  isRecording = computed(() => this._isRecording());\r\n  isPaused = computed(() => this._isPaused());\r\n  currentSession = computed(() => this._currentSession());\r\n  sessions = computed(() => this._sessions());\r\n  actionCount = computed(() => this._currentSession()?.actions.length ?? 0);\r\n\r\n  private listeners: Array<{ type: string; fn: EventListener }> = [];\r\n  private mutationObserver?: MutationObserver;\r\n\r\n  constructor(private zone: NgZone) {}\r\n\r\n  startRecording(name?: string, description?: string): void {\r\n    const session: RecordingSession = {\r\n      id: this.generateId(),\r\n      name: name || `Session ${new Date().toLocaleTimeString()}`,\r\n      description,\r\n      startTime: Date.now(),\r\n      startUrl: window.location.href,\r\n      actions: [],\r\n      tags: [],\r\n    };\r\n\r\n    this._currentSession.set(session);\r\n    this._isRecording.set(true);\r\n    this._isPaused.set(false);\r\n    this.attachListeners();\r\n    this.recordNavigation('navigation', window.location.href);\r\n  }\r\n\r\n  stopRecording(): RecordingSession | null {\r\n    const session = this._currentSession();\r\n    if (!session) return null;\r\n\r\n    const completed = { ...session, endTime: Date.now() };\r\n    this._sessions.update(s => [...s, completed]);\r\n    this._currentSession.set(null);\r\n    this._isRecording.set(false);\r\n    this._isPaused.set(false);\r\n    this.detachListeners();\r\n    return completed;\r\n  }\r\n\r\n  pauseRecording(): void {\r\n    if (this._isRecording()) {\r\n      this._isPaused.set(!this._isPaused());\r\n    }\r\n  }\r\n\r\n  clearCurrentSession(): void {\r\n    this._currentSession.update(s => s ? { ...s, actions: [] } : null);\r\n  }\r\n\r\n  removeAction(actionId: string): void {\r\n    this._currentSession.update(s =>\r\n      s ? { ...s, actions: s.actions.filter(a => a.id !== actionId) } : null\r\n    );\r\n  }\r\n\r\n  addNote(actionId: string, note: string): void {\r\n    this._currentSession.update(s =>\r\n      s ? {\r\n        ...s,\r\n        actions: s.actions.map(a => a.id === actionId ? { ...a, note } : a)\r\n      } : null\r\n    );\r\n  }\r\n\r\n  deleteSession(sessionId: string): void {\r\n    this._sessions.update(s => s.filter(x => x.id !== sessionId));\r\n  }\r\n\r\n  loadSession(session: RecordingSession): void {\r\n    this._currentSession.set(session);\r\n  }\r\n\r\n  // ─── Event Listeners ──────────────────────────────────────────────────────\r\n\r\n  private attachListeners(): void {\r\n    const opts = { capture: true, passive: true };\r\n\r\n    const onClick = (e: MouseEvent) => this.zone.run(() => this.handleClick(e));\r\n    const onDblClick = (e: MouseEvent) => this.zone.run(() => this.handleDblClick(e));\r\n    const onInput = (e: Event) => this.zone.run(() => this.handleInput(e));\r\n    const onChange = (e: Event) => this.zone.run(() => this.handleChange(e));\r\n    const onSubmit = (e: SubmitEvent) => this.zone.run(() => this.handleSubmit(e));\r\n    const onKeydown = (e: KeyboardEvent) => this.zone.run(() => this.handleKeydown(e));\r\n    const onScroll = (e: Event) => this.zone.run(() => this.handleScroll(e));\r\n\r\n    document.addEventListener('click', onClick, opts);\r\n    document.addEventListener('dblclick', onDblClick, opts);\r\n    document.addEventListener('input', onInput, opts);\r\n    document.addEventListener('change', onChange, opts);\r\n    document.addEventListener('submit', onSubmit, opts);\r\n    document.addEventListener('keydown', onKeydown, opts);\r\n    document.addEventListener('scroll', onScroll, { capture: true, passive: true });\r\n\r\n    this.listeners = [\r\n      { type: 'click', fn: onClick as EventListener },\r\n      { type: 'dblclick', fn: onDblClick as EventListener },\r\n      { type: 'input', fn: onInput as EventListener },\r\n      { type: 'change', fn: onChange as EventListener },\r\n      { type: 'submit', fn: onSubmit as EventListener },\r\n      { type: 'keydown', fn: onKeydown as EventListener },\r\n      { type: 'scroll', fn: onScroll as EventListener },\r\n    ];\r\n  }\r\n\r\n  private detachListeners(): void {\r\n    this.listeners.forEach(({ type, fn }) =>\r\n      document.removeEventListener(type, fn, true)\r\n    );\r\n    this.listeners = [];\r\n    this.mutationObserver?.disconnect();\r\n  }\r\n\r\n  // ─── Handlers ─────────────────────────────────────────────────────────────\r\n\r\n  private handleClick(e: MouseEvent): void {\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n    const el = e.target as HTMLElement;\r\n    const info = this.getElementInfo(el);\r\n    const selector = this.buildSelector(el);\r\n\r\n    this.addAction({\r\n      type: 'click',\r\n      selector,\r\n      selectorStrategy: this.getSelectorStrategy(el),\r\n      element: info,\r\n      description: `Click on ${this.describeElement(info)}`,\r\n    });\r\n  }\r\n\r\n  private handleDblClick(e: MouseEvent): void {\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n    const el = e.target as HTMLElement;\r\n    const info = this.getElementInfo(el);\r\n    const selector = this.buildSelector(el);\r\n\r\n    this.addAction({\r\n      type: 'dblclick',\r\n      selector,\r\n      selectorStrategy: this.getSelectorStrategy(el),\r\n      element: info,\r\n      description: `Double-click on ${this.describeElement(info)}`,\r\n    });\r\n  }\r\n\r\n  private handleInput(e: Event): void {\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n    const el = e.target as HTMLInputElement | HTMLTextAreaElement;\r\n    if (['checkbox', 'radio'].includes(el.type)) return; // handled by change\r\n    const info = this.getElementInfo(el);\r\n    const selector = this.buildSelector(el);\r\n\r\n    this.addAction({\r\n      type: 'input',\r\n      selector,\r\n      selectorStrategy: this.getSelectorStrategy(el),\r\n      element: info,\r\n      value: el.value,\r\n      description: `Type \"${el.value}\" in ${this.describeElement(info)}`,\r\n    });\r\n  }\r\n\r\n  private handleChange(e: Event): void {\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n    const el = e.target as HTMLSelectElement | HTMLInputElement;\r\n    const info = this.getElementInfo(el);\r\n    const selector = this.buildSelector(el);\r\n\r\n    if (el.tagName === 'SELECT') {\r\n      this.addAction({\r\n        type: 'select',\r\n        selector,\r\n        selectorStrategy: this.getSelectorStrategy(el),\r\n        element: info,\r\n        value: el.value,\r\n        description: `Select \"${el.value}\" in ${this.describeElement(info)}`,\r\n      });\r\n    } else if ((el as HTMLInputElement).type === 'checkbox') {\r\n      this.addAction({\r\n        type: 'click',\r\n        selector,\r\n        selectorStrategy: this.getSelectorStrategy(el),\r\n        element: info,\r\n        value: String((el as HTMLInputElement).checked),\r\n        description: `${(el as HTMLInputElement).checked ? 'Check' : 'Uncheck'} ${this.describeElement(info)}`,\r\n      });\r\n    }\r\n  }\r\n\r\n  private handleSubmit(e: SubmitEvent): void {\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n    const form = e.target as HTMLFormElement;\r\n    const info = this.getElementInfo(form);\r\n    const selector = this.buildSelector(form);\r\n\r\n    this.addAction({\r\n      type: 'submit',\r\n      selector,\r\n      selectorStrategy: this.getSelectorStrategy(form),\r\n      element: info,\r\n      description: `Submit form ${info.id ? '#' + info.id : info.name ? info.name : ''}`.trim(),\r\n    });\r\n  }\r\n\r\n  private lastScrollTime = 0;\r\n  private handleScroll(e: Event): void {\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n    const now = Date.now();\r\n    if (now - this.lastScrollTime < 1000) return; // debounce 1s\r\n    this.lastScrollTime = now;\r\n\r\n    const el = e.target as HTMLElement;\r\n    const isDoc = !el.tagName || el.tagName === 'HTML';\r\n    const scrollY = isDoc ? window.scrollY : el.scrollTop;\r\n    const scrollX = isDoc ? window.scrollX : el.scrollLeft;\r\n\r\n    const selector = el.tagName === 'HTML' || el.tagName === 'BODY'\r\n      ? 'window'\r\n      : this.buildSelector(el);\r\n\r\n    this.addAction({\r\n      type: 'scroll',\r\n      selector,\r\n      selectorStrategy: 'combined',\r\n      value: `${scrollX},${scrollY}`,\r\n      description: `Scroll to (${scrollX}, ${scrollY})`,\r\n    });\r\n  }\r\n\r\n  private handleKeydown(e: KeyboardEvent): void {\r\n    const specialKeys = ['Enter', 'Escape', 'Tab', 'F5', 'F12'];\r\n    if (!specialKeys.includes(e.key)) return;\r\n    if (!this.shouldRecord(e.target as Element)) return;\r\n\r\n    const el = e.target as HTMLElement;\r\n    const selector = this.buildSelector(el);\r\n\r\n    this.addAction({\r\n      type: 'keypress',\r\n      selector,\r\n      selectorStrategy: this.getSelectorStrategy(el),\r\n      value: e.key,\r\n      description: `Press \"${e.key}\" on ${el.tagName.toLowerCase()}`,\r\n    });\r\n  }\r\n\r\n  private recordNavigation(type: ActionType, navUrl: string): void {\r\n    const action: RecordedAction = {\r\n      id: this.generateId(),\r\n      timestamp: Date.now(),\r\n      url: navUrl,\r\n      type,\r\n      selector: 'window',\r\n      selectorStrategy: 'combined',\r\n      description: `Navigate to ${navUrl}`,\r\n    };\r\n    this._currentSession.update(s =>\r\n      s ? { ...s, actions: [...s.actions, action] } : s\r\n    );\r\n  }\r\n\r\n  // ─── Selector Building ────────────────────────────────────────────────────\r\n\r\n  private buildSelector(el: HTMLElement): string {\r\n    // Priority: data-testid > data-cy > id > name > aria-label > combined\r\n    const testId = el.getAttribute('data-testid');\r\n    if (testId) return `[data-testid=\"${testId}\"]`;\r\n\r\n    const cy = el.getAttribute('data-cy');\r\n    if (cy) return `[data-cy=\"${cy}\"]`;\r\n\r\n    if (el.id && !el.id.includes(':')) return `#${el.id}`;\r\n\r\n    const name = el.getAttribute('name');\r\n    if (name) return `${el.tagName.toLowerCase()}[name=\"${name}\"]`;\r\n\r\n    const ariaLabel = el.getAttribute('aria-label');\r\n    if (ariaLabel) return `[aria-label=\"${ariaLabel}\"]`;\r\n\r\n    // Class-based fallback\r\n    const relevantClasses = Array.from(el.classList)\r\n      .filter(c => !c.startsWith('ng-') && !c.startsWith('cdk-') && c.length > 0)\r\n      .slice(0, 3);\r\n    if (relevantClasses.length > 0) {\r\n      return `${el.tagName.toLowerCase()}.${relevantClasses.join('.')}`;\r\n    }\r\n\r\n    // Text content for buttons/links\r\n    if (['BUTTON', 'A'].includes(el.tagName)) {\r\n      const text = el.textContent?.trim().slice(0, 30);\r\n      if (text) return `${el.tagName.toLowerCase()}:contains(\"${text}\")`;\r\n    }\r\n\r\n    return el.tagName.toLowerCase();\r\n  }\r\n\r\n  private getSelectorStrategy(el: HTMLElement): RecordedAction['selectorStrategy'] {\r\n    if (el.getAttribute('data-testid')) return 'data-testid';\r\n    if (el.getAttribute('data-cy')) return 'data-cy';\r\n    if (el.id) return 'id';\r\n    if (el.getAttribute('name')) return 'name';\r\n    return 'class';\r\n  }\r\n\r\n  private getElementInfo(el: HTMLElement): ElementInfo {\r\n    return {\r\n      tagName: el.tagName.toLowerCase(),\r\n      id: el.id || undefined,\r\n      classes: Array.from(el.classList).filter(c => !c.startsWith('ng-')),\r\n      dataTestId: el.getAttribute('data-testid') || undefined,\r\n      dataCy: el.getAttribute('data-cy') || undefined,\r\n      name: el.getAttribute('name') || undefined,\r\n      type: el.getAttribute('type') || undefined,\r\n      placeholder: el.getAttribute('placeholder') || undefined,\r\n      text: el.textContent?.trim().slice(0, 50) || undefined,\r\n      href: (el as HTMLAnchorElement).href || undefined,\r\n      ariaLabel: el.getAttribute('aria-label') || undefined,\r\n    };\r\n  }\r\n\r\n  private describeElement(info: ElementInfo): string {\r\n    if (info.dataTestId) return `[data-testid=\"${info.dataTestId}\"]`;\r\n    if (info.dataCy) return `[data-cy=\"${info.dataCy}\"]`;\r\n    if (info.id) return `#${info.id}`;\r\n    if (info.ariaLabel) return `\"${info.ariaLabel}\"`;\r\n    if (info.placeholder) return `\"${info.placeholder}\" input`;\r\n    if (info.text) return `\"${info.text}\"`;\r\n    return info.tagName;\r\n  }\r\n\r\n  // ─── Helpers ──────────────────────────────────────────────────────────────\r\n\r\n  private shouldRecord(target: Element | null): boolean {\r\n    if (!this._isRecording() || this._isPaused()) return false;\r\n    if (!target) return false;\r\n    // Ignore the debug panel itself\r\n    if (target.closest('[data-debug-panel]')) return false;\r\n    return true;\r\n  }\r\n\r\n  private addAction(partial: Omit<RecordedAction, 'id' | 'timestamp' | 'url'>): void {\r\n    const action: RecordedAction = {\r\n      id: this.generateId(),\r\n      timestamp: Date.now(),\r\n      url: window.location.href,\r\n      ...partial,\r\n    };\r\n\r\n    // Deduplicate consecutive identical inputs (keep only latest value)\r\n    if (action.type === 'input') {\r\n      this._currentSession.update(s => {\r\n        if (!s) return s;\r\n        const last = s.actions[s.actions.length - 1];\r\n        if (last?.type === 'input' && last.selector === action.selector) {\r\n          return { ...s, actions: [...s.actions.slice(0, -1), action] };\r\n        }\r\n        return { ...s, actions: [...s.actions, action] };\r\n      });\r\n      return;\r\n    }\r\n\r\n    this._currentSession.update(s =>\r\n      s ? { ...s, actions: [...s.actions, action] } : s\r\n    );\r\n  }\r\n\r\n  private generateId(): string {\r\n    return Math.random().toString(36).slice(2, 11);\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,108 @@
1
+ import { Injectable, signal } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class RrwebRecorderService {
4
+ constructor(zone) {
5
+ this.zone = zone;
6
+ this._events = signal([]);
7
+ this._isRecording = signal(false);
8
+ this.events = this._events.asReadonly();
9
+ this.isRecording = this._isRecording.asReadonly();
10
+ }
11
+ async startRecording() {
12
+ // Dynamically import rrweb to avoid SSR issues and reduce initial bundle
13
+ const { record } = await import('rrweb');
14
+ this._events.set([]);
15
+ this._isRecording.set(true);
16
+ this.stopFn = record({
17
+ emit: (event) => {
18
+ this.zone.run(() => {
19
+ this._events.update(ev => [...ev, event]);
20
+ });
21
+ },
22
+ // Note: blockSelector is omitted — rrweb 2.0.0-alpha.4 calls node.matches()
23
+ // on TextNodes/CommentNodes which don't have that method, crashing the recorder.
24
+ maskTextSelector: 'input[type="password"]',
25
+ checkoutEveryNth: 200,
26
+ });
27
+ }
28
+ stopRecording() {
29
+ if (this.stopFn) {
30
+ this.stopFn();
31
+ this.stopFn = undefined;
32
+ }
33
+ this._isRecording.set(false);
34
+ return this._events();
35
+ }
36
+ getEvents() {
37
+ return this._events();
38
+ }
39
+ clearEvents() {
40
+ this._events.set([]);
41
+ }
42
+ hasEvents() {
43
+ return this._events().length > 0;
44
+ }
45
+ // ─── Replay ──────────────────────────────────────────────────────────────
46
+ async startReplay(container, events) {
47
+ const { Replayer } = await import('rrweb');
48
+ const eventsToReplay = events ?? this._events();
49
+ if (eventsToReplay.length === 0)
50
+ return;
51
+ // Destroy previous replayer
52
+ this.destroyReplayer();
53
+ this.replayer = new Replayer(eventsToReplay, {
54
+ root: container,
55
+ skipInactive: true,
56
+ showWarning: false,
57
+ showDebug: false,
58
+ blockClass: 'debug-panel',
59
+ });
60
+ this.replayer.play();
61
+ }
62
+ pauseReplay() {
63
+ this.replayer?.pause();
64
+ }
65
+ resumeReplay() {
66
+ this.replayer?.play();
67
+ }
68
+ destroyReplayer() {
69
+ if (this.replayer) {
70
+ try {
71
+ this.replayer.pause();
72
+ }
73
+ catch { }
74
+ this.replayer = undefined;
75
+ }
76
+ }
77
+ // ─── Export ───────────────────────────────────────────────────────────────
78
+ exportEvents() {
79
+ return JSON.stringify(this._events(), null, 2);
80
+ }
81
+ downloadEvents(filename = 'rrweb-session.json') {
82
+ const blob = new Blob([this.exportEvents()], { type: 'application/json' });
83
+ const url = URL.createObjectURL(blob);
84
+ const a = document.createElement('a');
85
+ a.href = url;
86
+ a.download = filename;
87
+ a.click();
88
+ URL.revokeObjectURL(url);
89
+ }
90
+ importEvents(json) {
91
+ try {
92
+ const events = JSON.parse(json);
93
+ this._events.set(events);
94
+ return events;
95
+ }
96
+ catch {
97
+ console.error('Invalid rrweb events JSON');
98
+ return [];
99
+ }
100
+ }
101
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RrwebRecorderService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable }); }
102
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RrwebRecorderService, providedIn: 'root' }); }
103
+ }
104
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RrwebRecorderService, decorators: [{
105
+ type: Injectable,
106
+ args: [{ providedIn: 'root' }]
107
+ }], ctorParameters: () => [{ type: i0.NgZone }] });
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rrweb-recorder.service.js","sourceRoot":"","sources":["../../../../../projects/debug-recorder/src/lib/services/rrweb-recorder.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, signal } from '@angular/core';\r\nimport type { eventWithTime } from '@rrweb/types';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class RrwebRecorderService {\r\n  private _events = signal<eventWithTime[]>([]);\r\n  private _isRecording = signal(false);\r\n  private stopFn?: () => void;\r\n  private replayer?: any;\r\n\r\n  events = this._events.asReadonly();\r\n  isRecording = this._isRecording.asReadonly();\r\n\r\n  constructor(private zone: NgZone) {}\r\n\r\n  async startRecording(): Promise<void> {\r\n    // Dynamically import rrweb to avoid SSR issues and reduce initial bundle\r\n    const { record } = await import('rrweb');\r\n    this._events.set([]);\r\n    this._isRecording.set(true);\r\n\r\n    this.stopFn = record({\r\n      emit: (event: eventWithTime) => {\r\n        this.zone.run(() => {\r\n          this._events.update(ev => [...ev, event]);\r\n        });\r\n      },\r\n      // Note: blockSelector is omitted — rrweb 2.0.0-alpha.4 calls node.matches()\r\n      // on TextNodes/CommentNodes which don't have that method, crashing the recorder.\r\n      maskTextSelector: 'input[type=\"password\"]',\r\n      checkoutEveryNth: 200,\r\n    });\r\n  }\r\n\r\n  stopRecording(): eventWithTime[] {\r\n    if (this.stopFn) {\r\n      this.stopFn();\r\n      this.stopFn = undefined;\r\n    }\r\n    this._isRecording.set(false);\r\n    return this._events();\r\n  }\r\n\r\n  getEvents(): eventWithTime[] {\r\n    return this._events();\r\n  }\r\n\r\n  clearEvents(): void {\r\n    this._events.set([]);\r\n  }\r\n\r\n  hasEvents(): boolean {\r\n    return this._events().length > 0;\r\n  }\r\n\r\n  // ─── Replay ──────────────────────────────────────────────────────────────\r\n\r\n  async startReplay(container: HTMLElement, events?: eventWithTime[]): Promise<void> {\r\n    const { Replayer } = await import('rrweb');\r\n    const eventsToReplay = events ?? this._events();\r\n\r\n    if (eventsToReplay.length === 0) return;\r\n\r\n    // Destroy previous replayer\r\n    this.destroyReplayer();\r\n\r\n    this.replayer = new Replayer(eventsToReplay, {\r\n      root: container,\r\n      skipInactive: true,\r\n      showWarning: false,\r\n      showDebug: false,\r\n      blockClass: 'debug-panel',\r\n    });\r\n\r\n    this.replayer.play();\r\n  }\r\n\r\n  pauseReplay(): void {\r\n    this.replayer?.pause();\r\n  }\r\n\r\n  resumeReplay(): void {\r\n    this.replayer?.play();\r\n  }\r\n\r\n  destroyReplayer(): void {\r\n    if (this.replayer) {\r\n      try { this.replayer.pause(); } catch {}\r\n      this.replayer = undefined;\r\n    }\r\n  }\r\n\r\n  // ─── Export ───────────────────────────────────────────────────────────────\r\n\r\n  exportEvents(): string {\r\n    return JSON.stringify(this._events(), null, 2);\r\n  }\r\n\r\n  downloadEvents(filename = 'rrweb-session.json'): void {\r\n    const blob = new Blob([this.exportEvents()], { type: 'application/json' });\r\n    const url = URL.createObjectURL(blob);\r\n    const a = document.createElement('a');\r\n    a.href = url;\r\n    a.download = filename;\r\n    a.click();\r\n    URL.revokeObjectURL(url);\r\n  }\r\n\r\n  importEvents(json: string): eventWithTime[] {\r\n    try {\r\n      const events = JSON.parse(json) as eventWithTime[];\r\n      this._events.set(events);\r\n      return events;\r\n    } catch {\r\n      console.error('Invalid rrweb events JSON');\r\n      return [];\r\n    }\r\n  }\r\n}\r\n"]}