2023-02-10 09:01:04 -05:00
|
|
|
import type { RenderableTreeNode, Tag } from '@markdoc/markdoc';
|
2023-02-06 16:11:15 -05:00
|
|
|
import { escape } from 'html-escaper';
|
|
|
|
|
|
|
|
// TODO: expose `AstroComponentFactory` type from core
|
|
|
|
type AstroComponentFactory = (props: Record<string, any>) => any & {
|
|
|
|
isAstroComponentFactory: true;
|
|
|
|
};
|
|
|
|
|
2023-02-10 09:01:04 -05:00
|
|
|
/**
|
|
|
|
* Copied from Markdoc Tag.isTag implementation
|
|
|
|
* to avoid dragging the whole 40kb Markdoc bundle into your build!
|
|
|
|
*/
|
|
|
|
function isTag(tag: any): tag is Tag {
|
|
|
|
return !!(tag?.$$mdtype === 'Tag');
|
|
|
|
}
|
|
|
|
|
2023-02-06 16:11:15 -05:00
|
|
|
export type ComponentRenderer =
|
|
|
|
| AstroComponentFactory
|
|
|
|
| {
|
|
|
|
component: AstroComponentFactory;
|
2023-02-10 09:01:04 -05:00
|
|
|
props?(params: {
|
|
|
|
attributes: Record<string, any>;
|
|
|
|
getTreeNode(): import('@markdoc/markdoc').Tag;
|
|
|
|
}): Record<string, any>;
|
2023-02-06 16:11:15 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
export type AstroNode =
|
|
|
|
| string
|
|
|
|
| {
|
|
|
|
component: AstroComponentFactory;
|
|
|
|
props: Record<string, any>;
|
|
|
|
children: AstroNode[];
|
|
|
|
}
|
|
|
|
| {
|
|
|
|
tag: string;
|
|
|
|
attributes: Record<string, any>;
|
|
|
|
children: AstroNode[];
|
|
|
|
};
|
|
|
|
|
|
|
|
export function createAstroNode(
|
|
|
|
node: RenderableTreeNode,
|
|
|
|
components: Record<string, ComponentRenderer> = {}
|
|
|
|
): AstroNode {
|
|
|
|
if (typeof node === 'string' || typeof node === 'number') {
|
|
|
|
return escape(String(node));
|
2023-02-10 09:01:04 -05:00
|
|
|
} else if (node === null || typeof node !== 'object' || !isTag(node)) {
|
2023-02-06 16:11:15 -05:00
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2023-02-08 09:53:35 -05:00
|
|
|
if (node.name in components) {
|
2023-02-06 16:11:15 -05:00
|
|
|
const componentRenderer = components[node.name];
|
|
|
|
const component =
|
2023-02-06 16:33:27 -05:00
|
|
|
'component' in componentRenderer ? componentRenderer.component : componentRenderer;
|
2023-02-06 16:11:15 -05:00
|
|
|
const props =
|
2023-02-08 09:53:35 -05:00
|
|
|
'props' in componentRenderer && typeof componentRenderer.props === 'function'
|
2023-02-06 16:11:15 -05:00
|
|
|
? componentRenderer.props({
|
|
|
|
attributes: node.attributes,
|
|
|
|
getTreeNode() {
|
|
|
|
return node;
|
|
|
|
},
|
|
|
|
})
|
|
|
|
: node.attributes;
|
|
|
|
|
|
|
|
const children = node.children.map((child) => createAstroNode(child, components));
|
|
|
|
|
|
|
|
return {
|
|
|
|
component,
|
|
|
|
props,
|
|
|
|
children,
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
tag: node.name,
|
|
|
|
attributes: node.attributes,
|
|
|
|
children: node.children.map((child) => createAstroNode(child, components)),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|