
* new collision api * isPointOnShape * removed redundant code * new collision methods in app * curve shape takes starting point * clean up geometry * curve rotation * freedraw * inside curve * improve ellipse inside check * ellipse distance func * curve inside * include frame name bounds * replace previous private methods for getting elements at x,y * arrow bound text hit detection * keep iframes on top * remove dependence on old collision methods from app * remove old collision functions * move some hit functions outside of app * code refactor * type * text collision from inside * fix context menu test * highest z-index collision * fix 1px away binding test * strictly less * remove unused imports * lint * 'ignore' resize flipping test * more lint fix * skip 'flips while resizing' test * more test * fix merge errors * fix selection in resize test * added a bit more comment --------- Co-authored-by: dwelle <5153846+dwelle@users.noreply.github.com>
123 lines
3.4 KiB
TypeScript
123 lines
3.4 KiB
TypeScript
import {
|
|
ExcalidrawElement,
|
|
NonDeletedExcalidrawElement,
|
|
NonDeleted,
|
|
} from "./types";
|
|
import { isInvisiblySmallElement } from "./sizeHelpers";
|
|
import { isLinearElementType } from "./typeChecks";
|
|
|
|
export {
|
|
newElement,
|
|
newTextElement,
|
|
updateTextElement,
|
|
refreshTextDimensions,
|
|
newLinearElement,
|
|
newImageElement,
|
|
duplicateElement,
|
|
} from "./newElement";
|
|
export {
|
|
getElementAbsoluteCoords,
|
|
getElementBounds,
|
|
getCommonBounds,
|
|
getDiamondPoints,
|
|
getArrowheadPoints,
|
|
getClosestElementBounds,
|
|
} from "./bounds";
|
|
|
|
export {
|
|
OMIT_SIDES_FOR_MULTIPLE_ELEMENTS,
|
|
getTransformHandlesFromCoords,
|
|
getTransformHandles,
|
|
} from "./transformHandles";
|
|
export {
|
|
resizeTest,
|
|
getCursorForResizingElement,
|
|
getElementWithTransformHandleType,
|
|
getTransformHandleTypeFromCoords,
|
|
} from "./resizeTest";
|
|
export {
|
|
transformElements,
|
|
getResizeOffsetXY,
|
|
getResizeArrowDirection,
|
|
} from "./resizeElements";
|
|
export {
|
|
dragSelectedElements,
|
|
getDragOffsetXY,
|
|
dragNewElement,
|
|
} from "./dragElements";
|
|
export { isTextElement, isExcalidrawElement } from "./typeChecks";
|
|
export { redrawTextBoundingBox } from "./textElement";
|
|
export {
|
|
getPerfectElementSize,
|
|
getLockedLinearCursorAlignSize,
|
|
isInvisiblySmallElement,
|
|
resizePerfectLineForNWHandler,
|
|
getNormalizedDimensions,
|
|
} from "./sizeHelpers";
|
|
export { showSelectedShapeActions } from "./showSelectedShapeActions";
|
|
|
|
/**
|
|
* @deprecated unsafe, use hashElementsVersion instead
|
|
*/
|
|
export const getSceneVersion = (elements: readonly ExcalidrawElement[]) =>
|
|
elements.reduce((acc, el) => acc + el.version, 0);
|
|
|
|
/**
|
|
* Hashes elements' versionNonce (using djb2 algo). Order of elements matters.
|
|
*/
|
|
export const hashElementsVersion = (
|
|
elements: readonly ExcalidrawElement[],
|
|
): number => {
|
|
let hash = 5381;
|
|
for (let i = 0; i < elements.length; i++) {
|
|
hash = (hash << 5) + hash + elements[i].versionNonce;
|
|
}
|
|
return hash >>> 0; // Ensure unsigned 32-bit integer
|
|
};
|
|
|
|
// string hash function (using djb2). Not cryptographically secure, use only
|
|
// for versioning and such.
|
|
export const hashString = (s: string): number => {
|
|
let hash: number = 5381;
|
|
for (let i = 0; i < s.length; i++) {
|
|
const char: number = s.charCodeAt(i);
|
|
hash = (hash << 5) + hash + char;
|
|
}
|
|
return hash >>> 0; // Ensure unsigned 32-bit integer
|
|
};
|
|
|
|
export const getVisibleElements = (elements: readonly ExcalidrawElement[]) =>
|
|
elements.filter(
|
|
(el) => !el.isDeleted && !isInvisiblySmallElement(el),
|
|
) as readonly NonDeletedExcalidrawElement[];
|
|
|
|
export const getNonDeletedElements = <T extends ExcalidrawElement>(
|
|
elements: readonly T[],
|
|
) =>
|
|
elements.filter((element) => !element.isDeleted) as readonly NonDeleted<T>[];
|
|
|
|
export const isNonDeletedElement = <T extends ExcalidrawElement>(
|
|
element: T,
|
|
): element is NonDeleted<T> => !element.isDeleted;
|
|
|
|
const _clearElements = (
|
|
elements: readonly ExcalidrawElement[],
|
|
): ExcalidrawElement[] =>
|
|
getNonDeletedElements(elements).map((element) =>
|
|
isLinearElementType(element.type)
|
|
? { ...element, lastCommittedPoint: null }
|
|
: element,
|
|
);
|
|
|
|
export const clearElementsForDatabase = (
|
|
elements: readonly ExcalidrawElement[],
|
|
) => _clearElements(elements);
|
|
|
|
export const clearElementsForExport = (
|
|
elements: readonly ExcalidrawElement[],
|
|
) => _clearElements(elements);
|
|
|
|
export const clearElementsForLocalStorage = (
|
|
elements: readonly ExcalidrawElement[],
|
|
) => _clearElements(elements);
|