@plait/core 0.77.3 → 0.78.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/fesm2022/plait-core.mjs.map +1 -1
- package/package.json +1 -3
- package/esm2022/constants/cursor.mjs +0 -13
- package/esm2022/constants/index.mjs +0 -20
- package/esm2022/constants/keycodes.mjs +0 -127
- package/esm2022/constants/selection.mjs +0 -6
- package/esm2022/constants/zoom.mjs +0 -4
- package/esm2022/context.mjs +0 -23
- package/esm2022/core/element/context-change.mjs +0 -13
- package/esm2022/core/element/context.mjs +0 -2
- package/esm2022/core/element/element-flavour.mjs +0 -140
- package/esm2022/core/element/element-ref.mjs +0 -2
- package/esm2022/core/list-render.mjs +0 -217
- package/esm2022/differs/default_iterable_differ.mjs +0 -614
- package/esm2022/differs/iterable_differs.mjs +0 -9
- package/esm2022/interfaces/board.mjs +0 -108
- package/esm2022/interfaces/custom-types.mjs +0 -5
- package/esm2022/interfaces/direction.mjs +0 -8
- package/esm2022/interfaces/element.mjs +0 -43
- package/esm2022/interfaces/group.mjs +0 -6
- package/esm2022/interfaces/history.mjs +0 -5
- package/esm2022/interfaces/index.mjs +0 -19
- package/esm2022/interfaces/node.mjs +0 -57
- package/esm2022/interfaces/operation.mjs +0 -96
- package/esm2022/interfaces/path-ref.mjs +0 -15
- package/esm2022/interfaces/path.mjs +0 -183
- package/esm2022/interfaces/plugin.mjs +0 -6
- package/esm2022/interfaces/point.mjs +0 -27
- package/esm2022/interfaces/pointer.mjs +0 -6
- package/esm2022/interfaces/rectangle-client.mjs +0 -171
- package/esm2022/interfaces/selection.mjs +0 -13
- package/esm2022/interfaces/svg-arc-command.mjs +0 -2
- package/esm2022/interfaces/theme.mjs +0 -49
- package/esm2022/interfaces/viewport.mjs +0 -7
- package/esm2022/plait-core.mjs +0 -5
- package/esm2022/plugins/create-board.mjs +0 -122
- package/esm2022/plugins/index.mjs +0 -11
- package/esm2022/plugins/with-board.mjs +0 -20
- package/esm2022/plugins/with-hand.mjs +0 -113
- package/esm2022/plugins/with-history.mjs +0 -91
- package/esm2022/plugins/with-hotkey.mjs +0 -96
- package/esm2022/plugins/with-i18n.mjs +0 -13
- package/esm2022/plugins/with-moving.mjs +0 -282
- package/esm2022/plugins/with-options.mjs +0 -13
- package/esm2022/plugins/with-related-fragment.mjs +0 -23
- package/esm2022/plugins/with-selection.mjs +0 -230
- package/esm2022/public-api.mjs +0 -16
- package/esm2022/testing/core/create-board.mjs +0 -15
- package/esm2022/testing/core/fake-weak-map.mjs +0 -18
- package/esm2022/testing/core/index.mjs +0 -3
- package/esm2022/testing/fake-events/event-objects.mjs +0 -131
- package/esm2022/testing/fake-events/index.mjs +0 -2
- package/esm2022/testing/index.mjs +0 -3
- package/esm2022/testing/test-element.mjs +0 -9
- package/esm2022/transforms/board.mjs +0 -137
- package/esm2022/transforms/element.mjs +0 -22
- package/esm2022/transforms/general.mjs +0 -146
- package/esm2022/transforms/group.mjs +0 -64
- package/esm2022/transforms/index.mjs +0 -17
- package/esm2022/transforms/node.mjs +0 -37
- package/esm2022/transforms/selection.mjs +0 -26
- package/esm2022/transforms/theme.mjs +0 -8
- package/esm2022/transforms/viewport.mjs +0 -8
- package/esm2022/transforms/z-index.mjs +0 -20
- package/esm2022/utils/angle.mjs +0 -164
- package/esm2022/utils/board.mjs +0 -18
- package/esm2022/utils/clipboard/clipboard.mjs +0 -40
- package/esm2022/utils/clipboard/common.mjs +0 -82
- package/esm2022/utils/clipboard/data-transfer.mjs +0 -33
- package/esm2022/utils/clipboard/index.mjs +0 -3
- package/esm2022/utils/clipboard/navigator-clipboard.mjs +0 -71
- package/esm2022/utils/clipboard/types.mjs +0 -13
- package/esm2022/utils/common.mjs +0 -79
- package/esm2022/utils/debug.mjs +0 -91
- package/esm2022/utils/dnd.mjs +0 -8
- package/esm2022/utils/dom/common.mjs +0 -75
- package/esm2022/utils/dom/environment.mjs +0 -2
- package/esm2022/utils/dom/foreign.mjs +0 -26
- package/esm2022/utils/dom/index.mjs +0 -4
- package/esm2022/utils/drawing/arrow.mjs +0 -23
- package/esm2022/utils/drawing/circle.mjs +0 -4
- package/esm2022/utils/drawing/line.mjs +0 -47
- package/esm2022/utils/drawing/rectangle.mjs +0 -36
- package/esm2022/utils/element.mjs +0 -90
- package/esm2022/utils/environment.mjs +0 -14
- package/esm2022/utils/fragment.mjs +0 -27
- package/esm2022/utils/group.mjs +0 -239
- package/esm2022/utils/helper.mjs +0 -68
- package/esm2022/utils/history.mjs +0 -96
- package/esm2022/utils/hotkeys.mjs +0 -109
- package/esm2022/utils/id-creator.mjs +0 -11
- package/esm2022/utils/index.mjs +0 -35
- package/esm2022/utils/iterable.mjs +0 -32
- package/esm2022/utils/math.mjs +0 -480
- package/esm2022/utils/mobile.mjs +0 -6
- package/esm2022/utils/moving-element.mjs +0 -17
- package/esm2022/utils/pointer.mjs +0 -13
- package/esm2022/utils/position.mjs +0 -9
- package/esm2022/utils/selected-element.mjs +0 -145
- package/esm2022/utils/selection.mjs +0 -151
- package/esm2022/utils/snap/snap-moving.mjs +0 -199
- package/esm2022/utils/snap/snap.mjs +0 -211
- package/esm2022/utils/to-image.mjs +0 -204
- package/esm2022/utils/to-point.mjs +0 -74
- package/esm2022/utils/tree.mjs +0 -22
- package/esm2022/utils/viewport.mjs +0 -227
- package/esm2022/utils/weak-maps.mjs +0 -27
- package/esm2022/utils/z-index.mjs +0 -166
|
@@ -1,227 +0,0 @@
|
|
|
1
|
-
import { SCROLL_BAR_WIDTH } from '../constants';
|
|
2
|
-
import { MAX_ZOOM, MIN_ZOOM } from '../constants/zoom';
|
|
3
|
-
import { PlaitBoard, Point } from '../interfaces';
|
|
4
|
-
import { BoardTransforms } from '../transforms/board';
|
|
5
|
-
import { getRectangleByElements } from './element';
|
|
6
|
-
import { approximately } from './math';
|
|
7
|
-
import { getViewBox, toHostPointFromViewBoxPoint, toViewBoxPoint } from './to-point';
|
|
8
|
-
import { BOARD_TO_VIEWPORT_ORIGINATION } from './weak-maps';
|
|
9
|
-
export const VIEWPORT_PADDING_RATIO = 0.75;
|
|
10
|
-
const IS_FROM_SCROLLING = new WeakMap();
|
|
11
|
-
const IS_FROM_VIEWPORT_CHANGE = new WeakMap();
|
|
12
|
-
export function getViewportContainerRect(board) {
|
|
13
|
-
const { hideScrollbar } = board.options;
|
|
14
|
-
const scrollBarWidth = hideScrollbar ? SCROLL_BAR_WIDTH : 0;
|
|
15
|
-
const viewportRect = PlaitBoard.getBoardContainer(board).getBoundingClientRect();
|
|
16
|
-
return {
|
|
17
|
-
width: viewportRect.width + scrollBarWidth,
|
|
18
|
-
height: viewportRect.height + scrollBarWidth
|
|
19
|
-
};
|
|
20
|
-
}
|
|
21
|
-
export function getElementHostBBox(board, zoom) {
|
|
22
|
-
const childrenRect = getRectangleByElements(board, board.children, true);
|
|
23
|
-
let left;
|
|
24
|
-
let right;
|
|
25
|
-
let top;
|
|
26
|
-
let bottom;
|
|
27
|
-
left = childrenRect.x;
|
|
28
|
-
right = childrenRect.x + childrenRect.width;
|
|
29
|
-
top = childrenRect.y;
|
|
30
|
-
bottom = childrenRect.y + childrenRect.height;
|
|
31
|
-
return {
|
|
32
|
-
left,
|
|
33
|
-
right,
|
|
34
|
-
top,
|
|
35
|
-
bottom
|
|
36
|
-
};
|
|
37
|
-
}
|
|
38
|
-
/**
|
|
39
|
-
* Normalize the scaling ratio, or return the corrected scaling ratio if the limit is exceeded
|
|
40
|
-
*/
|
|
41
|
-
export function clampZoomLevel(zoom, minZoom = MIN_ZOOM, maxZoom = MAX_ZOOM) {
|
|
42
|
-
return zoom < minZoom ? minZoom : zoom > maxZoom ? maxZoom : zoom;
|
|
43
|
-
}
|
|
44
|
-
/**
|
|
45
|
-
* Prepares element bounding box with minimum size constraints
|
|
46
|
-
*/
|
|
47
|
-
export function prepareElementBBox(board, zoom) {
|
|
48
|
-
const boardContainerRectangle = PlaitBoard.getBoardContainer(board).getBoundingClientRect();
|
|
49
|
-
const elementHostBBox = getElementHostBBox(board, zoom);
|
|
50
|
-
const containerWidth = boardContainerRectangle.width;
|
|
51
|
-
const containerHeight = boardContainerRectangle.height;
|
|
52
|
-
// Calculate bounding box dimensions
|
|
53
|
-
let width = elementHostBBox.right - elementHostBBox.left;
|
|
54
|
-
let height = elementHostBBox.bottom - elementHostBBox.top;
|
|
55
|
-
// If elementHostBBox dimensions are smaller than container dimensions,
|
|
56
|
-
// use half of container dimensions as minimum size
|
|
57
|
-
const minWidth = containerWidth / 2;
|
|
58
|
-
const minHeight = containerHeight / 2;
|
|
59
|
-
if (width < minWidth / zoom) {
|
|
60
|
-
// Center the content horizontally if applying minimum width
|
|
61
|
-
const center = elementHostBBox.left + width / 2;
|
|
62
|
-
elementHostBBox.left = center - minWidth / 2 / zoom;
|
|
63
|
-
elementHostBBox.right = center + minWidth / 2 / zoom;
|
|
64
|
-
width = minWidth / zoom;
|
|
65
|
-
}
|
|
66
|
-
if (height < minHeight / zoom) {
|
|
67
|
-
// Center the content vertically if applying minimum height
|
|
68
|
-
const center = elementHostBBox.top + height / 2;
|
|
69
|
-
elementHostBBox.top = center - minHeight / 2 / zoom;
|
|
70
|
-
elementHostBBox.bottom = center + minHeight / 2 / zoom;
|
|
71
|
-
height = minHeight / zoom;
|
|
72
|
-
}
|
|
73
|
-
return {
|
|
74
|
-
elementHostBBox,
|
|
75
|
-
containerWidth,
|
|
76
|
-
containerHeight,
|
|
77
|
-
width,
|
|
78
|
-
height
|
|
79
|
-
};
|
|
80
|
-
}
|
|
81
|
-
/**
|
|
82
|
-
* Calculates viewBox based on element bounding box with padding
|
|
83
|
-
*/
|
|
84
|
-
export function calculateViewBox(elementHostBBox, containerWidth, containerHeight, width, height, zoom, paddingRatio = VIEWPORT_PADDING_RATIO) {
|
|
85
|
-
const horizontalPaddingInViewBox = (containerWidth * paddingRatio) / zoom;
|
|
86
|
-
const verticalPaddingInViewBox = (containerHeight * paddingRatio) / zoom;
|
|
87
|
-
return [
|
|
88
|
-
elementHostBBox.left - horizontalPaddingInViewBox,
|
|
89
|
-
elementHostBBox.top - verticalPaddingInViewBox,
|
|
90
|
-
width + horizontalPaddingInViewBox * 2,
|
|
91
|
-
height + verticalPaddingInViewBox * 2
|
|
92
|
-
];
|
|
93
|
-
}
|
|
94
|
-
export function calcNewViewBox(board, zoom) {
|
|
95
|
-
const { elementHostBBox, containerWidth, containerHeight, width, height } = prepareElementBBox(board, zoom);
|
|
96
|
-
return calculateViewBox(elementHostBBox, containerWidth, containerHeight, width, height, zoom);
|
|
97
|
-
}
|
|
98
|
-
export function getViewBoxCenterPoint(board) {
|
|
99
|
-
const childrenRectangle = getRectangleByElements(board, board.children, true);
|
|
100
|
-
return [childrenRectangle.x + childrenRectangle.width / 2, childrenRectangle.y + childrenRectangle.height / 2];
|
|
101
|
-
}
|
|
102
|
-
export function setSVGViewBox(board, viewBox) {
|
|
103
|
-
const zoom = board.viewport.zoom;
|
|
104
|
-
const hostElement = PlaitBoard.getHost(board);
|
|
105
|
-
hostElement.style.display = 'block';
|
|
106
|
-
hostElement.style.width = `${viewBox[2] * zoom}px`;
|
|
107
|
-
hostElement.style.height = `${viewBox[3] * zoom}px`;
|
|
108
|
-
if (viewBox && viewBox[2] > 0 && viewBox[3] > 0) {
|
|
109
|
-
hostElement.setAttribute('viewBox', viewBox.join(' '));
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
export function updateViewportOffset(board) {
|
|
113
|
-
const origination = getViewportOrigination(board);
|
|
114
|
-
if (!origination) {
|
|
115
|
-
return;
|
|
116
|
-
}
|
|
117
|
-
const [scrollLeft, scrollTop] = toHostPointFromViewBoxPoint(board, origination);
|
|
118
|
-
updateViewportContainerScroll(board, scrollLeft, scrollTop);
|
|
119
|
-
}
|
|
120
|
-
export function updateViewportContainerScroll(board, left, top, isFromViewportChange = true) {
|
|
121
|
-
const viewportContainer = PlaitBoard.getViewportContainer(board);
|
|
122
|
-
const previousScrollLeft = viewportContainer.scrollLeft;
|
|
123
|
-
const previousScrollTop = viewportContainer.scrollTop;
|
|
124
|
-
// scrollTop assign 11.8 will get 11.5 in chrome
|
|
125
|
-
// scrollTop assign 11.8 will get 11 in firefox, safari
|
|
126
|
-
// scrollTop assign 11.4 will get 11 in chrome, firefox, safari
|
|
127
|
-
// use approximately method to determine the new value is valid updating to avoid debouncing
|
|
128
|
-
if (!approximately(viewportContainer.scrollLeft, left, 1) || !approximately(viewportContainer.scrollTop, top, 1)) {
|
|
129
|
-
viewportContainer.scrollLeft = left;
|
|
130
|
-
viewportContainer.scrollTop = top;
|
|
131
|
-
const offsetWidth = viewportContainer.offsetWidth;
|
|
132
|
-
const offsetHeight = viewportContainer.offsetHeight;
|
|
133
|
-
if (previousScrollLeft === viewportContainer.scrollLeft && previousScrollTop === viewportContainer.scrollTop) {
|
|
134
|
-
// The scroll event cannot be triggered, so the origination is modified directly based on the scroll distance.
|
|
135
|
-
updateViewportByScrolling(board, previousScrollLeft, previousScrollTop);
|
|
136
|
-
}
|
|
137
|
-
else {
|
|
138
|
-
const isValidLeftOrTop = left > 0 &&
|
|
139
|
-
top > 0 &&
|
|
140
|
-
left < viewportContainer.scrollWidth - offsetWidth &&
|
|
141
|
-
top < viewportContainer.scrollHeight - offsetHeight;
|
|
142
|
-
if (isFromViewportChange && isValidLeftOrTop) {
|
|
143
|
-
setIsFromViewportChange(board, true);
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
}
|
|
148
|
-
export function updateViewportByScrolling(board, scrollLeft, scrollTop) {
|
|
149
|
-
const origination = toViewBoxPoint(board, [scrollLeft, scrollTop]);
|
|
150
|
-
if (Point.isEquals(origination, getViewportOrigination(board))) {
|
|
151
|
-
return;
|
|
152
|
-
}
|
|
153
|
-
BoardTransforms.updateViewport(board, origination);
|
|
154
|
-
setIsFromScrolling(board, true);
|
|
155
|
-
}
|
|
156
|
-
export function initializeViewportContainer(board) {
|
|
157
|
-
const { width, height } = getViewportContainerRect(board);
|
|
158
|
-
const viewportContainer = PlaitBoard.getViewportContainer(board);
|
|
159
|
-
viewportContainer.style.width = `${width}px`;
|
|
160
|
-
viewportContainer.style.height = `${height}px`;
|
|
161
|
-
}
|
|
162
|
-
export function initializeViewBox(board) {
|
|
163
|
-
const zoom = board.viewport.zoom;
|
|
164
|
-
const viewBox = calcNewViewBox(board, zoom);
|
|
165
|
-
setSVGViewBox(board, viewBox);
|
|
166
|
-
}
|
|
167
|
-
export function updateViewBox(board) {
|
|
168
|
-
const zoom = board.viewport.zoom;
|
|
169
|
-
const { elementHostBBox, containerWidth, containerHeight, width, height } = prepareElementBBox(board, zoom);
|
|
170
|
-
// Use 0.5 ratio to check if contents are within current viewBox
|
|
171
|
-
const checkViewBox = calculateViewBox(elementHostBBox, containerWidth, containerHeight, width, height, zoom, 0.5 // Use smaller padding ratio for checking
|
|
172
|
-
);
|
|
173
|
-
// Get current viewBox
|
|
174
|
-
const currentViewBox = getViewBox(board);
|
|
175
|
-
// Only update if new viewBox is NOT contained within current viewBox
|
|
176
|
-
if (checkViewBox[0] < currentViewBox.x ||
|
|
177
|
-
checkViewBox[1] < currentViewBox.y ||
|
|
178
|
-
checkViewBox[0] + checkViewBox[2] > currentViewBox.x + currentViewBox.width ||
|
|
179
|
-
checkViewBox[1] + checkViewBox[3] > currentViewBox.y + currentViewBox.height) {
|
|
180
|
-
// Update with larger padding ratio
|
|
181
|
-
const newViewBox = calculateViewBox(elementHostBBox, containerWidth, containerHeight, width, height, zoom, VIEWPORT_PADDING_RATIO);
|
|
182
|
-
setSVGViewBox(board, newViewBox);
|
|
183
|
-
}
|
|
184
|
-
}
|
|
185
|
-
export function initializeViewportOffset(board) {
|
|
186
|
-
if (!board.viewport?.origination) {
|
|
187
|
-
const zoom = board.viewport.zoom;
|
|
188
|
-
const viewportContainerRect = PlaitBoard.getBoardContainer(board).getBoundingClientRect();
|
|
189
|
-
const viewBox = calcNewViewBox(board, zoom);
|
|
190
|
-
const centerX = viewBox[0] + viewBox[2] / 2;
|
|
191
|
-
const centerY = viewBox[1] + viewBox[3] / 2;
|
|
192
|
-
const origination = [centerX - viewportContainerRect.width / 2 / zoom, centerY - viewportContainerRect.height / 2 / zoom];
|
|
193
|
-
updateViewportOrigination(board, origination);
|
|
194
|
-
updateViewportOffset(board);
|
|
195
|
-
return;
|
|
196
|
-
}
|
|
197
|
-
updateViewportOffset(board);
|
|
198
|
-
}
|
|
199
|
-
export const updateViewportOrigination = (board, origination) => {
|
|
200
|
-
BOARD_TO_VIEWPORT_ORIGINATION.set(board, origination);
|
|
201
|
-
};
|
|
202
|
-
export const clearViewportOrigination = (board) => {
|
|
203
|
-
BOARD_TO_VIEWPORT_ORIGINATION.delete(board);
|
|
204
|
-
};
|
|
205
|
-
export const getViewportOrigination = (board) => {
|
|
206
|
-
const origination = BOARD_TO_VIEWPORT_ORIGINATION.get(board);
|
|
207
|
-
if (origination) {
|
|
208
|
-
return origination;
|
|
209
|
-
}
|
|
210
|
-
else {
|
|
211
|
-
return board.viewport.origination;
|
|
212
|
-
}
|
|
213
|
-
};
|
|
214
|
-
export const isFromScrolling = (board) => {
|
|
215
|
-
return !!IS_FROM_SCROLLING.get(board);
|
|
216
|
-
};
|
|
217
|
-
export const setIsFromScrolling = (board, state) => {
|
|
218
|
-
IS_FROM_SCROLLING.set(board, state);
|
|
219
|
-
};
|
|
220
|
-
export const isFromViewportChange = (board) => {
|
|
221
|
-
return !!IS_FROM_VIEWPORT_CHANGE.get(board);
|
|
222
|
-
};
|
|
223
|
-
export const setIsFromViewportChange = (board, state) => {
|
|
224
|
-
IS_FROM_VIEWPORT_CHANGE.set(board, state);
|
|
225
|
-
};
|
|
226
|
-
export function scrollToRectangle(board, client) { }
|
|
227
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewport.js","sourceRoot":"","sources":["../../../../packages/core/src/utils/viewport.ts"],"names":[],"mappings":"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","sourcesContent":["import { SCROLL_BAR_WIDTH } from '../constants';\nimport { MAX_ZOOM, MIN_ZOOM } from '../constants/zoom';\nimport { PlaitBoard, Point, RectangleClient } from '../interfaces';\nimport { BoardTransforms } from '../transforms/board';\nimport { getRectangleByElements } from './element';\nimport { approximately } from './math';\nimport { getViewBox, toHostPointFromViewBoxPoint, toViewBoxPoint } from './to-point';\nimport { BOARD_TO_VIEWPORT_ORIGINATION } from './weak-maps';\n\nexport const VIEWPORT_PADDING_RATIO = 0.75;\n\nexport interface ElementHostBBox {\n    left: number;\n    right: number;\n    top: number;\n    bottom: number;\n}\n\nconst IS_FROM_SCROLLING = new WeakMap<PlaitBoard, boolean>();\n\nconst IS_FROM_VIEWPORT_CHANGE = new WeakMap<PlaitBoard, boolean>();\n\nexport function getViewportContainerRect(board: PlaitBoard) {\n    const { hideScrollbar } = board.options;\n    const scrollBarWidth = hideScrollbar ? SCROLL_BAR_WIDTH : 0;\n    const viewportRect = PlaitBoard.getBoardContainer(board).getBoundingClientRect();\n\n    return {\n        width: viewportRect.width + scrollBarWidth,\n        height: viewportRect.height + scrollBarWidth\n    };\n}\n\nexport function getElementHostBBox(board: PlaitBoard, zoom: number): ElementHostBBox {\n    const childrenRect = getRectangleByElements(board, board.children, true);\n    let left: number;\n    let right: number;\n    let top: number;\n    let bottom: number;\n    left = childrenRect.x;\n    right = childrenRect.x + childrenRect.width;\n    top = childrenRect.y;\n    bottom = childrenRect.y + childrenRect.height;\n    return {\n        left,\n        right,\n        top,\n        bottom\n    };\n}\n\n/**\n * Normalize the scaling ratio, or return the corrected scaling ratio if the limit is exceeded\n */\nexport function clampZoomLevel(zoom: number, minZoom = MIN_ZOOM, maxZoom = MAX_ZOOM) {\n    return zoom < minZoom ? minZoom : zoom > maxZoom ? maxZoom : zoom;\n}\n\n/**\n * Prepares element bounding box with minimum size constraints\n */\nexport function prepareElementBBox(board: PlaitBoard, zoom: number): {\n    elementHostBBox: ElementHostBBox;\n    containerWidth: number;\n    containerHeight: number;\n    width: number;\n    height: number;\n} {\n    const boardContainerRectangle = PlaitBoard.getBoardContainer(board).getBoundingClientRect();\n    const elementHostBBox: ElementHostBBox = getElementHostBBox(board, zoom);\n    \n    const containerWidth = boardContainerRectangle.width;\n    const containerHeight = boardContainerRectangle.height;\n    \n    // Calculate bounding box dimensions\n    let width = elementHostBBox.right - elementHostBBox.left;\n    let height = elementHostBBox.bottom - elementHostBBox.top;\n    \n    // If elementHostBBox dimensions are smaller than container dimensions,\n    // use half of container dimensions as minimum size\n    const minWidth = containerWidth / 2;\n    const minHeight = containerHeight / 2;\n    \n    if (width < minWidth / zoom) {\n        // Center the content horizontally if applying minimum width\n        const center = elementHostBBox.left + width / 2;\n        elementHostBBox.left = center - minWidth / 2 / zoom;\n        elementHostBBox.right = center + minWidth / 2 / zoom;\n        width = minWidth / zoom;\n    }\n    \n    if (height < minHeight / zoom) {\n        // Center the content vertically if applying minimum height\n        const center = elementHostBBox.top + height / 2;\n        elementHostBBox.top = center - minHeight / 2 / zoom;\n        elementHostBBox.bottom = center + minHeight / 2 / zoom;\n        height = minHeight / zoom;\n    }\n    \n    return {\n        elementHostBBox,\n        containerWidth,\n        containerHeight,\n        width,\n        height\n    };\n}\n\n/**\n * Calculates viewBox based on element bounding box with padding\n */\nexport function calculateViewBox(\n    elementHostBBox: ElementHostBBox,\n    containerWidth: number,\n    containerHeight: number,\n    width: number,\n    height: number,\n    zoom: number,\n    paddingRatio: number = VIEWPORT_PADDING_RATIO\n): number[] {\n    const horizontalPaddingInViewBox = (containerWidth * paddingRatio) / zoom;\n    const verticalPaddingInViewBox = (containerHeight * paddingRatio) / zoom;\n    \n    return [\n        elementHostBBox.left - horizontalPaddingInViewBox,\n        elementHostBBox.top - verticalPaddingInViewBox,\n        width + horizontalPaddingInViewBox * 2,\n        height + verticalPaddingInViewBox * 2\n    ];\n}\n\nexport function calcNewViewBox(board: PlaitBoard, zoom: number) {\n    const { elementHostBBox, containerWidth, containerHeight, width, height } = prepareElementBBox(board, zoom);\n    \n    return calculateViewBox(\n        elementHostBBox,\n        containerWidth,\n        containerHeight,\n        width,\n        height,\n        zoom\n    );\n}\n\nexport function getViewBoxCenterPoint(board: PlaitBoard) {\n    const childrenRectangle = getRectangleByElements(board, board.children, true);\n    return [childrenRectangle.x + childrenRectangle.width / 2, childrenRectangle.y + childrenRectangle.height / 2] as Point;\n}\n\nexport function setSVGViewBox(board: PlaitBoard, viewBox: number[]) {\n    const zoom = board.viewport.zoom;\n    const hostElement = PlaitBoard.getHost(board);\n    hostElement.style.display = 'block';\n    hostElement.style.width = `${viewBox[2] * zoom}px`;\n    hostElement.style.height = `${viewBox[3] * zoom}px`;\n\n    if (viewBox && viewBox[2] > 0 && viewBox[3] > 0) {\n        hostElement.setAttribute('viewBox', viewBox.join(' '));\n    }\n}\n\nexport function updateViewportOffset(board: PlaitBoard) {\n    const origination = getViewportOrigination(board);\n    if (!origination) {\n        return;\n    }\n    const [scrollLeft, scrollTop] = toHostPointFromViewBoxPoint(board, origination);\n    updateViewportContainerScroll(board, scrollLeft, scrollTop);\n}\n\nexport function updateViewportContainerScroll(board: PlaitBoard, left: number, top: number, isFromViewportChange: boolean = true) {\n    const viewportContainer = PlaitBoard.getViewportContainer(board);\n    const previousScrollLeft = viewportContainer.scrollLeft;\n    const previousScrollTop = viewportContainer.scrollTop;\n    // scrollTop assign 11.8 will get 11.5 in chrome\n    // scrollTop assign 11.8 will get 11 in firefox, safari\n    // scrollTop assign 11.4 will get 11 in chrome, firefox, safari\n    // use approximately method to determine the new value is valid updating to avoid debouncing\n    if (!approximately(viewportContainer.scrollLeft, left, 1) || !approximately(viewportContainer.scrollTop, top, 1)) {\n        viewportContainer.scrollLeft = left;\n        viewportContainer.scrollTop = top;\n        const offsetWidth = viewportContainer.offsetWidth;\n        const offsetHeight = viewportContainer.offsetHeight;\n        if (previousScrollLeft === viewportContainer.scrollLeft && previousScrollTop === viewportContainer.scrollTop) {\n            // The scroll event cannot be triggered, so the origination is modified directly based on the scroll distance.\n            updateViewportByScrolling(board, previousScrollLeft, previousScrollTop);\n        } else {\n            const isValidLeftOrTop =\n                left > 0 &&\n                top > 0 &&\n                left < viewportContainer.scrollWidth - offsetWidth &&\n                top < viewportContainer.scrollHeight - offsetHeight;\n            if (isFromViewportChange && isValidLeftOrTop) {\n                setIsFromViewportChange(board, true);\n            }\n        }\n    }\n}\n\nexport function updateViewportByScrolling(board: PlaitBoard, scrollLeft: number, scrollTop: number) {\n    const origination = toViewBoxPoint(board, [scrollLeft, scrollTop]);\n    if (Point.isEquals(origination, getViewportOrigination(board))) {\n        return;\n    }\n    BoardTransforms.updateViewport(board, origination);\n    setIsFromScrolling(board, true);\n}\n\nexport function initializeViewportContainer(board: PlaitBoard) {\n    const { width, height } = getViewportContainerRect(board);\n    const viewportContainer = PlaitBoard.getViewportContainer(board);\n    viewportContainer.style.width = `${width}px`;\n    viewportContainer.style.height = `${height}px`;\n}\n\nexport function initializeViewBox(board: PlaitBoard) {\n    const zoom = board.viewport.zoom;\n    const viewBox = calcNewViewBox(board, zoom);\n    setSVGViewBox(board, viewBox);\n}\n\nexport function updateViewBox(board: PlaitBoard) {\n    const zoom = board.viewport.zoom;\n    const { elementHostBBox, containerWidth, containerHeight, width, height } = prepareElementBBox(board, zoom);\n    \n    // Use 0.5 ratio to check if contents are within current viewBox\n    const checkViewBox = calculateViewBox(\n        elementHostBBox,\n        containerWidth,\n        containerHeight,\n        width,\n        height,\n        zoom,\n        0.5 // Use smaller padding ratio for checking\n    );\n    \n    // Get current viewBox\n    const currentViewBox = getViewBox(board);\n    \n    // Only update if new viewBox is NOT contained within current viewBox\n    if (\n        checkViewBox[0] < currentViewBox.x ||\n        checkViewBox[1] < currentViewBox.y ||\n        checkViewBox[0] + checkViewBox[2] > currentViewBox.x + currentViewBox.width ||\n        checkViewBox[1] + checkViewBox[3] > currentViewBox.y + currentViewBox.height\n    ) {\n        // Update with larger padding ratio\n        const newViewBox = calculateViewBox(\n            elementHostBBox,\n            containerWidth,\n            containerHeight,\n            width,\n            height,\n            zoom,\n            VIEWPORT_PADDING_RATIO\n        );\n        setSVGViewBox(board, newViewBox);\n    }\n}\n\nexport function initializeViewportOffset(board: PlaitBoard) {\n    if (!board.viewport?.origination) {\n        const zoom = board.viewport.zoom;\n        const viewportContainerRect = PlaitBoard.getBoardContainer(board).getBoundingClientRect();\n        const viewBox = calcNewViewBox(board, zoom);\n        const centerX = viewBox[0] + viewBox[2] / 2;\n        const centerY = viewBox[1] + viewBox[3] / 2;\n        const origination = [centerX - viewportContainerRect.width / 2 / zoom, centerY - viewportContainerRect.height / 2 / zoom] as Point;\n        updateViewportOrigination(board, origination);\n        updateViewportOffset(board);\n        return;\n    }\n    updateViewportOffset(board);\n}\n\nexport const updateViewportOrigination = (board: PlaitBoard, origination: Point) => {\n    BOARD_TO_VIEWPORT_ORIGINATION.set(board, origination);\n};\n\nexport const clearViewportOrigination = (board: PlaitBoard) => {\n    BOARD_TO_VIEWPORT_ORIGINATION.delete(board);\n};\n\nexport const getViewportOrigination = (board: PlaitBoard) => {\n    const origination = BOARD_TO_VIEWPORT_ORIGINATION.get(board);\n    if (origination) {\n        return origination;\n    } else {\n        return board.viewport.origination;\n    }\n};\n\nexport const isFromScrolling = (board: PlaitBoard) => {\n    return !!IS_FROM_SCROLLING.get(board);\n};\n\nexport const setIsFromScrolling = (board: PlaitBoard, state: boolean) => {\n    IS_FROM_SCROLLING.set(board, state);\n};\n\nexport const isFromViewportChange = (board: PlaitBoard) => {\n    return !!IS_FROM_VIEWPORT_CHANGE.get(board);\n};\n\nexport const setIsFromViewportChange = (board: PlaitBoard, state: boolean) => {\n    IS_FROM_VIEWPORT_CHANGE.set(board, state);\n};\n\nexport function scrollToRectangle(board: PlaitBoard, client: RectangleClient) {}\n"]}
|
|
@@ -1,27 +0,0 @@
|
|
|
1
|
-
// record richtext type status
|
|
2
|
-
export const IS_BOARD_CACHE = new WeakMap();
|
|
3
|
-
export const FLUSHING = new WeakMap();
|
|
4
|
-
export const NODE_TO_INDEX = new WeakMap();
|
|
5
|
-
export const NODE_TO_PARENT = new WeakMap();
|
|
6
|
-
export const KEY_TO_ELEMENT_MAP = new WeakMap();
|
|
7
|
-
export const NODE_TO_G = new WeakMap();
|
|
8
|
-
export const NODE_TO_CONTAINER_G = new WeakMap();
|
|
9
|
-
export const IS_TEXT_EDITABLE = new WeakMap();
|
|
10
|
-
export const BOARD_TO_ON_CHANGE = new WeakMap();
|
|
11
|
-
export const BOARD_TO_AFTER_CHANGE = new WeakMap();
|
|
12
|
-
export const BOARD_TO_ROUGH_SVG = new WeakMap();
|
|
13
|
-
export const BOARD_TO_HOST = new WeakMap();
|
|
14
|
-
export const BOARD_TO_CONTEXT = new WeakMap();
|
|
15
|
-
export const IS_BOARD_ALIVE = new WeakMap();
|
|
16
|
-
export const BOARD_TO_ELEMENT_HOST = new WeakMap();
|
|
17
|
-
export const BOARD_TO_SELECTED_ELEMENT = new WeakMap();
|
|
18
|
-
export const BOARD_TO_MOVING_POINT_IN_BOARD = new WeakMap();
|
|
19
|
-
export const BOARD_TO_MOVING_POINT = new WeakMap();
|
|
20
|
-
export const BOARD_TO_VIEWPORT_ORIGINATION = new WeakMap();
|
|
21
|
-
export const BOARD_TO_IS_SELECTION_MOVING = new WeakMap();
|
|
22
|
-
// save no standard selected elements
|
|
23
|
-
export const BOARD_TO_TEMPORARY_ELEMENTS = new WeakMap();
|
|
24
|
-
export const BOARD_TO_MOVING_ELEMENT = new WeakMap();
|
|
25
|
-
export const PATH_REFS = new WeakMap();
|
|
26
|
-
export const ELEMENT_TO_REF = new WeakMap();
|
|
27
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2Vhay1tYXBzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcGFja2FnZXMvY29yZS9zcmMvdXRpbHMvd2Vhay1tYXBzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVNBLDhCQUE4QjtBQUM5QixNQUFNLENBQUMsTUFBTSxjQUFjLEdBQUcsSUFBSSxPQUFPLEVBQW1CLENBQUM7QUFFN0QsTUFBTSxDQUFDLE1BQU0sUUFBUSxHQUFHLElBQUksT0FBTyxFQUF1QixDQUFDO0FBRTNELE1BQU0sQ0FBQyxNQUFNLGFBQWEsR0FBRyxJQUFJLE9BQU8sRUFBcUIsQ0FBQztBQUU5RCxNQUFNLENBQUMsTUFBTSxjQUFjLEdBQUcsSUFBSSxPQUFPLEVBQXVCLENBQUM7QUFFakUsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsSUFBSSxPQUFPLEVBQXlDLENBQUM7QUFFdkYsTUFBTSxDQUFDLE1BQU0sU0FBUyxHQUFHLElBQUksT0FBTyxFQUEwQixDQUFDO0FBRS9ELE1BQU0sQ0FBQyxNQUFNLG1CQUFtQixHQUFHLElBQUksT0FBTyxFQUEwQixDQUFDO0FBRXpFLE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHLElBQUksT0FBTyxFQUF1QixDQUFDO0FBRW5FLE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksT0FBTyxFQUEwQixDQUFDO0FBRXhFLE1BQU0sQ0FBQyxNQUFNLHFCQUFxQixHQUFHLElBQUksT0FBTyxFQUEwQixDQUFDO0FBRTNFLE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksT0FBTyxFQUF3QixDQUFDO0FBRXRFLE1BQU0sQ0FBQyxNQUFNLGFBQWEsR0FBRyxJQUFJLE9BQU8sRUFBNkIsQ0FBQztBQUV0RSxNQUFNLENBQUMsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLE9BQU8sRUFBaUMsQ0FBQztBQUU3RSxNQUFNLENBQUMsTUFBTSxjQUFjLEdBQUcsSUFBSSxPQUFPLEVBQXVCLENBQUM7QUFFakUsTUFBTSxDQUFDLE1BQU0scUJBQXFCLEdBQUcsSUFBSSxPQUFPLEVBVzdDLENBQUM7QUFFSixNQUFNLENBQUMsTUFBTSx5QkFBeUIsR0FBRyxJQUFJLE9BQU8sRUFBOEIsQ0FBQztBQUVuRixNQUFNLENBQUMsTUFBTSw4QkFBOEIsR0FBRyxJQUFJLE9BQU8sRUFBcUIsQ0FBQztBQUUvRSxNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBRyxJQUFJLE9BQU8sRUFBcUIsQ0FBQztBQUV0RSxNQUFNLENBQUMsTUFBTSw2QkFBNkIsR0FBRyxJQUFJLE9BQU8sRUFBcUIsQ0FBQztBQUU5RSxNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyxJQUFJLE9BQU8sRUFBdUIsQ0FBQztBQUUvRSxxQ0FBcUM7QUFDckMsTUFBTSxDQUFDLE1BQU0sMkJBQTJCLEdBQUcsSUFBSSxPQUFPLEVBQTRELENBQUM7QUFFbkgsTUFBTSxDQUFDLE1BQU0sdUJBQXVCLEdBQUcsSUFBSSxPQUFPLEVBQThCLENBQUM7QUFFakYsTUFBTSxDQUFDLE1BQU0sU0FBUyxHQUFzQyxJQUFJLE9BQU8sRUFBRSxDQUFDO0FBRTFFLE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBMkMsSUFBSSxPQUFPLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJvdWdoU1ZHIH0gZnJvbSAncm91Z2hqcy9iaW4vc3ZnJztcbmltcG9ydCB7IFBsYWl0RWxlbWVudCB9IGZyb20gJy4uL2ludGVyZmFjZXMvZWxlbWVudCc7XG5pbXBvcnQgeyBQbGFpdEJvYXJkIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9ib2FyZCc7XG5pbXBvcnQgeyBQb2ludCB9IGZyb20gJy4uL2ludGVyZmFjZXMvcG9pbnQnO1xuaW1wb3J0IHsgQW5jZXN0b3IsIFBsYWl0Tm9kZSB9IGZyb20gJy4uL2ludGVyZmFjZXMvbm9kZSc7XG5pbXBvcnQgeyBQYXRoUmVmIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9wYXRoLXJlZic7XG5pbXBvcnQgeyBQbGFpdEVsZW1lbnRSZWYgfSBmcm9tICcuLi9jb3JlL2VsZW1lbnQvZWxlbWVudC1yZWYnO1xuaW1wb3J0IHsgUGxhaXRCb2FyZENvbnRleHQgfSBmcm9tICcuLi9jb250ZXh0JztcblxuLy8gcmVjb3JkIHJpY2h0ZXh0IHR5cGUgc3RhdHVzXG5leHBvcnQgY29uc3QgSVNfQk9BUkRfQ0FDSEUgPSBuZXcgV2Vha01hcDxPYmplY3QsIGJvb2xlYW4+KCk7XG5cbmV4cG9ydCBjb25zdCBGTFVTSElORyA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIGJvb2xlYW4+KCk7XG5cbmV4cG9ydCBjb25zdCBOT0RFX1RPX0lOREVYID0gbmV3IFdlYWtNYXA8UGxhaXROb2RlLCBudW1iZXI+KCk7XG5cbmV4cG9ydCBjb25zdCBOT0RFX1RPX1BBUkVOVCA9IG5ldyBXZWFrTWFwPFBsYWl0Tm9kZSwgQW5jZXN0b3I+KCk7XG5cbmV4cG9ydCBjb25zdCBLRVlfVE9fRUxFTUVOVF9NQVAgPSBuZXcgV2Vha01hcDxQbGFpdEJvYXJkLCBNYXA8U3RyaW5nLCBQbGFpdEVsZW1lbnQ+PigpO1xuXG5leHBvcnQgY29uc3QgTk9ERV9UT19HID0gbmV3IFdlYWtNYXA8UGxhaXROb2RlLCBTVkdHRWxlbWVudD4oKTtcblxuZXhwb3J0IGNvbnN0IE5PREVfVE9fQ09OVEFJTkVSX0cgPSBuZXcgV2Vha01hcDxQbGFpdE5vZGUsIFNWR0dFbGVtZW50PigpO1xuXG5leHBvcnQgY29uc3QgSVNfVEVYVF9FRElUQUJMRSA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIGJvb2xlYW4+KCk7XG5cbmV4cG9ydCBjb25zdCBCT0FSRF9UT19PTl9DSEFOR0UgPSBuZXcgV2Vha01hcDxQbGFpdEJvYXJkLCAoKSA9PiB2b2lkPigpO1xuXG5leHBvcnQgY29uc3QgQk9BUkRfVE9fQUZURVJfQ0hBTkdFID0gbmV3IFdlYWtNYXA8UGxhaXRCb2FyZCwgKCkgPT4gdm9pZD4oKTtcblxuZXhwb3J0IGNvbnN0IEJPQVJEX1RPX1JPVUdIX1NWRyA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIFJvdWdoU1ZHPigpO1xuXG5leHBvcnQgY29uc3QgQk9BUkRfVE9fSE9TVCA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIFNWR1NWR0VsZW1lbnQ+KCk7XG5cbmV4cG9ydCBjb25zdCBCT0FSRF9UT19DT05URVhUID0gbmV3IFdlYWtNYXA8UGxhaXRCb2FyZCwgUGxhaXRCb2FyZENvbnRleHQ+KCk7XG5cbmV4cG9ydCBjb25zdCBJU19CT0FSRF9BTElWRSA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIGJvb2xlYW4+KCk7XG5cbmV4cG9ydCBjb25zdCBCT0FSRF9UT19FTEVNRU5UX0hPU1QgPSBuZXcgV2Vha01hcDxcbiAgICBQbGFpdEJvYXJkLFxuICAgIHtcbiAgICAgICAgbG93ZXJIb3N0OiBTVkdHRWxlbWVudDtcbiAgICAgICAgaG9zdDogU1ZHR0VsZW1lbnQ7XG4gICAgICAgIHVwcGVySG9zdDogU1ZHR0VsZW1lbnQ7XG4gICAgICAgIHRvcEhvc3Q6IFNWR0dFbGVtZW50O1xuICAgICAgICBhY3RpdmVIb3N0OiBTVkdHRWxlbWVudDtcbiAgICAgICAgY29udGFpbmVyOiBIVE1MRWxlbWVudDtcbiAgICAgICAgdmlld3BvcnRDb250YWluZXI6IEhUTUxFbGVtZW50O1xuICAgIH1cbj4oKTtcblxuZXhwb3J0IGNvbnN0IEJPQVJEX1RPX1NFTEVDVEVEX0VMRU1FTlQgPSBuZXcgV2Vha01hcDxQbGFpdEJvYXJkLCBQbGFpdEVsZW1lbnRbXT4oKTtcblxuZXhwb3J0IGNvbnN0IEJPQVJEX1RPX01PVklOR19QT0lOVF9JTl9CT0FSRCA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIFBvaW50PigpO1xuXG5leHBvcnQgY29uc3QgQk9BUkRfVE9fTU9WSU5HX1BPSU5UID0gbmV3IFdlYWtNYXA8UGxhaXRCb2FyZCwgUG9pbnQ+KCk7XG5cbmV4cG9ydCBjb25zdCBCT0FSRF9UT19WSUVXUE9SVF9PUklHSU5BVElPTiA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIFBvaW50PigpO1xuXG5leHBvcnQgY29uc3QgQk9BUkRfVE9fSVNfU0VMRUNUSU9OX01PVklORyA9IG5ldyBXZWFrTWFwPFBsYWl0Qm9hcmQsIGJvb2xlYW4+KCk7XG5cbi8vIHNhdmUgbm8gc3RhbmRhcmQgc2VsZWN0ZWQgZWxlbWVudHNcbmV4cG9ydCBjb25zdCBCT0FSRF9UT19URU1QT1JBUllfRUxFTUVOVFMgPSBuZXcgV2Vha01hcDxQbGFpdEJvYXJkLCB7IGVsZW1lbnRzOiBQbGFpdEVsZW1lbnRbXTsgdGltZW91dElkOiBhbnkgfT4oKTtcblxuZXhwb3J0IGNvbnN0IEJPQVJEX1RPX01PVklOR19FTEVNRU5UID0gbmV3IFdlYWtNYXA8UGxhaXRCb2FyZCwgUGxhaXRFbGVtZW50W10+KCk7XG5cbmV4cG9ydCBjb25zdCBQQVRIX1JFRlM6IFdlYWtNYXA8UGxhaXRCb2FyZCwgU2V0PFBhdGhSZWY+PiA9IG5ldyBXZWFrTWFwKCk7XG5cbmV4cG9ydCBjb25zdCBFTEVNRU5UX1RPX1JFRjogV2Vha01hcDxQbGFpdEVsZW1lbnQsIFBsYWl0RWxlbWVudFJlZj4gPSBuZXcgV2Vha01hcCgpO1xuIl19
|
|
@@ -1,166 +0,0 @@
|
|
|
1
|
-
import { PlaitGroupElement } from '../interfaces';
|
|
2
|
-
import { getElementsIndices } from './common';
|
|
3
|
-
import { getEditingGroup, getElementsInGroup, getGroupByElement, getHighestGroup, isSelectedAllElementsInGroup } from './group';
|
|
4
|
-
import { findIndex, findLastIndex } from './helper';
|
|
5
|
-
import { sortElements } from './position';
|
|
6
|
-
import { getSelectedElements } from './selected-element';
|
|
7
|
-
export const getOneMoveOptions = (board, direction) => {
|
|
8
|
-
const indicesToMove = getElementsIndices(board, getSelectedElements(board));
|
|
9
|
-
let groupedIndices = toContiguousGroups(board, indicesToMove);
|
|
10
|
-
if (direction === 'up') {
|
|
11
|
-
groupedIndices = groupedIndices.reverse();
|
|
12
|
-
}
|
|
13
|
-
let moveContents = [];
|
|
14
|
-
groupedIndices.forEach((indices, i) => {
|
|
15
|
-
const leadingIndex = indices[0];
|
|
16
|
-
const trailingIndex = indices[indices.length - 1];
|
|
17
|
-
const boundaryIndex = direction === 'down' ? leadingIndex : trailingIndex;
|
|
18
|
-
const targetIndex = getTargetIndex(board, boundaryIndex, direction);
|
|
19
|
-
if (targetIndex === -1 || boundaryIndex === targetIndex) {
|
|
20
|
-
return;
|
|
21
|
-
}
|
|
22
|
-
if (direction === 'down') {
|
|
23
|
-
indices = indices.reverse();
|
|
24
|
-
}
|
|
25
|
-
moveContents.push(...indices.map(path => {
|
|
26
|
-
return {
|
|
27
|
-
element: board.children[path],
|
|
28
|
-
newPath: [targetIndex]
|
|
29
|
-
};
|
|
30
|
-
}));
|
|
31
|
-
});
|
|
32
|
-
return moveContents;
|
|
33
|
-
};
|
|
34
|
-
export const getAllMoveOptions = (board, direction) => {
|
|
35
|
-
const indicesToMove = getElementsIndices(board, getSelectedElements(board));
|
|
36
|
-
let groupedIndices = toContiguousGroups(board, indicesToMove);
|
|
37
|
-
let moveContents = [];
|
|
38
|
-
if (direction === 'down') {
|
|
39
|
-
groupedIndices = groupedIndices.reverse();
|
|
40
|
-
}
|
|
41
|
-
groupedIndices.forEach(indices => {
|
|
42
|
-
const leadingIndex = indices[0];
|
|
43
|
-
const trailingIndex = indices[indices.length - 1];
|
|
44
|
-
const boundaryIndex = direction === 'down' ? leadingIndex : trailingIndex;
|
|
45
|
-
const sourceElement = board.children[boundaryIndex];
|
|
46
|
-
const editingGroup = getEditingGroup(board, sourceElement);
|
|
47
|
-
let targetIndex = direction === 'down' ? 0 : board.children.length - 1;
|
|
48
|
-
if (editingGroup) {
|
|
49
|
-
const elementsInGroup = sortElements(board, getElementsInGroup(board, editingGroup, true, true));
|
|
50
|
-
targetIndex =
|
|
51
|
-
direction === 'down'
|
|
52
|
-
? board.children.indexOf(elementsInGroup[0])
|
|
53
|
-
: board.children.indexOf(elementsInGroup[elementsInGroup.length - 1]);
|
|
54
|
-
}
|
|
55
|
-
if (direction === 'down') {
|
|
56
|
-
indices = indices.reverse();
|
|
57
|
-
}
|
|
58
|
-
moveContents.push(...indices.map(path => {
|
|
59
|
-
return {
|
|
60
|
-
element: board.children[path],
|
|
61
|
-
newPath: [targetIndex]
|
|
62
|
-
};
|
|
63
|
-
}));
|
|
64
|
-
});
|
|
65
|
-
return moveContents;
|
|
66
|
-
};
|
|
67
|
-
export const canSetZIndex = (board) => {
|
|
68
|
-
const selectedElements = getSelectedElements(board).filter(item => board.canSetZIndex(item));
|
|
69
|
-
return selectedElements.length > 0;
|
|
70
|
-
};
|
|
71
|
-
const toContiguousGroups = (board, array) => {
|
|
72
|
-
let cursor = 0;
|
|
73
|
-
return array.reduce((acc, value, index) => {
|
|
74
|
-
if (index > 0) {
|
|
75
|
-
const currentElement = board.children[value];
|
|
76
|
-
const previousElement = board.children[array[index - 1]];
|
|
77
|
-
const isContiguous = value - 1 === array[index - 1]
|
|
78
|
-
? true
|
|
79
|
-
: board.children.every((item, childIndex) => {
|
|
80
|
-
if (childIndex > array[index - 1] && childIndex <= value - 1) {
|
|
81
|
-
return PlaitGroupElement.isGroup(item);
|
|
82
|
-
}
|
|
83
|
-
return true;
|
|
84
|
-
});
|
|
85
|
-
let isPartialSelectGroupElement = false;
|
|
86
|
-
if (previousElement?.groupId || (currentElement?.groupId && previousElement?.groupId !== currentElement?.groupId)) {
|
|
87
|
-
let isPartialSelectPreviousGroup = false;
|
|
88
|
-
let isPartialSelectCurrentElement = false;
|
|
89
|
-
if (previousElement.groupId) {
|
|
90
|
-
const highestGroup = getHighestGroup(board, previousElement);
|
|
91
|
-
isPartialSelectPreviousGroup = !isSelectedAllElementsInGroup(board, highestGroup);
|
|
92
|
-
}
|
|
93
|
-
if (currentElement.groupId) {
|
|
94
|
-
const highestGroup = getHighestGroup(board, currentElement);
|
|
95
|
-
isPartialSelectCurrentElement = !isSelectedAllElementsInGroup(board, highestGroup);
|
|
96
|
-
}
|
|
97
|
-
isPartialSelectGroupElement = isPartialSelectPreviousGroup || isPartialSelectCurrentElement;
|
|
98
|
-
}
|
|
99
|
-
if (!isContiguous || isPartialSelectGroupElement) {
|
|
100
|
-
cursor = ++cursor;
|
|
101
|
-
}
|
|
102
|
-
}
|
|
103
|
-
(acc[cursor] || (acc[cursor] = [])).push(value);
|
|
104
|
-
return acc;
|
|
105
|
-
}, []);
|
|
106
|
-
};
|
|
107
|
-
/**
|
|
108
|
-
* Returns next candidate index that's available to be moved to. Currently that
|
|
109
|
-
* is a non-deleted element, and not inside a group (unless we're editing it).
|
|
110
|
-
*/
|
|
111
|
-
const getTargetIndex = (board, boundaryIndex, direction) => {
|
|
112
|
-
if ((boundaryIndex === 0 && direction === 'down') || (boundaryIndex === board.children.length - 1 && direction === 'up')) {
|
|
113
|
-
return -1;
|
|
114
|
-
}
|
|
115
|
-
const indexFilter = (element) => {
|
|
116
|
-
if (element.isDeleted || PlaitGroupElement.isGroup(element)) {
|
|
117
|
-
return false;
|
|
118
|
-
}
|
|
119
|
-
return true;
|
|
120
|
-
};
|
|
121
|
-
const candidateIndex = direction === 'down'
|
|
122
|
-
? findLastIndex(board.children, el => indexFilter(el), Math.max(0, boundaryIndex - 1))
|
|
123
|
-
: findIndex(board.children, el => indexFilter(el), boundaryIndex + 1);
|
|
124
|
-
const nextElement = board.children[candidateIndex];
|
|
125
|
-
if (!nextElement) {
|
|
126
|
-
return -1;
|
|
127
|
-
}
|
|
128
|
-
const elements = [...board.children];
|
|
129
|
-
const sourceElement = elements[boundaryIndex];
|
|
130
|
-
const editingGroup = getEditingGroup(board, sourceElement);
|
|
131
|
-
const nextElementGroups = (getGroupByElement(board, nextElement, true) || []);
|
|
132
|
-
// candidate element is a sibling in current editing group → return
|
|
133
|
-
if (editingGroup && sourceElement?.groupId !== nextElement?.groupId) {
|
|
134
|
-
// candidate element is outside current editing group → prevent
|
|
135
|
-
if (!nextElementGroups.find(item => item.id === editingGroup.id)) {
|
|
136
|
-
return -1;
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
if (!nextElement.groupId) {
|
|
140
|
-
return candidateIndex;
|
|
141
|
-
}
|
|
142
|
-
let siblingGroup;
|
|
143
|
-
if (editingGroup) {
|
|
144
|
-
siblingGroup = nextElementGroups[nextElementGroups.indexOf(editingGroup) - 1];
|
|
145
|
-
}
|
|
146
|
-
else {
|
|
147
|
-
siblingGroup = nextElementGroups[nextElementGroups.length - 1];
|
|
148
|
-
}
|
|
149
|
-
if (siblingGroup) {
|
|
150
|
-
let elementsInSiblingGroup = getElementsInGroup(board, siblingGroup, true, false);
|
|
151
|
-
if (elementsInSiblingGroup.length) {
|
|
152
|
-
elementsInSiblingGroup.sort((a, b) => {
|
|
153
|
-
const indexA = board.children.findIndex(child => child.id === a.id);
|
|
154
|
-
const indexB = board.children.findIndex(child => child.id === b.id);
|
|
155
|
-
return indexA - indexB;
|
|
156
|
-
});
|
|
157
|
-
// assumes getElementsInGroup() returned elements are sorted
|
|
158
|
-
// by zIndex (ascending)
|
|
159
|
-
return direction === 'down'
|
|
160
|
-
? elements.indexOf(elementsInSiblingGroup[0])
|
|
161
|
-
: elements.indexOf(elementsInSiblingGroup[elementsInSiblingGroup.length - 1]);
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
return candidateIndex;
|
|
165
|
-
};
|
|
166
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"z-index.js","sourceRoot":"","sources":["../../../../packages/core/src/utils/z-index.ts"],"names":[],"mappings":"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","sourcesContent":["import { PlaitBoard, PlaitElement, PlaitGroupElement, PlaitGroup } from '../interfaces';\nimport { MoveNodeOption, getElementsIndices } from './common';\nimport { getEditingGroup, getElementsInGroup, getGroupByElement, getHighestGroup, isSelectedAllElementsInGroup } from './group';\nimport { findIndex, findLastIndex } from './helper';\nimport { sortElements } from './position';\nimport { getSelectedElements } from './selected-element';\n\nexport const getOneMoveOptions = (board: PlaitBoard, direction: 'down' | 'up'): MoveNodeOption[] => {\n    const indicesToMove = getElementsIndices(board, getSelectedElements(board));\n    let groupedIndices = toContiguousGroups(board, indicesToMove);\n    if (direction === 'up') {\n        groupedIndices = groupedIndices.reverse();\n    }\n    let moveContents: MoveNodeOption[] = [];\n    groupedIndices.forEach((indices, i) => {\n        const leadingIndex = indices[0];\n        const trailingIndex = indices[indices.length - 1];\n        const boundaryIndex = direction === 'down' ? leadingIndex : trailingIndex;\n        const targetIndex = getTargetIndex(board, boundaryIndex, direction);\n        if (targetIndex === -1 || boundaryIndex === targetIndex) {\n            return;\n        }\n        if (direction === 'down') {\n            indices = indices.reverse();\n        }\n        moveContents.push(\n            ...indices.map(path => {\n                return {\n                    element: board.children[path],\n                    newPath: [targetIndex]\n                };\n            })\n        );\n    });\n\n    return moveContents;\n};\n\nexport const getAllMoveOptions = (board: PlaitBoard, direction: 'down' | 'up'): MoveNodeOption[] => {\n    const indicesToMove = getElementsIndices(board, getSelectedElements(board));\n    let groupedIndices = toContiguousGroups(board, indicesToMove);\n    let moveContents: MoveNodeOption[] = [];\n    if (direction === 'down') {\n        groupedIndices = groupedIndices.reverse();\n    }\n    groupedIndices.forEach(indices => {\n        const leadingIndex = indices[0];\n        const trailingIndex = indices[indices.length - 1];\n        const boundaryIndex = direction === 'down' ? leadingIndex : trailingIndex;\n        const sourceElement = board.children[boundaryIndex];\n        const editingGroup = getEditingGroup(board, sourceElement);\n        let targetIndex = direction === 'down' ? 0 : board.children.length - 1;\n        if (editingGroup) {\n            const elementsInGroup = sortElements(board, getElementsInGroup(board, editingGroup, true, true));\n            targetIndex =\n                direction === 'down'\n                    ? board.children.indexOf(elementsInGroup[0])\n                    : board.children.indexOf(elementsInGroup[elementsInGroup.length - 1]);\n        }\n        if (direction === 'down') {\n            indices = indices.reverse();\n        }\n        moveContents.push(\n            ...indices.map(path => {\n                return {\n                    element: board.children[path],\n                    newPath: [targetIndex]\n                };\n            })\n        );\n    });\n\n    return moveContents;\n};\n\nexport const canSetZIndex = (board: PlaitBoard) => {\n    const selectedElements = getSelectedElements(board).filter(item => board.canSetZIndex(item));\n    return selectedElements.length > 0;\n};\n\nconst toContiguousGroups = (board: PlaitBoard, array: number[]) => {\n    let cursor = 0;\n    return array.reduce((acc, value, index) => {\n        if (index > 0) {\n            const currentElement = board.children[value];\n            const previousElement = board.children[array[index - 1]];\n            const isContiguous =\n                value - 1 === array[index - 1]\n                    ? true\n                    : board.children.every((item, childIndex) => {\n                          if (childIndex > array[index - 1] && childIndex <= value - 1) {\n                              return PlaitGroupElement.isGroup(item);\n                          }\n                          return true;\n                      });\n            let isPartialSelectGroupElement = false;\n            if (previousElement?.groupId || (currentElement?.groupId && previousElement?.groupId !== currentElement?.groupId)) {\n                let isPartialSelectPreviousGroup = false;\n                let isPartialSelectCurrentElement = false;\n                if (previousElement.groupId) {\n                    const highestGroup = getHighestGroup(board, previousElement);\n                    isPartialSelectPreviousGroup = !isSelectedAllElementsInGroup(board, highestGroup!);\n                }\n                if (currentElement.groupId) {\n                    const highestGroup = getHighestGroup(board, currentElement);\n                    isPartialSelectCurrentElement = !isSelectedAllElementsInGroup(board, highestGroup!);\n                }\n                isPartialSelectGroupElement = isPartialSelectPreviousGroup || isPartialSelectCurrentElement;\n            }\n            if (!isContiguous || isPartialSelectGroupElement) {\n                cursor = ++cursor;\n            }\n        }\n        (acc[cursor] || (acc[cursor] = [])).push(value);\n        return acc;\n    }, [] as number[][]);\n};\n\n/**\n * Returns next candidate index that's available to be moved to. Currently that\n *  is a non-deleted element, and not inside a group (unless we're editing it).\n */\nconst getTargetIndex = (board: PlaitBoard, boundaryIndex: number, direction: 'down' | 'up') => {\n    if ((boundaryIndex === 0 && direction === 'down') || (boundaryIndex === board.children.length - 1 && direction === 'up')) {\n        return -1;\n    }\n    const indexFilter = (element: PlaitElement) => {\n        if (element.isDeleted || PlaitGroupElement.isGroup(element)) {\n            return false;\n        }\n        return true;\n    };\n    const candidateIndex =\n        direction === 'down'\n            ? findLastIndex(board.children, el => indexFilter(el), Math.max(0, boundaryIndex - 1))\n            : findIndex(board.children, el => indexFilter(el), boundaryIndex + 1);\n\n    const nextElement = board.children[candidateIndex];\n    if (!nextElement) {\n        return -1;\n    }\n\n    const elements = [...board.children];\n    const sourceElement = elements[boundaryIndex];\n    const editingGroup = getEditingGroup(board, sourceElement);\n    const nextElementGroups = (getGroupByElement(board, nextElement, true) || []) as PlaitGroup[];\n    // candidate element is a sibling in current editing group → return\n    if (editingGroup && sourceElement?.groupId !== nextElement?.groupId) {\n        // candidate element is outside current editing group → prevent\n        if (!(nextElementGroups as PlaitGroup[]).find(item => item.id === editingGroup.id)) {\n            return -1;\n        }\n    }\n\n    if (!nextElement.groupId) {\n        return candidateIndex;\n    }\n\n    let siblingGroup: PlaitGroup;\n    if (editingGroup) {\n        siblingGroup = nextElementGroups[nextElementGroups.indexOf(editingGroup) - 1];\n    } else {\n        siblingGroup = nextElementGroups[nextElementGroups.length - 1];\n    }\n    if (siblingGroup) {\n        let elementsInSiblingGroup = getElementsInGroup(board, siblingGroup, true, false);\n        if (elementsInSiblingGroup.length) {\n            elementsInSiblingGroup.sort((a, b) => {\n                const indexA = board.children.findIndex(child => child.id === a.id);\n                const indexB = board.children.findIndex(child => child.id === b.id);\n                return indexA - indexB;\n            });\n            // assumes getElementsInGroup() returned elements are sorted\n            // by zIndex (ascending)\n            return direction === 'down'\n                ? elements.indexOf(elementsInSiblingGroup[0])\n                : elements.indexOf(elementsInSiblingGroup[elementsInSiblingGroup.length - 1]);\n        }\n    }\n\n    return candidateIndex;\n};\n"]}
|