@webreel/core 0.1.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.
Files changed (94) hide show
  1. package/README.md +188 -0
  2. package/assets/click-1.mp3 +0 -0
  3. package/assets/click-2.mp3 +0 -0
  4. package/assets/click-3.mp3 +0 -0
  5. package/assets/click-4.mp3 +0 -0
  6. package/assets/key-1.mp3 +0 -0
  7. package/assets/key-2.mp3 +0 -0
  8. package/assets/key-3.mp3 +0 -0
  9. package/assets/key-4.mp3 +0 -0
  10. package/dist/__tests__/actions.test.d.ts +2 -0
  11. package/dist/__tests__/actions.test.d.ts.map +1 -0
  12. package/dist/__tests__/actions.test.js +252 -0
  13. package/dist/__tests__/actions.test.js.map +1 -0
  14. package/dist/__tests__/chrome.test.d.ts +2 -0
  15. package/dist/__tests__/chrome.test.d.ts.map +1 -0
  16. package/dist/__tests__/chrome.test.js +29 -0
  17. package/dist/__tests__/chrome.test.js.map +1 -0
  18. package/dist/__tests__/cursor-motion.test.d.ts +2 -0
  19. package/dist/__tests__/cursor-motion.test.d.ts.map +1 -0
  20. package/dist/__tests__/cursor-motion.test.js +39 -0
  21. package/dist/__tests__/cursor-motion.test.js.map +1 -0
  22. package/dist/__tests__/ffmpeg.test.d.ts +2 -0
  23. package/dist/__tests__/ffmpeg.test.d.ts.map +1 -0
  24. package/dist/__tests__/ffmpeg.test.js +90 -0
  25. package/dist/__tests__/ffmpeg.test.js.map +1 -0
  26. package/dist/__tests__/media.test.d.ts +2 -0
  27. package/dist/__tests__/media.test.d.ts.map +1 -0
  28. package/dist/__tests__/media.test.js +98 -0
  29. package/dist/__tests__/media.test.js.map +1 -0
  30. package/dist/__tests__/overlays.test.d.ts +2 -0
  31. package/dist/__tests__/overlays.test.d.ts.map +1 -0
  32. package/dist/__tests__/overlays.test.js +109 -0
  33. package/dist/__tests__/overlays.test.js.map +1 -0
  34. package/dist/__tests__/recording-context.test.d.ts +2 -0
  35. package/dist/__tests__/recording-context.test.d.ts.map +1 -0
  36. package/dist/__tests__/recording-context.test.js +46 -0
  37. package/dist/__tests__/recording-context.test.js.map +1 -0
  38. package/dist/__tests__/timeline.test.d.ts +2 -0
  39. package/dist/__tests__/timeline.test.d.ts.map +1 -0
  40. package/dist/__tests__/timeline.test.js +88 -0
  41. package/dist/__tests__/timeline.test.js.map +1 -0
  42. package/dist/actions.d.ts +65 -0
  43. package/dist/actions.d.ts.map +1 -0
  44. package/dist/actions.js +729 -0
  45. package/dist/actions.js.map +1 -0
  46. package/dist/cdp.d.ts +3 -0
  47. package/dist/cdp.d.ts.map +1 -0
  48. package/dist/cdp.js +5 -0
  49. package/dist/cdp.js.map +1 -0
  50. package/dist/chrome.d.ts +12 -0
  51. package/dist/chrome.d.ts.map +1 -0
  52. package/dist/chrome.js +241 -0
  53. package/dist/chrome.js.map +1 -0
  54. package/dist/compositor.d.ts +8 -0
  55. package/dist/compositor.d.ts.map +1 -0
  56. package/dist/compositor.js +224 -0
  57. package/dist/compositor.js.map +1 -0
  58. package/dist/cursor-motion.d.ts +17 -0
  59. package/dist/cursor-motion.d.ts.map +1 -0
  60. package/dist/cursor-motion.js +138 -0
  61. package/dist/cursor-motion.js.map +1 -0
  62. package/dist/download.d.ts +6 -0
  63. package/dist/download.d.ts.map +1 -0
  64. package/dist/download.js +78 -0
  65. package/dist/download.js.map +1 -0
  66. package/dist/ffmpeg.d.ts +5 -0
  67. package/dist/ffmpeg.d.ts.map +1 -0
  68. package/dist/ffmpeg.js +106 -0
  69. package/dist/ffmpeg.js.map +1 -0
  70. package/dist/index.d.ts +12 -0
  71. package/dist/index.d.ts.map +1 -0
  72. package/dist/index.js +11 -0
  73. package/dist/index.js.map +1 -0
  74. package/dist/media.d.ts +23 -0
  75. package/dist/media.d.ts.map +1 -0
  76. package/dist/media.js +155 -0
  77. package/dist/media.js.map +1 -0
  78. package/dist/overlays.d.ts +21 -0
  79. package/dist/overlays.d.ts.map +1 -0
  80. package/dist/overlays.js +97 -0
  81. package/dist/overlays.js.map +1 -0
  82. package/dist/recorder.d.ts +41 -0
  83. package/dist/recorder.d.ts.map +1 -0
  84. package/dist/recorder.js +223 -0
  85. package/dist/recorder.js.map +1 -0
  86. package/dist/timeline.d.ts +82 -0
  87. package/dist/timeline.d.ts.map +1 -0
  88. package/dist/timeline.js +140 -0
  89. package/dist/timeline.js.map +1 -0
  90. package/dist/types.d.ts +90 -0
  91. package/dist/types.d.ts.map +1 -0
  92. package/dist/types.js +16 -0
  93. package/dist/types.js.map +1 -0
  94. package/package.json +51 -0
@@ -0,0 +1,90 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { mkdirSync, writeFileSync, rmSync } from "node:fs";
3
+ import { resolve } from "node:path";
4
+ import { tmpdir } from "node:os";
5
+ import { btbnAssetName, binaryName, findBinaryInDir } from "../ffmpeg.js";
6
+ describe("btbnAssetName", () => {
7
+ it("returns a string or null depending on platform", () => {
8
+ const result = btbnAssetName();
9
+ if (process.platform === "linux") {
10
+ expect(result).toMatch(/^ffmpeg-n7\.1-latest-linux/);
11
+ expect(result).toMatch(/\.tar\.xz$/);
12
+ }
13
+ else if (process.platform === "win32") {
14
+ expect(result).toMatch(/^ffmpeg-n7\.1-latest-win64/);
15
+ expect(result).toMatch(/\.zip$/);
16
+ }
17
+ else {
18
+ expect(result).toBeNull();
19
+ }
20
+ });
21
+ });
22
+ describe("binaryName", () => {
23
+ it("returns ffmpeg.exe on win32 and ffmpeg elsewhere", () => {
24
+ const result = binaryName();
25
+ if (process.platform === "win32") {
26
+ expect(result).toBe("ffmpeg.exe");
27
+ }
28
+ else {
29
+ expect(result).toBe("ffmpeg");
30
+ }
31
+ });
32
+ });
33
+ describe("findBinaryInDir", () => {
34
+ let testDir;
35
+ function setup() {
36
+ testDir = resolve(tmpdir(), `webreel-ffmpeg-test-${Date.now()}`);
37
+ mkdirSync(testDir, { recursive: true });
38
+ }
39
+ function cleanup() {
40
+ rmSync(testDir, { recursive: true, force: true });
41
+ }
42
+ it("returns null for non-existent directory", () => {
43
+ expect(findBinaryInDir("/does/not/exist", "ffmpeg")).toBeNull();
44
+ });
45
+ it("finds binary directly in the directory", () => {
46
+ setup();
47
+ try {
48
+ writeFileSync(resolve(testDir, "ffmpeg"), "");
49
+ expect(findBinaryInDir(testDir, "ffmpeg")).toBe(resolve(testDir, "ffmpeg"));
50
+ }
51
+ finally {
52
+ cleanup();
53
+ }
54
+ });
55
+ it("finds binary in a nested bin/ subdirectory", () => {
56
+ setup();
57
+ try {
58
+ const nested = resolve(testDir, "ffmpeg-7.1", "bin");
59
+ mkdirSync(nested, { recursive: true });
60
+ writeFileSync(resolve(nested, "ffmpeg"), "");
61
+ expect(findBinaryInDir(testDir, "ffmpeg")).toBe(resolve(testDir, "ffmpeg-7.1", "bin", "ffmpeg"));
62
+ }
63
+ finally {
64
+ cleanup();
65
+ }
66
+ });
67
+ it("finds binary flat inside a nested subdirectory", () => {
68
+ setup();
69
+ try {
70
+ const nested = resolve(testDir, "ffmpeg-build");
71
+ mkdirSync(nested, { recursive: true });
72
+ writeFileSync(resolve(nested, "ffmpeg"), "");
73
+ expect(findBinaryInDir(testDir, "ffmpeg")).toBe(resolve(testDir, "ffmpeg-build", "ffmpeg"));
74
+ }
75
+ finally {
76
+ cleanup();
77
+ }
78
+ });
79
+ it("returns null when binary is not found", () => {
80
+ setup();
81
+ try {
82
+ mkdirSync(resolve(testDir, "empty-dir"), { recursive: true });
83
+ expect(findBinaryInDir(testDir, "ffmpeg")).toBeNull();
84
+ }
85
+ finally {
86
+ cleanup();
87
+ }
88
+ });
89
+ });
90
+ //# sourceMappingURL=ffmpeg.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ffmpeg.test.js","sourceRoot":"","sources":["../../src/__tests__/ffmpeg.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=media.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/media.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,98 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { resolve, dirname } from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ import { resolveSfxPath, ensureSoundAssets, buildAudioMixArgs } from "../media.js";
5
+ const __dirname = dirname(fileURLToPath(import.meta.url));
6
+ const ASSETS_DIR = resolve(__dirname, "..", "..", "assets");
7
+ describe("resolveSfxPath", () => {
8
+ it("returns default asset path when value is undefined", () => {
9
+ expect(resolveSfxPath(undefined, "click")).toBe(resolve(ASSETS_DIR, "click-1.mp3"));
10
+ expect(resolveSfxPath(undefined, "key")).toBe(resolve(ASSETS_DIR, "key-1.mp3"));
11
+ });
12
+ it("returns numbered asset path for numeric values", () => {
13
+ expect(resolveSfxPath(1, "click")).toBe(resolve(ASSETS_DIR, "click-1.mp3"));
14
+ expect(resolveSfxPath(2, "click")).toBe(resolve(ASSETS_DIR, "click-2.mp3"));
15
+ expect(resolveSfxPath(3, "key")).toBe(resolve(ASSETS_DIR, "key-3.mp3"));
16
+ expect(resolveSfxPath(4, "key")).toBe(resolve(ASSETS_DIR, "key-4.mp3"));
17
+ });
18
+ it("returns custom path as-is for string values", () => {
19
+ expect(resolveSfxPath("/custom/click.mp3", "click")).toBe("/custom/click.mp3");
20
+ expect(resolveSfxPath("relative/key.wav", "key")).toBe("relative/key.wav");
21
+ });
22
+ });
23
+ describe("ensureSoundAssets", () => {
24
+ it("returns default paths when no config provided", () => {
25
+ const result = ensureSoundAssets();
26
+ expect(result.clickPath).toBe(resolve(ASSETS_DIR, "click-1.mp3"));
27
+ expect(result.keyPath).toBe(resolve(ASSETS_DIR, "key-1.mp3"));
28
+ });
29
+ it("returns default paths for empty config", () => {
30
+ const result = ensureSoundAssets({});
31
+ expect(result.clickPath).toBe(resolve(ASSETS_DIR, "click-1.mp3"));
32
+ expect(result.keyPath).toBe(resolve(ASSETS_DIR, "key-1.mp3"));
33
+ });
34
+ it("respects numeric overrides", () => {
35
+ const result = ensureSoundAssets({ click: 3, key: 2 });
36
+ expect(result.clickPath).toBe(resolve(ASSETS_DIR, "click-3.mp3"));
37
+ expect(result.keyPath).toBe(resolve(ASSETS_DIR, "key-2.mp3"));
38
+ });
39
+ it("respects custom string paths", () => {
40
+ const result = ensureSoundAssets({ click: "/my/click.mp3", key: "/my/key.mp3" });
41
+ expect(result.clickPath).toBe("/my/click.mp3");
42
+ expect(result.keyPath).toBe("/my/key.mp3");
43
+ });
44
+ });
45
+ describe("buildAudioMixArgs", () => {
46
+ it("produces correct structure with no events", () => {
47
+ const { inputArgs, filterComplex } = buildAudioMixArgs("input.mp4", [], 10);
48
+ expect(inputArgs).toContain("-i");
49
+ expect(inputArgs).toContain("input.mp4");
50
+ expect(inputArgs).toContain("anullsrc=r=44100:cl=mono");
51
+ expect(inputArgs).toContain("10.000");
52
+ expect(filterComplex).toContain("amix=inputs=1");
53
+ });
54
+ it("adds sound file inputs for each event", () => {
55
+ const events = [
56
+ { type: "click", timeMs: 500 },
57
+ { type: "key", timeMs: 1000 },
58
+ ];
59
+ const { inputArgs, filterComplex } = buildAudioMixArgs("input.mp4", events, 5);
60
+ const inputCount = inputArgs.filter((a) => a === "-i").length;
61
+ expect(inputCount).toBe(4);
62
+ expect(filterComplex).toContain("[2]");
63
+ expect(filterComplex).toContain("[3]");
64
+ expect(filterComplex).toContain("adelay=500|500");
65
+ expect(filterComplex).toContain("adelay=1000|1000");
66
+ expect(filterComplex).toContain("amix=inputs=3");
67
+ });
68
+ it("clamps event timeMs to duration", () => {
69
+ const events = [{ type: "click", timeMs: 99999 }];
70
+ const { filterComplex } = buildAudioMixArgs("input.mp4", events, 2);
71
+ expect(filterComplex).toContain("adelay=2000|2000");
72
+ });
73
+ it("includes volume and rate parameters in filter", () => {
74
+ const events = [{ type: "click", timeMs: 0 }];
75
+ const { filterComplex } = buildAudioMixArgs("input.mp4", events, 1);
76
+ expect(filterComplex).toMatch(/volume=\d+\.\d+/);
77
+ expect(filterComplex).toMatch(/asetrate=\d+/);
78
+ expect(filterComplex).toContain("aresample=44100");
79
+ });
80
+ it("uses click sound file for click events and key sound for key events", () => {
81
+ const events = [
82
+ { type: "click", timeMs: 0 },
83
+ { type: "key", timeMs: 100 },
84
+ ];
85
+ const { inputArgs } = buildAudioMixArgs("input.mp4", events, 1);
86
+ const clickPath = resolve(ASSETS_DIR, "click-1.mp3");
87
+ const keyPath = resolve(ASSETS_DIR, "key-1.mp3");
88
+ expect(inputArgs).toContain(clickPath);
89
+ expect(inputArgs).toContain(keyPath);
90
+ });
91
+ it("respects custom sfx config", () => {
92
+ const events = [{ type: "click", timeMs: 0 }];
93
+ const { inputArgs } = buildAudioMixArgs("input.mp4", events, 1, { click: 3 });
94
+ const clickPath = resolve(ASSETS_DIR, "click-3.mp3");
95
+ expect(inputArgs).toContain(clickPath);
96
+ });
97
+ });
98
+ //# sourceMappingURL=media.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media.test.js","sourceRoot":"","sources":["../../src/__tests__/media.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=overlays.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"overlays.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/overlays.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,109 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { DEFAULT_CURSOR_SVG, DEFAULT_CURSOR_SIZE, DEFAULT_HUD_THEME } from "../types.js";
3
+ import { injectOverlays, showKeys, hideKeys } from "../overlays.js";
4
+ function createMockClient() {
5
+ const calls = [];
6
+ const client = {
7
+ Runtime: {
8
+ evaluate: async (params) => {
9
+ calls.push({ method: "evaluate", expression: params.expression });
10
+ return { result: {} };
11
+ },
12
+ },
13
+ };
14
+ return { client, calls };
15
+ }
16
+ describe("injectOverlays", () => {
17
+ it("uses default cursor size when no theme provided", async () => {
18
+ const { client, calls } = createMockClient();
19
+ await injectOverlays(client);
20
+ expect(calls).toHaveLength(1);
21
+ expect(calls[0].expression).toContain(`width:${DEFAULT_CURSOR_SIZE}px`);
22
+ expect(calls[0].expression).toContain(`height:${DEFAULT_CURSOR_SIZE}px`);
23
+ });
24
+ it("uses custom cursor size from theme", async () => {
25
+ const { client, calls } = createMockClient();
26
+ await injectOverlays(client, { cursorSize: 48 });
27
+ expect(calls[0].expression).toContain("width:48px");
28
+ expect(calls[0].expression).toContain("height:48px");
29
+ });
30
+ it("applies top-left hotspot with zero margin offset", async () => {
31
+ const { client, calls } = createMockClient();
32
+ await injectOverlays(client, { cursorHotspot: "top-left" });
33
+ expect(calls[0].expression).toContain("margin-left:0px");
34
+ expect(calls[0].expression).toContain("margin-top:0px");
35
+ });
36
+ it("applies center hotspot with half-size margin offset", async () => {
37
+ const { client, calls } = createMockClient();
38
+ await injectOverlays(client, { cursorSize: 32, cursorHotspot: "center" });
39
+ expect(calls[0].expression).toContain("margin-left:-16px");
40
+ expect(calls[0].expression).toContain("margin-top:-16px");
41
+ });
42
+ it("uses default HUD theme values when not overridden", async () => {
43
+ const { client, calls } = createMockClient();
44
+ await injectOverlays(client);
45
+ const expr = calls[0].expression;
46
+ expect(expr).toContain(`background:${DEFAULT_HUD_THEME.background}`);
47
+ expect(expr).toContain(`border-radius:" + z(${DEFAULT_HUD_THEME.borderRadius})`);
48
+ });
49
+ it("uses custom HUD theme values", async () => {
50
+ const { client, calls } = createMockClient();
51
+ const theme = {
52
+ hud: {
53
+ background: "red",
54
+ color: "blue",
55
+ fontSize: 32,
56
+ borderRadius: 8,
57
+ position: "top",
58
+ },
59
+ };
60
+ await injectOverlays(client, theme);
61
+ const expr = calls[0].expression;
62
+ expect(expr).toContain("background:red");
63
+ expect(expr).toContain("color: blue");
64
+ expect(expr).toContain('border-radius:" + z(8)');
65
+ expect(expr).toContain('"top:"');
66
+ });
67
+ it("uses custom cursor SVG", async () => {
68
+ const { client, calls } = createMockClient();
69
+ const svg = '<svg><circle r="10"/></svg>';
70
+ await injectOverlays(client, { cursorSvg: svg });
71
+ expect(calls[0].expression).toContain(JSON.stringify(svg));
72
+ });
73
+ it("uses default cursor SVG when none provided", async () => {
74
+ const { client, calls } = createMockClient();
75
+ await injectOverlays(client);
76
+ expect(calls[0].expression).toContain(JSON.stringify(DEFAULT_CURSOR_SVG));
77
+ });
78
+ it("applies initial cursor position", async () => {
79
+ const { client, calls } = createMockClient();
80
+ await injectOverlays(client, undefined, { x: 100, y: 200 });
81
+ const expr = calls[0].expression;
82
+ expect(expr).toContain("translate(100px,200px)");
83
+ expect(expr).toContain('dataset.cx = "100"');
84
+ expect(expr).toContain('dataset.cy = "200"');
85
+ });
86
+ });
87
+ describe("showKeys", () => {
88
+ it("passes labels to the expression", async () => {
89
+ const { client, calls } = createMockClient();
90
+ await showKeys(client, ["Ctrl", "Z"]);
91
+ expect(calls[0].expression).toContain(JSON.stringify(["Ctrl", "Z"]));
92
+ });
93
+ it("expression includes HTML entity escaping for &, <, >", async () => {
94
+ const { client, calls } = createMockClient();
95
+ await showKeys(client, ["<test>"]);
96
+ const expr = calls[0].expression;
97
+ expect(expr).toContain("&amp;");
98
+ expect(expr).toContain("&lt;");
99
+ expect(expr).toContain("&gt;");
100
+ });
101
+ });
102
+ describe("hideKeys", () => {
103
+ it("sets opacity to 0", async () => {
104
+ const { client, calls } = createMockClient();
105
+ await hideKeys(client);
106
+ expect(calls[0].expression).toContain('opacity = "0"');
107
+ });
108
+ });
109
+ //# sourceMappingURL=overlays.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"overlays.test.js","sourceRoot":"","sources":["../../src/__tests__/overlays.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=recording-context.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recording-context.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/recording-context.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,46 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { RecordingContext } from "../actions.js";
3
+ describe("RecordingContext", () => {
4
+ it("defaults to preview mode", () => {
5
+ const ctx = new RecordingContext();
6
+ expect(ctx.mode).toBe("preview");
7
+ expect(ctx.isRecording).toBe(false);
8
+ });
9
+ it("isRecording is true when mode is record and timeline is set", () => {
10
+ const ctx = new RecordingContext();
11
+ ctx.setMode("record");
12
+ expect(ctx.isRecording).toBe(false);
13
+ ctx.setTimeline({ addEvent: () => { } });
14
+ expect(ctx.isRecording).toBe(true);
15
+ });
16
+ it("resetCursorPosition places cursor off-screen", () => {
17
+ const ctx = new RecordingContext();
18
+ ctx.resetCursorPosition(1920, 1080);
19
+ const pos = ctx.getCursorPosition();
20
+ const offScreen = pos.x < 0 || pos.x > 1920 || pos.y < 0 || pos.y > 1080;
21
+ expect(offScreen).toBe(true);
22
+ });
23
+ it("getCursorPosition returns initial off-screen position", () => {
24
+ const ctx = new RecordingContext();
25
+ const pos = ctx.getCursorPosition();
26
+ expect(pos.x).toBe(-40);
27
+ expect(pos.y).toBe(-40);
28
+ });
29
+ it("markEvent delegates to recorder when set", () => {
30
+ const ctx = new RecordingContext();
31
+ const events = [];
32
+ ctx.setRecorder({ addEvent: (type) => events.push(type) });
33
+ ctx.markEvent("click");
34
+ ctx.markEvent("key");
35
+ expect(events).toEqual(["click", "key"]);
36
+ });
37
+ it("markEvent delegates to timeline when recording", () => {
38
+ const ctx = new RecordingContext();
39
+ const events = [];
40
+ ctx.setMode("record");
41
+ ctx.setTimeline({ addEvent: (type) => events.push(type) });
42
+ ctx.markEvent("click");
43
+ expect(events).toEqual(["click"]);
44
+ });
45
+ });
46
+ //# sourceMappingURL=recording-context.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recording-context.test.js","sourceRoot":"","sources":["../../src/__tests__/recording-context.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,eAAe,CAAC;AAEjD,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;QAClC,MAAM,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;QACnC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACjC,MAAM,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6DAA6D,EAAE,GAAG,EAAE;QACrE,MAAM,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;QACnC,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACtB,MAAM,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAEpC,GAAG,CAAC,WAAW,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,GAAE,CAAC,EAAW,CAAC,CAAC;QACjD,MAAM,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8CAA8C,EAAE,GAAG,EAAE;QACtD,MAAM,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;QACnC,GAAG,CAAC,mBAAmB,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACpC,MAAM,GAAG,GAAG,GAAG,CAAC,iBAAiB,EAAE,CAAC;QACpC,MAAM,SAAS,GAAG,GAAG,CAAC,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI,IAAI,GAAG,CAAC,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC;QACzE,MAAM,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uDAAuD,EAAE,GAAG,EAAE;QAC/D,MAAM,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;QACnC,MAAM,GAAG,GAAG,GAAG,CAAC,iBAAiB,EAAE,CAAC;QACpC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC;QACxB,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC;IAC1B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;QACnC,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,GAAG,CAAC,WAAW,CAAC,EAAE,QAAQ,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC3D,GAAG,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;QACvB,GAAG,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QACrB,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;QACnC,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACtB,GAAG,CAAC,WAAW,CAAC,EAAE,QAAQ,EAAE,CAAC,IAAY,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAW,CAAC,CAAC;QAC5E,GAAG,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;QACvB,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=timeline.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeline.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/timeline.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,88 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { InteractionTimeline } from "../timeline.js";
3
+ describe("InteractionTimeline", () => {
4
+ it("starts with zero frames", () => {
5
+ const tl = new InteractionTimeline(1080, 1080);
6
+ expect(tl.getFrameCount()).toBe(0);
7
+ expect(tl.getEvents()).toEqual([]);
8
+ });
9
+ it("tick advances frame count", () => {
10
+ const tl = new InteractionTimeline(800, 600);
11
+ tl.tick();
12
+ tl.tick();
13
+ tl.tick();
14
+ expect(tl.getFrameCount()).toBe(3);
15
+ });
16
+ it("tickDuplicate also advances frame count", () => {
17
+ const tl = new InteractionTimeline(800, 600);
18
+ tl.tick();
19
+ tl.tickDuplicate();
20
+ expect(tl.getFrameCount()).toBe(2);
21
+ });
22
+ it("addEvent records events with correct timing", () => {
23
+ const tl = new InteractionTimeline(1080, 1080);
24
+ for (let i = 0; i < 60; i++)
25
+ tl.tick();
26
+ tl.addEvent("click");
27
+ const events = tl.getEvents();
28
+ expect(events).toHaveLength(1);
29
+ expect(events[0].type).toBe("click");
30
+ expect(events[0].timeMs).toBeCloseTo(1000, 0);
31
+ });
32
+ it("toJSON produces valid timeline data", () => {
33
+ const tl = new InteractionTimeline(1920, 1080, { zoom: 2 });
34
+ tl.tick();
35
+ const data = tl.toJSON();
36
+ expect(data.fps).toBe(60);
37
+ expect(data.width).toBe(1920);
38
+ expect(data.height).toBe(1080);
39
+ expect(data.zoom).toBe(2);
40
+ expect(data.frames).toHaveLength(1);
41
+ expect(data.theme.cursorSize).toBe(24);
42
+ });
43
+ it("load roundtrips correctly", () => {
44
+ const tl = new InteractionTimeline(800, 600, {
45
+ zoom: 1.5,
46
+ cursorSize: 32,
47
+ });
48
+ tl.setCursorPath([{ x: 100, y: 200 }]);
49
+ tl.tick();
50
+ tl.addEvent("key");
51
+ tl.tick();
52
+ const json = tl.toJSON();
53
+ const loaded = InteractionTimeline.load(json);
54
+ const reJson = loaded.toJSON();
55
+ expect(reJson.frames).toEqual(json.frames);
56
+ expect(reJson.events).toEqual(json.events);
57
+ expect(reJson.width).toBe(json.width);
58
+ expect(reJson.height).toBe(json.height);
59
+ expect(reJson.zoom).toBe(json.zoom);
60
+ });
61
+ it("showHud and hideHud affect frame data", () => {
62
+ const tl = new InteractionTimeline(1080, 1080);
63
+ tl.showHud(["Ctrl", "A"]);
64
+ tl.tick();
65
+ tl.hideHud();
66
+ tl.tick();
67
+ const data = tl.toJSON();
68
+ expect(data.frames[0].hud).toEqual({ labels: ["Ctrl", "A"] });
69
+ expect(data.frames[1].hud).toBeNull();
70
+ });
71
+ it("setCursorPath advances cursor through positions", () => {
72
+ const tl = new InteractionTimeline(1080, 1080);
73
+ tl.setCursorPath([
74
+ { x: 10, y: 20 },
75
+ { x: 30, y: 40 },
76
+ ]);
77
+ tl.tick();
78
+ tl.tick();
79
+ tl.tick();
80
+ const data = tl.toJSON();
81
+ expect(data.frames[0].cursor.x).toBe(10);
82
+ expect(data.frames[0].cursor.y).toBe(20);
83
+ expect(data.frames[1].cursor.x).toBe(30);
84
+ expect(data.frames[1].cursor.y).toBe(40);
85
+ expect(data.frames[2].cursor.x).toBe(30);
86
+ });
87
+ });
88
+ //# sourceMappingURL=timeline.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeline.test.js","sourceRoot":"","sources":["../../src/__tests__/timeline.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,65 @@
1
+ import type { CDPClient, BoundingBox } from "./types.js";
2
+ import type { InteractionTimeline } from "./timeline.js";
3
+ export declare class RecordingContext {
4
+ private _mode;
5
+ private _timeline;
6
+ private _recorder;
7
+ private _cursorX;
8
+ private _cursorY;
9
+ private _clickDwell;
10
+ get mode(): "record" | "preview";
11
+ setMode(mode: "record" | "preview"): void;
12
+ get timeline(): InteractionTimeline | null;
13
+ setTimeline(timeline: InteractionTimeline | null): void;
14
+ setRecorder(recorder: {
15
+ addEvent: (type: "click" | "key") => void;
16
+ } | null): void;
17
+ get cursorX(): number;
18
+ get cursorY(): number;
19
+ setCursorPosition(x: number, y: number): void;
20
+ get isRecording(): boolean;
21
+ resetCursorPosition(cssWidth?: number, cssHeight?: number): void;
22
+ setClickDwell(ms: number | undefined): void;
23
+ getClickDwellMs(): number;
24
+ getCursorPosition(): {
25
+ x: number;
26
+ y: number;
27
+ };
28
+ markEvent(type: "click" | "key"): void;
29
+ }
30
+ export declare function modKey(): string;
31
+ export declare function pause(ms?: number): Promise<void>;
32
+ export declare function navigate(client: CDPClient, url: string): Promise<void>;
33
+ export declare function waitForSelector(client: CDPClient, selector: string, timeoutMs?: number): Promise<void>;
34
+ export declare function waitForText(client: CDPClient, text: string, within?: string, timeoutMs?: number): Promise<void>;
35
+ export declare function findElementByText(client: CDPClient, text: string, within?: string): Promise<BoundingBox | null>;
36
+ export declare function findElementBySelector(client: CDPClient, selector: string, within?: string): Promise<BoundingBox | null>;
37
+ export declare function moveCursorTo(ctx: RecordingContext, client: CDPClient, x: number, y: number): Promise<void>;
38
+ export declare function resolveMod(mod: string): string;
39
+ export declare function modifierFlag(mod: string): number;
40
+ export declare function modifiersToFlag(mods?: string[]): number;
41
+ export declare function modLabel(mod: string): string;
42
+ interface ModKeyInfo {
43
+ key: string;
44
+ code: string;
45
+ keyCode: number;
46
+ location: number;
47
+ }
48
+ export declare function modKeyInfo(mod: string): ModKeyInfo | null;
49
+ export declare function clickAt(ctx: RecordingContext, client: CDPClient, x: number, y: number, modifiers?: string[]): Promise<void>;
50
+ export declare const KEY_CODES: Record<string, {
51
+ code: string;
52
+ keyCode: number;
53
+ }>;
54
+ export declare const SHORTCUT_COMMANDS: Record<string, string[]>;
55
+ export declare function resolveCommands(modifiers: string[], mainKey: string): string[] | undefined;
56
+ export declare function pressKey(ctx: RecordingContext, client: CDPClient, key: string, label?: string): Promise<void>;
57
+ export declare const CHAR_CODES: Record<string, {
58
+ code: string;
59
+ keyCode: number;
60
+ }>;
61
+ export declare function typeText(ctx: RecordingContext, client: CDPClient, text: string, delayMs?: number): Promise<void>;
62
+ export declare function dragFromTo(ctx: RecordingContext, client: CDPClient, fromBox: BoundingBox, toBox: BoundingBox): Promise<void>;
63
+ export declare function captureScreenshot(client: CDPClient, outputPath: string): Promise<void>;
64
+ export {};
65
+ //# sourceMappingURL=actions.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"actions.d.ts","sourceRoot":"","sources":["../src/actions.ts"],"names":[],"mappings":"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"}