import { NodeSpec, MarkSpec, Schema } from 'prosemirror-model' import { Extensions } from '../types' import splitExtensions from './splitExtensions' import getAttributesFromExtensions from './getAttributesFromExtensions' import getRenderedAttributes from './getRenderedAttributes' import isEmptyObject from './isEmptyObject' import injectExtensionAttributesToParseRule from './injectExtensionAttributesToParseRule' import callOrReturn from './callOrReturn' function cleanUpSchemaItem(data: T) { return Object.fromEntries(Object.entries(data).filter(([key, value]) => { if (key === 'attrs' && isEmptyObject(value)) { return false } return value !== null && value !== undefined })) as T } export default function getSchema(extensions: Extensions): Schema { const allAttributes = getAttributesFromExtensions(extensions) const { nodeExtensions, markExtensions } = splitExtensions(extensions) const topNode = nodeExtensions.find(extension => extension.topNode)?.name const nodes = Object.fromEntries(nodeExtensions.map(extension => { const extensionAttributes = allAttributes.filter(attribute => attribute.type === extension.name) const context = { options: extension.options } const schema: NodeSpec = cleanUpSchemaItem({ content: callOrReturn(extension.content, context), marks: callOrReturn(extension.marks, context), group: callOrReturn(extension.group, context), inline: callOrReturn(extension.inline, context), atom: callOrReturn(extension.atom, context), selectable: callOrReturn(extension.selectable, context), draggable: callOrReturn(extension.draggable, context), code: callOrReturn(extension.code, context), defining: callOrReturn(extension.defining, context), isolating: callOrReturn(extension.isolating, context), attrs: Object.fromEntries(extensionAttributes.map(extensionAttribute => { return [extensionAttribute.name, { default: extensionAttribute?.attribute?.default }] })), }) if (extension.parseHTML) { schema.parseDOM = extension.parseHTML .bind(context)() ?.map(parseRule => injectExtensionAttributesToParseRule(parseRule, extensionAttributes)) } if (extension.renderHTML) { schema.toDOM = node => (extension.renderHTML as Function)?.bind(context)({ node, attributes: getRenderedAttributes(node, extensionAttributes), }) } return [extension.name, schema] })) const marks = Object.fromEntries(markExtensions.map(extension => { const extensionAttributes = allAttributes.filter(attribute => attribute.type === extension.name) const context = { options: extension.options } const schema: MarkSpec = cleanUpSchemaItem({ inclusive: callOrReturn(extension.inclusive, context), excludes: callOrReturn(extension.excludes, context), group: callOrReturn(extension.group, context), spanning: callOrReturn(extension.spanning, context), attrs: Object.fromEntries(extensionAttributes.map(extensionAttribute => { return [extensionAttribute.name, { default: extensionAttribute?.attribute?.default }] })), }) if (extension.parseHTML) { schema.parseDOM = extension.parseHTML .bind(context)() ?.map(parseRule => injectExtensionAttributesToParseRule(parseRule, extensionAttributes)) } if (extension.renderHTML) { schema.toDOM = mark => (extension.renderHTML as Function)?.bind(context)({ mark, attributes: getRenderedAttributes(mark, extensionAttributes), }) } return [extension.name, schema] })) return new Schema({ topNode, nodes, marks, }) }