123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603 |
- /*************************************************************
- *
- * Copyright (c) 2018-2022 The MathJax Consortium
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- /**
- * @fileoverview Implements a lightweight DOM adaptor
- *
- * @author dpvc@mathjax.org (Davide Cervone)
- */
- import {AbstractDOMAdaptor} from '../core/DOMAdaptor.js';
- import {NodeMixin, Constructor} from './NodeMixin.js';
- import {LiteDocument} from './lite/Document.js';
- import {LiteElement, LiteNode} from './lite/Element.js';
- import {LiteText, LiteComment} from './lite/Text.js';
- import {LiteList} from './lite/List.js';
- import {LiteWindow} from './lite/Window.js';
- import {LiteParser} from './lite/Parser.js';
- import {Styles} from '../util/Styles.js';
- import {OptionList} from '../util/Options.js';
- /************************************************************/
- /**
- * Implements a lightweight DOMAdaptor on liteweight HTML elements
- */
- export class LiteBase extends AbstractDOMAdaptor<LiteElement, LiteText, LiteDocument> {
- /**
- * The document in which the HTML nodes will be created
- */
- public document: LiteDocument;
- /**
- * The window for the document
- */
- public window: LiteWindow;
- /**
- * The parser for serialized HTML
- */
- public parser: LiteParser;
- /**
- * @param {OptionList} options The options for the lite adaptor (e.g., fontSize)
- * @constructor
- */
- constructor() {
- super();
- this.parser = new LiteParser();
- this.window = new LiteWindow();
- }
- /**
- * @override
- */
- public parse(text: string, format?: string): LiteDocument {
- return this.parser.parseFromString(text, format, this);
- }
- /**
- * @override
- */
- protected create(kind: string, _ns: string = null) {
- return new LiteElement(kind);
- }
- /**
- * @override
- */
- public text(text: string) {
- return new LiteText(text);
- }
- /**
- * @param {string} text The text of the comment
- * @return {LiteComment} The comment node
- */
- public comment(text: string): LiteComment {
- return new LiteComment(text);
- }
- /**
- * @return {LiteDocument} A new document element
- */
- public createDocument(): LiteDocument {
- return new LiteDocument();
- }
- /**
- * @override
- */
- public head(doc: LiteDocument) {
- return doc.head;
- }
- /**
- * @override
- */
- public body(doc: LiteDocument) {
- return doc.body;
- }
- /**
- * @override
- */
- public root(doc: LiteDocument) {
- return doc.root;
- }
- /**
- * @override
- */
- public doctype(doc: LiteDocument) {
- return doc.type;
- }
- /**
- * @override
- */
- public tags(node: LiteElement, name: string, ns: string = null) {
- let stack = [] as LiteNode[];
- let tags = [] as LiteElement[];
- if (ns) {
- return tags; // we don't have namespaces
- }
- let n: LiteNode = node;
- while (n) {
- let kind = n.kind;
- if (kind !== '#text' && kind !== '#comment') {
- n = n as LiteElement;
- if (kind === name) {
- tags.push(n);
- }
- if (n.children.length) {
- stack = n.children.concat(stack);
- }
- }
- n = stack.shift();
- }
- return tags;
- }
- /**
- * @param {LiteElement} node The node to be searched
- * @param {string} id The id of the node to look for
- * @return {LiteElement} The child node having the given id
- */
- public elementById(node: LiteElement, id: string): LiteElement {
- let stack = [] as LiteNode[];
- let n = node as LiteNode;
- while (n) {
- if (n.kind !== '#text' && n.kind !== '#comment') {
- n = n as LiteElement;
- if (n.attributes['id'] === id) {
- return n;
- }
- if (n.children.length) {
- stack = n.children.concat(stack);
- }
- }
- n = stack.shift();
- }
- return null;
- }
- /**
- * @param {LiteElement} node The node to be searched
- * @param {string} name The name of the class to find
- * @return {LiteElement[]} The nodes with the given class
- */
- public elementsByClass(node: LiteElement, name: string): LiteElement[] {
- let stack = [] as LiteNode[];
- let tags = [] as LiteElement[];
- let n: LiteNode = node;
- while (n) {
- if (n.kind !== '#text' && n.kind !== '#comment') {
- n = n as LiteElement;
- const classes = (n.attributes['class'] || '').trim().split(/ +/);
- if (classes.includes(name)) {
- tags.push(n);
- }
- if (n.children.length) {
- stack = n.children.concat(stack);
- }
- }
- n = stack.shift();
- }
- return tags;
- }
- /**
- * @override
- */
- public getElements(nodes: (string | LiteElement | LiteElement[])[], document: LiteDocument) {
- let containers = [] as LiteElement[];
- const body = this.body(document);
- for (const node of nodes) {
- if (typeof(node) === 'string') {
- if (node.charAt(0) === '#') {
- const n = this.elementById(body, node.slice(1));
- if (n) {
- containers.push(n);
- }
- } else if (node.charAt(0) === '.') {
- containers = containers.concat(this.elementsByClass(body, node.slice(1)));
- } else if (node.match(/^[-a-z][-a-z0-9]*$/i)) {
- containers = containers.concat(this.tags(body, node));
- }
- } else if (Array.isArray(node)) {
- containers = containers.concat(node);
- } else if (node instanceof this.window.NodeList || node instanceof this.window.HTMLCollection) {
- containers = containers.concat((node as LiteList<LiteElement>).nodes);
- } else {
- containers.push(node);
- }
- }
- return containers;
- }
- /**
- * @override
- */
- public contains(container: LiteNode, node: LiteNode | LiteText) {
- while (node && node !== container) {
- node = this.parent(node);
- }
- return !!node;
- }
- /**
- * @override
- */
- public parent(node: LiteNode) {
- return node.parent;
- }
- /**
- * @param {LiteNode} node The node whose index is needed
- * @return {number} The index of the node it its parent's children array
- */
- public childIndex(node: LiteNode): number {
- return (node.parent ? node.parent.children.findIndex(n => n === node) : -1);
- }
- /**
- * @override
- */
- public append(node: LiteElement, child: LiteNode) {
- if (child.parent) {
- this.remove(child);
- }
- node.children.push(child);
- child.parent = node;
- return child;
- }
- /**
- * @override
- */
- public insert(nchild: LiteNode, ochild: LiteNode) {
- if (nchild.parent) {
- this.remove(nchild);
- }
- if (ochild && ochild.parent) {
- const i = this.childIndex(ochild);
- ochild.parent.children.splice(i, 0, nchild);
- nchild.parent = ochild.parent;
- }
- }
- /**
- * @override
- */
- public remove(child: LiteNode) {
- const i = this.childIndex(child);
- if (i >= 0) {
- child.parent.children.splice(i, 1);
- }
- child.parent = null;
- return child;
- }
- /**
- * @override
- */
- public replace(nnode: LiteNode, onode: LiteNode) {
- const i = this.childIndex(onode);
- if (i >= 0) {
- onode.parent.children[i] = nnode;
- nnode.parent = onode.parent;
- onode.parent = null;
- }
- return onode;
- }
- /**
- * @override
- */
- public clone(node: LiteElement) {
- const nnode = new LiteElement(node.kind);
- nnode.attributes = {...node.attributes};
- nnode.children = node.children.map(n => {
- if (n.kind === '#text') {
- return new LiteText((n as LiteText).value);
- } else if (n.kind === '#comment') {
- return new LiteComment((n as LiteComment).value);
- } else {
- const m = this.clone(n as LiteElement);
- m.parent = nnode;
- return m;
- }
- });
- return nnode;
- }
- /**
- * @override
- */
- public split(node: LiteText, n: number) {
- const text = new LiteText(node.value.slice(n));
- node.value = node.value.slice(0, n);
- node.parent.children.splice(this.childIndex(node) + 1, 0, text);
- text.parent = node.parent;
- return text;
- }
- /**
- * @override
- */
- public next(node: LiteNode) {
- const parent = node.parent;
- if (!parent) return null;
- const i = this.childIndex(node) + 1;
- return (i >= 0 && i < parent.children.length ? parent.children[i] : null);
- }
- /**
- * @override
- */
- public previous(node: LiteNode) {
- const parent = node.parent;
- if (!parent) return null;
- const i = this.childIndex(node) - 1;
- return (i >= 0 ? parent.children[i] : null);
- }
- /**
- * @override
- */
- public firstChild(node: LiteElement) {
- return node.children[0];
- }
- /**
- * @override
- */
- public lastChild(node: LiteElement) {
- return node.children[node.children.length - 1];
- }
- /**
- * @override
- */
- public childNodes(node: LiteElement) {
- return [...node.children];
- }
- /**
- * @override
- */
- public childNode(node: LiteElement, i: number) {
- return node.children[i];
- }
- /**
- * @override
- */
- public kind(node: LiteNode) {
- return node.kind;
- }
- /**
- * @override
- */
- public value(node: LiteNode | LiteText) {
- return (node.kind === '#text' ? (node as LiteText).value :
- node.kind === '#comment' ? (node as LiteComment).value.replace(/^<!(--)?((?:.|\n)*)\1>$/, '$2') : '');
- }
- /**
- * @override
- */
- public textContent(node: LiteElement): string {
- return node.children.reduce((s: string, n: LiteNode) => {
- return s + (n.kind === '#text' ? (n as LiteText).value :
- n.kind === '#comment' ? '' : this.textContent(n as LiteElement));
- }, '');
- }
- /**
- * @override
- */
- public innerHTML(node: LiteElement): string {
- return this.parser.serializeInner(this, node);
- }
- /**
- * @override
- */
- public outerHTML(node: LiteElement): string {
- return this.parser.serialize(this, node);
- }
- /**
- * @override
- */
- public serializeXML(node: LiteElement): string {
- return this.parser.serialize(this, node, true);
- }
- /**
- * @override
- */
- public setAttribute(node: LiteElement, name: string, value: string | number, ns: string = null) {
- if (typeof value !== 'string') {
- value = String(value);
- }
- if (ns) {
- name = ns.replace(/.*\//, '') + ':' + name.replace(/^.*:/, '');
- }
- node.attributes[name] = value;
- if (name === 'style') {
- node.styles = null;
- }
- }
- /**
- * @override
- */
- public getAttribute(node: LiteElement, name: string) {
- return node.attributes[name];
- }
- /**
- * @override
- */
- public removeAttribute(node: LiteElement, name: string) {
- delete node.attributes[name];
- }
- /**
- * @override
- */
- public hasAttribute(node: LiteElement, name: string) {
- return node.attributes.hasOwnProperty(name);
- }
- /**
- * @override
- */
- public allAttributes(node: LiteElement) {
- const attributes = node.attributes;
- const list = [];
- for (const name of Object.keys(attributes)) {
- list.push({name: name, value: attributes[name] as string});
- }
- return list;
- }
- /**
- * @override
- */
- public addClass(node: LiteElement, name: string) {
- const classes = (node.attributes['class'] as string || '').split(/ /);
- if (!classes.find(n => n === name)) {
- classes.push(name);
- node.attributes['class'] = classes.join(' ');
- }
- }
- /**
- * @override
- */
- public removeClass(node: LiteElement, name: string) {
- const classes = (node.attributes['class'] as string || '').split(/ /);
- const i = classes.findIndex(n => n === name);
- if (i >= 0) {
- classes.splice(i, 1);
- node.attributes['class'] = classes.join(' ');
- }
- }
- /**
- * @override
- */
- public hasClass(node: LiteElement, name: string) {
- const classes = (node.attributes['class'] as string || '').split(/ /);
- return !!classes.find(n => n === name);
- }
- /**
- * @override
- */
- public setStyle(node: LiteElement, name: string, value: string) {
- if (!node.styles) {
- node.styles = new Styles(this.getAttribute(node, 'style'));
- }
- node.styles.set(name, value);
- node.attributes['style'] = node.styles.cssText;
- }
- /**
- * @override
- */
- public getStyle(node: LiteElement, name: string) {
- if (!node.styles) {
- const style = this.getAttribute(node, 'style');
- if (!style) {
- return '';
- }
- node.styles = new Styles(style);
- }
- return node.styles.get(name);
- }
- /**
- * @override
- */
- public allStyles(node: LiteElement) {
- return this.getAttribute(node, 'style');
- }
- /**
- * @override
- */
- public insertRules(node: LiteElement, rules: string[]) {
- node.children = [this.text(rules.join('\n\n') + '\n\n' + this.textContent(node))];
- }
- /*******************************************************************/
- /*
- * The next four methods get overridden by the NodeMixin below
- */
- /**
- * @override
- */
- public fontSize(_node: LiteElement) {
- return 0;
- }
- /**
- * @override
- */
- public fontFamily(_node: LiteElement) {
- return '';
- }
- /**
- * @override
- */
- public nodeSize(_node: LiteElement, _em: number = 1, _local: boolean = null) {
- return [0, 0] as [number, number];
- }
- /**
- * @override
- */
- public nodeBBox(_node: LiteElement) {
- return {left: 0, right: 0, top: 0, bottom: 0};
- }
- }
- /**
- * The LiteAdaptor class (add in the NodeMixin methods and options)
- */
- export class LiteAdaptor extends NodeMixin<LiteElement, LiteText, LiteDocument, Constructor<LiteBase>>(LiteBase) {}
- /************************************************************/
- /**
- * The function to call to obtain a LiteAdaptor
- *
- * @param {OptionList} options The options for the adaptor
- * @return {LiteAdaptor} The newly created adaptor
- */
- export function liteAdaptor(options: OptionList = null): LiteAdaptor {
- return new LiteAdaptor(null, options);
- }
|