|
import { ElementType, isTag as isTagRaw } from "domelementtype"; |
|
|
|
|
|
|
|
|
|
export class Node { |
|
constructor() { |
|
|
|
this.parent = null; |
|
|
|
this.prev = null; |
|
|
|
this.next = null; |
|
|
|
this.startIndex = null; |
|
|
|
this.endIndex = null; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
get parentNode() { |
|
return this.parent; |
|
} |
|
set parentNode(parent) { |
|
this.parent = parent; |
|
} |
|
|
|
|
|
|
|
|
|
get previousSibling() { |
|
return this.prev; |
|
} |
|
set previousSibling(prev) { |
|
this.prev = prev; |
|
} |
|
|
|
|
|
|
|
|
|
get nextSibling() { |
|
return this.next; |
|
} |
|
set nextSibling(next) { |
|
this.next = next; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
cloneNode(recursive = false) { |
|
return cloneNode(this, recursive); |
|
} |
|
} |
|
|
|
|
|
|
|
export class DataNode extends Node { |
|
|
|
|
|
|
|
constructor(data) { |
|
super(); |
|
this.data = data; |
|
} |
|
|
|
|
|
|
|
|
|
get nodeValue() { |
|
return this.data; |
|
} |
|
set nodeValue(data) { |
|
this.data = data; |
|
} |
|
} |
|
|
|
|
|
|
|
export class Text extends DataNode { |
|
constructor() { |
|
super(...arguments); |
|
this.type = ElementType.Text; |
|
} |
|
get nodeType() { |
|
return 3; |
|
} |
|
} |
|
|
|
|
|
|
|
export class Comment extends DataNode { |
|
constructor() { |
|
super(...arguments); |
|
this.type = ElementType.Comment; |
|
} |
|
get nodeType() { |
|
return 8; |
|
} |
|
} |
|
|
|
|
|
|
|
export class ProcessingInstruction extends DataNode { |
|
constructor(name, data) { |
|
super(data); |
|
this.name = name; |
|
this.type = ElementType.Directive; |
|
} |
|
get nodeType() { |
|
return 1; |
|
} |
|
} |
|
|
|
|
|
|
|
export class NodeWithChildren extends Node { |
|
|
|
|
|
|
|
constructor(children) { |
|
super(); |
|
this.children = children; |
|
} |
|
|
|
|
|
get firstChild() { |
|
var _a; |
|
return (_a = this.children[0]) !== null && _a !== void 0 ? _a : null; |
|
} |
|
|
|
get lastChild() { |
|
return this.children.length > 0 |
|
? this.children[this.children.length - 1] |
|
: null; |
|
} |
|
|
|
|
|
|
|
|
|
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; |
|
} |
|
} |
|
|
|
|
|
|
|
export class Document extends NodeWithChildren { |
|
constructor() { |
|
super(...arguments); |
|
this.type = ElementType.Root; |
|
} |
|
get nodeType() { |
|
return 9; |
|
} |
|
} |
|
|
|
|
|
|
|
export class Element extends NodeWithChildren { |
|
|
|
|
|
|
|
|
|
|
|
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; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
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], |
|
}); |
|
}); |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
export function isTag(node) { |
|
return isTagRaw(node); |
|
} |
|
|
|
|
|
|
|
|
|
export function isCDATA(node) { |
|
return node.type === ElementType.CDATA; |
|
} |
|
|
|
|
|
|
|
|
|
export function isText(node) { |
|
return node.type === ElementType.Text; |
|
} |
|
|
|
|
|
|
|
|
|
export function isComment(node) { |
|
return node.type === ElementType.Comment; |
|
} |
|
|
|
|
|
|
|
|
|
export function isDirective(node) { |
|
return node.type === ElementType.Directive; |
|
} |
|
|
|
|
|
|
|
|
|
export function isDocument(node) { |
|
return node.type === ElementType.Root; |
|
} |
|
|
|
|
|
|
|
|
|
export function hasChildren(node) { |
|
return Object.prototype.hasOwnProperty.call(node, "children"); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
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; |
|
} |
|
|