import { ExcalidrawElement, ExcalidrawTextElement } from "../element/types"; import { isTextElement } from "../element/typeChecks"; import { getDiamondPoints, getArrowPoints, getElementAbsoluteCoords, } from "../element/bounds"; import { RoughCanvas } from "roughjs/bin/canvas"; import { Drawable } from "roughjs/bin/core"; import { Point } from "roughjs/bin/geometry"; import { RoughSVG } from "roughjs/bin/svg"; import { RoughGenerator } from "roughjs/bin/generator"; import { SceneState } from "../scene/types"; import { SVG_NS, distance } from "../utils"; import rough from "roughjs/bin/rough"; const CANVAS_PADDING = 20; function generateElementCanvas(element: ExcalidrawElement, zoom: number) { const canvas = document.createElement("canvas"); var context = canvas.getContext("2d")!; const isLinear = /\b(arrow|line)\b/.test(element.type); if (isLinear) { const [x1, y1, x2, y2] = getElementAbsoluteCoords(element); canvas.width = distance(x1, x2) * window.devicePixelRatio * zoom + CANVAS_PADDING * 2; canvas.height = distance(y1, y2) * window.devicePixelRatio * zoom + CANVAS_PADDING * 2; element.canvasOffsetX = element.x > x1 ? Math.floor(distance(element.x, x1)) * window.devicePixelRatio : 0; element.canvasOffsetY = element.y > y1 ? Math.floor(distance(element.y, y1)) * window.devicePixelRatio : 0; context.translate( element.canvasOffsetX * zoom, element.canvasOffsetY * zoom, ); } else { canvas.width = element.width * window.devicePixelRatio * zoom + CANVAS_PADDING * 2; canvas.height = element.height * window.devicePixelRatio * zoom + CANVAS_PADDING * 2; } context.translate(CANVAS_PADDING, CANVAS_PADDING); context.scale(window.devicePixelRatio * zoom, window.devicePixelRatio * zoom); const rc = rough.canvas(canvas); drawElementOnCanvas(element, rc, context); element.canvas = canvas; element.canvasZoom = zoom; context.translate(-CANVAS_PADDING, -CANVAS_PADDING); } function drawElementOnCanvas( element: ExcalidrawElement, rc: RoughCanvas, context: CanvasRenderingContext2D, ) { context.globalAlpha = element.opacity / 100; switch (element.type) { case "rectangle": case "diamond": case "ellipse": { rc.draw(element.shape as Drawable); break; } case "arrow": case "line": { (element.shape as Drawable[]).forEach(shape => rc.draw(shape)); break; } default: { if (isTextElement(element)) { const font = context.font; context.font = element.font; const fillStyle = context.fillStyle; context.fillStyle = element.strokeColor; // Canvas does not support multiline text by default const lines = element.text.replace(/\r\n?/g, "\n").split("\n"); const lineHeight = element.height / lines.length; const offset = element.height - element.baseline; for (let i = 0; i < lines.length; i++) { context.fillText(lines[i], 0, (i + 1) * lineHeight - offset); } context.fillStyle = fillStyle; context.font = font; } else { throw new Error(`Unimplemented type ${element.type}`); } } } context.globalAlpha = 1; } function generateElement( element: ExcalidrawElement, generator: RoughGenerator, sceneState?: SceneState, ) { if (!element.shape) { element.canvas = null; switch (element.type) { case "rectangle": element.shape = generator.rectangle( 0, 0, element.width, element.height, { stroke: element.strokeColor, fill: element.backgroundColor === "transparent" ? undefined : element.backgroundColor, fillStyle: element.fillStyle, strokeWidth: element.strokeWidth, roughness: element.roughness, seed: element.seed, }, ); break; case "diamond": { const [ topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY, ] = getDiamondPoints(element); element.shape = generator.polygon( [ [topX, topY], [rightX, rightY], [bottomX, bottomY], [leftX, leftY], ], { stroke: element.strokeColor, fill: element.backgroundColor === "transparent" ? undefined : element.backgroundColor, fillStyle: element.fillStyle, strokeWidth: element.strokeWidth, roughness: element.roughness, seed: element.seed, }, ); break; } case "ellipse": element.shape = generator.ellipse( element.width / 2, element.height / 2, element.width, element.height, { stroke: element.strokeColor, fill: element.backgroundColor === "transparent" ? undefined : element.backgroundColor, fillStyle: element.fillStyle, strokeWidth: element.strokeWidth, roughness: element.roughness, seed: element.seed, curveFitting: 1, }, ); break; case "line": case "arrow": { const options = { stroke: element.strokeColor, strokeWidth: element.strokeWidth, roughness: element.roughness, seed: element.seed, }; // points array can be empty in the beginning, so it is important to add // initial position to it const points: Point[] = element.points.length ? element.points : [[0, 0]]; // curve is always the first element // this simplifies finding the curve for an element element.shape = [generator.curve(points, options)]; // add lines only in arrow if (element.type === "arrow") { const [x2, y2, x3, y3, x4, y4] = getArrowPoints(element); element.shape.push( ...[ generator.line(x3, y3, x2, y2, options), generator.line(x4, y4, x2, y2, options), ], ); } break; } case "text": { // just to ensure we don't regenerate element.canvas on rerenders element.shape = []; break; } } } const zoom = sceneState ? sceneState.zoom : 1; if (!element.canvas || element.canvasZoom !== zoom) { generateElementCanvas(element, zoom); } } function drawElementFromCanvas( element: ExcalidrawElement | ExcalidrawTextElement, rc: RoughCanvas, context: CanvasRenderingContext2D, sceneState: SceneState, ) { context.scale(1 / window.devicePixelRatio, 1 / window.devicePixelRatio); context.translate( -CANVAS_PADDING / sceneState.zoom, -CANVAS_PADDING / sceneState.zoom, ); context.drawImage( element.canvas!, Math.floor( -element.canvasOffsetX + (Math.floor(element.x) + sceneState.scrollX) * window.devicePixelRatio, ), Math.floor( -element.canvasOffsetY + (Math.floor(element.y) + sceneState.scrollY) * window.devicePixelRatio, ), element.canvas!.width / sceneState.zoom, element.canvas!.height / sceneState.zoom, ); context.translate( CANVAS_PADDING / sceneState.zoom, CANVAS_PADDING / sceneState.zoom, ); context.scale(window.devicePixelRatio, window.devicePixelRatio); } export function renderElement( element: ExcalidrawElement, rc: RoughCanvas, context: CanvasRenderingContext2D, renderOptimizations: boolean, sceneState: SceneState, ) { const generator = rc.generator; switch (element.type) { case "selection": { context.translate( element.x + sceneState.scrollX, element.y + sceneState.scrollY, ); const fillStyle = context.fillStyle; context.fillStyle = "rgba(0, 0, 255, 0.10)"; context.fillRect(0, 0, element.width, element.height); context.fillStyle = fillStyle; break; } case "rectangle": case "diamond": case "ellipse": case "line": case "arrow": case "text": { generateElement(element, generator, sceneState); if (renderOptimizations) { drawElementFromCanvas(element, rc, context, sceneState); } else { const offsetX = Math.floor(element.x + sceneState.scrollX); const offsetY = Math.floor(element.y + sceneState.scrollY); context.translate(offsetX, offsetY); drawElementOnCanvas(element, rc, context); context.translate(-offsetX, -offsetY); } break; } default: { throw new Error(`Unimplemented type ${element.type}`); } } } export function renderElementToSvg( element: ExcalidrawElement, rsvg: RoughSVG, svgRoot: SVGElement, offsetX?: number, offsetY?: number, ) { const generator = rsvg.generator; switch (element.type) { case "selection": { // Since this is used only during editing experience, which is canvas based, // this should not happen throw new Error("Selection rendering is not supported for SVG"); } case "rectangle": case "diamond": case "ellipse": case "line": { generateElement(element, generator); const node = rsvg.draw(element.shape as Drawable); const opacity = element.opacity / 100; if (opacity !== 1) { node.setAttribute("stroke-opacity", `${opacity}`); node.setAttribute("fill-opacity", `${opacity}`); } node.setAttribute( "transform", `translate(${offsetX || 0} ${offsetY || 0})`, ); svgRoot.appendChild(node); break; } case "arrow": { generateElement(element, generator); const group = svgRoot.ownerDocument!.createElementNS(SVG_NS, "g"); const opacity = element.opacity / 100; (element.shape as Drawable[]).forEach(shape => { const node = rsvg.draw(shape); if (opacity !== 1) { node.setAttribute("stroke-opacity", `${opacity}`); node.setAttribute("fill-opacity", `${opacity}`); } node.setAttribute( "transform", `translate(${offsetX || 0} ${offsetY || 0})`, ); group.appendChild(node); }); svgRoot.appendChild(group); break; } default: { if (isTextElement(element)) { const opacity = element.opacity / 100; const node = svgRoot.ownerDocument!.createElementNS(SVG_NS, "g"); if (opacity !== 1) { node.setAttribute("stroke-opacity", `${opacity}`); node.setAttribute("fill-opacity", `${opacity}`); } node.setAttribute( "transform", `translate(${offsetX || 0} ${offsetY || 0})`, ); const lines = element.text.replace(/\r\n?/g, "\n").split("\n"); const lineHeight = element.height / lines.length; const offset = element.height - element.baseline; const fontSplit = element.font.split(" ").filter(d => !!d.trim()); let fontFamily = fontSplit[0]; let fontSize = "20px"; if (fontSplit.length > 1) { fontFamily = fontSplit[1]; fontSize = fontSplit[0]; } for (let i = 0; i < lines.length; i++) { const text = svgRoot.ownerDocument!.createElementNS(SVG_NS, "text"); text.textContent = lines[i]; text.setAttribute("x", "0"); text.setAttribute("y", `${(i + 1) * lineHeight - offset}`); text.setAttribute("font-family", fontFamily); text.setAttribute("font-size", fontSize); text.setAttribute("fill", element.strokeColor); node.appendChild(text); } svgRoot.appendChild(node); } else { throw new Error(`Unimplemented type ${element.type}`); } } } }