2020-01-13 11:04:28 -08:00
|
|
|
import { randomSeed } from "roughjs/bin/math";
|
2020-01-07 23:49:39 +04:00
|
|
|
import nanoid from "nanoid";
|
2020-02-01 15:49:18 +04:00
|
|
|
import { Point } from "roughjs/bin/geometry";
|
2020-01-06 19:34:22 +04:00
|
|
|
|
2020-01-21 00:16:22 +01:00
|
|
|
import { ExcalidrawElement, ExcalidrawTextElement } from "../element/types";
|
|
|
|
import { measureText } from "../utils";
|
|
|
|
|
2020-01-06 19:34:22 +04:00
|
|
|
export function newElement(
|
|
|
|
type: string,
|
|
|
|
x: number,
|
|
|
|
y: number,
|
|
|
|
strokeColor: string,
|
|
|
|
backgroundColor: string,
|
|
|
|
fillStyle: string,
|
|
|
|
strokeWidth: number,
|
|
|
|
roughness: number,
|
|
|
|
opacity: number,
|
|
|
|
width = 0,
|
2020-01-24 12:04:54 +02:00
|
|
|
height = 0,
|
2020-01-06 19:34:22 +04:00
|
|
|
) {
|
|
|
|
const element = {
|
2020-01-07 23:49:39 +04:00
|
|
|
id: nanoid(),
|
2020-01-07 19:04:52 +04:00
|
|
|
type,
|
|
|
|
x,
|
|
|
|
y,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
strokeColor,
|
|
|
|
backgroundColor,
|
|
|
|
fillStyle,
|
|
|
|
strokeWidth,
|
|
|
|
roughness,
|
|
|
|
opacity,
|
2020-01-12 04:00:00 +04:00
|
|
|
seed: randomSeed(),
|
2020-02-01 15:49:18 +04:00
|
|
|
points: [] as Point[],
|
2020-03-09 22:34:50 -07:00
|
|
|
version: 1,
|
2020-01-06 19:34:22 +04:00
|
|
|
};
|
|
|
|
return element;
|
|
|
|
}
|
2020-01-08 19:54:42 +01:00
|
|
|
|
2020-01-21 00:16:22 +01:00
|
|
|
export function newTextElement(
|
|
|
|
element: ExcalidrawElement,
|
|
|
|
text: string,
|
2020-01-24 12:04:54 +02:00
|
|
|
font: string,
|
2020-01-21 00:16:22 +01:00
|
|
|
) {
|
|
|
|
const metrics = measureText(text, font);
|
|
|
|
const textElement: ExcalidrawTextElement = {
|
|
|
|
...element,
|
|
|
|
type: "text",
|
|
|
|
text: text,
|
|
|
|
font: font,
|
|
|
|
// Center the text
|
|
|
|
x: element.x - metrics.width / 2,
|
|
|
|
y: element.y - metrics.height / 2,
|
|
|
|
width: metrics.width,
|
|
|
|
height: metrics.height,
|
2020-01-24 12:04:54 +02:00
|
|
|
baseline: metrics.baseline,
|
2020-01-21 00:16:22 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return textElement;
|
|
|
|
}
|
|
|
|
|
2020-02-19 22:28:11 +01:00
|
|
|
// Simplified deep clone for the purpose of cloning ExcalidrawElement only
|
|
|
|
// (doesn't clone Date, RegExp, Map, Set, Typed arrays etc.)
|
|
|
|
//
|
|
|
|
// Adapted from https://github.com/lukeed/klona
|
|
|
|
function _duplicateElement(val: any, depth: number = 0) {
|
|
|
|
if (val == null || typeof val !== "object") {
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Object.prototype.toString.call(val) === "[object Object]") {
|
|
|
|
const tmp =
|
|
|
|
typeof val.constructor === "function"
|
|
|
|
? Object.create(Object.getPrototypeOf(val))
|
|
|
|
: {};
|
2020-03-07 10:20:38 -05:00
|
|
|
for (const key in val) {
|
|
|
|
if (val.hasOwnProperty(key)) {
|
2020-02-19 22:28:11 +01:00
|
|
|
// don't copy top-level shape property, which we want to regenerate
|
2020-03-07 10:20:38 -05:00
|
|
|
if (depth === 0 && (key === "shape" || key === "canvas")) {
|
2020-02-19 22:28:11 +01:00
|
|
|
continue;
|
|
|
|
}
|
2020-03-07 10:20:38 -05:00
|
|
|
tmp[key] = _duplicateElement(val[key], depth + 1);
|
2020-02-19 22:28:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return tmp;
|
2020-02-09 23:57:14 +01:00
|
|
|
}
|
|
|
|
|
2020-02-19 22:28:11 +01:00
|
|
|
if (Array.isArray(val)) {
|
|
|
|
let k = val.length;
|
|
|
|
const arr = new Array(k);
|
|
|
|
while (k--) {
|
|
|
|
arr[k] = _duplicateElement(val[k], depth + 1);
|
|
|
|
}
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function duplicateElement(element: ReturnType<typeof newElement>) {
|
|
|
|
const copy = _duplicateElement(element);
|
2020-01-08 19:54:42 +01:00
|
|
|
copy.id = nanoid();
|
|
|
|
copy.seed = randomSeed();
|
|
|
|
return copy;
|
|
|
|
}
|