hedgedoc/src/components/markdown-renderer/renderer-utils.tsx
2020-09-05 10:34:43 +02:00

108 lines
3.4 KiB
TypeScript

import { diffArrays } from 'diff'
import { DomElement } from 'domhandler'
import React, { Fragment, ReactElement } from 'react'
import { convertNodeToElement, Transform } from 'react-html-parser'
import {
ComponentReplacer,
SubNodeTransform
} from './replace-components/ComponentReplacer'
export interface TextDifferenceResult {
lines: LineKeys[],
lastUsedLineId: number
}
export interface LineKeys {
line: string,
id: number
}
export const calculateNewLineNumberMapping = (newMarkdownLines: string[], oldLineKeys: LineKeys[], lastUsedLineId: number): TextDifferenceResult => {
const lineDifferences = diffArrays<string, LineKeys>(newMarkdownLines, oldLineKeys, {
comparator: (left:string|LineKeys, right:string|LineKeys) => {
const leftLine = (left as LineKeys).line ?? (left as string)
const rightLine = (right as LineKeys).line ?? (right as string)
return leftLine === rightLine
}
})
const newLines: LineKeys[] = []
lineDifferences
.filter((change) => change.added === undefined || !change.added)
.forEach((value) => {
if (value.removed) {
(value.value as string[])
.forEach(line => {
lastUsedLineId += 1
newLines.push({ line: line, id: lastUsedLineId })
})
} else {
(value.value as LineKeys[])
.forEach((line) => newLines.push(line))
}
})
return { lines: newLines, lastUsedLineId: lastUsedLineId }
}
export const calculateKeyFromLineMarker = (node: DomElement, lineKeys?: LineKeys[]): number|undefined => {
if (!node.attribs || lineKeys === undefined) {
return
}
const key = node.attribs['data-key']
if (key) {
return Number(key)
}
const lineMarker = node.prev
if (!lineMarker || !lineMarker.attribs) {
return
}
const lineInMarkdown = lineMarker.attribs['data-start-line']
if (lineInMarkdown === undefined) {
return
}
const line = Number(lineInMarkdown)
if (lineKeys[line] === undefined) {
return
}
return lineKeys[line].id
}
export const findNodeReplacement = (node: DomElement, index: number, allReplacers: ComponentReplacer[], subNodeTransform: SubNodeTransform): ReactElement|null|undefined => {
return allReplacers
.map((componentReplacer) => componentReplacer.getReplacement(node, subNodeTransform))
.find((replacement) => replacement !== undefined)
}
export const renderNativeNode = (node: DomElement, key: number, transform: Transform): ReactElement => {
if (node.attribs === undefined) {
node.attribs = {}
}
delete node.attribs['data-key']
return convertNodeToElement(node, key, transform)
}
export const buildTransformer = (lineKeys: (LineKeys[] | undefined), allReplacers: ComponentReplacer[]):Transform => {
const transform: Transform = (node, index) => {
const nativeRenderer = (subNode: DomElement, subKey: number) => renderNativeNode(subNode, subKey, transform)
const subNodeTransform:SubNodeTransform = (subNode, subIndex) => transform(subNode, subIndex, transform)
const key = calculateKeyFromLineMarker(node, lineKeys) ?? -index
const tryReplacement = findNodeReplacement(node, key, allReplacers, subNodeTransform)
if (tryReplacement === null) {
return null
} else if (tryReplacement === undefined) {
return nativeRenderer(node, key)
} else {
return <Fragment key={key}>{tryReplacement}</Fragment>
}
}
return transform
}