Skip to content

Commit

Permalink
feat: add support for providing the current node and parent node to m…
Browse files Browse the repository at this point in the history
…arks
  • Loading branch information
nperez0111 committed Dec 4, 2024
1 parent b53d25b commit bc7ad50
Show file tree
Hide file tree
Showing 2 changed files with 56 additions and 30 deletions.
52 changes: 26 additions & 26 deletions packages/static-renderer/src/json/renderer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,15 +29,29 @@ export type NodeProps<TNodeType = any, TChildren = any> = {
* The parent node of the current node
*/
parent?: TNodeType;
})=> TChildren;
}) => TChildren;
};

/**
* Props for a mark renderer
*/
export type MarkProps<TMarkType = any, TChildren = any> = {
export type MarkProps<TMarkType = any, TChildren = any, TNodeType = any> = {
/**
* The current mark to render
*/
mark: TMarkType;
/**
* The children of the current mark
*/
children?: TChildren;
/**
* The node the current mark is applied to
*/
node: TNodeType;
/**
* The node the current mark is applied to
*/
parent?: TNodeType;
};

export type TiptapStaticRendererOptions<
Expand All @@ -60,18 +74,14 @@ export type TiptapStaticRendererOptions<
/**
* A node renderer is a function that takes a node and its children and returns the rendered output
*/
TNodeRender extends (
ctx: NodeProps<TNodeType, TReturnType | TReturnType[]>
) => TReturnType = (
TNodeRender extends (ctx: NodeProps<TNodeType, TReturnType | TReturnType[]>) => TReturnType = (
ctx: NodeProps<TNodeType, TReturnType | TReturnType[]>
) => TReturnType,
/**
* A mark renderer is a function that takes a mark and its children and returns the rendered output
*/
TMarkRender extends (
ctx: MarkProps<TMarkType, TReturnType | TReturnType[]>
) => TReturnType = (
ctx: MarkProps<TMarkType, TReturnType | TReturnType[]>
TMarkRender extends (ctx: MarkProps<TMarkType, TReturnType | TReturnType[], TNodeType>) => TReturnType = (
ctx: MarkProps<TMarkType, TReturnType | TReturnType[], TNodeType>
) => TReturnType,
> = {
/**
Expand Down Expand Up @@ -124,18 +134,14 @@ cb: (node: TNodeType) => void) => void };
/**
* A node renderer is a function that takes a node and its children and returns the rendered output
*/
TNodeRender extends (
ctx: NodeProps<TNodeType, TReturnType | TReturnType[]>
) => TReturnType = (
TNodeRender extends (ctx: NodeProps<TNodeType, TReturnType | TReturnType[]>) => TReturnType = (
ctx: NodeProps<TNodeType, TReturnType | TReturnType[]>
) => TReturnType,
/**
* A mark renderer is a function that takes a mark and its children and returns the rendered output
*/
TMarkRender extends (
ctx: MarkProps<TMarkType, TReturnType | TReturnType[]>
) => TReturnType = (
ctx: MarkProps<TMarkType, TReturnType | TReturnType[]>
TMarkRender extends (ctx: MarkProps<TMarkType, TReturnType | TReturnType[], TNodeType>) => TReturnType = (
ctx: MarkProps<TMarkType, TReturnType | TReturnType[], TNodeType>
) => TReturnType,
>(
/**
Expand All @@ -149,21 +155,15 @@ cb: (node: TNodeType) => void) => void };
}
| {
component: TMarkRender;
props: MarkProps<TMarkType, TReturnType | TReturnType[]>;
props: MarkProps<TMarkType, TReturnType | TReturnType[], TNodeType>;
}
) => TReturnType,
{
nodeMapping,
markMapping,
unhandledNode,
unhandledMark,
}: TiptapStaticRendererOptions<
TReturnType,
TMarkType,
TNodeType,
TNodeRender,
TMarkRender
>,
}: TiptapStaticRendererOptions<TReturnType, TMarkType, TNodeType, TNodeRender, TMarkRender>,
) {
/**
* Render Tiptap JSON and all its children using the provided node and mark mappings.
Expand All @@ -181,7 +181,6 @@ cb: (node: TNodeType) => void) => void };
*/
parent?: TNodeType;
}): TReturnType {

const nodeType = typeof content.type === 'string' ? content.type : content.type.name
const NodeHandler = nodeMapping[nodeType] ?? unhandledNode

Expand All @@ -197,7 +196,7 @@ cb: (node: TNodeType) => void) => void };
renderElement: renderContent,
// Lazily compute the children to avoid unnecessary recursion
get children() {
// recursively render child content nodes
// recursively render child content nodes
const children: TReturnType[] = []

if (content.content) {
Expand Down Expand Up @@ -231,6 +230,7 @@ cb: (node: TNodeType) => void) => void };
props: {
mark,
parent,
node: content,
children: acc,
},
})
Expand Down
34 changes: 30 additions & 4 deletions packages/static-renderer/src/pm/markdown.example.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,12 +46,20 @@ const renderToMarkdown = ({ content }: { content: JSONContent | Node }) => rende
return `${new Array(level).fill('#').join('')} ${children}\n`
},
codeBlock({ node, children }) {
return `\n\`\`\`${node.attrs.language}\n${serializeChildrenToHTMLString(children)}\n\`\`\`\n`
return `\n\`\`\`${node.attrs.language}\n${serializeChildrenToHTMLString(
children,
)}\n\`\`\`\n`
},
blockquote({ children }) {
return `\n${serializeChildrenToHTMLString(children).trim().split('\n').map(a => `> ${a}`)
return `\n${serializeChildrenToHTMLString(children)
.trim()
.split('\n')
.map(a => `> ${a}`)
.join('\n')}`
},
image({ node }) {
return `![${node.attrs.alt}](${node.attrs.src})`
},
hardBreak() {
return '\n'
},
Expand All @@ -60,8 +68,23 @@ const renderToMarkdown = ({ content }: { content: JSONContent | Node }) => rende
bold({ children }) {
return `**${serializeChildrenToHTMLString(children)}**`
},
italic({ children }) {
return `*${serializeChildrenToHTMLString(children)}*`
italic({ children, node }) {
let isBoldToo = false

// Check if the node being wrapped also has a bold mark, if so, we need to use the bold markdown syntax
if (node?.marks.some(m => m.type.name === 'bold')) {
isBoldToo = true
}

if (isBoldToo) {
// If the content is bold, just wrap the bold content in italic markdown syntax with another set of asterisks
return `*${serializeChildrenToHTMLString(children)}*`
}

return `_${serializeChildrenToHTMLString(children)}_`
},
code({ children }) {
return `\`${serializeChildrenToHTMLString(children)}\``
},
},
},
Expand Down Expand Up @@ -127,6 +150,9 @@ console.log(
{
type: 'bold',
},
{
type: 'italic',
},
],
text: 'Tiptap',
},
Expand Down

0 comments on commit bc7ad50

Please sign in to comment.