0
Fork 0
mirror of https://github.com/fastmail/Squire.git synced 2025-01-05 06:10:07 -05:00
Squire/source/node/Node.ts
Neil Jenkins fe0dfdf6c4 Squire 2.0
This is a massive refactor to port Squire to TypeScript, fix a bunch of small
bugs and modernise our tooling. The development was done on an internal
repository, so apologies to anyone following externally for the commit dump;
updates from here should come as real commits again.

Co-authored-by: Joe Woods <woods@fastmailteam.com>
2023-01-23 13:18:29 +11:00

163 lines
3.7 KiB
TypeScript

import { isLeaf } from './Category';
// ---
const createElement = (
tag: string,
props?: Record<string, string> | null,
children?: Node[],
): HTMLElement => {
const el = document.createElement(tag);
if (props instanceof Array) {
children = props;
props = null;
}
if (props) {
for (const attr in props) {
const value = props[attr];
if (value !== undefined) {
el.setAttribute(attr, value);
}
}
}
if (children) {
children.forEach((node) => el.appendChild(node));
}
return el;
};
// --- Tests
const areAlike = (
node: HTMLElement | Node,
node2: HTMLElement | Node,
): boolean => {
if (isLeaf(node)) {
return false;
}
if (node.nodeType !== node2.nodeType || node.nodeName !== node2.nodeName) {
return false;
}
if (node instanceof HTMLElement && node2 instanceof HTMLElement) {
return (
node.nodeName !== 'A' &&
node.className === node2.className &&
node.style.cssText === node2.style.cssText
);
}
return true;
};
const hasTagAttributes = (
node: Node | Element,
tag: string,
attributes?: Record<string, string> | null,
): boolean => {
if (node.nodeName !== tag) {
return false;
}
for (const attr in attributes) {
if (
!('getAttribute' in node) ||
node.getAttribute(attr) !== attributes[attr]
) {
return false;
}
}
return true;
};
// --- Traversal
const getNearest = (
node: Node | null,
root: Element | DocumentFragment,
tag: string,
attributes?: Record<string, string> | null,
): Node | null => {
while (node && node !== root) {
if (hasTagAttributes(node, tag, attributes)) {
return node;
}
node = node.parentNode;
}
return null;
};
const getNodeBeforeOffset = (node: Node, offset: number): Node => {
let children = node.childNodes;
while (offset && node instanceof Element) {
node = children[offset - 1];
children = node.childNodes;
offset = children.length;
}
return node;
};
const getNodeAfterOffset = (node: Node, offset: number): Node | null => {
let returnNode: Node | null = node;
if (returnNode instanceof Element) {
const children = returnNode.childNodes;
if (offset < children.length) {
returnNode = children[offset];
} else {
while (returnNode && !returnNode.nextSibling) {
returnNode = returnNode.parentNode;
}
if (returnNode) {
returnNode = returnNode.nextSibling;
}
}
}
return returnNode;
};
const getLength = (node: Node): number => {
return node instanceof Element || node instanceof DocumentFragment
? node.childNodes.length
: node instanceof CharacterData
? node.length
: 0;
};
// --- Manipulation
const empty = (node: Node): DocumentFragment => {
const frag = document.createDocumentFragment();
let child = node.firstChild;
while (child) {
frag.appendChild(child);
child = node.firstChild;
}
return frag;
};
const detach = (node: Node): Node => {
const parent = node.parentNode;
if (parent) {
parent.removeChild(node);
}
return node;
};
const replaceWith = (node: Node, node2: Node): void => {
const parent = node.parentNode;
if (parent) {
parent.replaceChild(node2, node);
}
};
// --- Export
export {
areAlike,
createElement,
detach,
empty,
getLength,
getNearest,
getNodeAfterOffset,
getNodeBeforeOffset,
hasTagAttributes,
replaceWith,
};