import React from 'react'; import type { ReactNode, ReactElement } from 'react'; import { View } from 'react-native'; import getUniqueID from './util/getUniqueID'; import type { ASTNode, RenderRules, RenderFunction, MarkdownStyles } from '../types'; export function rootRenderRule(children: ReactNode[], styles: MarkdownStyles): ReactElement { return ( {children} ); } export default class AstRenderer { private _renderRules: RenderRules; private _style: MarkdownStyles; constructor(renderRules: RenderRules, style?: MarkdownStyles) { this._renderRules = renderRules; this._style = style || {}; } getRenderFunction = (type: string): RenderFunction => { const renderFunction = this._renderRules[type]; if (!renderFunction) { const fallback = this._renderRules['unknown']; if (!fallback) { throw new Error( `${type} renderRule not defined and no "unknown" fallback rule exists. Add a rule via ` ); } return fallback; } return renderFunction; }; renderNode = (node: ASTNode, parentNodes: ASTNode[]): ReactNode => { const renderFunction = this.getRenderFunction(node.type); const parents = [...parentNodes]; parents.unshift(node); if (node.type === 'text') { return renderFunction(node, [], parentNodes, this._style); } const children = node.children.map((value) => { return this.renderNode(value, parents); }); return renderFunction(node, children, parentNodes, this._style); }; render = (nodes: ASTNode[]): ReactElement => { const children = nodes.map((value) => this.renderNode(value, [])); return rootRenderRule(children, this._style); }; }