2023-03-01 19:20:48 +00:00
|
|
|
import type { AstroInstance } from 'astro';
|
|
|
|
import type { RenderableTreeNode } from '@markdoc/markdoc';
|
2023-02-14 14:25:56 +00:00
|
|
|
import Markdoc from '@markdoc/markdoc';
|
2023-03-06 15:05:59 +00:00
|
|
|
import { MarkdocError, isCapitalized } from '../dist/utils.js';
|
2023-02-06 21:11:15 +00:00
|
|
|
|
|
|
|
export type AstroNode =
|
|
|
|
| string
|
|
|
|
| {
|
2023-03-01 19:20:48 +00:00
|
|
|
component: AstroInstance['default'];
|
2023-02-06 21:11:15 +00:00
|
|
|
props: Record<string, any>;
|
|
|
|
children: AstroNode[];
|
|
|
|
}
|
|
|
|
| {
|
|
|
|
tag: string;
|
|
|
|
attributes: Record<string, any>;
|
|
|
|
children: AstroNode[];
|
|
|
|
};
|
|
|
|
|
|
|
|
export function createAstroNode(
|
|
|
|
node: RenderableTreeNode,
|
2023-03-01 19:20:48 +00:00
|
|
|
components: Record<string, AstroInstance['default']> = {}
|
2023-02-06 21:11:15 +00:00
|
|
|
): AstroNode {
|
|
|
|
if (typeof node === 'string' || typeof node === 'number') {
|
2023-02-14 17:30:42 +00:00
|
|
|
return String(node);
|
2023-02-14 14:25:38 +00:00
|
|
|
} else if (node === null || typeof node !== 'object' || !Markdoc.Tag.isTag(node)) {
|
2023-02-06 21:11:15 +00:00
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2023-03-06 15:05:59 +00:00
|
|
|
if (node.name in components) {
|
2023-03-01 19:20:48 +00:00
|
|
|
const component = components[node.name];
|
|
|
|
const props = node.attributes;
|
2023-02-06 21:11:15 +00:00
|
|
|
const children = node.children.map((child) => createAstroNode(child, components));
|
|
|
|
|
|
|
|
return {
|
|
|
|
component,
|
|
|
|
props,
|
|
|
|
children,
|
|
|
|
};
|
2023-03-02 16:34:01 +00:00
|
|
|
} else if (isCapitalized(node.name)) {
|
2023-03-02 18:53:46 +00:00
|
|
|
throw new MarkdocError({
|
|
|
|
message: `Unable to render ${JSON.stringify(node.name)}.`,
|
|
|
|
hint: 'Did you add this to the "components" prop on your <Content /> component?',
|
|
|
|
});
|
2023-02-06 21:11:15 +00:00
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
tag: node.name,
|
|
|
|
attributes: node.attributes,
|
|
|
|
children: node.children.map((child) => createAstroNode(child, components)),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|