508 lines
18 KiB
JavaScript
508 lines
18 KiB
JavaScript
"use strict";
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.AbstractWalker = void 0;
|
|
const auditory_description_1 = require("../audio/auditory_description");
|
|
const AuralRendering = require("../audio/aural_rendering");
|
|
const DomUtil = require("../common/dom_util");
|
|
const EngineConst = require("../common/engine_const");
|
|
const engine_setup_1 = require("../common/engine_setup");
|
|
const event_util_1 = require("../common/event_util");
|
|
const enrich_attr_1 = require("../enrich_mathml/enrich_attr");
|
|
const locale_1 = require("../l10n/locale");
|
|
const grammar_1 = require("../rule_engine/grammar");
|
|
const semantic_skeleton_1 = require("../semantic_tree/semantic_skeleton");
|
|
const SpeechGeneratorFactory = require("../speech_generator/speech_generator_factory");
|
|
const SpeechGeneratorUtil = require("../speech_generator/speech_generator_util");
|
|
const clearspeak_preferences_1 = require("../speech_rules/clearspeak_preferences");
|
|
const focus_1 = require("./focus");
|
|
const rebuild_stree_1 = require("./rebuild_stree");
|
|
const walker_1 = require("./walker");
|
|
const WalkerUtil = require("./walker_util");
|
|
const XpathUtil = require("../common/xpath_util");
|
|
class AbstractWalker {
|
|
constructor(node, generator, highlighter, xml) {
|
|
this.node = node;
|
|
this.generator = generator;
|
|
this.highlighter = highlighter;
|
|
this.modifier = false;
|
|
this.keyMapping = new Map([
|
|
[event_util_1.KeyCode.UP, this.up.bind(this)],
|
|
[event_util_1.KeyCode.DOWN, this.down.bind(this)],
|
|
[event_util_1.KeyCode.RIGHT, this.right.bind(this)],
|
|
[event_util_1.KeyCode.LEFT, this.left.bind(this)],
|
|
[event_util_1.KeyCode.TAB, this.repeat.bind(this)],
|
|
[event_util_1.KeyCode.DASH, this.expand.bind(this)],
|
|
[event_util_1.KeyCode.SPACE, this.depth.bind(this)],
|
|
[event_util_1.KeyCode.HOME, this.home.bind(this)],
|
|
[event_util_1.KeyCode.X, this.summary.bind(this)],
|
|
[event_util_1.KeyCode.Z, this.detail.bind(this)],
|
|
[event_util_1.KeyCode.V, this.virtualize.bind(this)],
|
|
[event_util_1.KeyCode.P, this.previous.bind(this)],
|
|
[event_util_1.KeyCode.U, this.undo.bind(this)],
|
|
[event_util_1.KeyCode.LESS, this.previousRules.bind(this)],
|
|
[event_util_1.KeyCode.GREATER, this.nextRules.bind(this)]
|
|
]);
|
|
this.cursors = [];
|
|
this.xml_ = null;
|
|
this.rebuilt_ = null;
|
|
this.focus_ = null;
|
|
this.active_ = false;
|
|
if (this.node.id) {
|
|
this.id = this.node.id;
|
|
}
|
|
else if (this.node.hasAttribute(AbstractWalker.SRE_ID_ATTR)) {
|
|
this.id = this.node.getAttribute(AbstractWalker.SRE_ID_ATTR);
|
|
}
|
|
else {
|
|
this.node.setAttribute(AbstractWalker.SRE_ID_ATTR, AbstractWalker.ID_COUNTER.toString());
|
|
this.id = AbstractWalker.ID_COUNTER++;
|
|
}
|
|
this.rootNode = WalkerUtil.getSemanticRoot(node);
|
|
this.rootId = this.rootNode.getAttribute(enrich_attr_1.Attribute.ID);
|
|
this.xmlString_ = xml;
|
|
this.moved = walker_1.WalkerMoves.ENTER;
|
|
}
|
|
getXml() {
|
|
if (!this.xml_) {
|
|
this.xml_ = DomUtil.parseInput(this.xmlString_);
|
|
}
|
|
return this.xml_;
|
|
}
|
|
getRebuilt() {
|
|
if (!this.rebuilt_) {
|
|
this.rebuildStree();
|
|
}
|
|
return this.rebuilt_;
|
|
}
|
|
isActive() {
|
|
return this.active_;
|
|
}
|
|
activate() {
|
|
if (this.isActive()) {
|
|
return;
|
|
}
|
|
this.generator.start();
|
|
this.toggleActive_();
|
|
}
|
|
deactivate() {
|
|
if (!this.isActive()) {
|
|
return;
|
|
}
|
|
walker_1.WalkerState.setState(this.id, this.primaryId());
|
|
this.generator.end();
|
|
this.toggleActive_();
|
|
}
|
|
getFocus(update = false) {
|
|
if (!this.focus_) {
|
|
this.focus_ = this.singletonFocus(this.rootId);
|
|
}
|
|
if (update) {
|
|
this.updateFocus();
|
|
}
|
|
return this.focus_;
|
|
}
|
|
setFocus(focus) {
|
|
this.focus_ = focus;
|
|
}
|
|
getDepth() {
|
|
return this.levels.depth() - 1;
|
|
}
|
|
isSpeech() {
|
|
return this.generator.modality === enrich_attr_1.Attribute.SPEECH;
|
|
}
|
|
focusDomNodes() {
|
|
return this.getFocus().getDomNodes();
|
|
}
|
|
focusSemanticNodes() {
|
|
return this.getFocus().getSemanticNodes();
|
|
}
|
|
speech() {
|
|
const nodes = this.focusDomNodes();
|
|
if (!nodes.length) {
|
|
return '';
|
|
}
|
|
const special = this.specialMove();
|
|
if (special !== null) {
|
|
return special;
|
|
}
|
|
switch (this.moved) {
|
|
case walker_1.WalkerMoves.DEPTH:
|
|
return this.depth_();
|
|
case walker_1.WalkerMoves.SUMMARY:
|
|
return this.summary_();
|
|
case walker_1.WalkerMoves.DETAIL:
|
|
return this.detail_();
|
|
default: {
|
|
const speech = [];
|
|
const snodes = this.focusSemanticNodes();
|
|
for (let i = 0, l = nodes.length; i < l; i++) {
|
|
const node = nodes[i];
|
|
const snode = snodes[i];
|
|
speech.push(node
|
|
? this.generator.getSpeech(node, this.getXml())
|
|
: SpeechGeneratorUtil.recomputeMarkup(snode));
|
|
}
|
|
return this.mergePrefix_(speech);
|
|
}
|
|
}
|
|
}
|
|
move(key) {
|
|
const direction = this.keyMapping.get(key);
|
|
if (!direction) {
|
|
return null;
|
|
}
|
|
const focus = direction();
|
|
if (!focus || focus === this.getFocus()) {
|
|
return false;
|
|
}
|
|
this.setFocus(focus);
|
|
if (this.moved === walker_1.WalkerMoves.HOME) {
|
|
this.levels = this.initLevels();
|
|
}
|
|
return true;
|
|
}
|
|
up() {
|
|
this.moved = walker_1.WalkerMoves.UP;
|
|
return this.getFocus();
|
|
}
|
|
down() {
|
|
this.moved = walker_1.WalkerMoves.DOWN;
|
|
return this.getFocus();
|
|
}
|
|
left() {
|
|
this.moved = walker_1.WalkerMoves.LEFT;
|
|
return this.getFocus();
|
|
}
|
|
right() {
|
|
this.moved = walker_1.WalkerMoves.RIGHT;
|
|
return this.getFocus();
|
|
}
|
|
repeat() {
|
|
this.moved = walker_1.WalkerMoves.REPEAT;
|
|
return this.getFocus().clone();
|
|
}
|
|
depth() {
|
|
this.moved = this.isSpeech() ? walker_1.WalkerMoves.DEPTH : walker_1.WalkerMoves.REPEAT;
|
|
return this.getFocus().clone();
|
|
}
|
|
home() {
|
|
this.moved = walker_1.WalkerMoves.HOME;
|
|
const focus = this.singletonFocus(this.rootId);
|
|
return focus;
|
|
}
|
|
getBySemanticId(id) {
|
|
return WalkerUtil.getBySemanticId(this.node, id);
|
|
}
|
|
primaryId() {
|
|
return this.getFocus().getSemanticPrimary().id.toString();
|
|
}
|
|
expand() {
|
|
const primary = this.getFocus().getDomPrimary();
|
|
const expandable = this.actionable_(primary);
|
|
if (!expandable) {
|
|
return this.getFocus();
|
|
}
|
|
this.moved = walker_1.WalkerMoves.EXPAND;
|
|
expandable.dispatchEvent(new Event('click'));
|
|
return this.getFocus().clone();
|
|
}
|
|
expandable(node) {
|
|
const parent = !!this.actionable_(node);
|
|
return parent && node.childNodes.length === 0;
|
|
}
|
|
collapsible(node) {
|
|
const parent = !!this.actionable_(node);
|
|
return parent && node.childNodes.length > 0;
|
|
}
|
|
restoreState() {
|
|
if (!this.highlighter) {
|
|
return;
|
|
}
|
|
const state = walker_1.WalkerState.getState(this.id);
|
|
if (!state) {
|
|
return;
|
|
}
|
|
let node = this.getRebuilt().nodeDict[state];
|
|
const path = [];
|
|
while (node) {
|
|
path.push(node.id);
|
|
node = node.parent;
|
|
}
|
|
path.pop();
|
|
while (path.length > 0) {
|
|
this.down();
|
|
const id = path.pop();
|
|
const focus = this.findFocusOnLevel(id);
|
|
if (!focus) {
|
|
break;
|
|
}
|
|
this.setFocus(focus);
|
|
}
|
|
this.moved = walker_1.WalkerMoves.ENTER;
|
|
}
|
|
updateFocus() {
|
|
this.setFocus(focus_1.Focus.factory(this.getFocus().getSemanticPrimary().id.toString(), this.getFocus()
|
|
.getSemanticNodes()
|
|
.map((x) => x.id.toString()), this.getRebuilt(), this.node));
|
|
}
|
|
rebuildStree() {
|
|
this.rebuilt_ = new rebuild_stree_1.RebuildStree(this.getXml());
|
|
this.rootId = this.rebuilt_.stree.root.id.toString();
|
|
this.generator.setRebuilt(this.rebuilt_);
|
|
this.skeleton = semantic_skeleton_1.SemanticSkeleton.fromTree(this.rebuilt_.stree);
|
|
this.skeleton.populate();
|
|
this.focus_ = this.singletonFocus(this.rootId);
|
|
this.levels = this.initLevels();
|
|
SpeechGeneratorUtil.connectMactions(this.node, this.getXml(), this.rebuilt_.xml);
|
|
}
|
|
previousLevel() {
|
|
const dnode = this.getFocus().getDomPrimary();
|
|
return dnode
|
|
? WalkerUtil.getAttribute(dnode, enrich_attr_1.Attribute.PARENT)
|
|
: this.getFocus().getSemanticPrimary().parent.id.toString();
|
|
}
|
|
nextLevel() {
|
|
const dnode = this.getFocus().getDomPrimary();
|
|
let children;
|
|
let content;
|
|
if (dnode) {
|
|
children = WalkerUtil.splitAttribute(WalkerUtil.getAttribute(dnode, enrich_attr_1.Attribute.CHILDREN));
|
|
content = WalkerUtil.splitAttribute(WalkerUtil.getAttribute(dnode, enrich_attr_1.Attribute.CONTENT));
|
|
const type = WalkerUtil.getAttribute(dnode, enrich_attr_1.Attribute.TYPE);
|
|
const role = WalkerUtil.getAttribute(dnode, enrich_attr_1.Attribute.ROLE);
|
|
return this.combineContentChildren(type, role, content, children);
|
|
}
|
|
const toIds = (x) => x.id.toString();
|
|
const snode = this.getRebuilt().nodeDict[this.primaryId()];
|
|
children = snode.childNodes.map(toIds);
|
|
content = snode.contentNodes.map(toIds);
|
|
if (children.length === 0) {
|
|
return [];
|
|
}
|
|
return this.combineContentChildren(snode.type, snode.role, content, children);
|
|
}
|
|
singletonFocus(id) {
|
|
this.getRebuilt();
|
|
const ids = this.retrieveVisuals(id);
|
|
return this.focusFromId(id, ids);
|
|
}
|
|
retrieveVisuals(id) {
|
|
if (!this.skeleton) {
|
|
return [id];
|
|
}
|
|
const num = parseInt(id, 10);
|
|
const semStree = this.skeleton.subtreeNodes(num);
|
|
if (!semStree.length) {
|
|
return [id];
|
|
}
|
|
semStree.unshift(num);
|
|
const mmlStree = {};
|
|
const result = [];
|
|
XpathUtil.updateEvaluator(this.getXml());
|
|
for (const child of semStree) {
|
|
if (mmlStree[child]) {
|
|
continue;
|
|
}
|
|
result.push(child.toString());
|
|
mmlStree[child] = true;
|
|
this.subtreeIds(child, mmlStree);
|
|
}
|
|
return result;
|
|
}
|
|
subtreeIds(id, nodes) {
|
|
const xmlRoot = XpathUtil.evalXPath(`//*[@data-semantic-id="${id}"]`, this.getXml());
|
|
const xpath = XpathUtil.evalXPath('*//@data-semantic-id', xmlRoot[0]);
|
|
xpath.forEach((x) => (nodes[parseInt(x.textContent, 10)] = true));
|
|
}
|
|
focusFromId(id, ids) {
|
|
return focus_1.Focus.factory(id, ids, this.getRebuilt(), this.node);
|
|
}
|
|
summary() {
|
|
this.moved = this.isSpeech() ? walker_1.WalkerMoves.SUMMARY : walker_1.WalkerMoves.REPEAT;
|
|
return this.getFocus().clone();
|
|
}
|
|
detail() {
|
|
this.moved = this.isSpeech() ? walker_1.WalkerMoves.DETAIL : walker_1.WalkerMoves.REPEAT;
|
|
return this.getFocus().clone();
|
|
}
|
|
specialMove() {
|
|
return null;
|
|
}
|
|
virtualize(opt_undo) {
|
|
this.cursors.push({
|
|
focus: this.getFocus(),
|
|
levels: this.levels,
|
|
undo: opt_undo || !this.cursors.length
|
|
});
|
|
this.levels = this.levels.clone();
|
|
return this.getFocus().clone();
|
|
}
|
|
previous() {
|
|
const previous = this.cursors.pop();
|
|
if (!previous) {
|
|
return this.getFocus();
|
|
}
|
|
this.levels = previous.levels;
|
|
return previous.focus;
|
|
}
|
|
undo() {
|
|
let previous;
|
|
do {
|
|
previous = this.cursors.pop();
|
|
} while (previous && !previous.undo);
|
|
if (!previous) {
|
|
return this.getFocus();
|
|
}
|
|
this.levels = previous.levels;
|
|
return previous.focus;
|
|
}
|
|
update(options) {
|
|
this.generator.setOptions(options);
|
|
(0, engine_setup_1.setup)(options).then(() => SpeechGeneratorFactory.generator('Tree').getSpeech(this.node, this.getXml()));
|
|
}
|
|
nextRules() {
|
|
const options = this.generator.getOptions();
|
|
if (options.modality !== 'speech') {
|
|
return this.getFocus();
|
|
}
|
|
EngineConst.DOMAIN_TO_STYLES[options.domain] = options.style;
|
|
options.domain =
|
|
options.domain === 'mathspeak' ? 'clearspeak' : 'mathspeak';
|
|
options.style = EngineConst.DOMAIN_TO_STYLES[options.domain];
|
|
this.update(options);
|
|
this.moved = walker_1.WalkerMoves.REPEAT;
|
|
return this.getFocus().clone();
|
|
}
|
|
nextStyle(domain, style) {
|
|
if (domain === 'mathspeak') {
|
|
const styles = ['default', 'brief', 'sbrief'];
|
|
const index = styles.indexOf(style);
|
|
if (index === -1) {
|
|
return style;
|
|
}
|
|
return index >= styles.length - 1 ? styles[0] : styles[index + 1];
|
|
}
|
|
if (domain === 'clearspeak') {
|
|
const prefs = clearspeak_preferences_1.ClearspeakPreferences.getLocalePreferences();
|
|
const loc = prefs['en'];
|
|
if (!loc) {
|
|
return 'default';
|
|
}
|
|
const smart = clearspeak_preferences_1.ClearspeakPreferences.relevantPreferences(this.getFocus().getSemanticPrimary());
|
|
const current = clearspeak_preferences_1.ClearspeakPreferences.findPreference(style, smart);
|
|
const options = loc[smart].map(function (x) {
|
|
return x.split('_')[1];
|
|
});
|
|
const index = options.indexOf(current);
|
|
if (index === -1) {
|
|
return style;
|
|
}
|
|
const next = index >= options.length - 1 ? options[0] : options[index + 1];
|
|
const result = clearspeak_preferences_1.ClearspeakPreferences.addPreference(style, smart, next);
|
|
return result;
|
|
}
|
|
return style;
|
|
}
|
|
previousRules() {
|
|
const options = this.generator.getOptions();
|
|
if (options.modality !== 'speech') {
|
|
return this.getFocus();
|
|
}
|
|
options.style = this.nextStyle(options.domain, options.style);
|
|
this.update(options);
|
|
this.moved = walker_1.WalkerMoves.REPEAT;
|
|
return this.getFocus().clone();
|
|
}
|
|
refocus() {
|
|
let focus = this.getFocus();
|
|
let last;
|
|
while (!focus.getNodes().length) {
|
|
last = this.levels.peek();
|
|
const up = this.up();
|
|
if (!up) {
|
|
break;
|
|
}
|
|
this.setFocus(up);
|
|
focus = this.getFocus(true);
|
|
}
|
|
this.levels.push(last);
|
|
this.setFocus(focus);
|
|
}
|
|
toggleActive_() {
|
|
this.active_ = !this.active_;
|
|
}
|
|
mergePrefix_(speech, pre = []) {
|
|
const prefix = this.isSpeech() ? this.prefix_() : '';
|
|
if (prefix) {
|
|
speech.unshift(prefix);
|
|
}
|
|
const postfix = this.isSpeech() ? this.postfix_() : '';
|
|
if (postfix) {
|
|
speech.push(postfix);
|
|
}
|
|
return AuralRendering.finalize(AuralRendering.merge(pre.concat(speech)));
|
|
}
|
|
prefix_() {
|
|
const nodes = this.getFocus().getDomNodes();
|
|
const snodes = this.getFocus().getSemanticNodes();
|
|
return nodes[0]
|
|
? WalkerUtil.getAttribute(nodes[0], enrich_attr_1.Attribute.PREFIX)
|
|
: SpeechGeneratorUtil.retrievePrefix(snodes[0]);
|
|
}
|
|
postfix_() {
|
|
const nodes = this.getFocus().getDomNodes();
|
|
return nodes[0]
|
|
? WalkerUtil.getAttribute(nodes[0], enrich_attr_1.Attribute.POSTFIX)
|
|
: '';
|
|
}
|
|
depth_() {
|
|
const oldDepth = grammar_1.Grammar.getInstance().getParameter('depth');
|
|
grammar_1.Grammar.getInstance().setParameter('depth', true);
|
|
const primary = this.getFocus().getDomPrimary();
|
|
const expand = this.expandable(primary)
|
|
? locale_1.LOCALE.MESSAGES.navigate.EXPANDABLE
|
|
: this.collapsible(primary)
|
|
? locale_1.LOCALE.MESSAGES.navigate.COLLAPSIBLE
|
|
: '';
|
|
const level = locale_1.LOCALE.MESSAGES.navigate.LEVEL + ' ' + this.getDepth();
|
|
const snodes = this.getFocus().getSemanticNodes();
|
|
const prefix = SpeechGeneratorUtil.retrievePrefix(snodes[0]);
|
|
const audio = [
|
|
new auditory_description_1.AuditoryDescription({ text: level, personality: {} }),
|
|
new auditory_description_1.AuditoryDescription({ text: prefix, personality: {} }),
|
|
new auditory_description_1.AuditoryDescription({ text: expand, personality: {} })
|
|
];
|
|
grammar_1.Grammar.getInstance().setParameter('depth', oldDepth);
|
|
return AuralRendering.finalize(AuralRendering.markup(audio));
|
|
}
|
|
actionable_(node) {
|
|
const parent = node === null || node === void 0 ? void 0 : node.parentNode;
|
|
return parent && this.highlighter.isMactionNode(parent) ? parent : null;
|
|
}
|
|
summary_() {
|
|
const sprimary = this.getFocus().getSemanticPrimary();
|
|
const sid = sprimary.id.toString();
|
|
const snode = this.getRebuilt().xml.getAttribute('id') === sid
|
|
? this.getRebuilt().xml
|
|
: DomUtil.querySelectorAllByAttrValue(this.getRebuilt().xml, 'id', sid)[0];
|
|
const summary = SpeechGeneratorUtil.retrieveSummary(snode);
|
|
const speech = this.mergePrefix_([summary]);
|
|
return speech;
|
|
}
|
|
detail_() {
|
|
const sprimary = this.getFocus().getSemanticPrimary();
|
|
const sid = sprimary.id.toString();
|
|
const snode = this.getRebuilt().xml.getAttribute('id') === sid
|
|
? this.getRebuilt().xml
|
|
: DomUtil.querySelectorAllByAttrValue(this.getRebuilt().xml, 'id', sid)[0];
|
|
const oldAlt = snode.getAttribute('alternative');
|
|
snode.removeAttribute('alternative');
|
|
const detail = SpeechGeneratorUtil.computeMarkup(snode);
|
|
const speech = this.mergePrefix_([detail]);
|
|
snode.setAttribute('alternative', oldAlt);
|
|
return speech;
|
|
}
|
|
}
|
|
exports.AbstractWalker = AbstractWalker;
|
|
AbstractWalker.ID_COUNTER = 0;
|
|
AbstractWalker.SRE_ID_ATTR = 'sre-explorer-id';
|