@bnomei/emdash-blocks
v0.2.0
Published
JSON block-list field widget for EmDash entries, with normalized block props and visibility state.
Maintainers
Readme
@bnomei/emdash-blocks
Structured block-list editor for EmDash JSON fields.
@bnomei/emdash-blocks is a native EmDash plugin for ordered
content blocks stored as plain JSON. It registers the block-builder:blocks
admin widget, lets schemas define project-specific block types and typed prop
controls, and exports runtime helpers for Astro renderers. Use it when an
EmDash site needs controlled page sections, marketing modules, email blocks, or
imported block data without moving that content into Portable Text.
What It Provides
- Native EmDash plugin factory:
blockBuilderPlugin(). - JSON field widget:
block-builder:blocks. - Stored value shape for ordered blocks with
id,type,hidden, andprops. - Admin UI built with Kumo UI with full light and dark mode support.
- Frontend helpers:
visibleBlocks(),blockProps(), andisBlockBuilderBlock().
Install
npm install @bnomei/emdash-blocksRegister the plugin in astro.config.mjs:
import emdash from "emdash/astro";
import { blockBuilderPlugin } from "@bnomei/emdash-blocks";
export default {
integrations: [
emdash({
plugins: [blockBuilderPlugin()],
}),
],
};Field Widget
Use the widget on an EmDash json field:
{
"slug": "blocks",
"label": "Blocks",
"type": "json",
"widget": "block-builder:blocks",
"options": {
"blockTypes": [
{ "type": "heading", "label": "Heading" },
{ "type": "text", "label": "Text" },
{ "type": "image", "label": "Image" }
]
}
}Use blockDefinitions when the editor should render typed prop controls instead
of the raw JSON fallback. TypeScript schemas can import BlockBuilderOptions
from the public package entry to type field options:
import type { BlockBuilderOptions } from "@bnomei/emdash-blocks";
const blockOptions: BlockBuilderOptions = {
blockDefinitions: [
{
type: "heading",
label: "Heading",
props: [
{ key: "text", label: "Text", type: "text" },
{
key: "level",
label: "Level",
type: "select",
defaultValue: "h2",
options: ["h1", "h2", "h3", "h4"],
},
],
},
],
};The same options can be used directly in JSON schemas:
{
"slug": "blocks",
"label": "Blocks",
"type": "json",
"widget": "block-builder:blocks",
"options": {
"blockDefinitions": [
{
"type": "heading",
"label": "Heading",
"props": [
{ "key": "text", "label": "Text", "type": "text" },
{
"key": "level",
"label": "Level",
"type": "select",
"defaultValue": "h2",
"options": ["h1", "h2", "h3", "h4"]
}
]
}
]
}
}Supported prop field types include all 16 EmDash field types: string, text,
url, number, integer, boolean, datetime, select, multiSelect,
portableText, image, file, reference, json, slug, and repeater.
The block builder also supports generic block-editor field types: writer,
markdown, textarea, color, media, and media-list. Text-like fields
render as text inputs or textareas, numeric fields normalize numeric values,
boolean renders as a switch, select and multiSelect render choice controls,
and portableText/writer render the built-in rich text editor with paragraph,
heading, quote, bold, italic, list, and link controls. media and media-list
render media-library selectors backed by EmDash media values. json and
repeater use focused JSON editors, with repeater stored as a JSON array until
a dedicated repeater UI is available. image, file, reference, and slug
render ID/text inputs that match EmDash's stored values for those field types.
Project-specific block types should be supplied through field options, or from a site-local helper/plugin that writes those options. Keep the publishable block builder package focused on generic block editing.
The package defaults include only generic starter blocks. Site-specific blocks
belong in options.blockDefinitions on each field schema.
Stored Value
type BlockBuilderValue = Array<{
id: string;
type: string;
hidden?: boolean;
props: Record<string, unknown>;
}>;Frontend Rendering
The package exports helpers for frontend renderers:
import { blockProps, isBlockBuilderBlock, visibleBlocks } from "@bnomei/emdash-blocks";
const blocks = visibleBlocks(entry.blocks);Renderers should use block.type to pick a component and block.props for that
component's data:
{
blocks.map((block) => (
<>
{block.type === "heading" && <HeadingBlock props={blockProps(block)} />}
{block.type === "text" && <TextBlock props={blockProps(block)} />}
</>
))
}visibleBlocks() accepts native Block Builder values and filters hidden blocks.
Migration from other systems should happen before values are saved into EmDash.
Package Surface
- ESM entry:
@bnomei/emdash-blocks. - Admin entry:
@bnomei/emdash-blocks/admin. - Type declarations are included from
dist/. - Peer dependencies:
emdash>=0.17.0,react^18.0.0 || ^19.0.0,react-dom^18.0.0 || ^19.0.0,@cloudflare/kumo^2.5.0, and@phosphor-icons/react^2.1.10.
Status
This package ships as a native EmDash plugin because the editor is a trusted React admin field widget. Package exports point at vp pack-built dist/ JavaScript and declarations.
Related Packages
@bnomei/emdash-fieldsprovides smaller JSON-backed field widgets.@bnomei/emdash-bentoprovides row and column layout editing and reuses Block Builder for nested blocks.
License
MIT.
Screenshot

