@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 +82 -0
- package/dist/export/ExportWorker.js +7 -1
- package/dist/export/exportVideo.js +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -0
- package/dist/media/video/demuxer/createDefaultDemuxerFactory.d.ts +3 -0
- package/dist/media/video/demuxer/createDefaultDemuxerFactory.js +6 -0
- package/dist/media/video/index.d.ts +1 -0
- package/dist/media/video/index.js +1 -0
- package/package.json +32 -3
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
|
+
[](https://www.npmjs.com/package/@elah/core)
|
|
8
|
+
[](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
|
-
|
|
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.
|
|
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';
|
|
@@ -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.
|
|
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": {
|