199 lines
8.3 KiB
JavaScript
199 lines
8.3 KiB
JavaScript
"use strict";
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.RebuildStree = void 0;
|
|
const enrich_attr_1 = require("../enrich_mathml/enrich_attr");
|
|
const semantic_attr_1 = require("../semantic_tree/semantic_attr");
|
|
const semantic_node_factory_1 = require("../semantic_tree/semantic_node_factory");
|
|
const semantic_processor_1 = require("../semantic_tree/semantic_processor");
|
|
const semantic_skeleton_1 = require("../semantic_tree/semantic_skeleton");
|
|
const semantic_tree_1 = require("../semantic_tree/semantic_tree");
|
|
const WalkerUtil = require("./walker_util");
|
|
class RebuildStree {
|
|
constructor(mathml) {
|
|
this.mathml = mathml;
|
|
this.factory = new semantic_node_factory_1.SemanticNodeFactory();
|
|
this.nodeDict = {};
|
|
this.mmlRoot = WalkerUtil.getSemanticRoot(mathml);
|
|
this.streeRoot = this.assembleTree(this.mmlRoot);
|
|
this.stree = semantic_tree_1.SemanticTree.fromNode(this.streeRoot, this.mathml);
|
|
this.xml = this.stree.xml();
|
|
semantic_processor_1.default.getInstance().setNodeFactory(this.factory);
|
|
}
|
|
static textContent(snode, node, ignore) {
|
|
if (!ignore && node.textContent) {
|
|
snode.textContent = node.textContent;
|
|
return;
|
|
}
|
|
const operator = WalkerUtil.splitAttribute(WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.OPERATOR));
|
|
if (operator.length > 1) {
|
|
snode.textContent = operator[1];
|
|
}
|
|
}
|
|
static isPunctuated(collapsed) {
|
|
return (!semantic_skeleton_1.SemanticSkeleton.simpleCollapseStructure(collapsed) &&
|
|
collapsed[1] &&
|
|
semantic_skeleton_1.SemanticSkeleton.contentCollapseStructure(collapsed[1]));
|
|
}
|
|
getTree() {
|
|
return this.stree;
|
|
}
|
|
assembleTree(node) {
|
|
const snode = this.makeNode(node);
|
|
const children = WalkerUtil.splitAttribute(WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.CHILDREN));
|
|
const content = WalkerUtil.splitAttribute(WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.CONTENT));
|
|
if (content.length === 0 && children.length === 0) {
|
|
RebuildStree.textContent(snode, node);
|
|
return snode;
|
|
}
|
|
if (content.length > 0) {
|
|
const fcontent = WalkerUtil.getBySemanticId(this.mathml, content[0]);
|
|
if (fcontent) {
|
|
RebuildStree.textContent(snode, fcontent, true);
|
|
}
|
|
}
|
|
snode.contentNodes = content.map((id) => this.setParent(id, snode));
|
|
snode.childNodes = children.map((id) => this.setParent(id, snode));
|
|
const collapsed = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.COLLAPSED);
|
|
return collapsed ? this.postProcess(snode, collapsed) : snode;
|
|
}
|
|
makeNode(node) {
|
|
const type = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.TYPE);
|
|
const role = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.ROLE);
|
|
const font = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.FONT);
|
|
const annotation = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.ANNOTATION) || '';
|
|
const attributes = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.ATTRIBUTES) || '';
|
|
const id = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.ID);
|
|
const embellished = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.EMBELLISHED);
|
|
const fencepointer = WalkerUtil.getAttribute(node, enrich_attr_1.Attribute.FENCEPOINTER);
|
|
const snode = this.createNode(parseInt(id, 10));
|
|
snode.type = type;
|
|
snode.role = role;
|
|
snode.font = font ? font : "unknown";
|
|
snode.parseAnnotation(annotation);
|
|
snode.parseAttributes(attributes);
|
|
if (fencepointer) {
|
|
snode.fencePointer = fencepointer;
|
|
}
|
|
if (embellished) {
|
|
snode.embellished = embellished;
|
|
}
|
|
return snode;
|
|
}
|
|
makePunctuation(id) {
|
|
const node = this.createNode(id);
|
|
node.updateContent((0, semantic_attr_1.invisibleComma)());
|
|
node.role = "dummy";
|
|
return node;
|
|
}
|
|
makePunctuated(snode, collapsed, role) {
|
|
const punctuated = this.createNode(collapsed[0]);
|
|
punctuated.type = "punctuated";
|
|
punctuated.embellished = snode.embellished;
|
|
punctuated.fencePointer = snode.fencePointer;
|
|
punctuated.role = role;
|
|
const cont = collapsed.splice(1, 1)[0].slice(1);
|
|
punctuated.contentNodes = cont.map(this.makePunctuation.bind(this));
|
|
this.collapsedChildren_(collapsed);
|
|
}
|
|
makeEmpty(snode, collapsed, role) {
|
|
const empty = this.createNode(collapsed);
|
|
empty.type = "empty";
|
|
empty.embellished = snode.embellished;
|
|
empty.fencePointer = snode.fencePointer;
|
|
empty.role = role;
|
|
}
|
|
makeIndex(snode, collapsed, role) {
|
|
if (RebuildStree.isPunctuated(collapsed)) {
|
|
this.makePunctuated(snode, collapsed, role);
|
|
collapsed = collapsed[0];
|
|
return;
|
|
}
|
|
if (semantic_skeleton_1.SemanticSkeleton.simpleCollapseStructure(collapsed) &&
|
|
!this.nodeDict[collapsed.toString()]) {
|
|
this.makeEmpty(snode, collapsed, role);
|
|
}
|
|
}
|
|
postProcess(snode, collapsed) {
|
|
const array = semantic_skeleton_1.SemanticSkeleton.fromString(collapsed).array;
|
|
if (snode.type === "subsup") {
|
|
const subscript = this.createNode(array[1][0]);
|
|
subscript.type = "subscript";
|
|
subscript.role = "subsup";
|
|
snode.type = "superscript";
|
|
subscript.embellished = snode.embellished;
|
|
subscript.fencePointer = snode.fencePointer;
|
|
this.makeIndex(snode, array[1][2], "rightsub");
|
|
this.makeIndex(snode, array[2], "rightsuper");
|
|
this.collapsedChildren_(array);
|
|
return snode;
|
|
}
|
|
if (snode.type === "subscript") {
|
|
this.makeIndex(snode, array[2], "rightsub");
|
|
this.collapsedChildren_(array);
|
|
return snode;
|
|
}
|
|
if (snode.type === "superscript") {
|
|
this.makeIndex(snode, array[2], "rightsuper");
|
|
this.collapsedChildren_(array);
|
|
return snode;
|
|
}
|
|
if (snode.type === "tensor") {
|
|
this.makeIndex(snode, array[2], "leftsub");
|
|
this.makeIndex(snode, array[3], "leftsuper");
|
|
this.makeIndex(snode, array[4], "rightsub");
|
|
this.makeIndex(snode, array[5], "rightsuper");
|
|
this.collapsedChildren_(array);
|
|
return snode;
|
|
}
|
|
if (snode.type === "punctuated") {
|
|
if (RebuildStree.isPunctuated(array)) {
|
|
const cont = array.splice(1, 1)[0].slice(1);
|
|
snode.contentNodes = cont.map(this.makePunctuation.bind(this));
|
|
}
|
|
return snode;
|
|
}
|
|
if (snode.type === "underover") {
|
|
const score = this.createNode(array[1][0]);
|
|
if (snode.childNodes[1].role === "overaccent") {
|
|
score.type = "overscore";
|
|
snode.type = "underscore";
|
|
}
|
|
else {
|
|
score.type = "underscore";
|
|
snode.type = "overscore";
|
|
}
|
|
score.role = "underover";
|
|
score.embellished = snode.embellished;
|
|
score.fencePointer = snode.fencePointer;
|
|
this.collapsedChildren_(array);
|
|
return snode;
|
|
}
|
|
return snode;
|
|
}
|
|
createNode(id) {
|
|
const node = this.factory.makeNode(id);
|
|
this.nodeDict[id.toString()] = node;
|
|
return node;
|
|
}
|
|
collapsedChildren_(collapsed) {
|
|
const recurseCollapsed = (coll) => {
|
|
const parent = this.nodeDict[coll[0]];
|
|
parent.childNodes = [];
|
|
for (let j = 1, l = coll.length; j < l; j++) {
|
|
const id = coll[j];
|
|
parent.childNodes.push(semantic_skeleton_1.SemanticSkeleton.simpleCollapseStructure(id)
|
|
? this.nodeDict[id]
|
|
: recurseCollapsed(id));
|
|
}
|
|
return parent;
|
|
};
|
|
recurseCollapsed(collapsed);
|
|
}
|
|
setParent(id, snode) {
|
|
const mml = WalkerUtil.getBySemanticId(this.mathml, id);
|
|
const sn = this.assembleTree(mml);
|
|
sn.parent = snode;
|
|
return sn;
|
|
}
|
|
}
|
|
exports.RebuildStree = RebuildStree;
|