mirror of
https://github.com/fastmail/Squire.git
synced 2025-01-07 23:40:15 -05:00
443 lines
14 KiB
TypeScript
443 lines
14 KiB
TypeScript
|
import { cleanupBRs } from '../Clean';
|
|||
|
import {
|
|||
|
split,
|
|||
|
fixCursor,
|
|||
|
mergeWithBlock,
|
|||
|
fixContainer,
|
|||
|
mergeContainers,
|
|||
|
} from '../node/MergeSplit';
|
|||
|
import { detach, getNearest, getLength } from '../node/Node';
|
|||
|
import { TreeIterator, SHOW_ELEMENT_OR_TEXT } from '../node/TreeIterator';
|
|||
|
import { isInline, isContainer, isLeaf } from '../node/Category';
|
|||
|
import { getNextBlock, isEmptyBlock, getPreviousBlock } from '../node/Block';
|
|||
|
import {
|
|||
|
getStartBlockOfRange,
|
|||
|
getEndBlockOfRange,
|
|||
|
rangeDoesEndAtBlockBoundary,
|
|||
|
rangeDoesStartAtBlockBoundary,
|
|||
|
} from './Block';
|
|||
|
import {
|
|||
|
moveRangeBoundariesDownTree,
|
|||
|
moveRangeBoundariesUpTree,
|
|||
|
} from './Boundaries';
|
|||
|
|
|||
|
// ---
|
|||
|
|
|||
|
function createRange(startContainer: Node, startOffset: number): Range;
|
|||
|
function createRange(
|
|||
|
startContainer: Node,
|
|||
|
startOffset: number,
|
|||
|
endContainer: Node,
|
|||
|
endOffset: number,
|
|||
|
): Range;
|
|||
|
function createRange(
|
|||
|
startContainer: Node,
|
|||
|
startOffset: number,
|
|||
|
endContainer?: Node,
|
|||
|
endOffset?: number,
|
|||
|
): Range {
|
|||
|
const range = document.createRange();
|
|||
|
range.setStart(startContainer, startOffset);
|
|||
|
if (endContainer && typeof endOffset === 'number') {
|
|||
|
range.setEnd(endContainer, endOffset);
|
|||
|
} else {
|
|||
|
range.setEnd(startContainer, startOffset);
|
|||
|
}
|
|||
|
return range;
|
|||
|
}
|
|||
|
|
|||
|
const insertNodeInRange = (range: Range, node: Node): void => {
|
|||
|
// Insert at start.
|
|||
|
let { startContainer, startOffset, endContainer, endOffset } = range;
|
|||
|
let children: NodeListOf<ChildNode>;
|
|||
|
|
|||
|
// If part way through a text node, split it.
|
|||
|
if (startContainer instanceof Text) {
|
|||
|
const parent = startContainer.parentNode!;
|
|||
|
children = parent.childNodes;
|
|||
|
if (startOffset === startContainer.length) {
|
|||
|
startOffset = Array.from(children).indexOf(startContainer) + 1;
|
|||
|
if (range.collapsed) {
|
|||
|
endContainer = parent;
|
|||
|
endOffset = startOffset;
|
|||
|
}
|
|||
|
} else {
|
|||
|
if (startOffset) {
|
|||
|
const afterSplit = startContainer.splitText(startOffset);
|
|||
|
if (endContainer === startContainer) {
|
|||
|
endOffset -= startOffset;
|
|||
|
endContainer = afterSplit;
|
|||
|
} else if (endContainer === parent) {
|
|||
|
endOffset += 1;
|
|||
|
}
|
|||
|
startContainer = afterSplit;
|
|||
|
}
|
|||
|
startOffset = Array.from(children).indexOf(
|
|||
|
startContainer as ChildNode,
|
|||
|
);
|
|||
|
}
|
|||
|
startContainer = parent;
|
|||
|
} else {
|
|||
|
children = startContainer.childNodes;
|
|||
|
}
|
|||
|
|
|||
|
const childCount = children.length;
|
|||
|
|
|||
|
if (startOffset === childCount) {
|
|||
|
startContainer.appendChild(node);
|
|||
|
} else {
|
|||
|
startContainer.insertBefore(node, children[startOffset]);
|
|||
|
}
|
|||
|
|
|||
|
if (startContainer === endContainer) {
|
|||
|
endOffset += children.length - childCount;
|
|||
|
}
|
|||
|
|
|||
|
range.setStart(startContainer, startOffset);
|
|||
|
range.setEnd(endContainer, endOffset);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Removes the contents of the range and returns it as a DocumentFragment.
|
|||
|
* The range at the end will be at the same position, with the edges just
|
|||
|
* before/after the split. If the start/end have the same parents, it will
|
|||
|
* be collapsed.
|
|||
|
*/
|
|||
|
const extractContentsOfRange = (
|
|||
|
range: Range,
|
|||
|
common: Node | null,
|
|||
|
root: Element,
|
|||
|
): DocumentFragment => {
|
|||
|
const frag = document.createDocumentFragment();
|
|||
|
if (range.collapsed) {
|
|||
|
return frag;
|
|||
|
}
|
|||
|
|
|||
|
if (!common) {
|
|||
|
common = range.commonAncestorContainer;
|
|||
|
}
|
|||
|
if (common instanceof Text) {
|
|||
|
common = common.parentNode!;
|
|||
|
}
|
|||
|
|
|||
|
const startContainer = range.startContainer;
|
|||
|
const startOffset = range.startOffset;
|
|||
|
|
|||
|
let endContainer = split(range.endContainer, range.endOffset, common, root);
|
|||
|
let endOffset = 0;
|
|||
|
|
|||
|
let node = split(startContainer, startOffset, common, root);
|
|||
|
while (node && node !== endContainer) {
|
|||
|
const next = node.nextSibling;
|
|||
|
frag.appendChild(node);
|
|||
|
node = next;
|
|||
|
}
|
|||
|
|
|||
|
// Merge text nodes if adjacent
|
|||
|
if (startContainer instanceof Text && endContainer instanceof Text) {
|
|||
|
startContainer.appendData(endContainer.data);
|
|||
|
detach(endContainer);
|
|||
|
endContainer = startContainer;
|
|||
|
endOffset = startOffset;
|
|||
|
}
|
|||
|
|
|||
|
range.setStart(startContainer, startOffset);
|
|||
|
if (endContainer) {
|
|||
|
range.setEnd(endContainer, endOffset);
|
|||
|
} else {
|
|||
|
// endContainer will be null if at end of parent's child nodes list.
|
|||
|
range.setEnd(common, common.childNodes.length);
|
|||
|
}
|
|||
|
|
|||
|
fixCursor(common);
|
|||
|
|
|||
|
return frag;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the next/prev node that's part of the same inline content.
|
|||
|
*/
|
|||
|
const getAdjacentInlineNode = (
|
|||
|
iterator: TreeIterator<Node>,
|
|||
|
method: 'nextNode' | 'previousPONode',
|
|||
|
node: Node,
|
|||
|
): Node | null => {
|
|||
|
iterator.currentNode = node;
|
|||
|
let nextNode: Node | null;
|
|||
|
while ((nextNode = iterator[method]())) {
|
|||
|
if (nextNode instanceof Text || isLeaf(nextNode)) {
|
|||
|
return nextNode;
|
|||
|
}
|
|||
|
if (!isInline(nextNode)) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
}
|
|||
|
return null;
|
|||
|
};
|
|||
|
|
|||
|
const deleteContentsOfRange = (
|
|||
|
range: Range,
|
|||
|
root: Element,
|
|||
|
): DocumentFragment => {
|
|||
|
const startBlock = getStartBlockOfRange(range, root);
|
|||
|
let endBlock = getEndBlockOfRange(range, root);
|
|||
|
const needsMerge = startBlock !== endBlock;
|
|||
|
|
|||
|
// Move boundaries up as much as possible without exiting block,
|
|||
|
// to reduce need to split.
|
|||
|
if (startBlock && endBlock) {
|
|||
|
moveRangeBoundariesDownTree(range);
|
|||
|
moveRangeBoundariesUpTree(range, startBlock, endBlock, root);
|
|||
|
}
|
|||
|
|
|||
|
// Remove selected range
|
|||
|
const frag = extractContentsOfRange(range, null, root);
|
|||
|
|
|||
|
// Move boundaries back down tree as far as possible.
|
|||
|
moveRangeBoundariesDownTree(range);
|
|||
|
|
|||
|
// If we split into two different blocks, merge the blocks.
|
|||
|
if (needsMerge) {
|
|||
|
// endBlock will have been split, so need to refetch
|
|||
|
endBlock = getEndBlockOfRange(range, root);
|
|||
|
if (startBlock && endBlock && startBlock !== endBlock) {
|
|||
|
mergeWithBlock(startBlock, endBlock, range, root);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Ensure block has necessary children
|
|||
|
if (startBlock) {
|
|||
|
fixCursor(startBlock);
|
|||
|
}
|
|||
|
|
|||
|
// Ensure root has a block-level element in it.
|
|||
|
const child = root.firstChild;
|
|||
|
if (!child || child.nodeName === 'BR') {
|
|||
|
fixCursor(root);
|
|||
|
if (root.firstChild) {
|
|||
|
range.selectNodeContents(root.firstChild);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
range.collapse(true);
|
|||
|
|
|||
|
// Now we may need to swap a space for a nbsp if the browser is going
|
|||
|
// to swallow it due to HTML whitespace rules:
|
|||
|
const startContainer = range.startContainer;
|
|||
|
const startOffset = range.startOffset;
|
|||
|
const iterator = new TreeIterator(root, SHOW_ELEMENT_OR_TEXT);
|
|||
|
|
|||
|
// Find the character after cursor point
|
|||
|
let afterNode: Node | null = startContainer;
|
|||
|
let afterOffset = startOffset;
|
|||
|
if (!(afterNode instanceof Text) || afterOffset === afterNode.data.length) {
|
|||
|
afterNode = getAdjacentInlineNode(iterator, 'nextNode', afterNode);
|
|||
|
afterOffset = 0;
|
|||
|
}
|
|||
|
|
|||
|
// Find the character before cursor point
|
|||
|
let beforeNode: Node | null = startContainer;
|
|||
|
let beforeOffset = startOffset - 1;
|
|||
|
if (!(beforeNode instanceof Text) || beforeOffset === -1) {
|
|||
|
beforeNode = getAdjacentInlineNode(
|
|||
|
iterator,
|
|||
|
'previousPONode',
|
|||
|
afterNode ||
|
|||
|
(startContainer instanceof Text
|
|||
|
? startContainer
|
|||
|
: startContainer.childNodes[startOffset] || startContainer),
|
|||
|
);
|
|||
|
if (beforeNode instanceof Text) {
|
|||
|
beforeOffset = beforeNode.data.length;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// If range starts at block boundary and character after cursor point
|
|||
|
// is a space, replace with nbsp
|
|||
|
let node = null;
|
|||
|
let offset = 0;
|
|||
|
if (
|
|||
|
afterNode instanceof Text &&
|
|||
|
afterNode.data.charAt(afterOffset) === ' ' &&
|
|||
|
rangeDoesStartAtBlockBoundary(range, root)
|
|||
|
) {
|
|||
|
node = afterNode;
|
|||
|
offset = afterOffset;
|
|||
|
} else if (
|
|||
|
beforeNode instanceof Text &&
|
|||
|
beforeNode.data.charAt(beforeOffset) === ' '
|
|||
|
) {
|
|||
|
// If character before cursor point is a space, replace with nbsp
|
|||
|
// if either:
|
|||
|
// a) There is a space after it; or
|
|||
|
// b) The point after is the end of the block
|
|||
|
if (
|
|||
|
(afterNode instanceof Text &&
|
|||
|
afterNode.data.charAt(afterOffset) === ' ') ||
|
|||
|
rangeDoesEndAtBlockBoundary(range, root)
|
|||
|
) {
|
|||
|
node = beforeNode;
|
|||
|
offset = beforeOffset;
|
|||
|
}
|
|||
|
}
|
|||
|
if (node) {
|
|||
|
node.replaceData(offset, 1, ' '); // nbsp
|
|||
|
}
|
|||
|
// Range needs to be put back in place
|
|||
|
range.setStart(startContainer, startOffset);
|
|||
|
range.collapse(true);
|
|||
|
|
|||
|
return frag;
|
|||
|
};
|
|||
|
|
|||
|
// Contents of range will be deleted.
|
|||
|
// After method, range will be around inserted content
|
|||
|
const insertTreeFragmentIntoRange = (
|
|||
|
range: Range,
|
|||
|
frag: DocumentFragment,
|
|||
|
root: Element,
|
|||
|
): void => {
|
|||
|
const firstInFragIsInline = frag.firstChild && isInline(frag.firstChild);
|
|||
|
let node: Node | null;
|
|||
|
|
|||
|
// Fixup content: ensure no top-level inline, and add cursor fix elements.
|
|||
|
fixContainer(frag, root);
|
|||
|
node = frag;
|
|||
|
while ((node = getNextBlock(node, root))) {
|
|||
|
fixCursor(node);
|
|||
|
}
|
|||
|
|
|||
|
// Delete any selected content.
|
|||
|
if (!range.collapsed) {
|
|||
|
deleteContentsOfRange(range, root);
|
|||
|
}
|
|||
|
|
|||
|
// Move range down into text nodes.
|
|||
|
moveRangeBoundariesDownTree(range);
|
|||
|
range.collapse(false); // collapse to end
|
|||
|
|
|||
|
// Where will we split up to? First blockquote parent, otherwise root.
|
|||
|
const stopPoint =
|
|||
|
getNearest(range.endContainer, root, 'BLOCKQUOTE') || root;
|
|||
|
|
|||
|
// Merge the contents of the first block in the frag with the focused block.
|
|||
|
// If there are contents in the block after the focus point, collect this
|
|||
|
// up to insert in the last block later. This preserves the style that was
|
|||
|
// present in this bit of the page.
|
|||
|
//
|
|||
|
// If the block being inserted into is empty though, replace it instead of
|
|||
|
// merging if the fragment had block contents.
|
|||
|
// e.g. <blockquote><p>Foo</p></blockquote>
|
|||
|
// This seems a reasonable approximation of user intent.
|
|||
|
let block = getStartBlockOfRange(range, root);
|
|||
|
let blockContentsAfterSplit: DocumentFragment | null = null;
|
|||
|
const firstBlockInFrag = getNextBlock(frag, frag);
|
|||
|
const replaceBlock = !firstInFragIsInline && !!block && isEmptyBlock(block);
|
|||
|
if (
|
|||
|
block &&
|
|||
|
firstBlockInFrag &&
|
|||
|
!replaceBlock &&
|
|||
|
// Don't merge table cells or PRE elements into block
|
|||
|
!getNearest(firstBlockInFrag, frag, 'PRE') &&
|
|||
|
!getNearest(firstBlockInFrag, frag, 'TABLE')
|
|||
|
) {
|
|||
|
moveRangeBoundariesUpTree(range, block, block, root);
|
|||
|
range.collapse(true); // collapse to start
|
|||
|
let container = range.endContainer;
|
|||
|
let offset = range.endOffset;
|
|||
|
// Remove trailing <br> – we don't want this considered content to be
|
|||
|
// inserted again later
|
|||
|
cleanupBRs(block as HTMLElement, root, false);
|
|||
|
if (isInline(container)) {
|
|||
|
// Split up to block parent.
|
|||
|
const nodeAfterSplit = split(
|
|||
|
container,
|
|||
|
offset,
|
|||
|
getPreviousBlock(container, root) || root,
|
|||
|
root,
|
|||
|
) as Node;
|
|||
|
container = nodeAfterSplit.parentNode!;
|
|||
|
offset = Array.from(container.childNodes).indexOf(
|
|||
|
nodeAfterSplit as ChildNode,
|
|||
|
);
|
|||
|
}
|
|||
|
if (/*isBlock( container ) && */ offset !== getLength(container)) {
|
|||
|
// Collect any inline contents of the block after the range point
|
|||
|
blockContentsAfterSplit = document.createDocumentFragment();
|
|||
|
while ((node = container.childNodes[offset])) {
|
|||
|
blockContentsAfterSplit.appendChild(node);
|
|||
|
}
|
|||
|
}
|
|||
|
// And merge the first block in.
|
|||
|
mergeWithBlock(container, firstBlockInFrag, range, root);
|
|||
|
|
|||
|
// And where we will insert
|
|||
|
offset =
|
|||
|
Array.from(container.parentNode!.childNodes).indexOf(
|
|||
|
container as ChildNode,
|
|||
|
) + 1;
|
|||
|
container = container.parentNode!;
|
|||
|
range.setEnd(container, offset);
|
|||
|
}
|
|||
|
|
|||
|
// Is there still any content in the fragment?
|
|||
|
if (getLength(frag)) {
|
|||
|
if (replaceBlock && block) {
|
|||
|
range.setEndBefore(block);
|
|||
|
range.collapse(false);
|
|||
|
detach(block);
|
|||
|
}
|
|||
|
moveRangeBoundariesUpTree(range, stopPoint, stopPoint, root);
|
|||
|
// Now split after block up to blockquote (if a parent) or root
|
|||
|
let nodeAfterSplit = split(
|
|||
|
range.endContainer,
|
|||
|
range.endOffset,
|
|||
|
stopPoint,
|
|||
|
root,
|
|||
|
) as Node | null;
|
|||
|
const nodeBeforeSplit = nodeAfterSplit
|
|||
|
? nodeAfterSplit.previousSibling
|
|||
|
: stopPoint.lastChild;
|
|||
|
stopPoint.insertBefore(frag, nodeAfterSplit);
|
|||
|
if (nodeAfterSplit) {
|
|||
|
range.setEndBefore(nodeAfterSplit);
|
|||
|
} else {
|
|||
|
range.setEnd(stopPoint, getLength(stopPoint));
|
|||
|
}
|
|||
|
block = getEndBlockOfRange(range, root);
|
|||
|
|
|||
|
// Get a reference that won't be invalidated if we merge containers.
|
|||
|
moveRangeBoundariesDownTree(range);
|
|||
|
const container = range.endContainer;
|
|||
|
const offset = range.endOffset;
|
|||
|
|
|||
|
// Merge inserted containers with edges of split
|
|||
|
if (nodeAfterSplit && isContainer(nodeAfterSplit)) {
|
|||
|
mergeContainers(nodeAfterSplit, root);
|
|||
|
}
|
|||
|
nodeAfterSplit = nodeBeforeSplit && nodeBeforeSplit.nextSibling;
|
|||
|
if (nodeAfterSplit && isContainer(nodeAfterSplit)) {
|
|||
|
mergeContainers(nodeAfterSplit, root);
|
|||
|
}
|
|||
|
range.setEnd(container, offset);
|
|||
|
}
|
|||
|
|
|||
|
// Insert inline content saved from before.
|
|||
|
if (blockContentsAfterSplit && block) {
|
|||
|
const tempRange = range.cloneRange();
|
|||
|
mergeWithBlock(block, blockContentsAfterSplit, tempRange, root);
|
|||
|
range.setEnd(tempRange.endContainer, tempRange.endOffset);
|
|||
|
}
|
|||
|
moveRangeBoundariesDownTree(range);
|
|||
|
};
|
|||
|
|
|||
|
// ---
|
|||
|
|
|||
|
export {
|
|||
|
createRange,
|
|||
|
deleteContentsOfRange,
|
|||
|
extractContentsOfRange,
|
|||
|
insertNodeInRange,
|
|||
|
insertTreeFragmentIntoRange,
|
|||
|
};
|