/* * Copyright [2021] [wisemapping] * * Licensed under WiseMapping Public License, Version 1.0 (the "License"). * It is basically the Apache License, Version 2.0 (the "License") plus the * "powered by wisemapping" text requirement on every single page; * you may not use this file except in compliance with the License. * You may obtain a copy of the license at * * http://www.wisemapping.org/license * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import SizeType from '../SizeType'; import Exporter from './Exporter'; class SVGExporter extends Exporter { private svgElement: Element; private static prolog = '\n'; private static regexpTranslate = /translate\((-?[0-9]+.[0-9]+),(-?[0-9]+.[0-9]+)\)/; private static padding = 30; private adjustToFit: boolean; private static MAX_SUPPORTED_SIZE = 2500; constructor(svgElement: Element, adjustToFit = true) { super('svg', 'image/svg+xml'); this.svgElement = svgElement; this.adjustToFit = adjustToFit; } export(): Promise { // Replace all images for in-line images ... let svgTxt: string = new XMLSerializer().serializeToString(this.svgElement); svgTxt = SVGExporter.prolog + svgTxt; // Are namespace declared ?. Otherwise, force the declaration ... if (svgTxt.indexOf('xmlns:xlink=') === -1) { svgTxt = svgTxt.replace('rect')); const translates: SizeType[] = rectElems.map((rect: Element) => { const g = rect.parentElement; const transformStr = g?.getAttribute('transform'); let result: SizeType = { width: 0, height: 0 }; if (transformStr) { // Looking to parse translate(220.00000,279.00000) scale(1.00000,1.00000) const match = transformStr.match(SVGExporter.regexpTranslate); if (match !== null) { result = { width: Number.parseFloat(match[1]), height: Number.parseFloat(match[2]) }; // Add rect size ... const widthAttr = rect.getAttribute('width'); if (result.width > 0 && widthAttr) { const rectWidth = Number.parseFloat(widthAttr); result.width += rectWidth; } const heightAttr = rect.getAttribute('height'); if (result.height > 0 && heightAttr) { const rectHeight = Number.parseFloat(heightAttr); result.height += rectHeight; } } } return result; }); // Find max and mins ... const widths = translates.map((t) => t.width).sort((a, b) => a - b); const heights = translates.map((t) => t.height).sort((a, b) => a - b); const minX = widths[0] - SVGExporter.padding; const minY = heights[0] - SVGExporter.padding; const maxX = widths[widths.length - 1] + SVGExporter.padding; const maxY = heights[heights.length - 1] + SVGExporter.padding; return { minX, maxX, minY, maxY, }; } getImgSize(): SizeType { const { minX, maxX, minY, maxY } = this._calcualteDimensions(); let width: number = maxX + Math.abs(minX); let height: number = maxY + Math.abs(minY); // Prevents an image too big. Failures seen during export in couple of browsers if (Math.max(width, height) > SVGExporter.MAX_SUPPORTED_SIZE) { const scale = Math.max(width, height) / SVGExporter.MAX_SUPPORTED_SIZE; width /= scale; height /= scale; } return { width, height }; } private _normalizeToFit(document: Document): Document { const { minX, maxX, minY, maxY } = this._calcualteDimensions(); const svgElem = document.firstChild as Element; const width = maxX + Math.abs(minX); const height = maxY + Math.abs(minY); svgElem.setAttribute('viewBox', `${minX} ${minY} ${width} ${height}`); svgElem.setAttribute('preserveAspectRatio', 'xMinYMin'); // Get image size ... const imgSize = this.getImgSize(); svgElem.setAttribute('width', imgSize.width.toFixed(0)); svgElem.setAttribute('height', imgSize.height.toFixed(0)); return document; } private static parseXMLString = (xmlStr: string, mimeType: DOMParserSupportedType) => { const parser = new DOMParser(); const xmlDoc = parser.parseFromString(xmlStr, mimeType); // Is there any parsing error ?. if (xmlDoc.getElementsByTagName('parsererror').length > 0) { const xmmStr = new XMLSerializer().serializeToString(xmlDoc); console.log(xmmStr); throw new Error(`Unexpected error parsing: ${xmlStr}.Error: ${xmmStr}`); } return xmlDoc; }; } export default SVGExporter;