screenwright 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 (106) hide show
  1. package/LICENSE +21 -0
  2. package/assets/click-ripple.svg +3 -0
  3. package/assets/cursor-default.svg +10 -0
  4. package/assets/cursor-pointer.svg +40 -0
  5. package/dist/bin/screenwright.d.ts +3 -0
  6. package/dist/bin/screenwright.d.ts.map +1 -0
  7. package/dist/bin/screenwright.js +18 -0
  8. package/dist/bin/screenwright.js.map +1 -0
  9. package/dist/src/commands/compose.d.ts +3 -0
  10. package/dist/src/commands/compose.d.ts.map +1 -0
  11. package/dist/src/commands/compose.js +154 -0
  12. package/dist/src/commands/compose.js.map +1 -0
  13. package/dist/src/commands/generate.d.ts +3 -0
  14. package/dist/src/commands/generate.d.ts.map +1 -0
  15. package/dist/src/commands/generate.js +70 -0
  16. package/dist/src/commands/generate.js.map +1 -0
  17. package/dist/src/commands/init.d.ts +3 -0
  18. package/dist/src/commands/init.d.ts.map +1 -0
  19. package/dist/src/commands/init.js +49 -0
  20. package/dist/src/commands/init.js.map +1 -0
  21. package/dist/src/commands/preview.d.ts +3 -0
  22. package/dist/src/commands/preview.d.ts.map +1 -0
  23. package/dist/src/commands/preview.js +62 -0
  24. package/dist/src/commands/preview.js.map +1 -0
  25. package/dist/src/composition/CursorOverlay.d.ts +11 -0
  26. package/dist/src/composition/CursorOverlay.d.ts.map +1 -0
  27. package/dist/src/composition/CursorOverlay.js +29 -0
  28. package/dist/src/composition/CursorOverlay.js.map +1 -0
  29. package/dist/src/composition/DemoVideo.d.ts +8 -0
  30. package/dist/src/composition/DemoVideo.d.ts.map +1 -0
  31. package/dist/src/composition/DemoVideo.js +18 -0
  32. package/dist/src/composition/DemoVideo.js.map +1 -0
  33. package/dist/src/composition/NarrationTrack.d.ts +9 -0
  34. package/dist/src/composition/NarrationTrack.d.ts.map +1 -0
  35. package/dist/src/composition/NarrationTrack.js +8 -0
  36. package/dist/src/composition/NarrationTrack.js.map +1 -0
  37. package/dist/src/composition/cursor-path.d.ts +31 -0
  38. package/dist/src/composition/cursor-path.d.ts.map +1 -0
  39. package/dist/src/composition/cursor-path.js +78 -0
  40. package/dist/src/composition/cursor-path.js.map +1 -0
  41. package/dist/src/composition/remotion-root.d.ts +3 -0
  42. package/dist/src/composition/remotion-root.d.ts.map +1 -0
  43. package/dist/src/composition/remotion-root.js +34 -0
  44. package/dist/src/composition/remotion-root.js.map +1 -0
  45. package/dist/src/composition/render.d.ts +8 -0
  46. package/dist/src/composition/render.d.ts.map +1 -0
  47. package/dist/src/composition/render.js +24 -0
  48. package/dist/src/composition/render.js.map +1 -0
  49. package/dist/src/config/config-schema.d.ts +40 -0
  50. package/dist/src/config/config-schema.d.ts.map +1 -0
  51. package/dist/src/config/config-schema.js +13 -0
  52. package/dist/src/config/config-schema.js.map +1 -0
  53. package/dist/src/config/defaults.d.ts +4 -0
  54. package/dist/src/config/defaults.d.ts.map +1 -0
  55. package/dist/src/config/defaults.js +17 -0
  56. package/dist/src/config/defaults.js.map +1 -0
  57. package/dist/src/generator/prompts.d.ts +3 -0
  58. package/dist/src/generator/prompts.d.ts.map +1 -0
  59. package/dist/src/generator/prompts.js +125 -0
  60. package/dist/src/generator/prompts.js.map +1 -0
  61. package/dist/src/generator/scenario-generator.d.ts +46 -0
  62. package/dist/src/generator/scenario-generator.d.ts.map +1 -0
  63. package/dist/src/generator/scenario-generator.js +86 -0
  64. package/dist/src/generator/scenario-generator.js.map +1 -0
  65. package/dist/src/index.d.ts +7 -0
  66. package/dist/src/index.d.ts.map +1 -0
  67. package/dist/src/index.js +2 -0
  68. package/dist/src/index.js.map +1 -0
  69. package/dist/src/runtime/action-helpers.d.ts +19 -0
  70. package/dist/src/runtime/action-helpers.d.ts.map +1 -0
  71. package/dist/src/runtime/action-helpers.js +138 -0
  72. package/dist/src/runtime/action-helpers.js.map +1 -0
  73. package/dist/src/runtime/instrumented-page.d.ts +21 -0
  74. package/dist/src/runtime/instrumented-page.d.ts.map +1 -0
  75. package/dist/src/runtime/instrumented-page.js +48 -0
  76. package/dist/src/runtime/instrumented-page.js.map +1 -0
  77. package/dist/src/runtime/timeline-collector.d.ts +20 -0
  78. package/dist/src/runtime/timeline-collector.d.ts.map +1 -0
  79. package/dist/src/runtime/timeline-collector.js +40 -0
  80. package/dist/src/runtime/timeline-collector.js.map +1 -0
  81. package/dist/src/timeline/schema.d.ts +297 -0
  82. package/dist/src/timeline/schema.d.ts.map +1 -0
  83. package/dist/src/timeline/schema.js +72 -0
  84. package/dist/src/timeline/schema.js.map +1 -0
  85. package/dist/src/timeline/types.d.ts +67 -0
  86. package/dist/src/timeline/types.d.ts.map +1 -0
  87. package/dist/src/timeline/types.js +2 -0
  88. package/dist/src/timeline/types.js.map +1 -0
  89. package/dist/src/version.d.ts +2 -0
  90. package/dist/src/version.d.ts.map +1 -0
  91. package/dist/src/version.js +2 -0
  92. package/dist/src/version.js.map +1 -0
  93. package/dist/src/voiceover/narration-timing.d.ts +12 -0
  94. package/dist/src/voiceover/narration-timing.d.ts.map +1 -0
  95. package/dist/src/voiceover/narration-timing.js +25 -0
  96. package/dist/src/voiceover/narration-timing.js.map +1 -0
  97. package/dist/src/voiceover/piper-engine.d.ts +6 -0
  98. package/dist/src/voiceover/piper-engine.d.ts.map +1 -0
  99. package/dist/src/voiceover/piper-engine.js +48 -0
  100. package/dist/src/voiceover/piper-engine.js.map +1 -0
  101. package/dist/src/voiceover/voice-models.d.ts +18 -0
  102. package/dist/src/voiceover/voice-models.d.ts.map +1 -0
  103. package/dist/src/voiceover/voice-models.js +123 -0
  104. package/dist/src/voiceover/voice-models.js.map +1 -0
  105. package/dist/tsconfig.tsbuildinfo +1 -0
  106. package/package.json +69 -0
@@ -0,0 +1,19 @@
1
+ import type { Page } from 'playwright';
2
+ import type { TimelineCollector } from './timeline-collector.js';
3
+ export interface ActionOptions {
4
+ narration?: string;
5
+ }
6
+ export interface ScreenwrightHelpers {
7
+ page: Page;
8
+ scene(title: string, description?: string): Promise<void>;
9
+ navigate(url: string, opts?: ActionOptions): Promise<void>;
10
+ click(selector: string, opts?: ActionOptions): Promise<void>;
11
+ fill(selector: string, value: string, opts?: ActionOptions): Promise<void>;
12
+ hover(selector: string, opts?: ActionOptions): Promise<void>;
13
+ press(key: string, opts?: ActionOptions): Promise<void>;
14
+ wait(ms: number): Promise<void>;
15
+ narrate(text: string): Promise<void>;
16
+ }
17
+ export declare function calculateMoveDuration(fromX: number, fromY: number, toX: number, toY: number): number;
18
+ export declare function createHelpers(page: Page, collector: TimelineCollector): ScreenwrightHelpers;
19
+ //# sourceMappingURL=action-helpers.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"action-helpers.d.ts","sourceRoot":"","sources":["../../../src/runtime/action-helpers.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,YAAY,CAAC;AACvC,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC;AAEjE,MAAM,WAAW,aAAa;IAC5B,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,mBAAmB;IAClC,IAAI,EAAE,IAAI,CAAC;IACX,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,WAAW,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC1D,QAAQ,CAAC,GAAG,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC3D,KAAK,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7D,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC3E,KAAK,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7D,KAAK,CAAC,GAAG,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IACxD,IAAI,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAChC,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;CACtC;AAWD,wBAAgB,qBAAqB,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,CAGpG;AAED,wBAAgB,aAAa,CAAC,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,iBAAiB,GAAG,mBAAmB,CA6I3F"}
@@ -0,0 +1,138 @@
1
+ const NARRATION_WPM = 150;
2
+ const POST_ACTION_DELAY_MS = 500;
3
+ const CHAR_TYPE_DELAY_MS = 50;
4
+ function estimateNarrationMs(text) {
5
+ const words = text.split(/\s+/).length;
6
+ return Math.round((words / NARRATION_WPM) * 60 * 1000);
7
+ }
8
+ export function calculateMoveDuration(fromX, fromY, toX, toY) {
9
+ const distance = Math.sqrt((toX - fromX) ** 2 + (toY - fromY) ** 2);
10
+ return Math.min(1200, Math.max(300, Math.round(200 * Math.log2(distance / 10 + 1))));
11
+ }
12
+ export function createHelpers(page, collector) {
13
+ let lastX = 640;
14
+ let lastY = 360;
15
+ async function emitNarration(text) {
16
+ const estimatedMs = estimateNarrationMs(text);
17
+ collector.emit({ type: 'narration', text });
18
+ collector.emit({ type: 'wait', durationMs: estimatedMs, reason: 'narration_sync' });
19
+ await page.waitForTimeout(estimatedMs);
20
+ }
21
+ async function moveCursorTo(toX, toY) {
22
+ const moveDurationMs = calculateMoveDuration(lastX, lastY, toX, toY);
23
+ collector.emit({
24
+ type: 'cursor_target',
25
+ fromX: lastX, fromY: lastY,
26
+ toX, toY,
27
+ moveDurationMs,
28
+ easing: 'bezier',
29
+ });
30
+ await page.waitForTimeout(moveDurationMs);
31
+ lastX = toX;
32
+ lastY = toY;
33
+ }
34
+ async function resolveCenter(selector) {
35
+ const locator = page.locator(selector).first();
36
+ await locator.waitFor({ state: 'visible', timeout: 10000 });
37
+ const box = await locator.boundingBox();
38
+ if (!box)
39
+ return { x: lastX, y: lastY };
40
+ return { x: Math.round(box.x + box.width / 2), y: Math.round(box.y + box.height / 2) };
41
+ }
42
+ return {
43
+ page,
44
+ async scene(title, description) {
45
+ collector.emit({ type: 'scene', title, description });
46
+ },
47
+ async navigate(url, opts) {
48
+ if (opts?.narration)
49
+ await emitNarration(opts.narration);
50
+ collector.emit({
51
+ type: 'action',
52
+ action: 'navigate',
53
+ selector: url,
54
+ durationMs: 0,
55
+ boundingBox: null,
56
+ });
57
+ await page.goto(url, { waitUntil: 'domcontentloaded' });
58
+ collector.emit({ type: 'wait', durationMs: 1000, reason: 'page_load' });
59
+ await page.waitForTimeout(1000);
60
+ },
61
+ async click(selector, opts) {
62
+ if (opts?.narration)
63
+ await emitNarration(opts.narration);
64
+ const center = await resolveCenter(selector);
65
+ await moveCursorTo(center.x, center.y);
66
+ const locator = page.locator(selector).first();
67
+ const box = await locator.boundingBox();
68
+ collector.emit({
69
+ type: 'action',
70
+ action: 'click',
71
+ selector,
72
+ durationMs: 200,
73
+ boundingBox: box ? { x: Math.round(box.x), y: Math.round(box.y), width: Math.round(box.width), height: Math.round(box.height) } : null,
74
+ });
75
+ await locator.click();
76
+ await page.waitForTimeout(POST_ACTION_DELAY_MS);
77
+ },
78
+ async fill(selector, value, opts) {
79
+ if (opts?.narration)
80
+ await emitNarration(opts.narration);
81
+ const center = await resolveCenter(selector);
82
+ await moveCursorTo(center.x, center.y);
83
+ const locator = page.locator(selector).first();
84
+ const box = await locator.boundingBox();
85
+ await locator.click();
86
+ collector.emit({
87
+ type: 'action',
88
+ action: 'fill',
89
+ selector,
90
+ value,
91
+ durationMs: value.length * CHAR_TYPE_DELAY_MS,
92
+ boundingBox: box ? { x: Math.round(box.x), y: Math.round(box.y), width: Math.round(box.width), height: Math.round(box.height) } : null,
93
+ });
94
+ for (const char of value) {
95
+ await page.keyboard.type(char, { delay: CHAR_TYPE_DELAY_MS });
96
+ }
97
+ await page.waitForTimeout(POST_ACTION_DELAY_MS);
98
+ },
99
+ async hover(selector, opts) {
100
+ if (opts?.narration)
101
+ await emitNarration(opts.narration);
102
+ const center = await resolveCenter(selector);
103
+ await moveCursorTo(center.x, center.y);
104
+ const locator = page.locator(selector).first();
105
+ const box = await locator.boundingBox();
106
+ collector.emit({
107
+ type: 'action',
108
+ action: 'hover',
109
+ selector,
110
+ durationMs: 200,
111
+ boundingBox: box ? { x: Math.round(box.x), y: Math.round(box.y), width: Math.round(box.width), height: Math.round(box.height) } : null,
112
+ });
113
+ await locator.hover();
114
+ await page.waitForTimeout(POST_ACTION_DELAY_MS);
115
+ },
116
+ async press(key, opts) {
117
+ if (opts?.narration)
118
+ await emitNarration(opts.narration);
119
+ collector.emit({
120
+ type: 'action',
121
+ action: 'press',
122
+ selector: key,
123
+ durationMs: 100,
124
+ boundingBox: null,
125
+ });
126
+ await page.keyboard.press(key);
127
+ await page.waitForTimeout(POST_ACTION_DELAY_MS);
128
+ },
129
+ async wait(ms) {
130
+ collector.emit({ type: 'wait', durationMs: ms, reason: 'pacing' });
131
+ await page.waitForTimeout(ms);
132
+ },
133
+ async narrate(text) {
134
+ await emitNarration(text);
135
+ },
136
+ };
137
+ }
138
+ //# sourceMappingURL=action-helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"action-helpers.js","sourceRoot":"","sources":["../../../src/runtime/action-helpers.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ import type { Timeline } from '../timeline/types.js';
2
+ import { type ScreenwrightHelpers } from './action-helpers.js';
3
+ export type ScenarioFn = (sw: ScreenwrightHelpers) => Promise<void>;
4
+ export interface RunOptions {
5
+ scenarioFile: string;
6
+ testFile: string;
7
+ viewport?: {
8
+ width: number;
9
+ height: number;
10
+ };
11
+ colorScheme?: 'light' | 'dark';
12
+ locale?: string;
13
+ timezoneId?: string;
14
+ }
15
+ export interface RunResult {
16
+ timeline: Timeline;
17
+ videoFile: string;
18
+ tempDir: string;
19
+ }
20
+ export declare function runScenario(scenario: ScenarioFn, opts: RunOptions): Promise<RunResult>;
21
+ //# sourceMappingURL=instrumented-page.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"instrumented-page.d.ts","sourceRoot":"","sources":["../../../src/runtime/instrumented-page.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAErD,OAAO,EAAiB,KAAK,mBAAmB,EAAE,MAAM,qBAAqB,CAAC;AAE9E,MAAM,MAAM,UAAU,GAAG,CAAC,EAAE,EAAE,mBAAmB,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;AAEpE,MAAM,WAAW,UAAU;IACzB,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC;IAC7C,WAAW,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IAC/B,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,SAAS;IACxB,QAAQ,EAAE,QAAQ,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,wBAAsB,WAAW,CAAC,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,UAAU,GAAG,OAAO,CAAC,SAAS,CAAC,CAoD5F"}
@@ -0,0 +1,48 @@
1
+ import { chromium } from 'playwright';
2
+ import { mkdtemp, writeFile } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import { TimelineCollector } from './timeline-collector.js';
6
+ import { createHelpers } from './action-helpers.js';
7
+ export async function runScenario(scenario, opts) {
8
+ const viewport = opts.viewport ?? { width: 1280, height: 720 };
9
+ const tempDir = await mkdtemp(join(tmpdir(), 'screenwright-'));
10
+ const browser = await chromium.launch({
11
+ args: ['--disable-gpu', '--font-render-hinting=none', '--disable-lcd-text'],
12
+ });
13
+ const context = await browser.newContext({
14
+ viewport,
15
+ deviceScaleFactor: 1,
16
+ colorScheme: opts.colorScheme ?? 'light',
17
+ locale: opts.locale ?? 'en-US',
18
+ timezoneId: opts.timezoneId ?? 'America/New_York',
19
+ recordVideo: { dir: tempDir, size: viewport },
20
+ });
21
+ const page = await context.newPage();
22
+ const collector = new TimelineCollector();
23
+ collector.start();
24
+ const sw = createHelpers(page, collector);
25
+ await scenario(sw);
26
+ // Close page to finalize video
27
+ await page.close();
28
+ const video = page.video();
29
+ const videoFile = video ? await video.path() : '';
30
+ const videoDurationMs = collector.getEvents().reduce((max, e) => {
31
+ const ts = e.timestampMs + ('durationMs' in e ? (e.durationMs ?? 0) : 0);
32
+ return Math.max(max, ts);
33
+ }, 0);
34
+ const timeline = collector.finalize({
35
+ testFile: opts.testFile,
36
+ scenarioFile: opts.scenarioFile,
37
+ recordedAt: new Date().toISOString(),
38
+ viewport,
39
+ videoDurationMs,
40
+ videoFile,
41
+ });
42
+ const timelinePath = join(tempDir, 'timeline.json');
43
+ await writeFile(timelinePath, JSON.stringify(timeline, null, 2));
44
+ await context.close();
45
+ await browser.close();
46
+ return { timeline, videoFile, tempDir };
47
+ }
48
+ //# sourceMappingURL=instrumented-page.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"instrumented-page.js","sourceRoot":"","sources":["../../../src/runtime/instrumented-page.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AACtC,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AACtD,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC;AAC5D,OAAO,EAAE,aAAa,EAA4B,MAAM,qBAAqB,CAAC;AAmB9E,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,QAAoB,EAAE,IAAgB;IACtE,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC;IAC/D,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,eAAe,CAAC,CAAC,CAAC;IAE/D,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;QACpC,IAAI,EAAE,CAAC,eAAe,EAAE,4BAA4B,EAAE,oBAAoB,CAAC;KAC5E,CAAC,CAAC;IAEH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CAAC;QACvC,QAAQ;QACR,iBAAiB,EAAE,CAAC;QACpB,WAAW,EAAE,IAAI,CAAC,WAAW,IAAI,OAAO;QACxC,MAAM,EAAE,IAAI,CAAC,MAAM,IAAI,OAAO;QAC9B,UAAU,EAAE,IAAI,CAAC,UAAU,IAAI,kBAAkB;QACjD,WAAW,EAAE,EAAE,GAAG,EAAE,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE;KAC9C,CAAC,CAAC;IAEH,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,OAAO,EAAE,CAAC;IACrC,MAAM,SAAS,GAAG,IAAI,iBAAiB,EAAE,CAAC;IAE1C,SAAS,CAAC,KAAK,EAAE,CAAC;IAClB,MAAM,EAAE,GAAG,aAAa,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;IAE1C,MAAM,QAAQ,CAAC,EAAE,CAAC,CAAC;IAEnB,+BAA+B;IAC/B,MAAM,IAAI,CAAC,KAAK,EAAE,CAAC;IAEnB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;IAC3B,MAAM,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IAElD,MAAM,eAAe,GAAG,SAAS,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;QAC9D,MAAM,EAAE,GAAG,CAAC,CAAC,WAAW,GAAG,CAAC,YAAY,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACzE,OAAO,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IAC3B,CAAC,EAAE,CAAC,CAAC,CAAC;IAEN,MAAM,QAAQ,GAAG,SAAS,CAAC,QAAQ,CAAC;QAClC,QAAQ,EAAE,IAAI,CAAC,QAAQ;QACvB,YAAY,EAAE,IAAI,CAAC,YAAY;QAC/B,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACpC,QAAQ;QACR,eAAe;QACf,SAAS;KACV,CAAC,CAAC;IAEH,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,EAAE,eAAe,CAAC,CAAC;IACpD,MAAM,SAAS,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IAEjE,MAAM,OAAO,CAAC,KAAK,EAAE,CAAC;IACtB,MAAM,OAAO,CAAC,KAAK,EAAE,CAAC;IAEtB,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,OAAO,EAAE,CAAC;AAC1C,CAAC"}
@@ -0,0 +1,20 @@
1
+ import type { Timeline, TimelineMetadata, TimelineEvent } from '../timeline/types.js';
2
+ type PartialEvent = {
3
+ type: string;
4
+ id?: string;
5
+ timestampMs?: number;
6
+ [key: string]: unknown;
7
+ };
8
+ export declare class TimelineCollector {
9
+ private events;
10
+ private counter;
11
+ private startTime;
12
+ start(): void;
13
+ private elapsed;
14
+ nextId(): string;
15
+ emit(event: PartialEvent): string;
16
+ getEvents(): readonly TimelineEvent[];
17
+ finalize(metadata: TimelineMetadata): Timeline;
18
+ }
19
+ export {};
20
+ //# sourceMappingURL=timeline-collector.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeline-collector.d.ts","sourceRoot":"","sources":["../../../src/runtime/timeline-collector.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,gBAAgB,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAGtF,KAAK,YAAY,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,EAAE,CAAC,EAAE,MAAM,CAAC;IAAC,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;CAAE,CAAC;AAEhG,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,OAAO,CAAK;IACpB,OAAO,CAAC,SAAS,CAAuB;IAExC,KAAK,IAAI,IAAI;IAIb,OAAO,CAAC,OAAO;IAKf,MAAM,IAAI,MAAM;IAIhB,IAAI,CAAC,KAAK,EAAE,YAAY,GAAG,MAAM;IAQjC,SAAS,IAAI,SAAS,aAAa,EAAE;IAIrC,QAAQ,CAAC,QAAQ,EAAE,gBAAgB,GAAG,QAAQ;CAc/C"}
@@ -0,0 +1,40 @@
1
+ import { timelineSchema } from '../timeline/schema.js';
2
+ export class TimelineCollector {
3
+ events = [];
4
+ counter = 0;
5
+ startTime = null;
6
+ start() {
7
+ this.startTime = performance.now();
8
+ }
9
+ elapsed() {
10
+ if (this.startTime === null)
11
+ throw new Error('TimelineCollector not started');
12
+ return Math.round(performance.now() - this.startTime);
13
+ }
14
+ nextId() {
15
+ return `ev-${String(++this.counter).padStart(3, '0')}`;
16
+ }
17
+ emit(event) {
18
+ const id = event.id ?? this.nextId();
19
+ const timestampMs = event.timestampMs ?? this.elapsed();
20
+ const full = { ...event, id, timestampMs };
21
+ this.events.push(full);
22
+ return id;
23
+ }
24
+ getEvents() {
25
+ return this.events;
26
+ }
27
+ finalize(metadata) {
28
+ const timeline = {
29
+ version: 1,
30
+ metadata,
31
+ events: [...this.events],
32
+ };
33
+ const result = timelineSchema.safeParse(timeline);
34
+ if (!result.success) {
35
+ throw new Error(`Invalid timeline: ${result.error.message}`);
36
+ }
37
+ return timeline;
38
+ }
39
+ }
40
+ //# sourceMappingURL=timeline-collector.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeline-collector.js","sourceRoot":"","sources":["../../../src/runtime/timeline-collector.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAIvD,MAAM,OAAO,iBAAiB;IACpB,MAAM,GAAoB,EAAE,CAAC;IAC7B,OAAO,GAAG,CAAC,CAAC;IACZ,SAAS,GAAkB,IAAI,CAAC;IAExC,KAAK;QACH,IAAI,CAAC,SAAS,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;IACrC,CAAC;IAEO,OAAO;QACb,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI;YAAE,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;QAC9E,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC;IACxD,CAAC;IAED,MAAM;QACJ,OAAO,MAAM,MAAM,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;IACzD,CAAC;IAED,IAAI,CAAC,KAAmB;QACtB,MAAM,EAAE,GAAG,KAAK,CAAC,EAAE,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;QACrC,MAAM,WAAW,GAAG,KAAK,CAAC,WAAW,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;QACxD,MAAM,IAAI,GAAG,EAAE,GAAG,KAAK,EAAE,EAAE,EAAE,WAAW,EAAmB,CAAC;QAC5D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACvB,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,QAAQ,CAAC,QAA0B;QACjC,MAAM,QAAQ,GAAa;YACzB,OAAO,EAAE,CAAC;YACV,QAAQ;YACR,MAAM,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC;SACzB,CAAC;QAEF,MAAM,MAAM,GAAG,cAAc,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QAClD,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,qBAAqB,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QAC/D,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF"}
@@ -0,0 +1,297 @@
1
+ import { z } from 'zod';
2
+ export declare const timelineSchema: z.ZodObject<{
3
+ version: z.ZodLiteral<1>;
4
+ metadata: z.ZodObject<{
5
+ testFile: z.ZodString;
6
+ scenarioFile: z.ZodString;
7
+ recordedAt: z.ZodString;
8
+ viewport: z.ZodObject<{
9
+ width: z.ZodNumber;
10
+ height: z.ZodNumber;
11
+ }, "strip", z.ZodTypeAny, {
12
+ width: number;
13
+ height: number;
14
+ }, {
15
+ width: number;
16
+ height: number;
17
+ }>;
18
+ videoDurationMs: z.ZodNumber;
19
+ videoFile: z.ZodString;
20
+ }, "strip", z.ZodTypeAny, {
21
+ testFile: string;
22
+ scenarioFile: string;
23
+ recordedAt: string;
24
+ viewport: {
25
+ width: number;
26
+ height: number;
27
+ };
28
+ videoDurationMs: number;
29
+ videoFile: string;
30
+ }, {
31
+ testFile: string;
32
+ scenarioFile: string;
33
+ recordedAt: string;
34
+ viewport: {
35
+ width: number;
36
+ height: number;
37
+ };
38
+ videoDurationMs: number;
39
+ videoFile: string;
40
+ }>;
41
+ events: z.ZodArray<z.ZodDiscriminatedUnion<"type", [z.ZodObject<{
42
+ type: z.ZodLiteral<"scene">;
43
+ id: z.ZodString;
44
+ timestampMs: z.ZodNumber;
45
+ title: z.ZodString;
46
+ description: z.ZodOptional<z.ZodString>;
47
+ }, "strip", z.ZodTypeAny, {
48
+ type: "scene";
49
+ id: string;
50
+ title: string;
51
+ timestampMs: number;
52
+ description?: string | undefined;
53
+ }, {
54
+ type: "scene";
55
+ id: string;
56
+ title: string;
57
+ timestampMs: number;
58
+ description?: string | undefined;
59
+ }>, z.ZodObject<{
60
+ type: z.ZodLiteral<"action">;
61
+ id: z.ZodString;
62
+ timestampMs: z.ZodNumber;
63
+ action: z.ZodEnum<["click", "fill", "hover", "select", "press", "navigate"]>;
64
+ selector: z.ZodString;
65
+ value: z.ZodOptional<z.ZodString>;
66
+ durationMs: z.ZodNumber;
67
+ boundingBox: z.ZodNullable<z.ZodObject<{
68
+ x: z.ZodNumber;
69
+ y: z.ZodNumber;
70
+ width: z.ZodNumber;
71
+ height: z.ZodNumber;
72
+ }, "strip", z.ZodTypeAny, {
73
+ width: number;
74
+ height: number;
75
+ x: number;
76
+ y: number;
77
+ }, {
78
+ width: number;
79
+ height: number;
80
+ x: number;
81
+ y: number;
82
+ }>>;
83
+ }, "strip", z.ZodTypeAny, {
84
+ type: "action";
85
+ action: "fill" | "click" | "hover" | "select" | "press" | "navigate";
86
+ id: string;
87
+ timestampMs: number;
88
+ selector: string;
89
+ durationMs: number;
90
+ boundingBox: {
91
+ width: number;
92
+ height: number;
93
+ x: number;
94
+ y: number;
95
+ } | null;
96
+ value?: string | undefined;
97
+ }, {
98
+ type: "action";
99
+ action: "fill" | "click" | "hover" | "select" | "press" | "navigate";
100
+ id: string;
101
+ timestampMs: number;
102
+ selector: string;
103
+ durationMs: number;
104
+ boundingBox: {
105
+ width: number;
106
+ height: number;
107
+ x: number;
108
+ y: number;
109
+ } | null;
110
+ value?: string | undefined;
111
+ }>, z.ZodObject<{
112
+ type: z.ZodLiteral<"cursor_target">;
113
+ id: z.ZodString;
114
+ timestampMs: z.ZodNumber;
115
+ fromX: z.ZodNumber;
116
+ fromY: z.ZodNumber;
117
+ toX: z.ZodNumber;
118
+ toY: z.ZodNumber;
119
+ moveDurationMs: z.ZodNumber;
120
+ easing: z.ZodLiteral<"bezier">;
121
+ }, "strip", z.ZodTypeAny, {
122
+ type: "cursor_target";
123
+ id: string;
124
+ timestampMs: number;
125
+ fromX: number;
126
+ fromY: number;
127
+ toX: number;
128
+ toY: number;
129
+ moveDurationMs: number;
130
+ easing: "bezier";
131
+ }, {
132
+ type: "cursor_target";
133
+ id: string;
134
+ timestampMs: number;
135
+ fromX: number;
136
+ fromY: number;
137
+ toX: number;
138
+ toY: number;
139
+ moveDurationMs: number;
140
+ easing: "bezier";
141
+ }>, z.ZodObject<{
142
+ type: z.ZodLiteral<"narration">;
143
+ id: z.ZodString;
144
+ timestampMs: z.ZodNumber;
145
+ text: z.ZodString;
146
+ audioDurationMs: z.ZodOptional<z.ZodNumber>;
147
+ audioFile: z.ZodOptional<z.ZodString>;
148
+ }, "strip", z.ZodTypeAny, {
149
+ type: "narration";
150
+ id: string;
151
+ timestampMs: number;
152
+ text: string;
153
+ audioDurationMs?: number | undefined;
154
+ audioFile?: string | undefined;
155
+ }, {
156
+ type: "narration";
157
+ id: string;
158
+ timestampMs: number;
159
+ text: string;
160
+ audioDurationMs?: number | undefined;
161
+ audioFile?: string | undefined;
162
+ }>, z.ZodObject<{
163
+ type: z.ZodLiteral<"wait">;
164
+ id: z.ZodString;
165
+ timestampMs: z.ZodNumber;
166
+ durationMs: z.ZodNumber;
167
+ reason: z.ZodEnum<["pacing", "narration_sync", "page_load"]>;
168
+ }, "strip", z.ZodTypeAny, {
169
+ type: "wait";
170
+ id: string;
171
+ timestampMs: number;
172
+ durationMs: number;
173
+ reason: "pacing" | "narration_sync" | "page_load";
174
+ }, {
175
+ type: "wait";
176
+ id: string;
177
+ timestampMs: number;
178
+ durationMs: number;
179
+ reason: "pacing" | "narration_sync" | "page_load";
180
+ }>]>, "many">;
181
+ }, "strip", z.ZodTypeAny, {
182
+ metadata: {
183
+ testFile: string;
184
+ scenarioFile: string;
185
+ recordedAt: string;
186
+ viewport: {
187
+ width: number;
188
+ height: number;
189
+ };
190
+ videoDurationMs: number;
191
+ videoFile: string;
192
+ };
193
+ version: 1;
194
+ events: ({
195
+ type: "scene";
196
+ id: string;
197
+ title: string;
198
+ timestampMs: number;
199
+ description?: string | undefined;
200
+ } | {
201
+ type: "action";
202
+ action: "fill" | "click" | "hover" | "select" | "press" | "navigate";
203
+ id: string;
204
+ timestampMs: number;
205
+ selector: string;
206
+ durationMs: number;
207
+ boundingBox: {
208
+ width: number;
209
+ height: number;
210
+ x: number;
211
+ y: number;
212
+ } | null;
213
+ value?: string | undefined;
214
+ } | {
215
+ type: "cursor_target";
216
+ id: string;
217
+ timestampMs: number;
218
+ fromX: number;
219
+ fromY: number;
220
+ toX: number;
221
+ toY: number;
222
+ moveDurationMs: number;
223
+ easing: "bezier";
224
+ } | {
225
+ type: "narration";
226
+ id: string;
227
+ timestampMs: number;
228
+ text: string;
229
+ audioDurationMs?: number | undefined;
230
+ audioFile?: string | undefined;
231
+ } | {
232
+ type: "wait";
233
+ id: string;
234
+ timestampMs: number;
235
+ durationMs: number;
236
+ reason: "pacing" | "narration_sync" | "page_load";
237
+ })[];
238
+ }, {
239
+ metadata: {
240
+ testFile: string;
241
+ scenarioFile: string;
242
+ recordedAt: string;
243
+ viewport: {
244
+ width: number;
245
+ height: number;
246
+ };
247
+ videoDurationMs: number;
248
+ videoFile: string;
249
+ };
250
+ version: 1;
251
+ events: ({
252
+ type: "scene";
253
+ id: string;
254
+ title: string;
255
+ timestampMs: number;
256
+ description?: string | undefined;
257
+ } | {
258
+ type: "action";
259
+ action: "fill" | "click" | "hover" | "select" | "press" | "navigate";
260
+ id: string;
261
+ timestampMs: number;
262
+ selector: string;
263
+ durationMs: number;
264
+ boundingBox: {
265
+ width: number;
266
+ height: number;
267
+ x: number;
268
+ y: number;
269
+ } | null;
270
+ value?: string | undefined;
271
+ } | {
272
+ type: "cursor_target";
273
+ id: string;
274
+ timestampMs: number;
275
+ fromX: number;
276
+ fromY: number;
277
+ toX: number;
278
+ toY: number;
279
+ moveDurationMs: number;
280
+ easing: "bezier";
281
+ } | {
282
+ type: "narration";
283
+ id: string;
284
+ timestampMs: number;
285
+ text: string;
286
+ audioDurationMs?: number | undefined;
287
+ audioFile?: string | undefined;
288
+ } | {
289
+ type: "wait";
290
+ id: string;
291
+ timestampMs: number;
292
+ durationMs: number;
293
+ reason: "pacing" | "narration_sync" | "page_load";
294
+ })[];
295
+ }>;
296
+ export type ValidatedTimeline = z.infer<typeof timelineSchema>;
297
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../../src/timeline/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AA+DxB,eAAO,MAAM,cAAc;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAczB,CAAC;AAEH,MAAM,MAAM,iBAAiB,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,cAAc,CAAC,CAAC"}