@plait/core 0.11.0 → 0.12.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -18,4 +18,5 @@ export * from './element';
18
18
  export * from './viewport';
19
19
  export * from './common';
20
20
  export * from './moving-element';
21
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wbGFpdC9zcmMvdXRpbHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxvQkFBb0IsQ0FBQztBQUNuQyxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsYUFBYSxDQUFDO0FBQzVCLGNBQWMsUUFBUSxDQUFDO0FBQ3ZCLGNBQWMsV0FBVyxDQUFDO0FBQzFCLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsVUFBVSxDQUFDO0FBQ3pCLGNBQWMsa0JBQWtCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2JvYXJkJztcbmV4cG9ydCAqIGZyb20gJy4vZG9tL2NvbW1vbic7XG5leHBvcnQgKiBmcm9tICcuL2RvbS9mb3JlaWduJztcbmV4cG9ydCAqIGZyb20gJy4vZW52aXJvbm1lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9oZWxwZXInO1xuZXhwb3J0ICogZnJvbSAnLi9oaXN0b3J5JztcbmV4cG9ydCAqIGZyb20gJy4vaG90a2V5cyc7XG5leHBvcnQgKiBmcm9tICcuL2lkLWNyZWF0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9tYXRoJztcbmV4cG9ydCAqIGZyb20gJy4vd2Vhay1tYXBzJztcbmV4cG9ydCAqIGZyb20gJy4vc2VsZWN0ZWQtZWxlbWVudCc7XG5leHBvcnQgKiBmcm9tICcuL2RyYXcvcmVjdGFuZ2xlJztcbmV4cG9ydCAqIGZyb20gJy4vZHJhdy9hcnJvdyc7XG5leHBvcnQgKiBmcm9tICcuL2RyYXcvY2lyY2xlJztcbmV4cG9ydCAqIGZyb20gJy4vZHJhdy9saW5lJztcbmV4cG9ydCAqIGZyb20gJy4vdHJlZSc7XG5leHBvcnQgKiBmcm9tICcuL2VsZW1lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi92aWV3cG9ydCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbW1vbic7XG5leHBvcnQgKiBmcm9tICcuL21vdmluZy1lbGVtZW50JztcbiJdfQ==
21
+ export * from './to-image';
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wbGFpdC9zcmMvdXRpbHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxvQkFBb0IsQ0FBQztBQUNuQyxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsYUFBYSxDQUFDO0FBQzVCLGNBQWMsUUFBUSxDQUFDO0FBQ3ZCLGNBQWMsV0FBVyxDQUFDO0FBQzFCLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsVUFBVSxDQUFDO0FBQ3pCLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxZQUFZLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2JvYXJkJztcbmV4cG9ydCAqIGZyb20gJy4vZG9tL2NvbW1vbic7XG5leHBvcnQgKiBmcm9tICcuL2RvbS9mb3JlaWduJztcbmV4cG9ydCAqIGZyb20gJy4vZW52aXJvbm1lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9oZWxwZXInO1xuZXhwb3J0ICogZnJvbSAnLi9oaXN0b3J5JztcbmV4cG9ydCAqIGZyb20gJy4vaG90a2V5cyc7XG5leHBvcnQgKiBmcm9tICcuL2lkLWNyZWF0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9tYXRoJztcbmV4cG9ydCAqIGZyb20gJy4vd2Vhay1tYXBzJztcbmV4cG9ydCAqIGZyb20gJy4vc2VsZWN0ZWQtZWxlbWVudCc7XG5leHBvcnQgKiBmcm9tICcuL2RyYXcvcmVjdGFuZ2xlJztcbmV4cG9ydCAqIGZyb20gJy4vZHJhdy9hcnJvdyc7XG5leHBvcnQgKiBmcm9tICcuL2RyYXcvY2lyY2xlJztcbmV4cG9ydCAqIGZyb20gJy4vZHJhdy9saW5lJztcbmV4cG9ydCAqIGZyb20gJy4vdHJlZSc7XG5leHBvcnQgKiBmcm9tICcuL2VsZW1lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi92aWV3cG9ydCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbW1vbic7XG5leHBvcnQgKiBmcm9tICcuL21vdmluZy1lbGVtZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdG8taW1hZ2UnO1xuIl19
@@ -0,0 +1,106 @@
1
+ import { PlaitBoard } from '../interfaces';
2
+ import { getRectangleByElements } from './element';
3
+ function cloneCSSStyle(nativeNode, clonedNode) {
4
+ const targetStyle = clonedNode.style;
5
+ if (!targetStyle) {
6
+ return;
7
+ }
8
+ const sourceStyle = window.getComputedStyle(nativeNode);
9
+ if (sourceStyle.cssText) {
10
+ targetStyle.cssText = sourceStyle.cssText;
11
+ targetStyle.transformOrigin = sourceStyle.transformOrigin;
12
+ }
13
+ else {
14
+ Array.from(sourceStyle).forEach(name => {
15
+ let value = sourceStyle.getPropertyValue(name);
16
+ targetStyle.setProperty(name, value, sourceStyle.getPropertyPriority(name));
17
+ });
18
+ }
19
+ }
20
+ function createCanvas(width, height, fillStyle) {
21
+ const canvas = document.createElement('canvas');
22
+ const ctx = canvas.getContext('2d');
23
+ canvas.width = width;
24
+ canvas.height = height;
25
+ canvas.style.width = `${width}px`;
26
+ canvas.style.height = `${height}px`;
27
+ ctx.strokeStyle = '#ffffff';
28
+ ctx.fillStyle = fillStyle;
29
+ ctx.fillRect(0, 0, width, height);
30
+ return {
31
+ canvas,
32
+ ctx
33
+ };
34
+ }
35
+ function isElementNode(node) {
36
+ return node.nodeType === Node.ELEMENT_NODE;
37
+ }
38
+ function cloneSvg(board, options) {
39
+ const elementHostBox = getRectangleByElements(board, board.children, true);
40
+ const { width, height, x, y } = elementHostBox;
41
+ const { padding = 4, inlineStyleClassNames } = options;
42
+ const sourceSvg = PlaitBoard.getHost(board);
43
+ const cloneSvgElement = sourceSvg.cloneNode(true);
44
+ cloneSvgElement.style.width = `${width}px`;
45
+ cloneSvgElement.style.height = `${height}px`;
46
+ cloneSvgElement.style.backgroundColor = '';
47
+ cloneSvgElement.setAttribute('width', `${width}`);
48
+ cloneSvgElement.setAttribute('height', `${height}`);
49
+ cloneSvgElement.setAttribute('viewBox', [x - padding, y - padding, width + 2 * padding, height + 2 * padding].join(','));
50
+ if (inlineStyleClassNames) {
51
+ const sourceNodes = Array.from(sourceSvg.querySelectorAll(inlineStyleClassNames));
52
+ const cloneNodes = Array.from(cloneSvgElement.querySelectorAll(inlineStyleClassNames));
53
+ sourceNodes.forEach((node, index) => {
54
+ const cloneNode = cloneNodes[index];
55
+ const childElements = Array.from(node.querySelectorAll('*')).filter(isElementNode);
56
+ const cloneChildElements = Array.from(cloneNode.querySelectorAll('*')).filter(isElementNode);
57
+ sourceNodes.push(...childElements);
58
+ cloneNodes.push(...cloneChildElements);
59
+ });
60
+ sourceNodes.forEach((node, index) => {
61
+ const cloneNode = cloneNodes[index];
62
+ cloneCSSStyle(node, cloneNode);
63
+ });
64
+ }
65
+ return cloneSvgElement;
66
+ }
67
+ function loadImage(src) {
68
+ return new Promise((resolve, reject) => {
69
+ const img = new Image();
70
+ img.onload = () => resolve(img);
71
+ img.onerror = () => reject(new Error('Failed to load image'));
72
+ img.src = src;
73
+ });
74
+ }
75
+ export async function toImage(board, options) {
76
+ if (!board) {
77
+ return undefined;
78
+ }
79
+ const elementHostBox = getRectangleByElements(board, board.children, true);
80
+ const { ratio = 2, fillStyle = 'transparent' } = options;
81
+ const { width, height } = elementHostBox;
82
+ const ratioWidth = width * ratio;
83
+ const ratioHeight = height * ratio;
84
+ const cloneSvgElement = cloneSvg(board, options);
85
+ const { canvas, ctx } = createCanvas(ratioWidth, ratioHeight, fillStyle);
86
+ const svgStr = new XMLSerializer().serializeToString(cloneSvgElement);
87
+ const imgSrc = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svgStr)}`;
88
+ try {
89
+ const img = await loadImage(imgSrc);
90
+ ctx.drawImage(img, 0, 0, ratioWidth, ratioHeight);
91
+ const url = canvas.toDataURL('image/png');
92
+ return url;
93
+ }
94
+ catch (error) {
95
+ console.error('Error converting SVG to image:', error);
96
+ return undefined;
97
+ }
98
+ }
99
+ export function downloadImage(url, name) {
100
+ const a = document.createElement('a');
101
+ a.href = url;
102
+ a.download = name;
103
+ a.click();
104
+ a.remove();
105
+ }
106
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"to-image.js","sourceRoot":"","sources":["../../../../packages/plait/src/utils/to-image.ts"],"names":[],"mappings":"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","sourcesContent":["import { PlaitBoard } from '../interfaces';\nimport { getRectangleByElements } from './element';\n\nexport interface ToImageOptions {\n    name?: string;\n    ratio?: number;\n    padding?: number;\n    fillStyle?: string;\n    // 逗号类名列表。 该列表必须采用 class1,class2,... 的形式。\n    inlineStyleClassNames?: string;\n}\n\nfunction cloneCSSStyle<T extends HTMLElement>(nativeNode: T, clonedNode: T) {\n    const targetStyle = clonedNode.style;\n    if (!targetStyle) {\n        return;\n    }\n\n    const sourceStyle = window.getComputedStyle(nativeNode);\n    if (sourceStyle.cssText) {\n        targetStyle.cssText = sourceStyle.cssText;\n        targetStyle.transformOrigin = sourceStyle.transformOrigin;\n    } else {\n        Array.from(sourceStyle).forEach(name => {\n            let value = sourceStyle.getPropertyValue(name);\n            targetStyle.setProperty(name, value, sourceStyle.getPropertyPriority(name));\n        });\n    }\n}\n\nfunction createCanvas(width: number, height: number, fillStyle: string) {\n    const canvas = document.createElement('canvas');\n    const ctx = canvas.getContext('2d')!;\n\n    canvas.width = width;\n    canvas.height = height;\n    canvas.style.width = `${width}px`;\n    canvas.style.height = `${height}px`;\n    ctx.strokeStyle = '#ffffff';\n    ctx.fillStyle = fillStyle;\n    ctx.fillRect(0, 0, width, height);\n\n    return {\n        canvas,\n        ctx\n    };\n}\n\nfunction isElementNode(node: Node): node is HTMLElement {\n    return node.nodeType === Node.ELEMENT_NODE;\n}\n\nfunction cloneSvg(board: PlaitBoard, options: ToImageOptions) {\n    const elementHostBox = getRectangleByElements(board, board.children, true);\n    const { width, height, x, y } = elementHostBox;\n    const { padding = 4, inlineStyleClassNames } = options;\n    const sourceSvg = PlaitBoard.getHost(board);\n    const cloneSvgElement = sourceSvg.cloneNode(true) as SVGElement;\n\n    cloneSvgElement.style.width = `${width}px`;\n    cloneSvgElement.style.height = `${height}px`;\n    cloneSvgElement.style.backgroundColor = '';\n    cloneSvgElement.setAttribute('width', `${width}`);\n    cloneSvgElement.setAttribute('height', `${height}`);\n    cloneSvgElement.setAttribute('viewBox', [x - padding, y - padding, width + 2 * padding, height + 2 * padding].join(','));\n\n    if (inlineStyleClassNames) {\n        const sourceNodes = Array.from(sourceSvg.querySelectorAll(inlineStyleClassNames));\n        const cloneNodes = Array.from(cloneSvgElement.querySelectorAll(inlineStyleClassNames));\n\n        sourceNodes.forEach((node, index) => {\n            const cloneNode = cloneNodes[index];\n            const childElements = Array.from(node.querySelectorAll('*')).filter(isElementNode) as HTMLElement[];\n            const cloneChildElements = Array.from(cloneNode.querySelectorAll('*')).filter(isElementNode) as HTMLElement[];\n            sourceNodes.push(...childElements);\n            cloneNodes.push(...cloneChildElements);\n        });\n        sourceNodes.forEach((node, index) => {\n            const cloneNode = cloneNodes[index];\n            cloneCSSStyle(node as HTMLElement, cloneNode as HTMLElement);\n        });\n    }\n\n    return cloneSvgElement;\n}\n\nfunction loadImage(src: string): Promise<HTMLImageElement> {\n    return new Promise((resolve, reject) => {\n        const img = new Image();\n        img.onload = () => resolve(img);\n        img.onerror = () => reject(new Error('Failed to load image'));\n        img.src = src;\n    });\n}\n\nexport async function toImage(board: PlaitBoard, options: ToImageOptions): Promise<string | undefined> {\n    if (!board) {\n        return undefined;\n    }\n\n    const elementHostBox = getRectangleByElements(board, board.children, true);\n    const { ratio = 2, fillStyle = 'transparent' } = options;\n    const { width, height } = elementHostBox;\n    const ratioWidth = width * ratio;\n    const ratioHeight = height * ratio;\n    const cloneSvgElement = cloneSvg(board, options);\n    const { canvas, ctx } = createCanvas(ratioWidth, ratioHeight, fillStyle);\n\n    const svgStr = new XMLSerializer().serializeToString(cloneSvgElement);\n    const imgSrc = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svgStr)}`;\n\n    try {\n        const img = await loadImage(imgSrc);\n        ctx.drawImage(img, 0, 0, ratioWidth, ratioHeight);\n        const url = canvas.toDataURL('image/png');\n        return url;\n    } catch (error) {\n        console.error('Error converting SVG to image:', error);\n        return undefined;\n    }\n}\n\nexport function downloadImage(url: string, name: string) {\n    const a = document.createElement('a');\n    a.href = url;\n    a.download = name;\n    a.click();\n    a.remove();\n}\n"]}
@@ -5,6 +5,7 @@ import { timer, Subject, fromEvent } from 'rxjs';
5
5
  import { takeUntil, filter, tap } from 'rxjs/operators';
6
6
  import produce, { createDraft, finishDraft, isDraft } from 'immer';
7
7
  import { isKeyHotkey, isHotkey } from 'is-hotkey';
8
+ import { __awaiter } from 'tslib';
8
9
  import * as i1 from '@angular/common';
9
10
  import { CommonModule } from '@angular/common';
10
11
 
@@ -1320,6 +1321,112 @@ const cacheMovingElements = (board, elements) => {
1320
1321
  BOARD_TO_MOVING_ELEMENT.set(board, elements);
1321
1322
  };
1322
1323
 
1324
+ function cloneCSSStyle(nativeNode, clonedNode) {
1325
+ const targetStyle = clonedNode.style;
1326
+ if (!targetStyle) {
1327
+ return;
1328
+ }
1329
+ const sourceStyle = window.getComputedStyle(nativeNode);
1330
+ if (sourceStyle.cssText) {
1331
+ targetStyle.cssText = sourceStyle.cssText;
1332
+ targetStyle.transformOrigin = sourceStyle.transformOrigin;
1333
+ }
1334
+ else {
1335
+ Array.from(sourceStyle).forEach(name => {
1336
+ let value = sourceStyle.getPropertyValue(name);
1337
+ targetStyle.setProperty(name, value, sourceStyle.getPropertyPriority(name));
1338
+ });
1339
+ }
1340
+ }
1341
+ function createCanvas(width, height, fillStyle) {
1342
+ const canvas = document.createElement('canvas');
1343
+ const ctx = canvas.getContext('2d');
1344
+ canvas.width = width;
1345
+ canvas.height = height;
1346
+ canvas.style.width = `${width}px`;
1347
+ canvas.style.height = `${height}px`;
1348
+ ctx.strokeStyle = '#ffffff';
1349
+ ctx.fillStyle = fillStyle;
1350
+ ctx.fillRect(0, 0, width, height);
1351
+ return {
1352
+ canvas,
1353
+ ctx
1354
+ };
1355
+ }
1356
+ function isElementNode(node) {
1357
+ return node.nodeType === Node.ELEMENT_NODE;
1358
+ }
1359
+ function cloneSvg(board, options) {
1360
+ const elementHostBox = getRectangleByElements(board, board.children, true);
1361
+ const { width, height, x, y } = elementHostBox;
1362
+ const { padding = 4, inlineStyleClassNames } = options;
1363
+ const sourceSvg = PlaitBoard.getHost(board);
1364
+ const cloneSvgElement = sourceSvg.cloneNode(true);
1365
+ cloneSvgElement.style.width = `${width}px`;
1366
+ cloneSvgElement.style.height = `${height}px`;
1367
+ cloneSvgElement.style.backgroundColor = '';
1368
+ cloneSvgElement.setAttribute('width', `${width}`);
1369
+ cloneSvgElement.setAttribute('height', `${height}`);
1370
+ cloneSvgElement.setAttribute('viewBox', [x - padding, y - padding, width + 2 * padding, height + 2 * padding].join(','));
1371
+ if (inlineStyleClassNames) {
1372
+ const sourceNodes = Array.from(sourceSvg.querySelectorAll(inlineStyleClassNames));
1373
+ const cloneNodes = Array.from(cloneSvgElement.querySelectorAll(inlineStyleClassNames));
1374
+ sourceNodes.forEach((node, index) => {
1375
+ const cloneNode = cloneNodes[index];
1376
+ const childElements = Array.from(node.querySelectorAll('*')).filter(isElementNode);
1377
+ const cloneChildElements = Array.from(cloneNode.querySelectorAll('*')).filter(isElementNode);
1378
+ sourceNodes.push(...childElements);
1379
+ cloneNodes.push(...cloneChildElements);
1380
+ });
1381
+ sourceNodes.forEach((node, index) => {
1382
+ const cloneNode = cloneNodes[index];
1383
+ cloneCSSStyle(node, cloneNode);
1384
+ });
1385
+ }
1386
+ return cloneSvgElement;
1387
+ }
1388
+ function loadImage(src) {
1389
+ return new Promise((resolve, reject) => {
1390
+ const img = new Image();
1391
+ img.onload = () => resolve(img);
1392
+ img.onerror = () => reject(new Error('Failed to load image'));
1393
+ img.src = src;
1394
+ });
1395
+ }
1396
+ function toImage(board, options) {
1397
+ return __awaiter(this, void 0, void 0, function* () {
1398
+ if (!board) {
1399
+ return undefined;
1400
+ }
1401
+ const elementHostBox = getRectangleByElements(board, board.children, true);
1402
+ const { ratio = 2, fillStyle = 'transparent' } = options;
1403
+ const { width, height } = elementHostBox;
1404
+ const ratioWidth = width * ratio;
1405
+ const ratioHeight = height * ratio;
1406
+ const cloneSvgElement = cloneSvg(board, options);
1407
+ const { canvas, ctx } = createCanvas(ratioWidth, ratioHeight, fillStyle);
1408
+ const svgStr = new XMLSerializer().serializeToString(cloneSvgElement);
1409
+ const imgSrc = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svgStr)}`;
1410
+ try {
1411
+ const img = yield loadImage(imgSrc);
1412
+ ctx.drawImage(img, 0, 0, ratioWidth, ratioHeight);
1413
+ const url = canvas.toDataURL('image/png');
1414
+ return url;
1415
+ }
1416
+ catch (error) {
1417
+ console.error('Error converting SVG to image:', error);
1418
+ return undefined;
1419
+ }
1420
+ });
1421
+ }
1422
+ function downloadImage(url, name) {
1423
+ const a = document.createElement('a');
1424
+ a.href = url;
1425
+ a.download = name;
1426
+ a.click();
1427
+ a.remove();
1428
+ }
1429
+
1323
1430
  const PlaitElement = {
1324
1431
  isRootElement(value) {
1325
1432
  const parent = NODE_TO_PARENT.get(value);
@@ -2829,5 +2936,5 @@ const clearNodeWeakMap = (object) => {
2829
2936
  * Generated bundle index. Do not edit.
2830
2937
  */
2831
2938
 
2832
- export { BOARD_TO_COMPONENT, BOARD_TO_ELEMENT_HOST, BOARD_TO_HOST, BOARD_TO_IS_SELECTION_MOVING, BOARD_TO_MOVING_ELEMENT, BOARD_TO_MOVING_POINT, BOARD_TO_ON_CHANGE, BOARD_TO_ROUGH_SVG, BOARD_TO_SELECTED_ELEMENT, BOARD_TO_TEMPORARY_ELEMENTS, BOARD_TO_VIEWPORT_ORIGINATION, BoardTransforms, CLIP_BOARD_FORMAT_KEY, ColorfulThemeColor, DarkThemeColor, DefaultThemeColor, ELEMENT_TO_COMPONENT, FLUSHING, IS_APPLE, IS_BOARD_CACHE, IS_CHROME, IS_CHROME_LEGACY, IS_EDGE_LEGACY, IS_FIREFOX, IS_IOS, IS_SAFARI, IS_TEXT_EDITABLE, MAX_RADIUS, MERGING, NODE_TO_INDEX, NODE_TO_PARENT, NS, PATH_REFS, POINTER_BUTTON, Path, PlaitBoard, PlaitBoardComponent, PlaitChildrenElement, PlaitElement, PlaitElementComponent, PlaitHistoryBoard, PlaitIslandBaseComponent, PlaitModule, PlaitNode, PlaitOperation, PlaitPluginElementComponent, PlaitPluginKey, PlaitPointerType, Point, RectangleClient, RetroThemeColor, SAVING, SCROLL_BAR_WIDTH, SELECTION_BORDER_COLOR, SELECTION_FILL_COLOR, Selection, SoftThemeColor, StarryThemeColor, ThemeColorMode, ThemeColors, Transforms, Viewport, addMovingElements, addSelectedElement, arrowPoints, cacheMovingElements, cacheSelectedElements, clampZoomLevel, clearNodeWeakMap, clearSelectedElement, clearSelectionMoving, clearViewportOrigination, createForeignObject, createG, createPath, createSVG, createSelectionOuterG, createTestingBoard, createText, debounce, deleteTemporaryElements, depthFirstRecursion, distanceBetweenPointAndPoint, distanceBetweenPointAndRectangle, distanceBetweenPointAndSegment, drawAbstractRoundRectangle, drawArrow, drawCircle, drawLine, drawLinearPath, drawRoundRectangle, fakeNodeWeakMap, getBoardRectangle, getElementHostBBox, getHitElementOfRoot, getHitElements, getMovingElements, getRealScrollBarWidth, getRectangleByElements, getSelectedElements, getTemporaryElements, getViewBox, getViewBoxCenterPoint, getViewportContainerRect, getViewportOrigination, hasBeforeContextChange, hasInputOrTextareaTarget, hasOnBoardChange, hasOnContextChanged, hotkeys, idCreator, initializeViewBox, initializeViewportContainer, initializeViewportOffset, inverse, isDOMElement, isDOMNode, isFromScrolling, isFromViewportChange, isHitElements, isInPlaitBoard, isMainPointer, isNullOrUndefined, isSecondaryPointer, isSelectedElement, isSelectionMoving, isSetViewportOperation, normalizePoint, removeMovingElements, removeSelectedElement, rotate, scrollToRectangle, setIsFromScrolling, setIsFromViewportChange, setSVGViewBox, setSelectionMoving, shouldClear, shouldMerge, shouldSave, throttleRAF, toPoint, transformPoint, transformPoints, updateForeignObject, updateViewportContainerScroll, updateViewportOffset, updateViewportOrigination, withMoving, withOptions, withSelection };
2939
+ export { BOARD_TO_COMPONENT, BOARD_TO_ELEMENT_HOST, BOARD_TO_HOST, BOARD_TO_IS_SELECTION_MOVING, BOARD_TO_MOVING_ELEMENT, BOARD_TO_MOVING_POINT, BOARD_TO_ON_CHANGE, BOARD_TO_ROUGH_SVG, BOARD_TO_SELECTED_ELEMENT, BOARD_TO_TEMPORARY_ELEMENTS, BOARD_TO_VIEWPORT_ORIGINATION, BoardTransforms, CLIP_BOARD_FORMAT_KEY, ColorfulThemeColor, DarkThemeColor, DefaultThemeColor, ELEMENT_TO_COMPONENT, FLUSHING, IS_APPLE, IS_BOARD_CACHE, IS_CHROME, IS_CHROME_LEGACY, IS_EDGE_LEGACY, IS_FIREFOX, IS_IOS, IS_SAFARI, IS_TEXT_EDITABLE, MAX_RADIUS, MERGING, NODE_TO_INDEX, NODE_TO_PARENT, NS, PATH_REFS, POINTER_BUTTON, Path, PlaitBoard, PlaitBoardComponent, PlaitChildrenElement, PlaitElement, PlaitElementComponent, PlaitHistoryBoard, PlaitIslandBaseComponent, PlaitModule, PlaitNode, PlaitOperation, PlaitPluginElementComponent, PlaitPluginKey, PlaitPointerType, Point, RectangleClient, RetroThemeColor, SAVING, SCROLL_BAR_WIDTH, SELECTION_BORDER_COLOR, SELECTION_FILL_COLOR, Selection, SoftThemeColor, StarryThemeColor, ThemeColorMode, ThemeColors, Transforms, Viewport, addMovingElements, addSelectedElement, arrowPoints, cacheMovingElements, cacheSelectedElements, clampZoomLevel, clearNodeWeakMap, clearSelectedElement, clearSelectionMoving, clearViewportOrigination, createForeignObject, createG, createPath, createSVG, createSelectionOuterG, createTestingBoard, createText, debounce, deleteTemporaryElements, depthFirstRecursion, distanceBetweenPointAndPoint, distanceBetweenPointAndRectangle, distanceBetweenPointAndSegment, downloadImage, drawAbstractRoundRectangle, drawArrow, drawCircle, drawLine, drawLinearPath, drawRoundRectangle, fakeNodeWeakMap, getBoardRectangle, getElementHostBBox, getHitElementOfRoot, getHitElements, getMovingElements, getRealScrollBarWidth, getRectangleByElements, getSelectedElements, getTemporaryElements, getViewBox, getViewBoxCenterPoint, getViewportContainerRect, getViewportOrigination, hasBeforeContextChange, hasInputOrTextareaTarget, hasOnBoardChange, hasOnContextChanged, hotkeys, idCreator, initializeViewBox, initializeViewportContainer, initializeViewportOffset, inverse, isDOMElement, isDOMNode, isFromScrolling, isFromViewportChange, isHitElements, isInPlaitBoard, isMainPointer, isNullOrUndefined, isSecondaryPointer, isSelectedElement, isSelectionMoving, isSetViewportOperation, normalizePoint, removeMovingElements, removeSelectedElement, rotate, scrollToRectangle, setIsFromScrolling, setIsFromViewportChange, setSVGViewBox, setSelectionMoving, shouldClear, shouldMerge, shouldSave, throttleRAF, toImage, toPoint, transformPoint, transformPoints, updateForeignObject, updateViewportContainerScroll, updateViewportOffset, updateViewportOrigination, withMoving, withOptions, withSelection };
2833
2940
  //# sourceMappingURL=plait-core.mjs.map