@plait/core 0.8.0 → 0.10.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.
@@ -2,7 +2,7 @@ import { PlaitBoard } from '../interfaces/board';
2
2
  import { toPoint } from '../utils/dom/common';
3
3
  import { getRectangleByElements } from '../utils/element';
4
4
  import { distanceBetweenPointAndRectangle } from '../utils/math';
5
- import { clampZoomLevel, clearViewportOrigination, getViewBox, getViewportContainerRect, getViewportOrigination } from '../utils/viewport';
5
+ import { clampZoomLevel, clearViewportOrigination, getViewBox, getViewportOrigination } from '../utils/viewport';
6
6
  import { BOARD_TO_COMPONENT } from '../utils/weak-maps';
7
7
  import { setViewport } from './viewport';
8
8
  import { depthFirstRecursion } from '../utils';
@@ -26,8 +26,8 @@ function updateZoom(board, newZoom, isCenter = true) {
26
26
  const mousePoint = PlaitBoard.getMovingPoint(board);
27
27
  const nativeElement = PlaitBoard.getBoardNativeElement(board);
28
28
  const nativeElementRect = nativeElement.getBoundingClientRect();
29
- const viewportContainerRect = getViewportContainerRect(board);
30
- let focusPoint = [viewportContainerRect.width / 2, viewportContainerRect.height / 2];
29
+ const boardContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
30
+ let focusPoint = [boardContainerRect.width / 2, boardContainerRect.height / 2];
31
31
  if (!isCenter && mousePoint && distanceBetweenPointAndRectangle(mousePoint[0], mousePoint[1], nativeElementRect) === 0) {
32
32
  focusPoint = toPoint(mousePoint[0], mousePoint[1], nativeElement);
33
33
  }
@@ -39,13 +39,18 @@ function updateZoom(board, newZoom, isCenter = true) {
39
39
  updateViewport(board, newOrigination, newZoom);
40
40
  }
41
41
  function fitViewport(board) {
42
- const nativeElementRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
43
- const viewportContainerRect = getViewportContainerRect(board);
42
+ const { hideScrollbar } = board.options;
43
+ let scrollBarWidth = 0;
44
+ if (!hideScrollbar) {
45
+ const viewportContainer = PlaitBoard.getViewportContainer(board);
46
+ scrollBarWidth = viewportContainer.offsetWidth - viewportContainer.clientWidth;
47
+ }
48
+ const boardContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
44
49
  const elementHostBox = getRectangleByElements(board, board.children, true);
45
50
  const zoom = board.viewport.zoom;
46
51
  const autoFitPadding = 16;
47
- const viewportWidth = nativeElementRect.width - 2 * autoFitPadding;
48
- const viewportHeight = nativeElementRect.height - 2 * autoFitPadding;
52
+ const viewportWidth = boardContainerRect.width - 2 * autoFitPadding;
53
+ const viewportHeight = boardContainerRect.height - 2 * autoFitPadding;
49
54
  let newZoom = zoom;
50
55
  if (viewportWidth < elementHostBox.width || viewportHeight < elementHostBox.height) {
51
56
  newZoom = Math.min(viewportWidth / elementHostBox.width, viewportHeight / elementHostBox.height);
@@ -57,8 +62,8 @@ function fitViewport(board) {
57
62
  const centerX = viewBox[0] + viewBox[2] / 2;
58
63
  const centerY = viewBox[1] + viewBox[3] / 2;
59
64
  const newOrigination = [
60
- centerX - viewportContainerRect.width / 2 / newZoom,
61
- centerY - viewportContainerRect.height / 2 / newZoom
65
+ centerX - boardContainerRect.width / 2 / newZoom + scrollBarWidth / 2 / zoom,
66
+ centerY - boardContainerRect.height / 2 / newZoom + scrollBarWidth / 2 / zoom
62
67
  ];
63
68
  updateViewport(board, newOrigination, newZoom);
64
69
  }
@@ -80,4 +85,4 @@ export const BoardTransforms = {
80
85
  updateZoom,
81
86
  updateThemeColor
82
87
  };
83
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.js","sourceRoot":"","sources":["../../../../packages/plait/src/transforms/board.ts"],"names":[],"mappings":"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","sourcesContent":["import { ThemeColorMode } from '../interfaces/theme';\nimport { PlaitBoard } from '../interfaces/board';\nimport { Point } from '../interfaces/point';\nimport { PlaitPointerType } from '../interfaces/pointer';\nimport { toPoint } from '../utils/dom/common';\nimport { getRectangleByElements } from '../utils/element';\nimport { distanceBetweenPointAndRectangle } from '../utils/math';\nimport { clampZoomLevel, clearViewportOrigination, getViewBox, getViewportContainerRect, getViewportOrigination } from '../utils/viewport';\nimport { BOARD_TO_COMPONENT } from '../utils/weak-maps';\nimport { setViewport } from './viewport';\nimport { depthFirstRecursion } from '../utils';\nimport { PlaitElement } from '../interfaces/element';\nimport { setTheme } from './theme';\n\nfunction updateViewport(board: PlaitBoard, origination: Point, zoom?: number) {\n    zoom = zoom ?? board.viewport.zoom;\n    setViewport(board, {\n        ...board.viewport,\n        zoom,\n        origination\n    });\n    clearViewportOrigination(board);\n}\n\nconst updatePointerType = <T extends string = PlaitPointerType>(board: PlaitBoard, pointer: T) => {\n    board.pointer = pointer;\n    const boardComponent = BOARD_TO_COMPONENT.get(board);\n    boardComponent?.markForCheck();\n};\n\nfunction updateZoom(board: PlaitBoard, newZoom: number, isCenter = true) {\n    newZoom = clampZoomLevel(newZoom);\n\n    const mousePoint = PlaitBoard.getMovingPoint(board);\n    const nativeElement = PlaitBoard.getBoardNativeElement(board);\n    const nativeElementRect = nativeElement.getBoundingClientRect();\n    const viewportContainerRect = getViewportContainerRect(board);\n    let focusPoint = [viewportContainerRect.width / 2, viewportContainerRect.height / 2];\n\n    if (!isCenter && mousePoint && distanceBetweenPointAndRectangle(mousePoint[0], mousePoint[1], nativeElementRect) === 0) {\n        focusPoint = toPoint(mousePoint[0], mousePoint[1], (nativeElement as unknown) as SVGElement);\n    }\n\n    const zoom = board.viewport.zoom;\n    const origination = getViewportOrigination(board);\n    const centerX = origination![0] + focusPoint[0] / zoom;\n    const centerY = origination![1] + focusPoint[1] / zoom;\n    const newOrigination = [centerX - focusPoint[0] / newZoom, centerY - focusPoint[1] / newZoom] as Point;\n    updateViewport(board, newOrigination, newZoom);\n}\n\nfunction fitViewport(board: PlaitBoard) {\n    const nativeElementRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    const viewportContainerRect = getViewportContainerRect(board);\n    const elementHostBox = getRectangleByElements(board, board.children, true);\n    const zoom = board.viewport.zoom;\n    const autoFitPadding = 16;\n    const viewportWidth = nativeElementRect.width - 2 * autoFitPadding;\n    const viewportHeight = nativeElementRect.height - 2 * autoFitPadding;\n\n    let newZoom = zoom;\n    if (viewportWidth < elementHostBox.width || viewportHeight < elementHostBox.height) {\n        newZoom = Math.min(viewportWidth / elementHostBox.width, viewportHeight / elementHostBox.height);\n    } else {\n        newZoom = 1;\n    }\n\n    const viewBox = getViewBox(board, newZoom);\n    const centerX = viewBox[0] + viewBox[2] / 2;\n    const centerY = viewBox[1] + viewBox[3] / 2;\n    const newOrigination = [\n        centerX - viewportContainerRect.width / 2 / newZoom,\n        centerY - viewportContainerRect.height / 2 / newZoom\n    ] as Point;\n    updateViewport(board, newOrigination, newZoom);\n}\n\n/**\n * apply theme to every element (remove element custom properties)\n * invoke applyThemeColor\n */\nfunction updateThemeColor(board: PlaitBoard, mode: ThemeColorMode) {\n    mode = mode ?? board.theme.themeColorMode;\n    setTheme(board, { themeColorMode: mode });\n\n    depthFirstRecursion((board as unknown) as PlaitElement, element => {\n        board.applyTheme(element);\n    });\n}\n\nexport const BoardTransforms = {\n    updatePointerType,\n    updateViewport,\n    fitViewport,\n    updateZoom,\n    updateThemeColor\n};\n"]}
88
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.js","sourceRoot":"","sources":["../../../../packages/plait/src/transforms/board.ts"],"names":[],"mappings":"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","sourcesContent":["import { ThemeColorMode } from '../interfaces/theme';\nimport { PlaitBoard } from '../interfaces/board';\nimport { Point } from '../interfaces/point';\nimport { PlaitPointerType } from '../interfaces/pointer';\nimport { toPoint } from '../utils/dom/common';\nimport { getRectangleByElements } from '../utils/element';\nimport { distanceBetweenPointAndRectangle } from '../utils/math';\nimport { clampZoomLevel, clearViewportOrigination, getViewBox, getViewportOrigination } from '../utils/viewport';\nimport { BOARD_TO_COMPONENT } from '../utils/weak-maps';\nimport { setViewport } from './viewport';\nimport { depthFirstRecursion } from '../utils';\nimport { PlaitElement } from '../interfaces/element';\nimport { setTheme } from './theme';\n\nfunction updateViewport(board: PlaitBoard, origination: Point, zoom?: number) {\n    zoom = zoom ?? board.viewport.zoom;\n    setViewport(board, {\n        ...board.viewport,\n        zoom,\n        origination\n    });\n    clearViewportOrigination(board);\n}\n\nconst updatePointerType = <T extends string = PlaitPointerType>(board: PlaitBoard, pointer: T) => {\n    board.pointer = pointer;\n    const boardComponent = BOARD_TO_COMPONENT.get(board);\n    boardComponent?.markForCheck();\n};\n\nfunction updateZoom(board: PlaitBoard, newZoom: number, isCenter = true) {\n    newZoom = clampZoomLevel(newZoom);\n\n    const mousePoint = PlaitBoard.getMovingPoint(board);\n    const nativeElement = PlaitBoard.getBoardNativeElement(board);\n    const nativeElementRect = nativeElement.getBoundingClientRect();\n    const boardContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    let focusPoint = [boardContainerRect.width / 2, boardContainerRect.height / 2];\n\n    if (!isCenter && mousePoint && distanceBetweenPointAndRectangle(mousePoint[0], mousePoint[1], nativeElementRect) === 0) {\n        focusPoint = toPoint(mousePoint[0], mousePoint[1], (nativeElement as unknown) as SVGElement);\n    }\n\n    const zoom = board.viewport.zoom;\n    const origination = getViewportOrigination(board);\n    const centerX = origination![0] + focusPoint[0] / zoom;\n    const centerY = origination![1] + focusPoint[1] / zoom;\n    const newOrigination = [centerX - focusPoint[0] / newZoom, centerY - focusPoint[1] / newZoom] as Point;\n    updateViewport(board, newOrigination, newZoom);\n}\n\nfunction fitViewport(board: PlaitBoard) {\n    const { hideScrollbar } = board.options;\n    let scrollBarWidth = 0;\n    if (!hideScrollbar) {\n        const viewportContainer = PlaitBoard.getViewportContainer(board);\n        scrollBarWidth = viewportContainer.offsetWidth - viewportContainer.clientWidth;\n    }\n\n    const boardContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    const elementHostBox = getRectangleByElements(board, board.children, true);\n    const zoom = board.viewport.zoom;\n    const autoFitPadding = 16;\n    const viewportWidth = boardContainerRect.width - 2 * autoFitPadding;\n    const viewportHeight = boardContainerRect.height - 2 * autoFitPadding;\n\n    let newZoom = zoom;\n    if (viewportWidth < elementHostBox.width || viewportHeight < elementHostBox.height) {\n        newZoom = Math.min(viewportWidth / elementHostBox.width, viewportHeight / elementHostBox.height);\n    } else {\n        newZoom = 1;\n    }\n\n    const viewBox = getViewBox(board, newZoom);\n    const centerX = viewBox[0] + viewBox[2] / 2;\n    const centerY = viewBox[1] + viewBox[3] / 2;\n    const newOrigination = [\n        centerX - boardContainerRect.width / 2 / newZoom + scrollBarWidth / 2 / zoom,\n        centerY - boardContainerRect.height / 2 / newZoom + scrollBarWidth / 2 / zoom\n    ] as Point;\n    updateViewport(board, newOrigination, newZoom);\n}\n\n/**\n * apply theme to every element (remove element custom properties)\n * invoke applyThemeColor\n */\nfunction updateThemeColor(board: PlaitBoard, mode: ThemeColorMode) {\n    mode = mode ?? board.theme.themeColorMode;\n    setTheme(board, { themeColorMode: mode });\n\n    depthFirstRecursion((board as unknown) as PlaitElement, element => {\n        board.applyTheme(element);\n    });\n}\n\nexport const BoardTransforms = {\n    updatePointerType,\n    updateViewport,\n    fitViewport,\n    updateZoom,\n    updateThemeColor\n};\n"]}
@@ -60,17 +60,15 @@ export function clampZoomLevel(zoom, minZoom = 0.2, maxZoom = 4) {
60
60
  return zoom < minZoom ? minZoom : zoom > maxZoom ? maxZoom : zoom;
61
61
  }
62
62
  export function getViewBox(board, zoom) {
63
- const { hideScrollbar } = board.options;
64
- const scrollBarWidth = hideScrollbar ? 0 : SCROLL_BAR_WIDTH;
65
- const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
63
+ const boardContainerRectangle = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
66
64
  const elementHostBBox = getElementHostBBox(board, zoom);
67
- const horizontalPadding = viewportContainerRect.width / 2;
68
- const verticalPadding = viewportContainerRect.height / 2;
65
+ const horizontalPadding = boardContainerRectangle.width / 2;
66
+ const verticalPadding = boardContainerRectangle.height / 2;
69
67
  const viewBox = [
70
68
  elementHostBBox.left - horizontalPadding / zoom,
71
69
  elementHostBBox.top - verticalPadding / zoom,
72
- elementHostBBox.right - elementHostBBox.left + (horizontalPadding * 2 - scrollBarWidth) / zoom,
73
- elementHostBBox.bottom - elementHostBBox.top + (verticalPadding * 2 - scrollBarWidth) / zoom
70
+ elementHostBBox.right - elementHostBBox.left + (horizontalPadding * 2) / zoom,
71
+ elementHostBBox.bottom - elementHostBBox.top + (verticalPadding * 2) / zoom
74
72
  ];
75
73
  return viewBox;
76
74
  }
@@ -155,4 +153,4 @@ export const setIsFromViewportChange = (board, state) => {
155
153
  IS_FROM_VIEWPORT_CHANGE.set(board, state);
156
154
  };
157
155
  export function scrollToRectangle(board, client) { }
158
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewport.js","sourceRoot":"","sources":["../../../../packages/plait/src/utils/viewport.ts"],"names":[],"mappings":"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","sourcesContent":["import { SCROLL_BAR_WIDTH } from '../constants';\nimport { PlaitBoard, Point, RectangleClient } from '../interfaces';\nimport { getRectangleByElements } from './element';\nimport { BOARD_TO_VIEWPORT_ORIGINATION } from './weak-maps';\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.getBoardNativeElement(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) {\n    const childrenRect = getRectangleByElements(board, board.children, true);\n    const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    const containerWidth = viewportContainerRect.width / zoom;\n    const containerHeight = viewportContainerRect.height / zoom;\n    let left: number;\n    let right: number;\n    let top: number;\n    let bottom: number;\n\n    if (childrenRect.width < containerWidth) {\n        const centerX = childrenRect.x + childrenRect.width / 2;\n        const halfContainerWidth = containerWidth / 2;\n        left = centerX - halfContainerWidth;\n        right = centerX + halfContainerWidth;\n    } else {\n        left = childrenRect.x;\n        right = childrenRect.x + childrenRect.width;\n    }\n    if (childrenRect.height < containerHeight) {\n        const centerY = childrenRect.y + childrenRect.height / 2;\n        const halfContainerHeight = containerHeight / 2;\n        top = centerY - halfContainerHeight;\n        bottom = centerY + halfContainerHeight;\n    } else {\n        top = childrenRect.y;\n        bottom = childrenRect.y + childrenRect.height;\n    }\n    return {\n        left,\n        right,\n        top,\n        bottom\n    };\n}\n\n/**\n * 验证缩放比是否符合限制，如果超出限制，则返回合适的缩放比\n * @param zoom 缩放比\n * @param minZoom 最小缩放比\n * @param maxZoom 最大缩放比\n * @returns 正确的缩放比\n */\nexport function clampZoomLevel(zoom: number, minZoom = 0.2, maxZoom = 4) {\n    return zoom < minZoom ? minZoom : zoom > maxZoom ? maxZoom : zoom;\n}\n\nexport function getViewBox(board: PlaitBoard, zoom: number) {\n    const { hideScrollbar } = board.options;\n    const scrollBarWidth = hideScrollbar ? 0 : SCROLL_BAR_WIDTH;\n\n    const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    const elementHostBBox = getElementHostBBox(board, zoom);\n\n    const horizontalPadding = viewportContainerRect.width / 2;\n    const verticalPadding = viewportContainerRect.height / 2;\n    const viewBox = [\n        elementHostBBox.left - horizontalPadding / zoom,\n        elementHostBBox.top - verticalPadding / zoom,\n        elementHostBBox.right - elementHostBBox.left + (horizontalPadding * 2 - scrollBarWidth) / zoom,\n        elementHostBBox.bottom - elementHostBBox.top + (verticalPadding * 2 - scrollBarWidth) / zoom\n    ];\n    return viewBox;\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) return;\n\n    const { zoom } = board.viewport;\n    const viewBox = getViewBox(board, zoom);\n    const scrollLeft = (origination![0] - viewBox[0]) * zoom;\n    const scrollTop = (origination![1] - viewBox[1]) * zoom;\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    if (viewportContainer.scrollLeft !== left || viewportContainer.scrollTop !== top) {\n        viewportContainer.scrollLeft = left;\n        viewportContainer.scrollTop = top;\n        isFromViewportChange && setIsFromViewportChange(board, true);\n    }\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 = getViewBox(board, zoom);\n    setSVGViewBox(board, viewBox);\n}\n\nexport function initializeViewportOffset(board: PlaitBoard) {\n    if (!board.viewport?.origination) {\n        const zoom = board.viewport.zoom;\n        const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n        const viewBox = getViewBox(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"]}
156
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewport.js","sourceRoot":"","sources":["../../../../packages/plait/src/utils/viewport.ts"],"names":[],"mappings":"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","sourcesContent":["import { SCROLL_BAR_WIDTH } from '../constants';\nimport { PlaitBoard, Point, RectangleClient } from '../interfaces';\nimport { getRectangleByElements } from './element';\nimport { BOARD_TO_VIEWPORT_ORIGINATION } from './weak-maps';\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.getBoardNativeElement(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) {\n    const childrenRect = getRectangleByElements(board, board.children, true);\n    const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    const containerWidth = viewportContainerRect.width / zoom;\n    const containerHeight = viewportContainerRect.height / zoom;\n    let left: number;\n    let right: number;\n    let top: number;\n    let bottom: number;\n\n    if (childrenRect.width < containerWidth) {\n        const centerX = childrenRect.x + childrenRect.width / 2;\n        const halfContainerWidth = containerWidth / 2;\n        left = centerX - halfContainerWidth;\n        right = centerX + halfContainerWidth;\n    } else {\n        left = childrenRect.x;\n        right = childrenRect.x + childrenRect.width;\n    }\n    if (childrenRect.height < containerHeight) {\n        const centerY = childrenRect.y + childrenRect.height / 2;\n        const halfContainerHeight = containerHeight / 2;\n        top = centerY - halfContainerHeight;\n        bottom = centerY + halfContainerHeight;\n    } else {\n        top = childrenRect.y;\n        bottom = childrenRect.y + childrenRect.height;\n    }\n    return {\n        left,\n        right,\n        top,\n        bottom\n    };\n}\n\n/**\n * 验证缩放比是否符合限制，如果超出限制，则返回合适的缩放比\n * @param zoom 缩放比\n * @param minZoom 最小缩放比\n * @param maxZoom 最大缩放比\n * @returns 正确的缩放比\n */\nexport function clampZoomLevel(zoom: number, minZoom = 0.2, maxZoom = 4) {\n    return zoom < minZoom ? minZoom : zoom > maxZoom ? maxZoom : zoom;\n}\n\nexport function getViewBox(board: PlaitBoard, zoom: number) {\n    const boardContainerRectangle = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n    const elementHostBBox = getElementHostBBox(board, zoom);\n\n    const horizontalPadding = boardContainerRectangle.width / 2;\n    const verticalPadding = boardContainerRectangle.height / 2;\n    const viewBox = [\n        elementHostBBox.left - horizontalPadding / zoom,\n        elementHostBBox.top - verticalPadding / zoom,\n        elementHostBBox.right - elementHostBBox.left + (horizontalPadding * 2) / zoom,\n        elementHostBBox.bottom - elementHostBBox.top + (verticalPadding * 2) / zoom\n    ];\n    return viewBox;\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) return;\n\n    const { zoom } = board.viewport;\n    const viewBox = getViewBox(board, zoom);\n    const scrollLeft = (origination![0] - viewBox[0]) * zoom;\n    const scrollTop = (origination![1] - viewBox[1]) * zoom;\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    if (viewportContainer.scrollLeft !== left || viewportContainer.scrollTop !== top) {\n        viewportContainer.scrollLeft = left;\n        viewportContainer.scrollTop = top;\n        isFromViewportChange && setIsFromViewportChange(board, true);\n    }\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 = getViewBox(board, zoom);\n    setSVGViewBox(board, viewBox);\n}\n\nexport function initializeViewportOffset(board: PlaitBoard) {\n    if (!board.viewport?.origination) {\n        const zoom = board.viewport.zoom;\n        const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();\n        const viewBox = getViewBox(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"]}
@@ -1495,17 +1495,15 @@ function clampZoomLevel(zoom, minZoom = 0.2, maxZoom = 4) {
1495
1495
  return zoom < minZoom ? minZoom : zoom > maxZoom ? maxZoom : zoom;
1496
1496
  }
1497
1497
  function getViewBox(board, zoom) {
1498
- const { hideScrollbar } = board.options;
1499
- const scrollBarWidth = hideScrollbar ? 0 : SCROLL_BAR_WIDTH;
1500
- const viewportContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
1498
+ const boardContainerRectangle = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
1501
1499
  const elementHostBBox = getElementHostBBox(board, zoom);
1502
- const horizontalPadding = viewportContainerRect.width / 2;
1503
- const verticalPadding = viewportContainerRect.height / 2;
1500
+ const horizontalPadding = boardContainerRectangle.width / 2;
1501
+ const verticalPadding = boardContainerRectangle.height / 2;
1504
1502
  const viewBox = [
1505
1503
  elementHostBBox.left - horizontalPadding / zoom,
1506
1504
  elementHostBBox.top - verticalPadding / zoom,
1507
- elementHostBBox.right - elementHostBBox.left + (horizontalPadding * 2 - scrollBarWidth) / zoom,
1508
- elementHostBBox.bottom - elementHostBBox.top + (verticalPadding * 2 - scrollBarWidth) / zoom
1505
+ elementHostBBox.right - elementHostBBox.left + (horizontalPadding * 2) / zoom,
1506
+ elementHostBBox.bottom - elementHostBBox.top + (verticalPadding * 2) / zoom
1509
1507
  ];
1510
1508
  return viewBox;
1511
1509
  }
@@ -1616,8 +1614,8 @@ function updateZoom(board, newZoom, isCenter = true) {
1616
1614
  const mousePoint = PlaitBoard.getMovingPoint(board);
1617
1615
  const nativeElement = PlaitBoard.getBoardNativeElement(board);
1618
1616
  const nativeElementRect = nativeElement.getBoundingClientRect();
1619
- const viewportContainerRect = getViewportContainerRect(board);
1620
- let focusPoint = [viewportContainerRect.width / 2, viewportContainerRect.height / 2];
1617
+ const boardContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
1618
+ let focusPoint = [boardContainerRect.width / 2, boardContainerRect.height / 2];
1621
1619
  if (!isCenter && mousePoint && distanceBetweenPointAndRectangle(mousePoint[0], mousePoint[1], nativeElementRect) === 0) {
1622
1620
  focusPoint = toPoint(mousePoint[0], mousePoint[1], nativeElement);
1623
1621
  }
@@ -1629,13 +1627,18 @@ function updateZoom(board, newZoom, isCenter = true) {
1629
1627
  updateViewport(board, newOrigination, newZoom);
1630
1628
  }
1631
1629
  function fitViewport(board) {
1632
- const nativeElementRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
1633
- const viewportContainerRect = getViewportContainerRect(board);
1630
+ const { hideScrollbar } = board.options;
1631
+ let scrollBarWidth = 0;
1632
+ if (!hideScrollbar) {
1633
+ const viewportContainer = PlaitBoard.getViewportContainer(board);
1634
+ scrollBarWidth = viewportContainer.offsetWidth - viewportContainer.clientWidth;
1635
+ }
1636
+ const boardContainerRect = PlaitBoard.getBoardNativeElement(board).getBoundingClientRect();
1634
1637
  const elementHostBox = getRectangleByElements(board, board.children, true);
1635
1638
  const zoom = board.viewport.zoom;
1636
1639
  const autoFitPadding = 16;
1637
- const viewportWidth = nativeElementRect.width - 2 * autoFitPadding;
1638
- const viewportHeight = nativeElementRect.height - 2 * autoFitPadding;
1640
+ const viewportWidth = boardContainerRect.width - 2 * autoFitPadding;
1641
+ const viewportHeight = boardContainerRect.height - 2 * autoFitPadding;
1639
1642
  let newZoom = zoom;
1640
1643
  if (viewportWidth < elementHostBox.width || viewportHeight < elementHostBox.height) {
1641
1644
  newZoom = Math.min(viewportWidth / elementHostBox.width, viewportHeight / elementHostBox.height);
@@ -1647,8 +1650,8 @@ function fitViewport(board) {
1647
1650
  const centerX = viewBox[0] + viewBox[2] / 2;
1648
1651
  const centerY = viewBox[1] + viewBox[3] / 2;
1649
1652
  const newOrigination = [
1650
- centerX - viewportContainerRect.width / 2 / newZoom,
1651
- centerY - viewportContainerRect.height / 2 / newZoom
1653
+ centerX - boardContainerRect.width / 2 / newZoom + scrollBarWidth / 2 / zoom,
1654
+ centerY - boardContainerRect.height / 2 / newZoom + scrollBarWidth / 2 / zoom
1652
1655
  ];
1653
1656
  updateViewport(board, newOrigination, newZoom);
1654
1657
  }
@@ -1751,6 +1754,7 @@ function createBoard(children, options) {
1751
1754
  onChange: () => { },
1752
1755
  mousedown: (event) => { },
1753
1756
  mousemove: (event) => { },
1757
+ mouseleave: (event) => { },
1754
1758
  globalMousemove: (event) => { },
1755
1759
  mouseup: (event) => { },
1756
1760
  globalMouseup: (event) => { },
@@ -1948,7 +1952,6 @@ function withSelection(board) {
1948
1952
  const range = { anchor: point, focus: point };
1949
1953
  const hitElements = getHitElements(board, { ranges: [range] });
1950
1954
  if (PlaitBoard.isPointer(board, PlaitPointerType.selection) &&
1951
- getHitElements(board, { ranges: [range] }).length === 0 &&
1952
1955
  hitElements.length === 0 &&
1953
1956
  options.isMultiple &&
1954
1957
  !options.isDisabledSelect) {
@@ -2150,19 +2153,22 @@ function withMoving(board) {
2150
2153
  const endPoint = transformPoint(board, toPoint(event.x, event.y, host));
2151
2154
  offsetX = endPoint[0] - startPoint[0];
2152
2155
  offsetY = endPoint[1] - startPoint[1];
2153
- throttleRAF(() => {
2154
- const currentElements = activeElements.map(activeElement => {
2155
- const [x, y] = activeElement === null || activeElement === void 0 ? void 0 : activeElement.points[0];
2156
- const index = board.children.findIndex(item => item.id === activeElement.id);
2157
- Transforms.setNode(board, {
2158
- points: [[x + offsetX, y + offsetY]]
2159
- }, [index]);
2160
- MERGING.set(board, true);
2161
- return PlaitNode.get(board, [index]);
2156
+ const offsetBuffer = 5;
2157
+ if (Math.abs(offsetX) > offsetBuffer || Math.abs(offsetY) > offsetBuffer) {
2158
+ throttleRAF(() => {
2159
+ const currentElements = activeElements.map(activeElement => {
2160
+ const [x, y] = activeElement === null || activeElement === void 0 ? void 0 : activeElement.points[0];
2161
+ const index = board.children.findIndex(item => item.id === activeElement.id);
2162
+ Transforms.setNode(board, {
2163
+ points: [[x + offsetX, y + offsetY]]
2164
+ }, [index]);
2165
+ MERGING.set(board, true);
2166
+ return PlaitNode.get(board, [index]);
2167
+ });
2168
+ PlaitBoard.getBoardNativeElement(board).classList.add('element-moving');
2169
+ addMovingElements(board, currentElements);
2162
2170
  });
2163
- PlaitBoard.getBoardNativeElement(board).classList.add('element-moving');
2164
- addMovingElements(board, currentElements);
2165
- });
2171
+ }
2166
2172
  }
2167
2173
  if (isPreventDefault) {
2168
2174
  // 阻止 move 过程中触发画布滚动行为
@@ -2439,7 +2445,6 @@ class PlaitBoardComponent {
2439
2445
  this.initializeHookListener();
2440
2446
  this.viewportScrollListener();
2441
2447
  this.elementResizeListener();
2442
- this.mouseLeaveListener();
2443
2448
  });
2444
2449
  BOARD_TO_COMPONENT.set(this.board, this);
2445
2450
  BOARD_TO_ROUGH_SVG.set(this.board, roughSVG);
@@ -2510,6 +2515,12 @@ class PlaitBoardComponent {
2510
2515
  BOARD_TO_MOVING_POINT.set(this.board, [event.x, event.y]);
2511
2516
  this.board.mousemove(event);
2512
2517
  });
2518
+ fromEvent(this.host, 'mouseleave')
2519
+ .pipe(takeUntil(this.destroy$))
2520
+ .subscribe((event) => {
2521
+ BOARD_TO_MOVING_POINT.delete(this.board);
2522
+ this.board.mouseleave(event);
2523
+ });
2513
2524
  fromEvent(document, 'mousemove')
2514
2525
  .pipe(takeUntil(this.destroy$))
2515
2526
  .subscribe((event) => {
@@ -2619,13 +2630,6 @@ class PlaitBoardComponent {
2619
2630
  });
2620
2631
  this.resizeObserver.observe(this.nativeElement);
2621
2632
  }
2622
- mouseLeaveListener() {
2623
- fromEvent(this.host, 'mouseleave')
2624
- .pipe(takeUntil(this.destroy$))
2625
- .subscribe((event) => {
2626
- BOARD_TO_MOVING_POINT.delete(this.board);
2627
- });
2628
- }
2629
2633
  initializeIslands() {
2630
2634
  var _a;
2631
2635
  (_a = this.islands) === null || _a === void 0 ? void 0 : _a.forEach(island => {