123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338 |
- import { ElementType, isTag as isTagRaw } from "domelementtype";
- /**
- * This object will be used as the prototype for Nodes when creating a
- * DOM-Level-1-compliant structure.
- */
- export class Node {
- constructor() {
- /** Parent of the node */
- this.parent = null;
- /** Previous sibling */
- this.prev = null;
- /** Next sibling */
- this.next = null;
- /** The start index of the node. Requires `withStartIndices` on the handler to be `true. */
- this.startIndex = null;
- /** The end index of the node. Requires `withEndIndices` on the handler to be `true. */
- this.endIndex = null;
- }
- // Read-write aliases for properties
- /**
- * Same as {@link parent}.
- * [DOM spec](https://dom.spec.whatwg.org)-compatible alias.
- */
- get parentNode() {
- return this.parent;
- }
- set parentNode(parent) {
- this.parent = parent;
- }
- /**
- * Same as {@link prev}.
- * [DOM spec](https://dom.spec.whatwg.org)-compatible alias.
- */
- get previousSibling() {
- return this.prev;
- }
- set previousSibling(prev) {
- this.prev = prev;
- }
- /**
- * Same as {@link next}.
- * [DOM spec](https://dom.spec.whatwg.org)-compatible alias.
- */
- get nextSibling() {
- return this.next;
- }
- set nextSibling(next) {
- this.next = next;
- }
- /**
- * Clone this node, and optionally its children.
- *
- * @param recursive Clone child nodes as well.
- * @returns A clone of the node.
- */
- cloneNode(recursive = false) {
- return cloneNode(this, recursive);
- }
- }
- /**
- * A node that contains some data.
- */
- export class DataNode extends Node {
- /**
- * @param data The content of the data node
- */
- constructor(data) {
- super();
- this.data = data;
- }
- /**
- * Same as {@link data}.
- * [DOM spec](https://dom.spec.whatwg.org)-compatible alias.
- */
- get nodeValue() {
- return this.data;
- }
- set nodeValue(data) {
- this.data = data;
- }
- }
- /**
- * Text within the document.
- */
- export class Text extends DataNode {
- constructor() {
- super(...arguments);
- this.type = ElementType.Text;
- }
- get nodeType() {
- return 3;
- }
- }
- /**
- * Comments within the document.
- */
- export class Comment extends DataNode {
- constructor() {
- super(...arguments);
- this.type = ElementType.Comment;
- }
- get nodeType() {
- return 8;
- }
- }
- /**
- * Processing instructions, including doc types.
- */
- export class ProcessingInstruction extends DataNode {
- constructor(name, data) {
- super(data);
- this.name = name;
- this.type = ElementType.Directive;
- }
- get nodeType() {
- return 1;
- }
- }
- /**
- * A `Node` that can have children.
- */
- export class NodeWithChildren extends Node {
- /**
- * @param children Children of the node. Only certain node types can have children.
- */
- constructor(children) {
- super();
- this.children = children;
- }
- // Aliases
- /** First child of the node. */
- get firstChild() {
- var _a;
- return (_a = this.children[0]) !== null && _a !== void 0 ? _a : null;
- }
- /** Last child of the node. */
- get lastChild() {
- return this.children.length > 0
- ? this.children[this.children.length - 1]
- : null;
- }
- /**
- * Same as {@link children}.
- * [DOM spec](https://dom.spec.whatwg.org)-compatible alias.
- */
- get childNodes() {
- return this.children;
- }
- set childNodes(children) {
- this.children = children;
- }
- }
- export class CDATA extends NodeWithChildren {
- constructor() {
- super(...arguments);
- this.type = ElementType.CDATA;
- }
- get nodeType() {
- return 4;
- }
- }
- /**
- * The root node of the document.
- */
- export class Document extends NodeWithChildren {
- constructor() {
- super(...arguments);
- this.type = ElementType.Root;
- }
- get nodeType() {
- return 9;
- }
- }
- /**
- * An element within the DOM.
- */
- export class Element extends NodeWithChildren {
- /**
- * @param name Name of the tag, eg. `div`, `span`.
- * @param attribs Object mapping attribute names to attribute values.
- * @param children Children of the node.
- */
- constructor(name, attribs, children = [], type = name === "script"
- ? ElementType.Script
- : name === "style"
- ? ElementType.Style
- : ElementType.Tag) {
- super(children);
- this.name = name;
- this.attribs = attribs;
- this.type = type;
- }
- get nodeType() {
- return 1;
- }
- // DOM Level 1 aliases
- /**
- * Same as {@link name}.
- * [DOM spec](https://dom.spec.whatwg.org)-compatible alias.
- */
- get tagName() {
- return this.name;
- }
- set tagName(name) {
- this.name = name;
- }
- get attributes() {
- return Object.keys(this.attribs).map((name) => {
- var _a, _b;
- return ({
- name,
- value: this.attribs[name],
- namespace: (_a = this["x-attribsNamespace"]) === null || _a === void 0 ? void 0 : _a[name],
- prefix: (_b = this["x-attribsPrefix"]) === null || _b === void 0 ? void 0 : _b[name],
- });
- });
- }
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node is a `Element`, `false` otherwise.
- */
- export function isTag(node) {
- return isTagRaw(node);
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node has the type `CDATA`, `false` otherwise.
- */
- export function isCDATA(node) {
- return node.type === ElementType.CDATA;
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node has the type `Text`, `false` otherwise.
- */
- export function isText(node) {
- return node.type === ElementType.Text;
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node has the type `Comment`, `false` otherwise.
- */
- export function isComment(node) {
- return node.type === ElementType.Comment;
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node has the type `ProcessingInstruction`, `false` otherwise.
- */
- export function isDirective(node) {
- return node.type === ElementType.Directive;
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node has the type `ProcessingInstruction`, `false` otherwise.
- */
- export function isDocument(node) {
- return node.type === ElementType.Root;
- }
- /**
- * @param node Node to check.
- * @returns `true` if the node has children, `false` otherwise.
- */
- export function hasChildren(node) {
- return Object.prototype.hasOwnProperty.call(node, "children");
- }
- /**
- * Clone a node, and optionally its children.
- *
- * @param recursive Clone child nodes as well.
- * @returns A clone of the node.
- */
- export function cloneNode(node, recursive = false) {
- let result;
- if (isText(node)) {
- result = new Text(node.data);
- }
- else if (isComment(node)) {
- result = new Comment(node.data);
- }
- else if (isTag(node)) {
- const children = recursive ? cloneChildren(node.children) : [];
- const clone = new Element(node.name, { ...node.attribs }, children);
- children.forEach((child) => (child.parent = clone));
- if (node.namespace != null) {
- clone.namespace = node.namespace;
- }
- if (node["x-attribsNamespace"]) {
- clone["x-attribsNamespace"] = { ...node["x-attribsNamespace"] };
- }
- if (node["x-attribsPrefix"]) {
- clone["x-attribsPrefix"] = { ...node["x-attribsPrefix"] };
- }
- result = clone;
- }
- else if (isCDATA(node)) {
- const children = recursive ? cloneChildren(node.children) : [];
- const clone = new CDATA(children);
- children.forEach((child) => (child.parent = clone));
- result = clone;
- }
- else if (isDocument(node)) {
- const children = recursive ? cloneChildren(node.children) : [];
- const clone = new Document(children);
- children.forEach((child) => (child.parent = clone));
- if (node["x-mode"]) {
- clone["x-mode"] = node["x-mode"];
- }
- result = clone;
- }
- else if (isDirective(node)) {
- const instruction = new ProcessingInstruction(node.name, node.data);
- if (node["x-name"] != null) {
- instruction["x-name"] = node["x-name"];
- instruction["x-publicId"] = node["x-publicId"];
- instruction["x-systemId"] = node["x-systemId"];
- }
- result = instruction;
- }
- else {
- throw new Error(`Not implemented yet: ${node.type}`);
- }
- result.startIndex = node.startIndex;
- result.endIndex = node.endIndex;
- if (node.sourceCodeLocation != null) {
- result.sourceCodeLocation = node.sourceCodeLocation;
- }
- return result;
- }
- function cloneChildren(childs) {
- const children = childs.map((child) => cloneNode(child, true));
- for (let i = 1; i < children.length; i++) {
- children[i].prev = children[i - 1];
- children[i - 1].next = children[i];
- }
- return children;
- }
|