@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.
- package/esm2020/utils/index.mjs +2 -1
- package/esm2020/utils/to-image.mjs +106 -0
- package/fesm2015/plait-core.mjs +108 -1
- package/fesm2015/plait-core.mjs.map +1 -1
- package/fesm2020/plait-core.mjs +105 -1
- package/fesm2020/plait-core.mjs.map +1 -1
- package/package.json +1 -1
- package/utils/index.d.ts +1 -0
- package/utils/to-image.d.ts +10 -0
package/esm2020/utils/index.mjs
CHANGED
|
@@ -18,4 +18,5 @@ export * from './element';
|
|
|
18
18
|
export * from './viewport';
|
|
19
19
|
export * from './common';
|
|
20
20
|
export * from './moving-element';
|
|
21
|
-
|
|
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"]}
|
package/fesm2015/plait-core.mjs
CHANGED
|
@@ -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
|