Skip to main content
Version: 0.61

Authoring doc stories

The doc blocks from @unpunnyfuns/swatchbook-blocks are React components that render against the active token graph. Use them in MDX stories to compose reference pages — a ColorPalette, a TypographyScale, a TokenTable filtered to a subpath — all of which update live as the toolbar changes axes.

Prerequisite

Install the blocks package:

npm install -D @unpunnyfuns/swatchbook-blocks

The blocks read the active token graph from a SwatchbookProvider. The addon's preview decorator mounts that provider for every story, so blocks render inside MDX as long as @unpunnyfuns/swatchbook-addon is registered in your Storybook.

A minimal docs page

src/docs/colors.mdx
import { Meta } from '@storybook/addon-docs/blocks';
import { ColorPalette, ColorTable, TokenTable } from '@unpunnyfuns/swatchbook-blocks';

<Meta title="Docs/Colors" />

# Colors

## Primitive palette

<ColorPalette filter="color.palette.**" />

## Semantic slots

<ColorTable filter="color.**" />

## Everything as a table

<TokenTable filter="color.**" type="color" />

ColorTable groups variants under their base path — color.accent.bg + color.accent.bg-hover collapse to one accent.bg row with a default / hover pill selector, and each row expands inline to show all variants side-by-side. <TokenTable> is the flat view for comparison. Visit Docs / Colors; swatches and values update as you flip the toolbar.

A dashboard composition

One page with diagnostics, tree, and table gives a top-to-bottom view — project health, browsable hierarchy, searchable lookup:

src/docs/tokens.mdx
import { Meta } from '@storybook/addon-docs/blocks';
import { Diagnostics, TokenNavigator, TokenTable } from '@unpunnyfuns/swatchbook-blocks';

<Meta title="Docs/Tokens" />

# Design Tokens

<Diagnostics />

## Tree
<TokenNavigator />

## Table
<TokenTable />

Diagnostics auto-collapses on clean loads, so it's safe at the top of every page without adding noise. Scope the tree or table with a prop: <TokenNavigator root="color" />, <TokenTable filter="space.**" />.

Blocks at a glance

BlockWhat it renders
TokenDetailA single token's full picture — alias chain, aliased-by tree, per-axis variance, composite preview.
TokenTableFilterable table of tokens. Supports path glob and $type filters.
TokenNavigatorExpandable tree; scope to a subtree or $type.
ColorPaletteSwatch grid grouped by sub-path.
ColorTableColor-specific table; variant suffixes (-hover, -disabled) collapse onto the base row with a pill selector + inline expand.
TypographyScaleEach typography token as a sample line using its own value.
FontFamilyPreviewPangram per fontFamily primitive.
FontWeightScaleSame "Aa" sample at each fontWeight.
StrokeStylePreviewBorder rendered per strokeStyle primitive.
DiagnosticsCollapsible list of parser / resolver warnings; auto-opens on non-zero.
Motion / shadow / border / gradient previewsInside TokenDetail's composite rendering, or standalone blocks.

See the blocks reference for the full prop surface.

Typed token access from components

If you need a token's resolved value (not just a CSS var reference), use useToken. It reads from the virtual graph + active tuple and returns { value, cssVar, type, description } with typed paths:

import { useToken } from '@unpunnyfuns/swatchbook-addon/hooks';

function Badge({ level }: { level: 'info' | 'warn' }) {
const tokenName = level === 'info' ? 'color.accent.bg' : 'color.warning.bg';
const bg = useToken(tokenName);
return <span style={{ background: bg.cssVar }}>{bg.description}</span>;
}

cssVar is stable across themes; value reflects the active tuple. Paths autocomplete from a generated .swatchbook/tokens.d.ts once the addon has run.

Overriding axes on a specific doc

Force a specific tuple on a single doc page:

<Meta title="Docs/Colors (Dark)" parameters={{ swatchbook: { axes: { mode: 'Dark' } } }} />

The toolbar still reflects the story's active tuple, so readers see what they're looking at. Any axis you omit falls back to its default.

What MDX can't do

Storybook's preview hooks (useGlobals, useArgs, useChannel) throw when called from an MDX doc block — the preview HooksContext only exists while a story is rendering. The blocks in this package subscribe to Storybook's channel directly instead. If you write a custom block for your project, same pattern:

import { addons } from 'storybook/preview-api';

const channel = addons.getChannel();
channel.on('globalsUpdated', (payload) => {
// react to axis / theme changes
});

See also