mirror of
https://github.com/hedgedoc/hedgedoc.git
synced 2025-05-14 15:14:56 -04:00
Remove keys from replacers (#539)
This commit is contained in:
parent
63b24fa5bf
commit
e58e53a116
20 changed files with 77 additions and 78 deletions
108
src/components/markdown-renderer/renderer-utils.tsx
Normal file
108
src/components/markdown-renderer/renderer-utils.tsx
Normal file
|
@ -0,0 +1,108 @@
|
|||
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
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue