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
|
@@ -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"]}
|