sakuga 0.0.1
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/LICENSE +21 -0
- package/README.md +9 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +233 -0
- package/dist/index.d.ts +82 -0
- package/dist/index.js +237 -0
- package/dist/shared/chunk-ks7faw1s.js +1782 -0
- package/package.json +69 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
import {
|
|
2
|
+
CanvasContextUnavailable,
|
|
3
|
+
DEFAULT_BLOCK_DURATION,
|
|
4
|
+
DEFAULT_FPS,
|
|
5
|
+
DEFAULT_HEIGHT,
|
|
6
|
+
DEFAULT_THEME,
|
|
7
|
+
DEFAULT_TRANSITION_DURATION_MS,
|
|
8
|
+
DEFAULT_WIDTH,
|
|
9
|
+
MissingCanvasFactory,
|
|
10
|
+
NoEncodableVideoCodec,
|
|
11
|
+
OutputBufferMissing,
|
|
12
|
+
WebCodecsUnavailable,
|
|
13
|
+
blendColors,
|
|
14
|
+
buildScene,
|
|
15
|
+
buildTransitionDiff,
|
|
16
|
+
buildTransitionTokens,
|
|
17
|
+
diffLayoutTokens,
|
|
18
|
+
easeInOutCubic,
|
|
19
|
+
parseMarkdownCodeBlocks,
|
|
20
|
+
renderFrame,
|
|
21
|
+
resolveTheme
|
|
22
|
+
} from "./shared/chunk-ks7faw1s.js";
|
|
23
|
+
// src/lib/render-api.ts
|
|
24
|
+
import { Effect as Effect2 } from "effect";
|
|
25
|
+
|
|
26
|
+
// src/lib/browser.ts
|
|
27
|
+
import { Effect, Ref, Stream } from "effect";
|
|
28
|
+
import {
|
|
29
|
+
BufferTarget,
|
|
30
|
+
Mp4OutputFormat,
|
|
31
|
+
Output,
|
|
32
|
+
QUALITY_HIGH,
|
|
33
|
+
VideoSample,
|
|
34
|
+
VideoSampleSource,
|
|
35
|
+
WebMOutputFormat,
|
|
36
|
+
getFirstEncodableVideoCodec
|
|
37
|
+
} from "mediabunny";
|
|
38
|
+
var getCanvasContext = (canvas) => Effect.gen(function* () {
|
|
39
|
+
const context = canvas.getContext("2d");
|
|
40
|
+
if (!context) {
|
|
41
|
+
return yield* Effect.fail(new CanvasContextUnavailable({ reason: "Unable to acquire 2D canvas context." }));
|
|
42
|
+
}
|
|
43
|
+
return context;
|
|
44
|
+
});
|
|
45
|
+
var ensureWebCodecs = () => Effect.gen(function* () {
|
|
46
|
+
if (typeof VideoEncoder === "undefined" || typeof VideoFrame === "undefined") {
|
|
47
|
+
return yield* Effect.fail(new WebCodecsUnavailable({
|
|
48
|
+
reason: "WebCodecs VideoEncoder is not available in this browser."
|
|
49
|
+
}));
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
var computeFrameCounts = (transitionDurationMs, fps) => {
|
|
53
|
+
const frameDuration = 1 / fps;
|
|
54
|
+
const blockFrames = Math.max(1, Math.round(DEFAULT_BLOCK_DURATION * fps));
|
|
55
|
+
const transitionFrames = Math.max(1, Math.round(transitionDurationMs / 1000 * fps));
|
|
56
|
+
return {
|
|
57
|
+
blockFrames,
|
|
58
|
+
frameDuration,
|
|
59
|
+
transitionFrames
|
|
60
|
+
};
|
|
61
|
+
};
|
|
62
|
+
var makeOutput = (format) => Effect.sync(() => {
|
|
63
|
+
const resolvedFormat = format === "webm" ? new WebMOutputFormat : new Mp4OutputFormat;
|
|
64
|
+
const target = new BufferTarget;
|
|
65
|
+
const outputInstance = new Output({
|
|
66
|
+
format: resolvedFormat,
|
|
67
|
+
target
|
|
68
|
+
});
|
|
69
|
+
return {
|
|
70
|
+
mimeType: resolvedFormat.mimeType,
|
|
71
|
+
output: outputInstance,
|
|
72
|
+
target
|
|
73
|
+
};
|
|
74
|
+
});
|
|
75
|
+
var makeVideoSource = (codec) => Effect.sync(() => new VideoSampleSource({
|
|
76
|
+
bitrate: QUALITY_HIGH,
|
|
77
|
+
codec
|
|
78
|
+
}));
|
|
79
|
+
var renderAndWriteFrame = (context, videoSource, frameDuration, frameIndexRef, width, height) => (frame) => Effect.gen(function* () {
|
|
80
|
+
const frameIndex = yield* Ref.get(frameIndexRef);
|
|
81
|
+
renderFrame(context, width, height, frame);
|
|
82
|
+
const imageData = context.getImageData(0, 0, width, height);
|
|
83
|
+
const sample = new VideoSample(imageData.data, {
|
|
84
|
+
codedHeight: height,
|
|
85
|
+
codedWidth: width,
|
|
86
|
+
duration: frameDuration,
|
|
87
|
+
format: "RGBA",
|
|
88
|
+
timestamp: frameIndex * frameDuration
|
|
89
|
+
});
|
|
90
|
+
yield* Effect.tryPromise({
|
|
91
|
+
catch: (error) => error instanceof Error ? error : new Error(String(error)),
|
|
92
|
+
try: () => videoSource.add(sample)
|
|
93
|
+
});
|
|
94
|
+
sample.close();
|
|
95
|
+
yield* Ref.set(frameIndexRef, frameIndex + 1);
|
|
96
|
+
});
|
|
97
|
+
var buildSceneFrames = (scene, blockFrames) => Stream.range(1, blockFrames).pipe(Stream.map(() => ({
|
|
98
|
+
background: scene.background,
|
|
99
|
+
kind: "scene",
|
|
100
|
+
opacity: 1,
|
|
101
|
+
positionX: scene.blockX,
|
|
102
|
+
positionY: scene.blockY,
|
|
103
|
+
scene
|
|
104
|
+
})));
|
|
105
|
+
var buildTransitionFrames = (scene, nextScene, transitionFrames) => {
|
|
106
|
+
const diff = buildTransitionDiff(scene, nextScene);
|
|
107
|
+
return Stream.range(1, transitionFrames).pipe(Stream.map((index) => {
|
|
108
|
+
const rawProgress = index / transitionFrames;
|
|
109
|
+
const progress = easeInOutCubic(rawProgress);
|
|
110
|
+
const blendedBackground = blendColors(scene.background, nextScene.background, progress);
|
|
111
|
+
const tokens = buildTransitionTokens(diff, progress);
|
|
112
|
+
return {
|
|
113
|
+
background: blendedBackground,
|
|
114
|
+
kind: "transition",
|
|
115
|
+
tokens
|
|
116
|
+
};
|
|
117
|
+
}));
|
|
118
|
+
};
|
|
119
|
+
var buildFramesStream = (scenes, blockFrames, transitionFrames) => Stream.fromIterable(scenes).pipe(Stream.zipWithIndex, Stream.flatMap(([scene, index]) => {
|
|
120
|
+
const nextScene = scenes[index + 1];
|
|
121
|
+
const base = buildSceneFrames(scene, blockFrames);
|
|
122
|
+
return nextScene ? Stream.concat(base, buildTransitionFrames(scene, nextScene, transitionFrames)) : base;
|
|
123
|
+
}));
|
|
124
|
+
var resolveFormat = (format, height, width) => Effect.gen(function* () {
|
|
125
|
+
if (format !== "auto") {
|
|
126
|
+
return {
|
|
127
|
+
codec: format === "mp4" ? "avc" : "vp9",
|
|
128
|
+
container: format
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
const codec = yield* Effect.tryPromise({
|
|
132
|
+
catch: () => new NoEncodableVideoCodec({
|
|
133
|
+
reason: "No encodable video codec available in this browser."
|
|
134
|
+
}),
|
|
135
|
+
try: () => getFirstEncodableVideoCodec(["avc", "vp9", "vp8"], {
|
|
136
|
+
bitrate: QUALITY_HIGH,
|
|
137
|
+
height,
|
|
138
|
+
width
|
|
139
|
+
})
|
|
140
|
+
});
|
|
141
|
+
if (!codec) {
|
|
142
|
+
return yield* Effect.fail(new NoEncodableVideoCodec({
|
|
143
|
+
reason: "No encodable video codec available in this browser."
|
|
144
|
+
}));
|
|
145
|
+
}
|
|
146
|
+
return {
|
|
147
|
+
codec,
|
|
148
|
+
container: codec === "avc" ? "mp4" : "webm"
|
|
149
|
+
};
|
|
150
|
+
});
|
|
151
|
+
var resolveCanvas = (canvas, height, width, createCanvas) => Effect.gen(function* () {
|
|
152
|
+
if (canvas) {
|
|
153
|
+
const context2 = yield* getCanvasContext(canvas);
|
|
154
|
+
return {
|
|
155
|
+
canvas,
|
|
156
|
+
context: context2
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
if (!createCanvas) {
|
|
160
|
+
return yield* Effect.fail(new MissingCanvasFactory({
|
|
161
|
+
reason: "Browser canvas factory is required when no canvas is provided."
|
|
162
|
+
}));
|
|
163
|
+
}
|
|
164
|
+
const created = createCanvas(height, width);
|
|
165
|
+
const context = yield* getCanvasContext(created);
|
|
166
|
+
return {
|
|
167
|
+
canvas: created,
|
|
168
|
+
context
|
|
169
|
+
};
|
|
170
|
+
});
|
|
171
|
+
var renderVideoBrowser = (theme, codeBlocks, options = {}) => Effect.gen(function* () {
|
|
172
|
+
yield* ensureWebCodecs();
|
|
173
|
+
const width = options.width ?? DEFAULT_WIDTH;
|
|
174
|
+
const height = options.height ?? DEFAULT_HEIGHT;
|
|
175
|
+
const transitionDurationMs = options.transitionDurationMs ?? DEFAULT_TRANSITION_DURATION_MS;
|
|
176
|
+
const format = options.format ?? "auto";
|
|
177
|
+
const { context } = yield* resolveCanvas(options.canvas, height, width, options.createCanvas);
|
|
178
|
+
const resolved = yield* resolveFormat(format, height, width);
|
|
179
|
+
const outputInfo = yield* makeOutput(resolved.container);
|
|
180
|
+
const frameCounts = computeFrameCounts(transitionDurationMs, DEFAULT_FPS);
|
|
181
|
+
const frameIndexRef = yield* Ref.make(0);
|
|
182
|
+
const scenes = yield* Effect.forEach(codeBlocks, (codeBlock) => buildScene(context, codeBlock, theme, width, height), { concurrency: options.concurrency });
|
|
183
|
+
const videoSource = yield* makeVideoSource(resolved.codec);
|
|
184
|
+
return yield* Effect.scoped(Effect.gen(function* () {
|
|
185
|
+
yield* Effect.sync(() => {
|
|
186
|
+
outputInfo.output.addVideoTrack(videoSource, { frameRate: DEFAULT_FPS });
|
|
187
|
+
});
|
|
188
|
+
yield* Effect.tryPromise({
|
|
189
|
+
catch: (error) => error instanceof Error ? error : new Error(String(error)),
|
|
190
|
+
try: () => outputInfo.output.start()
|
|
191
|
+
});
|
|
192
|
+
const frameStream = buildFramesStream(scenes, frameCounts.blockFrames, frameCounts.transitionFrames);
|
|
193
|
+
yield* Stream.runForEach(frameStream, renderAndWriteFrame(context, videoSource, frameCounts.frameDuration, frameIndexRef, width, height));
|
|
194
|
+
yield* Effect.tryPromise({
|
|
195
|
+
catch: (error) => error instanceof Error ? error : new Error(String(error)),
|
|
196
|
+
try: () => outputInfo.output.finalize()
|
|
197
|
+
});
|
|
198
|
+
videoSource.close();
|
|
199
|
+
const buffer = outputInfo.target.buffer;
|
|
200
|
+
if (!buffer) {
|
|
201
|
+
return yield* Effect.fail(new OutputBufferMissing({
|
|
202
|
+
reason: "Output buffer missing after finalize."
|
|
203
|
+
}));
|
|
204
|
+
}
|
|
205
|
+
const output = {
|
|
206
|
+
data: new Uint8Array(buffer),
|
|
207
|
+
extension: resolved.container,
|
|
208
|
+
mimeType: outputInfo.mimeType
|
|
209
|
+
};
|
|
210
|
+
return output;
|
|
211
|
+
}));
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
// src/lib/render-api.ts
|
|
215
|
+
var render = (options) => Effect2.runPromise(Effect2.gen(function* () {
|
|
216
|
+
const theme = options.theme ?? DEFAULT_THEME;
|
|
217
|
+
const resolvedTheme = yield* resolveTheme(theme);
|
|
218
|
+
const blocks = yield* parseMarkdownCodeBlocks(options.markdown);
|
|
219
|
+
const result = yield* renderVideoBrowser(resolvedTheme, blocks, {
|
|
220
|
+
canvas: options.canvas,
|
|
221
|
+
concurrency: options.concurrency,
|
|
222
|
+
createCanvas: options.createCanvas,
|
|
223
|
+
format: options.format,
|
|
224
|
+
height: options.height,
|
|
225
|
+
transitionDurationMs: options.transitionDurationMs,
|
|
226
|
+
width: options.width
|
|
227
|
+
});
|
|
228
|
+
return result;
|
|
229
|
+
}));
|
|
230
|
+
export {
|
|
231
|
+
render,
|
|
232
|
+
parseMarkdownCodeBlocks,
|
|
233
|
+
diffLayoutTokens
|
|
234
|
+
};
|
|
235
|
+
|
|
236
|
+
//# debugId=30C1AF8D08558A9464756E2164756E21
|
|
237
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/lib/render-api.ts", "src/lib/browser.ts"],
  "sourcesContent": [
    "import { Effect } from \"effect\"\nimport { renderVideoBrowser, type RenderVideoBrowserOptions } from \"./browser\"\nimport { DEFAULT_THEME } from \"./constants\"\nimport { parseMarkdownCodeBlocks } from \"./markdown\"\nimport { resolveTheme } from \"./theme\"\n\nexport type RenderOptions = RenderVideoBrowserOptions & {\n  markdown: string\n  theme?: string\n}\n\nexport type RenderResult =\n  ReturnType<typeof renderVideoBrowser> extends Effect.Effect<infer A, unknown> ? A : never\n\nexport const render = (options: RenderOptions) =>\n  Effect.runPromise(\n    Effect.gen(function* () {\n      const theme = options.theme ?? DEFAULT_THEME\n      const resolvedTheme = yield* resolveTheme(theme)\n      const blocks = yield* parseMarkdownCodeBlocks(options.markdown)\n\n      const result = yield* renderVideoBrowser(resolvedTheme, blocks, {\n        canvas: options.canvas,\n        concurrency: options.concurrency,\n        createCanvas: options.createCanvas,\n        format: options.format,\n        height: options.height,\n        transitionDurationMs: options.transitionDurationMs,\n        width: options.width,\n      })\n\n      return result\n    })\n  )\n",
    "import { Effect, Ref, Stream } from \"effect\"\nimport {\n  BufferTarget,\n  Mp4OutputFormat,\n  Output,\n  QUALITY_HIGH,\n  VideoSample,\n  VideoSampleSource,\n  WebMOutputFormat,\n  getFirstEncodableVideoCodec,\n} from \"mediabunny\"\nimport type { CanvasContext } from \"./context\"\nimport type { CodeBlock, RenderFrame, Scene } from \"./types\"\nimport { blendColors } from \"./color\"\nimport {\n  DEFAULT_BLOCK_DURATION,\n  DEFAULT_FPS,\n  DEFAULT_HEIGHT,\n  DEFAULT_TRANSITION_DURATION_MS,\n  DEFAULT_WIDTH,\n} from \"./constants\"\nimport {\n  CanvasContextUnavailable,\n  MissingCanvasFactory,\n  NoEncodableVideoCodec,\n  OutputBufferMissing,\n  WebCodecsUnavailable,\n} from \"./errors\"\nimport { renderFrame } from \"./render\"\nimport { buildScene } from \"./scene\"\nimport { buildTransitionDiff, buildTransitionTokens, easeInOutCubic } from \"./transition\"\n\nexport type BrowserFormat = \"mp4\" | \"webm\" | \"auto\"\n\nexport type CanvasLike = {\n  getContext: (type: \"2d\") => CanvasContext | null\n  height: number\n  width: number\n}\n\nexport type CanvasFactory = (height: number, width: number) => CanvasLike\n\nexport type BrowserOutput = {\n  data: Uint8Array\n  extension: \"mp4\" | \"webm\"\n  mimeType: string\n}\n\nexport type RenderVideoBrowserOptions = {\n  canvas?: CanvasLike\n  concurrency?: number\n  createCanvas?: CanvasFactory\n  format?: BrowserFormat\n  height?: number\n  transitionDurationMs?: number\n  width?: number\n}\n\nconst getCanvasContext = (canvas: CanvasLike) =>\n  Effect.gen(function* () {\n    const context = canvas.getContext(\"2d\")\n    if (!context) {\n      return yield* Effect.fail(\n        new CanvasContextUnavailable({ reason: \"Unable to acquire 2D canvas context.\" })\n      )\n    }\n\n    return context\n  })\n\nconst ensureWebCodecs = () =>\n  Effect.gen(function* () {\n    if (typeof VideoEncoder === \"undefined\" || typeof VideoFrame === \"undefined\") {\n      return yield* Effect.fail(\n        new WebCodecsUnavailable({\n          reason: \"WebCodecs VideoEncoder is not available in this browser.\",\n        })\n      )\n    }\n  })\n\nconst computeFrameCounts = (transitionDurationMs: number, fps: number) => {\n  const frameDuration = 1 / fps\n  const blockFrames = Math.max(1, Math.round(DEFAULT_BLOCK_DURATION * fps))\n  const transitionFrames = Math.max(1, Math.round((transitionDurationMs / 1000) * fps))\n\n  return {\n    blockFrames,\n    frameDuration,\n    transitionFrames,\n  }\n}\n\nconst makeOutput = (format: \"mp4\" | \"webm\") =>\n  Effect.sync(() => {\n    const resolvedFormat = format === \"webm\" ? new WebMOutputFormat() : new Mp4OutputFormat()\n    const target = new BufferTarget()\n\n    const outputInstance = new Output({\n      format: resolvedFormat,\n      target,\n    })\n\n    return {\n      mimeType: resolvedFormat.mimeType,\n      output: outputInstance,\n      target,\n    }\n  })\n\nconst makeVideoSource = (codec: ResolvedCodec) =>\n  Effect.sync(\n    () =>\n      new VideoSampleSource({\n        bitrate: QUALITY_HIGH,\n        codec,\n      })\n  )\n\nconst renderAndWriteFrame =\n  (\n    context: CanvasContext,\n    videoSource: VideoSampleSource,\n    frameDuration: number,\n    frameIndexRef: Ref.Ref<number>,\n    width: number,\n    height: number\n  ) =>\n  (frame: RenderFrame) =>\n    Effect.gen(function* () {\n      const frameIndex = yield* Ref.get(frameIndexRef)\n      renderFrame(context, width, height, frame)\n\n      const imageData = context.getImageData(0, 0, width, height)\n      const sample = new VideoSample(imageData.data, {\n        codedHeight: height,\n        codedWidth: width,\n        duration: frameDuration,\n        format: \"RGBA\",\n        timestamp: frameIndex * frameDuration,\n      })\n\n      yield* Effect.tryPromise({\n        catch: (error) => (error instanceof Error ? error : new Error(String(error))),\n        try: () => videoSource.add(sample),\n      })\n\n      sample.close()\n      yield* Ref.set(frameIndexRef, frameIndex + 1)\n    })\n\nconst buildSceneFrames = (scene: Scene, blockFrames: number) =>\n  Stream.range(1, blockFrames).pipe(\n    Stream.map(\n      () =>\n        ({\n          background: scene.background,\n          kind: \"scene\",\n          opacity: 1,\n          positionX: scene.blockX,\n          positionY: scene.blockY,\n          scene,\n        }) satisfies RenderFrame\n    )\n  )\n\nconst buildTransitionFrames = (scene: Scene, nextScene: Scene, transitionFrames: number) => {\n  const diff = buildTransitionDiff(scene, nextScene)\n\n  return Stream.range(1, transitionFrames).pipe(\n    Stream.map((index) => {\n      const rawProgress = index / transitionFrames\n      const progress = easeInOutCubic(rawProgress)\n      const blendedBackground = blendColors(scene.background, nextScene.background, progress)\n      const tokens = buildTransitionTokens(diff, progress)\n\n      return {\n        background: blendedBackground,\n        kind: \"transition\",\n        tokens,\n      } satisfies RenderFrame\n    })\n  )\n}\n\nconst buildFramesStream = (scenes: Scene[], blockFrames: number, transitionFrames: number) =>\n  Stream.fromIterable(scenes).pipe(\n    Stream.zipWithIndex,\n    Stream.flatMap(([scene, index]) => {\n      const nextScene = scenes[index + 1]\n      const base = buildSceneFrames(scene, blockFrames)\n      return nextScene\n        ? Stream.concat(base, buildTransitionFrames(scene, nextScene, transitionFrames))\n        : base\n    })\n  )\n\ntype ResolvedCodec = \"avc\" | \"vp9\" | \"vp8\" | \"hevc\" | \"av1\"\n\ntype ResolvedFormat = {\n  codec: ResolvedCodec\n  container: \"mp4\" | \"webm\"\n}\n\nconst resolveFormat = (format: BrowserFormat, height: number, width: number) =>\n  Effect.gen(function* () {\n    if (format !== \"auto\") {\n      return {\n        codec: format === \"mp4\" ? \"avc\" : \"vp9\",\n        container: format,\n      } satisfies ResolvedFormat\n    }\n\n    const codec = yield* Effect.tryPromise({\n      catch: () =>\n        new NoEncodableVideoCodec({\n          reason: \"No encodable video codec available in this browser.\",\n        }),\n      try: () =>\n        getFirstEncodableVideoCodec([\"avc\", \"vp9\", \"vp8\"], {\n          bitrate: QUALITY_HIGH,\n          height,\n          width,\n        }),\n    })\n\n    if (!codec) {\n      return yield* Effect.fail(\n        new NoEncodableVideoCodec({\n          reason: \"No encodable video codec available in this browser.\",\n        })\n      )\n    }\n\n    return {\n      codec,\n      container: codec === \"avc\" ? \"mp4\" : \"webm\",\n    } satisfies ResolvedFormat\n  })\n\nconst resolveCanvas = (\n  canvas: CanvasLike | undefined,\n  height: number,\n  width: number,\n  createCanvas?: CanvasFactory\n) =>\n  Effect.gen(function* () {\n    if (canvas) {\n      const context = yield* getCanvasContext(canvas)\n\n      return {\n        canvas,\n        context,\n      }\n    }\n\n    if (!createCanvas) {\n      return yield* Effect.fail(\n        new MissingCanvasFactory({\n          reason: \"Browser canvas factory is required when no canvas is provided.\",\n        })\n      )\n    }\n\n    const created = createCanvas(height, width)\n    const context = yield* getCanvasContext(created)\n\n    return {\n      canvas: created,\n      context,\n    }\n  })\n\nexport const renderVideoBrowser = (\n  theme: string,\n  codeBlocks: CodeBlock[],\n  options: RenderVideoBrowserOptions = {}\n) =>\n  Effect.gen(function* () {\n    yield* ensureWebCodecs()\n\n    const width = options.width ?? DEFAULT_WIDTH\n    const height = options.height ?? DEFAULT_HEIGHT\n    const transitionDurationMs = options.transitionDurationMs ?? DEFAULT_TRANSITION_DURATION_MS\n    const format = options.format ?? \"auto\"\n\n    const { context } = yield* resolveCanvas(options.canvas, height, width, options.createCanvas)\n    const resolved = yield* resolveFormat(format, height, width)\n    const outputInfo = yield* makeOutput(resolved.container)\n    const frameCounts = computeFrameCounts(transitionDurationMs, DEFAULT_FPS)\n    const frameIndexRef = yield* Ref.make(0)\n\n    const scenes = yield* Effect.forEach(\n      codeBlocks,\n      (codeBlock) => buildScene(context, codeBlock, theme as never, width, height),\n      { concurrency: options.concurrency }\n    )\n\n    const videoSource = yield* makeVideoSource(resolved.codec)\n\n    return yield* Effect.scoped(\n      Effect.gen(function* () {\n        yield* Effect.sync(() => {\n          outputInfo.output.addVideoTrack(videoSource, { frameRate: DEFAULT_FPS })\n        })\n\n        yield* Effect.tryPromise({\n          catch: (error) => (error instanceof Error ? error : new Error(String(error))),\n          try: () => outputInfo.output.start(),\n        })\n\n        const frameStream = buildFramesStream(\n          scenes,\n          frameCounts.blockFrames,\n          frameCounts.transitionFrames\n        )\n\n        yield* Stream.runForEach(\n          frameStream,\n          renderAndWriteFrame(\n            context,\n            videoSource,\n            frameCounts.frameDuration,\n            frameIndexRef,\n            width,\n            height\n          )\n        )\n\n        yield* Effect.tryPromise({\n          catch: (error) => (error instanceof Error ? error : new Error(String(error))),\n          try: () => outputInfo.output.finalize(),\n        })\n\n        videoSource.close()\n\n        const buffer = outputInfo.target.buffer\n        if (!buffer) {\n          return yield* Effect.fail(\n            new OutputBufferMissing({\n              reason: \"Output buffer missing after finalize.\",\n            })\n          )\n        }\n\n        const output: BrowserOutput = {\n          data: new Uint8Array(buffer),\n          extension: resolved.container,\n          mimeType: outputInfo.mimeType,\n        }\n\n        return output\n      })\n    )\n  })\n"
  ],
  "mappings": "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",
  "debugId": "30C1AF8D08558A9464756E2164756E21",
  "names": []
}
|