Thanks for your interest in contributing to Hyperframes! This guide will help you get started.
- Fork the repository
- Clone your fork:
git clone https://github.com/YOUR_USERNAME/hyperframes.git - Install dependencies:
bun install - Create a branch:
git checkout -b my-feature
bun install # Install all dependencies
bun run dev # Run the studio (composition editor)
bun run build # Build all packages
bun run --filter '*' typecheck # Type-check all packages
bun run lint # Lint all packages
bun run format:check # Check formattingbun run --filter @hyperframes/core test # Core unit tests (vitest)
bun run --filter @hyperframes/engine test # Engine unit tests (vitest)
bun run --filter @hyperframes/core test:hyperframe-runtime-ci # Runtime contract testsbun run lint # Run oxlint
bun run lint:fix # Run oxlint with auto-fix
bun run format # Format all files with oxfmt
bun run format:check # Check formatting without writingGit hooks (via lefthook) run automatically after bun install and enforce linting + formatting on staged files before each commit.
We aim for honest types — code that lies to the compiler eventually lies to users. The underlying convention is:
- Avoid
any. Useunknownand narrow it where possible. - Avoid
as Ttype assertions. They suppress type-checker warnings without telling the compiler anything new. Prefer:- Type guards (
function isFoo(x): x is Foo) instanceof/typeofnarrowing- Centralized narrowing helpers (e.g.
resolveIframe) - Properly-typed interfaces at the source
- Type guards (
- Acceptable
asuse, with a comment explaining why:as const— literal narrowing; always safeas unknown as T— explicit double-cast at hard type-system boundaries (e.g. parsing untrusted JSON, FFI/postMessage). Pair with a one-line justification.
- Avoid
!non-null assertions outside of post-if-checked code paths. Use??defaults or guard clauses instead.
If you must add a cast, add a comment:
// `postMessage` data is `unknown`; the runtime guarantees this shape.
const event = data as unknown as RuntimeEvent;The registry at registry/ contains reusable items installable via hyperframes add <name>. Each item lives in its own directory under registry/blocks/ or registry/components/.
registry/blocks/<name>/
registry-item.json # Manifest (name, type, description, tags, files)
<name>.html # The composition HTML
registry/components/<name>/
registry-item.json # Manifest (no dimensions/duration for components)
<name>.html # The snippet HTML to paste into a composition
demo.html # Required — standalone demo showing the effect
Every component must ship a companion demo.html. This file:
- Is a complete, standalone HTML document (with
<!doctype html>, GSAP CDN, etc.) - Shows the component effect applied to representative content
- Registers a GSAP timeline on
window.__timelinesso it can be previewed in the Studio and rendered by the CI preview pipeline - Uses
data-composition-id="<name>-demo"to avoid ID collisions
Blocks don't need demo.html — they are already standalone compositions.
- Create
registry/<blocks|components>/<name>/registry-item.jsonfollowing the schema - Add the item to
registry/registry.json - For components: include a
demo.html - Run
npx hyperframes lintandnpx hyperframes validateon your HTML - Test the install flow:
hyperframes add <name> --dir /tmp/test-project
When you add a new block or component, its documentation page is generated automatically — you don't need to write MDX by hand.
Run the codegen script after adding items:
npx tsx scripts/generate-catalog-pages.tsThis produces:
docs/catalog/blocks/<name>.mdx— per-block detail pagedocs/catalog/components/<name>.mdx— per-component detail pagedocs/public/catalog-index.json— flat manifest for the catalog grid page- Updates
docs/docs.jsonnavigation with the new pages
The script wipes docs/catalog/ before regenerating, so deleted items are automatically cleaned up.
- Use conventional commit format for all commits (e.g.,
feat: add timeline export,fix: resolve seek overflow). Enforced by a git hook. - CI must pass before merge (build, typecheck, tests, semantic PR title)
- PRs require at least 1 approval
| Package | Description |
|---|---|
@hyperframes/core |
Types, HTML generation, runtime, linter |
@hyperframes/engine |
Seekable page-to-video capture engine |
@hyperframes/producer |
Full rendering pipeline (capture + encode) |
@hyperframes/studio |
Composition editor UI |
hyperframes |
CLI for creating, previewing, and rendering |
All packages use fixed versioning — every release bumps all packages to the same version.
bun run set-version 0.2.0 # bumps all packages, commits, and creates git tag
git push origin main --tags # triggers the publish workflowThe set-version script automatically creates a chore: release v<version> commit and a v<version> git tag. Pushing the tag triggers CI to publish all packages to npm and create a GitHub Release.
Use a pre-release suffix to publish to a named npm dist-tag instead of latest:
bun run set-version 0.2.0-alpha.1 # first alpha
git push origin v0.2.0-alpha.1 # publishes to npm with --tag alpha
bun run set-version 0.2.0-alpha.2 # iterate
bun run set-version 0.2.0-beta.1 # promote to beta (--tag beta)
bun run set-version 0.2.0-rc.1 # release candidate (--tag rc)
bun run set-version 0.2.0 # final stable release (--tag latest)Consumers install pre-releases with npm install @hyperframes/core@alpha (or @beta, @rc). The latest tag is never touched by pre-releases, so npm install @hyperframes/core always gets the last stable version.
Pre-releases also create GitHub Releases marked as pre-release.
If you need to bump versions without committing (e.g., for a release PR), pass --no-tag:
bun run set-version 0.2.0 --no-tag # updates package.json files only- Use GitHub Issues for bug reports and feature requests
- Search existing issues before creating a new one
- Include reproduction steps for bugs
We welcome contributions that use AI tools (GitHub Copilot, Claude, ChatGPT, etc.). If you used AI to help write a PR, there is no need to disclose it — we review all code on its merits. However:
- You are responsible for the correctness of any code you submit, regardless of how it was generated.
- AI-generated tests must actually test meaningful behavior, not just assert truthy values.
- Do not submit AI-generated code you don't understand. If you can't explain what a change does during review, it will be rejected.
Hyperframes uses a BDFL (Benevolent Dictator for Life) governance model. The core maintainers at HeyGen have final say on the project's direction, API design, and what gets merged. This keeps the project focused and moving fast.
Community input is valued and encouraged — open issues, propose RFCs, and discuss in PRs. But final decisions rest with the maintainers.
This project follows the Contributor Covenant Code of Conduct. By participating, you are expected to uphold this code.
By contributing, you agree that your contributions will be licensed under the project's license. See LICENSE for details.