mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-01-06 07:43:40 +08:00
229 lines
6.2 KiB
TypeScript
229 lines
6.2 KiB
TypeScript
|
/**
|
||
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
||
|
*
|
||
|
* This source code is licensed under the MIT license found in the
|
||
|
* LICENSE file in the root directory of this source tree.
|
||
|
*
|
||
|
*/
|
||
|
import type {LexicalEditor, LexicalNode} from 'lexical';
|
||
|
|
||
|
import {$isTextNode} from 'lexical';
|
||
|
|
||
|
import {CSS_TO_STYLES} from './constants';
|
||
|
|
||
|
function getDOMTextNode(element: Node | null): Text | null {
|
||
|
let node = element;
|
||
|
|
||
|
while (node != null) {
|
||
|
if (node.nodeType === Node.TEXT_NODE) {
|
||
|
return node as Text;
|
||
|
}
|
||
|
|
||
|
node = node.firstChild;
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
function getDOMIndexWithinParent(node: ChildNode): [ParentNode, number] {
|
||
|
const parent = node.parentNode;
|
||
|
|
||
|
if (parent == null) {
|
||
|
throw new Error('Should never happen');
|
||
|
}
|
||
|
|
||
|
return [parent, Array.from(parent.childNodes).indexOf(node)];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates a selection range for the DOM.
|
||
|
* @param editor - The lexical editor.
|
||
|
* @param anchorNode - The anchor node of a selection.
|
||
|
* @param _anchorOffset - The amount of space offset from the anchor to the focus.
|
||
|
* @param focusNode - The current focus.
|
||
|
* @param _focusOffset - The amount of space offset from the focus to the anchor.
|
||
|
* @returns The range of selection for the DOM that was created.
|
||
|
*/
|
||
|
export function createDOMRange(
|
||
|
editor: LexicalEditor,
|
||
|
anchorNode: LexicalNode,
|
||
|
_anchorOffset: number,
|
||
|
focusNode: LexicalNode,
|
||
|
_focusOffset: number,
|
||
|
): Range | null {
|
||
|
const anchorKey = anchorNode.getKey();
|
||
|
const focusKey = focusNode.getKey();
|
||
|
const range = document.createRange();
|
||
|
let anchorDOM: Node | Text | null = editor.getElementByKey(anchorKey);
|
||
|
let focusDOM: Node | Text | null = editor.getElementByKey(focusKey);
|
||
|
let anchorOffset = _anchorOffset;
|
||
|
let focusOffset = _focusOffset;
|
||
|
|
||
|
if ($isTextNode(anchorNode)) {
|
||
|
anchorDOM = getDOMTextNode(anchorDOM);
|
||
|
}
|
||
|
|
||
|
if ($isTextNode(focusNode)) {
|
||
|
focusDOM = getDOMTextNode(focusDOM);
|
||
|
}
|
||
|
|
||
|
if (
|
||
|
anchorNode === undefined ||
|
||
|
focusNode === undefined ||
|
||
|
anchorDOM === null ||
|
||
|
focusDOM === null
|
||
|
) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
if (anchorDOM.nodeName === 'BR') {
|
||
|
[anchorDOM, anchorOffset] = getDOMIndexWithinParent(anchorDOM as ChildNode);
|
||
|
}
|
||
|
|
||
|
if (focusDOM.nodeName === 'BR') {
|
||
|
[focusDOM, focusOffset] = getDOMIndexWithinParent(focusDOM as ChildNode);
|
||
|
}
|
||
|
|
||
|
const firstChild = anchorDOM.firstChild;
|
||
|
|
||
|
if (
|
||
|
anchorDOM === focusDOM &&
|
||
|
firstChild != null &&
|
||
|
firstChild.nodeName === 'BR' &&
|
||
|
anchorOffset === 0 &&
|
||
|
focusOffset === 0
|
||
|
) {
|
||
|
focusOffset = 1;
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
range.setStart(anchorDOM, anchorOffset);
|
||
|
range.setEnd(focusDOM, focusOffset);
|
||
|
} catch (e) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
if (
|
||
|
range.collapsed &&
|
||
|
(anchorOffset !== focusOffset || anchorKey !== focusKey)
|
||
|
) {
|
||
|
// Range is backwards, we need to reverse it
|
||
|
range.setStart(focusDOM, focusOffset);
|
||
|
range.setEnd(anchorDOM, anchorOffset);
|
||
|
}
|
||
|
|
||
|
return range;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates DOMRects, generally used to help the editor find a specific location on the screen.
|
||
|
* @param editor - The lexical editor
|
||
|
* @param range - A fragment of a document that can contain nodes and parts of text nodes.
|
||
|
* @returns The selectionRects as an array.
|
||
|
*/
|
||
|
export function createRectsFromDOMRange(
|
||
|
editor: LexicalEditor,
|
||
|
range: Range,
|
||
|
): Array<ClientRect> {
|
||
|
const rootElement = editor.getRootElement();
|
||
|
|
||
|
if (rootElement === null) {
|
||
|
return [];
|
||
|
}
|
||
|
const rootRect = rootElement.getBoundingClientRect();
|
||
|
const computedStyle = getComputedStyle(rootElement);
|
||
|
const rootPadding =
|
||
|
parseFloat(computedStyle.paddingLeft) +
|
||
|
parseFloat(computedStyle.paddingRight);
|
||
|
const selectionRects = Array.from(range.getClientRects());
|
||
|
let selectionRectsLength = selectionRects.length;
|
||
|
//sort rects from top left to bottom right.
|
||
|
selectionRects.sort((a, b) => {
|
||
|
const top = a.top - b.top;
|
||
|
// Some rects match position closely, but not perfectly,
|
||
|
// so we give a 3px tolerance.
|
||
|
if (Math.abs(top) <= 3) {
|
||
|
return a.left - b.left;
|
||
|
}
|
||
|
return top;
|
||
|
});
|
||
|
let prevRect;
|
||
|
for (let i = 0; i < selectionRectsLength; i++) {
|
||
|
const selectionRect = selectionRects[i];
|
||
|
// Exclude rects that overlap preceding Rects in the sorted list.
|
||
|
const isOverlappingRect =
|
||
|
prevRect &&
|
||
|
prevRect.top <= selectionRect.top &&
|
||
|
prevRect.top + prevRect.height > selectionRect.top &&
|
||
|
prevRect.left + prevRect.width > selectionRect.left;
|
||
|
// Exclude selections that span the entire element
|
||
|
const selectionSpansElement =
|
||
|
selectionRect.width + rootPadding === rootRect.width;
|
||
|
if (isOverlappingRect || selectionSpansElement) {
|
||
|
selectionRects.splice(i--, 1);
|
||
|
selectionRectsLength--;
|
||
|
continue;
|
||
|
}
|
||
|
prevRect = selectionRect;
|
||
|
}
|
||
|
return selectionRects;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates an object containing all the styles and their values provided in the CSS string.
|
||
|
* @param css - The CSS string of styles and their values.
|
||
|
* @returns The styleObject containing all the styles and their values.
|
||
|
*/
|
||
|
export function getStyleObjectFromRawCSS(css: string): Record<string, string> {
|
||
|
const styleObject: Record<string, string> = {};
|
||
|
const styles = css.split(';');
|
||
|
|
||
|
for (const style of styles) {
|
||
|
if (style !== '') {
|
||
|
const [key, value] = style.split(/:([^]+)/); // split on first colon
|
||
|
if (key && value) {
|
||
|
styleObject[key.trim()] = value.trim();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return styleObject;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Given a CSS string, returns an object from the style cache.
|
||
|
* @param css - The CSS property as a string.
|
||
|
* @returns The value of the given CSS property.
|
||
|
*/
|
||
|
export function getStyleObjectFromCSS(css: string): Record<string, string> {
|
||
|
let value = CSS_TO_STYLES.get(css);
|
||
|
if (value === undefined) {
|
||
|
value = getStyleObjectFromRawCSS(css);
|
||
|
CSS_TO_STYLES.set(css, value);
|
||
|
}
|
||
|
|
||
|
if (__DEV__) {
|
||
|
// Freeze the value in DEV to prevent accidental mutations
|
||
|
Object.freeze(value);
|
||
|
}
|
||
|
|
||
|
return value;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the CSS styles from the style object.
|
||
|
* @param styles - The style object containing the styles to get.
|
||
|
* @returns A string containing the CSS styles and their values.
|
||
|
*/
|
||
|
export function getCSSFromStyleObject(styles: Record<string, string>): string {
|
||
|
let css = '';
|
||
|
|
||
|
for (const style in styles) {
|
||
|
if (style) {
|
||
|
css += `${style}: ${styles[style]};`;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return css;
|
||
|
}
|