@elah/core 0.1.0 → 0.2.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 ADDED
@@ -0,0 +1,82 @@
1
+ # @elah/core
2
+
3
+ Framework-agnostic video timeline engine. No React. No renderer. Just the pure logic layer — project state, playback, frame resolution, and media management.
4
+
5
+ Used internally by `@elah/timeline` and `@elah/editor`, but can be consumed directly for custom rendering pipelines or headless environments.
6
+
7
+ [![npm](https://img.shields.io/npm/v/@elah/core)](https://www.npmjs.com/package/@elah/core)
8
+ [![license](https://img.shields.io/badge/license-ECL--1.0-blue)](https://github.com/elahlabs/elah/blob/main/LICENSE)
9
+
10
+ ---
11
+
12
+ ## Install
13
+
14
+ ```bash
15
+ npm install @elah/core
16
+ ```
17
+
18
+ ---
19
+
20
+ ## What's inside
21
+
22
+ | Module | Description |
23
+ |---|---|
24
+ | `TimelineEngine` | Manages project state — tracks, clips, undo/redo |
25
+ | `PlaybackEngine` | Frame-accurate playback clock |
26
+ | `resolveTimeline` | Pure function — project → active scene at a given frame |
27
+ | `GpuRenderer` | WebGL2 renderer for video, image, and text layers |
28
+ | `useTracksStore` | Zustand mirror of project state for React |
29
+ | `usePlaybackStore` | Zustand mirror of playback state for React |
30
+ | `useSelectionStore` | Zustand mirror of selection state for React |
31
+ | `useMediaLibrary` | Media asset library with thumbnail generation |
32
+ | `importFiles` | Import local files into the media library |
33
+ | `exportVideo` | Export the timeline to MP4 via a web worker |
34
+
35
+ ---
36
+
37
+ ## Quick start
38
+
39
+ ```ts
40
+ import { TimelineEngine, PlaybackEngine, createVideoClip } from '@elah/core'
41
+
42
+ const engine = new TimelineEngine({ fps: 30, stage: { width: 1920, height: 1080 } })
43
+ const playback = new PlaybackEngine(engine)
44
+
45
+ // Add a video clip
46
+ engine.addClip('track-1', createVideoClip({ src: 'video.mp4', startFrame: 0, durationFrames: 90 }))
47
+
48
+ // Resolve the scene at a given frame
49
+ import { resolveTimeline } from '@elah/core'
50
+ const scene = resolveTimeline(engine.getProject(), { currentFrame: 15 })
51
+ ```
52
+
53
+ ---
54
+
55
+ ## Clip factories
56
+
57
+ ```ts
58
+ import { createVideoClip, createAudioClip, createTextClip, createImageClip } from '@elah/core'
59
+ ```
60
+
61
+ ---
62
+
63
+ ## Export
64
+
65
+ ```ts
66
+ import { exportVideo } from '@elah/core'
67
+
68
+ const blob = await exportVideo(engine.getProject(), {
69
+ videoBitrate: 8_000_000,
70
+ onProgress: ({ frame, totalFrames }) => console.log(frame, '/', totalFrames),
71
+ })
72
+ ```
73
+
74
+ ---
75
+
76
+ ## Links
77
+
78
+ - [Website](https://www.elah.dev)
79
+ - [GitHub](https://github.com/elahlabs/elah)
80
+ - [Full SDK — @elah/editor](https://www.npmjs.com/package/@elah/editor)
81
+ - [License](https://github.com/elahlabs/elah/blob/main/LICENSE)
82
+ - [Commercial licensing](mailto:contact@elah.dev)
@@ -275,7 +275,13 @@ async function renderFrame(ctx, scene, videoSinks, imageBitmaps, transitionSnaps
275
275
  });
276
276
  }
277
277
  const seekStart = isDebugFrame ? performance.now() : 0;
278
- const wrapped = await sink.getCanvas(sourceTimeSec);
278
+ let wrapped = null;
279
+ try {
280
+ wrapped = await sink.getCanvas(sourceTimeSec);
281
+ }
282
+ catch (err) {
283
+ xlog('render:frame0', `video layer — getCanvas() failed (frame skipped): ${String(err)}`);
284
+ }
279
285
  if (isDebugFrame) {
280
286
  xlog('render:frame0', `video layer — getCanvas()`, {
281
287
  gotFrame: !!wrapped,
@@ -67,7 +67,7 @@ export async function exportVideo(project, options = {}) {
67
67
  return Promise.reject(signal.reason ?? new DOMException('Export aborted', 'AbortError'));
68
68
  return new Promise((resolve, reject) => {
69
69
  mlog('EXPORT', 'spawning ExportWorker (module worker)');
70
- const worker = new Worker(new URL('./ExportWorker.ts', import.meta.url), { type: 'module' });
70
+ const worker = new Worker(new URL('./ExportWorker.js', import.meta.url), { type: 'module' });
71
71
  const abort = () => {
72
72
  worker.terminate();
73
73
  mlog('EXPORT', 'aborted by signal');
package/dist/index.d.ts CHANGED
@@ -14,6 +14,7 @@ export type { TextLayout } from './renderer/gpu/layers/textLayout';
14
14
  export type { DemuxerBackend as MediabunnyDemuxer } from './media/video/demuxer/MediabunnyDemuxer';
15
15
  export { GpuDebugCounters } from './renderer/gpu/debug/GpuDebugCounters';
16
16
  export type { CounterSnapshot } from './renderer/gpu/debug/GpuDebugCounters';
17
+ export { createDefaultDemuxerFactory } from './media/video/demuxer/createDefaultDemuxerFactory';
17
18
  export { createMediabunnyBackend, isMediabunnyCompatible } from './media/video/demuxer/createMediabunnyBackend';
18
19
  export type { MediabunnyModule, CreateMediabunnyBackendOpts } from './media/video/demuxer/createMediabunnyBackend';
19
20
  export type { DemuxerBackend, DemuxerFactory } from './media/video/demuxer/MediabunnyDemuxer';
package/dist/index.js CHANGED
@@ -6,6 +6,7 @@ export { resolveDrawRect, transformFromContainRect } from './renderer/gpu/layers
6
6
  export { computeContainViewport } from './renderer/gpu/viewport';
7
7
  export { computeTextLayout, SIDE_MARGIN } from './renderer/gpu/layers/textLayout';
8
8
  export { GpuDebugCounters } from './renderer/gpu/debug/GpuDebugCounters';
9
+ export { createDefaultDemuxerFactory } from './media/video/demuxer/createDefaultDemuxerFactory';
9
10
  export { createMediabunnyBackend, isMediabunnyCompatible } from './media/video/demuxer/createMediabunnyBackend';
10
11
  export { createVideoFrameProvider, MockVideoFrameProvider, SyntheticVideoFrameProvider } from './media/video';
11
12
  export { AudioPlaybackController } from './media/audio/AudioPlaybackController';
@@ -0,0 +1,3 @@
1
+ import { type CreateMediabunnyBackendOpts } from './createMediabunnyBackend';
2
+ import type { DemuxerFactory } from './MediabunnyDemuxer';
3
+ export declare function createDefaultDemuxerFactory(opts?: CreateMediabunnyBackendOpts): DemuxerFactory;
@@ -0,0 +1,6 @@
1
+ import * as mediabunny from 'mediabunny';
2
+ import { createMediabunnyBackend } from './createMediabunnyBackend';
3
+ const mb = mediabunny;
4
+ export function createDefaultDemuxerFactory(opts = {}) {
5
+ return () => createMediabunnyBackend(mb, opts);
6
+ }
@@ -6,3 +6,4 @@ export { DecoderBackedVideoFrameProvider } from './DecoderBackedVideoFrameProvid
6
6
  export { FrameCache } from './FrameCache';
7
7
  export type { DemuxerFactory, DemuxerBackend } from './demuxer/MediabunnyDemuxer';
8
8
  export { createMediabunnyBackend, isMediabunnyCompatible } from './demuxer/createMediabunnyBackend';
9
+ export { createDefaultDemuxerFactory } from './demuxer/createDefaultDemuxerFactory';
@@ -3,3 +3,4 @@ export { StreamingFrameProducer } from './StreamingFrameProducer';
3
3
  export { DecoderBackedVideoFrameProvider } from './DecoderBackedVideoFrameProvider';
4
4
  export { FrameCache } from './FrameCache';
5
5
  export { createMediabunnyBackend, isMediabunnyCompatible } from './demuxer/createMediabunnyBackend';
6
+ export { createDefaultDemuxerFactory } from './demuxer/createDefaultDemuxerFactory';
package/package.json CHANGED
@@ -1,10 +1,11 @@
1
1
  {
2
2
  "name": "@elah/core",
3
- "version": "0.1.0",
4
- "description": "Framework-agnostic video timeline engine",
3
+ "version": "0.2.0",
4
+ "description": "Framework-agnostic video timeline engine — playback, frame resolution, WebGL2 renderer, and media management",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",
7
7
  "types": "./dist/index.d.ts",
8
+ "sideEffects": false,
8
9
  "exports": {
9
10
  ".": {
10
11
  "import": "./dist/index.js",
@@ -12,7 +13,9 @@
12
13
  }
13
14
  },
14
15
  "files": [
15
- "dist"
16
+ "dist",
17
+ "README.md",
18
+ "LICENSE"
16
19
  ],
17
20
  "scripts": {
18
21
  "typecheck": "tsc --noEmit",
@@ -20,8 +23,34 @@
20
23
  "test:watch": "vitest",
21
24
  "build": "tsc --noEmit && tsc -p tsconfig.build.json"
22
25
  },
26
+ "keywords": [
27
+ "video-editor",
28
+ "timeline",
29
+ "video-engine",
30
+ "webgl",
31
+ "webgl2",
32
+ "gpu-renderer",
33
+ "playback",
34
+ "frame-accurate",
35
+ "media",
36
+ "export",
37
+ "mp4",
38
+ "typescript",
39
+ "browser",
40
+ "react"
41
+ ],
42
+ "homepage": "https://www.elah.dev",
43
+ "repository": {
44
+ "type": "git",
45
+ "url": "https://github.com/elahlabs/elah.git"
46
+ },
47
+ "bugs": {
48
+ "url": "https://github.com/elahlabs/elah/issues"
49
+ },
50
+ "license": "SEE LICENSE IN LICENSE",
23
51
  "dependencies": {
24
52
  "immer": "^10.1.1",
53
+ "mediabunny": "^1.45.3",
25
54
  "zustand": "^5.0.3"
26
55
  },
27
56
  "devDependencies": {