/* * Copyright (C) 2019 Alex Yatskov * Author: Alex Yatskov * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ class QueryParser { constructor(search) { this.search = search; this.pendingLookup = false; this.clickScanPrevent = false; this.queryParser = document.querySelector('#query-parser'); this.queryParser.addEventListener('mousedown', (e) => this.onMouseDown(e)); this.queryParser.addEventListener('mouseup', (e) => this.onMouseUp(e)); } onError(error) { logError(error, false); } onMouseDown(e) { if (Frontend.isMouseButton('primary', e)) { this.clickScanPrevent = false; } } onMouseUp(e) { if ( this.search.options.scanning.clickGlossary && !this.clickScanPrevent && Frontend.isMouseButton('primary', e) ) { const selectText = this.search.options.scanning.selectText; this.onTermLookup(e, {disableScroll: true, selectText}); } } onMouseMove(e) { if (this.pendingLookup || Frontend.isMouseButton('primary', e)) { return; } const scanningOptions = this.search.options.scanning; const scanningModifier = scanningOptions.modifier; if (!( Frontend.isScanningModifierPressed(scanningModifier, e) || (scanningOptions.middleMouse && Frontend.isMouseButton('auxiliary', e)) )) { return; } const selectText = this.search.options.scanning.selectText; this.onTermLookup(e, {disableScroll: true, disableHistory: true, selectText}); } onMouseLeave(e) { this.clickScanPrevent = true; clearTimeout(e.target.dataset.timer); delete e.target.dataset.timer; } onTermLookup(e, params) { this.pendingLookup = true; (async () => { await this.search.onTermLookup(e, params); this.pendingLookup = false; })(); } async setText(text) { this.search.setSpinnerVisible(true); await this.setPreview(text); const results = {}; if (this.search.options.parsing.enableScanningParser) { results['scan'] = await apiTextParse(text, this.search.getOptionsContext()); } if (this.search.options.parsing.enableMecabParser) { let mecabResults = await apiTextParseMecab(text, this.search.getOptionsContext()); for (const mecabDictName in mecabResults) { results[`mecab-${mecabDictName}`] = mecabResults[mecabDictName]; } } const contents = await Promise.all(Object.values(results).map(result => { return apiTemplateRender('query-parser.html', { terms: result.map((term) => { return term.filter(part => part.text.trim()).map((part) => { return { text: Array.from(part.text), reading: part.reading, raw: !part.reading || !part.reading.trim(), }; }); }) }); })); this.queryParser.innerHTML = contents.join('
'); for (const charElement of this.queryParser.querySelectorAll('.query-parser-char')) { this.activateScanning(charElement); } this.search.setSpinnerVisible(false); } async setPreview(text) { const previewTerms = []; while (text.length > 0) { const tempText = text.slice(0, 2); previewTerms.push([{text: Array.from(tempText)}]); text = text.slice(2); } this.queryParser.innerHTML = await apiTemplateRender('query-parser.html', { terms: previewTerms, preview: true }); for (const charElement of this.queryParser.querySelectorAll('.query-parser-char')) { this.activateScanning(charElement); } } activateScanning(element) { element.addEventListener('mousemove', (e) => { clearTimeout(e.target.dataset.timer); if (this.search.options.scanning.modifier === 'none') { e.target.dataset.timer = setTimeout(() => { this.onMouseMove(e); delete e.target.dataset.timer; }, this.search.options.scanning.delay); } else { this.onMouseMove(e); } }); element.addEventListener('mouseleave', (e) => { this.onMouseLeave(e); }); } }