From 4097f58eafd2cedb306e0ab44f86b96f17533246 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Krist=C3=B3f=20Marussy?= Date: Sat, 23 Nov 2024 23:09:20 +0100 Subject: [PATCH] fix(frontend): PDF text positioning --- .eslintrc.cjs | 1 + .../src/graph/export/exportDiagram.tsx | 43 +++++++++++++++++++ 2 files changed, 44 insertions(+) diff --git a/.eslintrc.cjs b/.eslintrc.cjs index 514300d9..cff7ff97 100644 --- a/.eslintrc.cjs +++ b/.eslintrc.cjs @@ -83,6 +83,7 @@ module.exports = { // In typescript, some class methods implementing an inderface do not use `this`: // https://github.com/typescript-eslint/typescript-eslint/issues/1103 'class-methods-use-this': 'off', + eqeqeq: 'error', // Disable rules with a high performance cost. // See https://typescript-eslint.io/linting/troubleshooting/performance-troubleshooting/ 'import/default': 'off', diff --git a/subprojects/frontend/src/graph/export/exportDiagram.tsx b/subprojects/frontend/src/graph/export/exportDiagram.tsx index 33d93105..8854feda 100644 --- a/subprojects/frontend/src/graph/export/exportDiagram.tsx +++ b/subprojects/frontend/src/graph/export/exportDiagram.tsx @@ -296,6 +296,48 @@ async function serializePNG( }); } +/** + * Compensate for text baseline alignment as a workaround for the lack of support + * for the `dominant-baseline` SVG property for positioning text. + * + * We remove the `dominant-baseline` attribute from the already in the DOM and + * measure the vertical shift to compensate for it in the serialized SVG. + * This causes frequent DOM layout calculations, but we only do this when creating + * a PDF and make sure to restore the original layout at the end. + * + * @param svg The original SVG document currently present in the DOM. + * @param copyOfSVG The SVG document to be serialized as PDF, not in the DOM. + */ +function fixTextBaseline(svg: SVGSVGElement, copyOfSVG: SVGSVGElement): void { + const originalText = svg.querySelectorAll( + 'text[dominant-baseline]', + ); + const copiedText = copyOfSVG.querySelectorAll( + 'text[dominant-baseline]', + ); + copiedText.forEach((text, i) => { + const original = originalText[i]; + if (original === undefined) { + return; + } + const y = parseFloat(original.getAttribute('y') ?? ''); + const dominantBaseline = original.getAttribute('dominant-baseline'); + if (dominantBaseline === null) { + return; + } + const bbox = original.getBBox(); + let shiftedBBox: DOMRect; + try { + original.setAttribute('dominant-baseline', 'auto'); + shiftedBBox = original.getBBox(); + } finally { + original.setAttribute('dominant-baseline', dominantBaseline); + } + text.setAttribute('y', String(y + bbox.y - shiftedBBox.y)); + text.setAttribute('dominant-baseline', 'auto'); + }); +} + let serializePDFCached: | ((svg: SVGSVGElement, embedFonts: boolean) => Promise) | undefined; @@ -386,6 +428,7 @@ export default async function exportDiagram( ); if (settings.format === 'pdf') { + fixTextBaseline(svg, copyOfSVG); const pdf = await serializePDF(copyOfSVG, settings); await saveBlob(pdf, `${graph.name}.pdf`, { id: EXPORT_ID,