From sanity
Renders and serializes Portable Text to React, Svelte, Vue, Astro, HTML, Markdown, and plain text with framework-specific component mapping patterns.
How this skill is triggered — by the user, by Claude, or both
Slash command
/sanity:portable-text-serializationThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Render Portable Text content across frameworks using the `@portabletext/*` library family. Each library follows the same component-mapping pattern: you provide a `components` object that maps PT node types to framework-specific renderers.
Render Portable Text content across frameworks using the @portabletext/* library family. Each library follows the same component-mapping pattern: you provide a components object that maps PT node types to framework-specific renderers.
PT is an array of blocks. Each block has _type, optional style, children (spans), markDefs, listItem, and level.
Root array
├── block (_type: "block")
│ ├── style: "normal" | "h1" | "h2" | "blockquote" | ...
│ ├── children: [span, span, ...]
│ │ └── span: { _type: "span", text: "...", marks: ["strong", "<markDefKey>"] }
│ ├── markDefs: [{ _key, _type: "link", href: "..." }, ...]
│ ├── listItem: "bullet" | "number" (optional)
│ └── level: 1, 2, 3... (optional, for nested lists)
├── custom block (_type: "image" | "code" | any custom type)
└── ...more blocks
Marks come in two forms:
marks[] like "strong", "em", "underline", "code"marks[] referencing entries in markDefs[] (e.g., links, internal references)Every @portabletext/* library accepts a components object with these keys:
| Key | Renders | Props/Data |
|---|---|---|
types | Custom block/inline types (image, code, CTA) | value (the block data) |
marks | Decorators + annotations | children + value (mark data) |
block | Block styles (h1, normal, blockquote) | children |
list | List wrappers (ul, ol) | children |
listItem | List items | children |
hardBreak | Line breaks within a block | — |
Read the rule file matching your framework:
rules/react.md — @portabletext/react or next-sanityrules/svelte.md — @portabletext/svelterules/vue.md — @portabletext/vuerules/astro.md — astro-portabletextrules/html.md — @portabletext/to-htmlrules/markdown.md — @portabletext/markdownrules/plain-text.md — @portabletext/toolkitThese are listed on portabletext.org but don't have dedicated rule files:
| Target | Package |
|---|---|
| React Native | @portabletext/react-native-portabletext |
| React PDF | @portabletext/react-pdf-portabletext |
| Solid | solid-portabletext |
| Qwik | portabletext-qwik |
| Shopify Liquid | portable-text-to-liquid |
| PHP | sanity-php (SanityBlockContent class) |
| Python | portabletext-html |
| C# / .NET | dotnet-portable-text |
| Dart / Flutter | flutter_sanity_portable_text |
PT renderers only handle standard blocks by default. Custom types (image, code, callToAction, etc.) require explicit component mappings — they won't render otherwise.
In React/Vue, define components outside the render function or memoize it. Recreating on every render causes unnecessary re-renders.
All libraries accept onMissingComponent to control behavior when encountering unknown types:
false — suppress warningsAlways expand references inside custom blocks:
body[]{
...,
_type == "image" => {
...,
asset->
},
markDefs[]{
...,
_type == "internalLink" => {
...,
"slug": @.reference->slug.current
}
}
}
npx claudepluginhub sanity-io/agent-toolkit --plugin sanityConverts HTML and Markdown into Sanity Portable Text blocks using @portabletext/markdown (markdownToPortableText) and @portabletext/block-tools (htmlToBlocks). Also covers manual block construction per the Portable Text spec.
Generates React components for Optimizely CMS content types and display templates, mapping properties to React props and following SDK patterns.
Builds content-focused websites with Astro's zero-JS islands architecture, multi-framework components (React/Vue/Svelte), and Markdown/MDX support. Triggers on .astro files, Astro.props, content collections.