diff options
Diffstat (limited to 'ext')
54 files changed, 1473 insertions, 1276 deletions
| diff --git a/ext/bg/background.html b/ext/bg/background.html index 7fd1c477..44abe8fd 100644 --- a/ext/bg/background.html +++ b/ext/bg/background.html @@ -22,9 +22,9 @@          <script src="/mixed/js/dom.js"></script>          <script src="/bg/js/anki.js"></script> -        <script src="/bg/js/api.js"></script> +        <script src="/bg/js/anki-note-builder.js"></script>          <script src="/bg/js/mecab.js"></script> -        <script src="/bg/js/audio.js"></script> +        <script src="/bg/js/audio-uri-builder.js"></script>          <script src="/bg/js/backend-api-forwarder.js"></script>          <script src="/bg/js/clipboard-monitor.js"></script>          <script src="/bg/js/conditions.js"></script> @@ -39,7 +39,7 @@          <script src="/bg/js/request.js"></script>          <script src="/bg/js/translator.js"></script>          <script src="/bg/js/util.js"></script> -        <script src="/mixed/js/audio.js"></script> +        <script src="/mixed/js/audio-system.js"></script>          <script src="/bg/js/backend.js"></script>      </body> diff --git a/ext/bg/data/default-anki-field-templates.handlebars b/ext/bg/data/default-anki-field-templates.handlebars new file mode 100644 index 00000000..0442f7c5 --- /dev/null +++ b/ext/bg/data/default-anki-field-templates.handlebars @@ -0,0 +1,161 @@ +{{#*inline "glossary-single"}} +    {{~#unless brief~}} +        {{~#if definitionTags~}}<i>({{#each definitionTags}}{{name}}{{#unless @last}}, {{/unless}}{{/each}})</i> {{/if~}} +        {{~#if only~}}({{#each only}}{{{.}}}{{#unless @last}}, {{/unless}}{{/each}} only) {{/if~}} +    {{~/unless~}} +    {{~#if glossary.[1]~}} +        {{~#if compactGlossaries~}} +            {{#each glossary}}{{#multiLine}}{{.}}{{/multiLine}}{{#unless @last}} | {{/unless}}{{/each}} +        {{~else~}} +            <ul>{{#each glossary}}<li>{{#multiLine}}{{.}}{{/multiLine}}</li>{{/each}}</ul> +        {{~/if~}} +    {{~else~}} +        {{~#multiLine}}{{glossary.[0]}}{{/multiLine~}} +    {{~/if~}} +{{/inline}} + +{{#*inline "audio"}}{{/inline}} + +{{#*inline "character"}} +    {{~definition.character~}} +{{/inline}} + +{{#*inline "dictionary"}} +    {{~definition.dictionary~}} +{{/inline}} + +{{#*inline "expression"}} +    {{~#if merge~}} +        {{~#if modeTermKana~}} +            {{~#each definition.reading~}} +                {{{.}}} +                {{~#unless @last}}、{{/unless~}} +            {{~else~}} +                {{~#each definition.expression~}} +                    {{{.}}} +                    {{~#unless @last}}、{{/unless~}} +                {{~/each~}} +            {{~/each~}} +        {{~else~}} +            {{~#each definition.expression~}} +                {{{.}}} +                {{~#unless @last}}、{{/unless~}} +            {{~/each~}} +        {{~/if~}} +    {{~else~}} +        {{~#if modeTermKana~}} +            {{~#if definition.reading~}} +                {{definition.reading}} +            {{~else~}} +                {{definition.expression}} +            {{~/if~}} +        {{~else~}} +            {{definition.expression}} +        {{~/if~}} +    {{~/if~}} +{{/inline}} + +{{#*inline "furigana"}} +    {{~#if merge~}} +        {{~#each definition.expressions~}} +            <span class="expression-{{termFrequency}}">{{~#furigana}}{{{.}}}{{/furigana~}}</span> +            {{~#unless @last}}、{{/unless~}} +        {{~/each~}} +    {{~else~}} +        {{#furigana}}{{{definition}}}{{/furigana}} +    {{~/if~}} +{{/inline}} + +{{#*inline "furigana-plain"}} +    {{~#if merge~}} +        {{~#each definition.expressions~}} +            <span class="expression-{{termFrequency}}">{{~#furiganaPlain}}{{{.}}}{{/furiganaPlain~}}</span> +            {{~#unless @last}}、{{/unless~}} +        {{~/each~}} +    {{~else~}} +        {{#furiganaPlain}}{{{definition}}}{{/furiganaPlain}} +    {{~/if~}} +{{/inline}} + +{{#*inline "glossary"}} +    <div style="text-align: left;"> +    {{~#if modeKanji~}} +        {{~#if definition.glossary.[1]~}} +            <ol>{{#each definition.glossary}}<li>{{.}}</li>{{/each}}</ol> +        {{~else~}} +            {{definition.glossary.[0]}} +        {{~/if~}} +    {{~else~}} +        {{~#if group~}} +            {{~#if definition.definitions.[1]~}} +                <ol>{{#each definition.definitions}}<li>{{> glossary-single brief=../brief compactGlossaries=../compactGlossaries}}</li>{{/each}}</ol> +            {{~else~}} +                {{~> glossary-single definition.definitions.[0] brief=brief compactGlossaries=compactGlossaries~}} +            {{~/if~}} +        {{~else if merge~}} +            {{~#if definition.definitions.[1]~}} +                <ol>{{#each definition.definitions}}<li>{{> glossary-single brief=../brief compactGlossaries=../compactGlossaries}}</li>{{/each}}</ol> +            {{~else~}} +                {{~> glossary-single definition.definitions.[0] brief=brief compactGlossaries=compactGlossaries~}} +            {{~/if~}} +        {{~else~}} +            {{~> glossary-single definition brief=brief compactGlossaries=compactGlossaries~}} +        {{~/if~}} +    {{~/if~}} +    </div> +{{/inline}} + +{{#*inline "glossary-brief"}} +    {{~> glossary brief=true ~}} +{{/inline}} + +{{#*inline "kunyomi"}} +    {{~#each definition.kunyomi}}{{.}}{{#unless @last}}, {{/unless}}{{/each~}} +{{/inline}} + +{{#*inline "onyomi"}} +    {{~#each definition.onyomi}}{{.}}{{#unless @last}}, {{/unless}}{{/each~}} +{{/inline}} + +{{#*inline "reading"}} +    {{~#unless modeTermKana~}} +        {{~#if merge~}} +            {{~#each definition.reading~}} +                {{{.}}} +                {{~#unless @last}}、{{/unless~}} +            {{~/each~}} +        {{~else~}} +            {{~definition.reading~}} +        {{~/if~}} +    {{~/unless~}} +{{/inline}} + +{{#*inline "sentence"}} +    {{~#if definition.cloze}}{{definition.cloze.sentence}}{{/if~}} +{{/inline}} + +{{#*inline "cloze-prefix"}} +    {{~#if definition.cloze}}{{definition.cloze.prefix}}{{/if~}} +{{/inline}} + +{{#*inline "cloze-body"}} +    {{~#if definition.cloze}}{{definition.cloze.body}}{{/if~}} +{{/inline}} + +{{#*inline "cloze-suffix"}} +    {{~#if definition.cloze}}{{definition.cloze.suffix}}{{/if~}} +{{/inline}} + +{{#*inline "tags"}} +    {{~#each definition.definitionTags}}{{name}}{{#unless @last}}, {{/unless}}{{/each~}} +{{/inline}} + +{{#*inline "url"}} +    <a href="{{definition.url}}">{{definition.url}}</a> +{{/inline}} + +{{#*inline "screenshot"}} +    <img src="{{definition.screenshotFileName}}" /> +{{/inline}} + +{{~> (lookup . "marker") ~}}
\ No newline at end of file diff --git a/ext/bg/js/anki-note-builder.js b/ext/bg/js/anki-note-builder.js new file mode 100644 index 00000000..d0ff8205 --- /dev/null +++ b/ext/bg/js/anki-note-builder.js @@ -0,0 +1,100 @@ +/* + * Copyright (C) 2020  Alex Yatskov <alex@foosoft.net> + * Author: Alex Yatskov <alex@foosoft.net> + * + * 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 <https://www.gnu.org/licenses/>. + */ + +class AnkiNoteBuilder { +    constructor({renderTemplate}) { +        this._renderTemplate = renderTemplate; +    } + +    async createNote(definition, mode, options, templates) { +        const isKanji = (mode === 'kanji'); +        const tags = options.anki.tags; +        const modeOptions = isKanji ? options.anki.kanji : options.anki.terms; +        const modeOptionsFieldEntries = Object.entries(modeOptions.fields); + +        const note = { +            fields: {}, +            tags, +            deckName: modeOptions.deck, +            modelName: modeOptions.model +        }; + +        for (const [fieldName, fieldValue] of modeOptionsFieldEntries) { +            note.fields[fieldName] = await this.formatField(fieldValue, definition, mode, options, templates, null); +        } + +        if (!isKanji && definition.audio) { +            const audioFields = []; + +            for (const [fieldName, fieldValue] of modeOptionsFieldEntries) { +                if (fieldValue.includes('{audio}')) { +                    audioFields.push(fieldName); +                } +            } + +            if (audioFields.length > 0) { +                note.audio = { +                    url: definition.audio.url, +                    filename: definition.audio.filename, +                    skipHash: '7e2c2f954ef6051373ba916f000168dc', // hash of audio data that should be skipped +                    fields: audioFields +                }; +            } +        } + +        return note; +    } + +    async formatField(field, definition, mode, options, templates, errors=null) { +        const data = { +            marker: null, +            definition, +            group: options.general.resultOutputMode === 'group', +            merge: options.general.resultOutputMode === 'merge', +            modeTermKanji: mode === 'term-kanji', +            modeTermKana: mode === 'term-kana', +            modeKanji: mode === 'kanji', +            compactGlossaries: options.general.compactGlossaries +        }; +        const pattern = /\{([\w-]+)\}/g; +        return await AnkiNoteBuilder.stringReplaceAsync(field, pattern, async (g0, marker) => { +            data.marker = marker; +            try { +                return await this._renderTemplate(templates, data); +            } catch (e) { +                if (errors) { errors.push(e); } +                return `{${marker}-render-error}`; +            } +        }); +    } + +    static stringReplaceAsync(str, regex, replacer) { +        let match; +        let index = 0; +        const parts = []; +        while ((match = regex.exec(str)) !== null) { +            parts.push(str.substring(index, match.index), replacer(...match, match.index, str)); +            index = regex.lastIndex; +        } +        if (parts.length === 0) { +            return Promise.resolve(str); +        } +        parts.push(str.substring(index)); +        return Promise.all(parts).then((v) => v.join('')); +    } +} diff --git a/ext/bg/js/anki.js b/ext/bg/js/anki.js index 39c6ad51..a70388bd 100644 --- a/ext/bg/js/anki.js +++ b/ext/bg/js/anki.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global requestJson*/ +/* global + * requestJson + */  /*   * AnkiConnect diff --git a/ext/bg/js/api.js b/ext/bg/js/api.js deleted file mode 100644 index 0c244ffa..00000000 --- a/ext/bg/js/api.js +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright (C) 2019-2020  Alex Yatskov <alex@foosoft.net> - * Author: Alex Yatskov <alex@foosoft.net> - * - * 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 <https://www.gnu.org/licenses/>. - */ - - -function apiTemplateRender(template, data) { -    return _apiInvoke('templateRender', {data, template}); -} - -function apiAudioGetUrl(definition, source, optionsContext) { -    return _apiInvoke('audioGetUrl', {definition, source, optionsContext}); -} - -function apiClipboardGet() { -    return _apiInvoke('clipboardGet'); -} - -function _apiInvoke(action, params={}) { -    const data = {action, params}; -    return new Promise((resolve, reject) => { -        try { -            const callback = (response) => { -                if (response !== null && typeof response === 'object') { -                    if (typeof response.error !== 'undefined') { -                        reject(jsonToError(response.error)); -                    } else { -                        resolve(response.result); -                    } -                } else { -                    const message = response === null ? 'Unexpected null response' : `Unexpected response of type ${typeof response}`; -                    reject(new Error(`${message} (${JSON.stringify(data)})`)); -                } -            }; -            const backend = window.yomichanBackend; -            backend.onMessage({action, params}, null, callback); -        } catch (e) { -            reject(e); -            yomichan.triggerOrphaned(e); -        } -    }); -} diff --git a/ext/bg/js/audio.js b/ext/bg/js/audio-uri-builder.js index d300570b..499c3441 100644 --- a/ext/bg/js/audio.js +++ b/ext/bg/js/audio-uri-builder.js @@ -16,10 +16,53 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global jpIsStringEntirelyKana, audioGetFromSources*/ +/* global + * jpIsStringEntirelyKana + */ + +class AudioUriBuilder { +    constructor() { +        this._getUrlHandlers = new Map([ +            ['jpod101', this._getUriJpod101.bind(this)], +            ['jpod101-alternate', this._getUriJpod101Alternate.bind(this)], +            ['jisho', this._getUriJisho.bind(this)], +            ['text-to-speech', this._getUriTextToSpeech.bind(this)], +            ['text-to-speech-reading', this._getUriTextToSpeechReading.bind(this)], +            ['custom', this._getUriCustom.bind(this)] +        ]); +    } + +    normalizeUrl(url, baseUrl, basePath) { +        if (url) { +            if (url[0] === '/') { +                if (url.length >= 2 && url[1] === '/') { +                    // Begins with "//" +                    url = baseUrl.substring(0, baseUrl.indexOf(':') + 1) + url; +                } else { +                    // Begins with "/" +                    url = baseUrl + url; +                } +            } else if (!/^[a-z][a-z0-9\-+.]*:/i.test(url)) { +                // No URI scheme => relative path +                url = baseUrl + basePath + url; +            } +        } +        return url; +    } -const audioUrlBuilders = new Map([ -    ['jpod101', async (definition) => { +    async getUri(definition, source, options) { +        const handler = this._getUrlHandlers.get(source); +        if (typeof handler === 'function') { +            try { +                return await handler(definition, options); +            } catch (e) { +                // NOP +            } +        } +        return null; +    } + +    async _getUriJpod101(definition) {          let kana = definition.reading;          let kanji = definition.expression; @@ -37,8 +80,9 @@ const audioUrlBuilders = new Map([          }          return `https://assets.languagepod101.com/dictionary/japanese/audiomp3.php?${params.join('&')}`; -    }], -    ['jpod101-alternate', async (definition) => { +    } + +    async _getUriJpod101Alternate(definition) {          const response = await new Promise((resolve, reject) => {              const xhr = new XMLHttpRequest();              xhr.open('POST', 'https://www.japanesepod101.com/learningcenter/reference/dictionary_post'); @@ -54,7 +98,7 @@ const audioUrlBuilders = new Map([                  const url = row.querySelector('audio>source[src]').getAttribute('src');                  const reading = row.getElementsByClassName('dc-vocab_kana').item(0).textContent;                  if (url && reading && (!definition.reading || definition.reading === reading)) { -                    return audioUrlNormalize(url, 'https://www.japanesepod101.com', '/learningcenter/reference/'); +                    return this.normalizeUrl(url, 'https://www.japanesepod101.com', '/learningcenter/reference/');                  }              } catch (e) {                  // NOP @@ -62,8 +106,9 @@ const audioUrlBuilders = new Map([          }          throw new Error('Failed to find audio URL'); -    }], -    ['jisho', async (definition) => { +    } + +    async _getUriJisho(definition) {          const response = await new Promise((resolve, reject) => {              const xhr = new XMLHttpRequest();              xhr.open('GET', `https://jisho.org/search/${definition.expression}`); @@ -78,7 +123,7 @@ const audioUrlBuilders = new Map([              if (audio !== null) {                  const url = audio.getElementsByTagName('source').item(0).getAttribute('src');                  if (url) { -                    return audioUrlNormalize(url, 'https://jisho.org', '/search/'); +                    return this.normalizeUrl(url, 'https://jisho.org', '/search/');                  }              }          } catch (e) { @@ -86,101 +131,28 @@ const audioUrlBuilders = new Map([          }          throw new Error('Failed to find audio URL'); -    }], -    ['text-to-speech', async (definition, options) => { +    } + +    async _getUriTextToSpeech(definition, options) {          const voiceURI = options.audio.textToSpeechVoice;          if (!voiceURI) {              throw new Error('No voice');          }          return `tts:?text=${encodeURIComponent(definition.expression)}&voice=${encodeURIComponent(voiceURI)}`; -    }], -    ['text-to-speech-reading', async (definition, options) => { +    } + +    async _getUriTextToSpeechReading(definition, options) {          const voiceURI = options.audio.textToSpeechVoice;          if (!voiceURI) {              throw new Error('No voice');          }          return `tts:?text=${encodeURIComponent(definition.reading || definition.expression)}&voice=${encodeURIComponent(voiceURI)}`; -    }], -    ['custom', async (definition, options) => { -        const customSourceUrl = options.audio.customSourceUrl; -        return customSourceUrl.replace(/\{([^}]*)\}/g, (m0, m1) => (hasOwn(definition, m1) ? `${definition[m1]}` : m0)); -    }] -]); - -async function audioGetUrl(definition, mode, options, download) { -    const handler = audioUrlBuilders.get(mode); -    if (typeof handler === 'function') { -        try { -            return await handler(definition, options, download); -        } catch (e) { -            // NOP -        } -    } -    return null; -} - -function audioUrlNormalize(url, baseUrl, basePath) { -    if (url) { -        if (url[0] === '/') { -            if (url.length >= 2 && url[1] === '/') { -                // Begins with "//" -                url = baseUrl.substring(0, baseUrl.indexOf(':') + 1) + url; -            } else { -                // Begins with "/" -                url = baseUrl + url; -            } -        } else if (!/^[a-z][a-z0-9\-+.]*:/i.test(url)) { -            // No URI scheme => relative path -            url = baseUrl + basePath + url; -        } -    } -    return url; -} - -function audioBuildFilename(definition) { -    if (definition.reading || definition.expression) { -        let filename = 'yomichan'; -        if (definition.reading) { -            filename += `_${definition.reading}`; -        } -        if (definition.expression) { -            filename += `_${definition.expression}`; -        } - -        return filename += '.mp3'; -    } -    return null; -} - -async function audioInject(definition, fields, sources, optionsContext) { -    let usesAudio = false; -    for (const name in fields) { -        if (fields[name].includes('{audio}')) { -            usesAudio = true; -            break; -        } -    } - -    if (!usesAudio) { -        return true;      } -    try { -        const expressions = definition.expressions; -        const audioSourceDefinition = Array.isArray(expressions) ? expressions[0] : definition; - -        const {url} = await audioGetFromSources(audioSourceDefinition, sources, optionsContext, true); -        if (url !== null) { -            const filename = audioBuildFilename(audioSourceDefinition); -            if (filename !== null) { -                definition.audio = {url, filename}; -            } -        } - -        return true; -    } catch (e) { -        return false; +    async _getUriCustom(definition, options) { +        const customSourceUrl = options.audio.customSourceUrl; +        return customSourceUrl.replace(/\{([^}]*)\}/g, (m0, m1) => (hasOwn(definition, m1) ? `${definition[m1]}` : m0));      }  } diff --git a/ext/bg/js/backend.js b/ext/bg/js/backend.js index e3bf7bda..978c5a4a 100644 --- a/ext/bg/js/backend.js +++ b/ext/bg/js/backend.js @@ -16,30 +16,51 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global optionsSave, utilIsolate -conditionsTestValue, profileConditionsDescriptor, profileOptionsGetDefaultFieldTemplates -handlebarsRenderDynamic -requestText, requestJson, optionsLoad -dictConfigured, dictTermsSort, dictEnabledSet, dictNoteFormat -audioGetUrl, audioInject -jpConvertReading, jpDistributeFuriganaInflected, jpKatakanaToHiragana -Translator, AnkiConnect, AnkiNull, Mecab, BackendApiForwarder, JsonSchema, ClipboardMonitor*/ +/* global + * AnkiConnect + * AnkiNoteBuilder + * AnkiNull + * AudioSystem + * AudioUriBuilder + * BackendApiForwarder + * ClipboardMonitor + * JsonSchema + * Mecab + * Translator + * conditionsTestValue + * dictConfigured + * dictEnabledSet + * dictTermsSort + * handlebarsRenderDynamic + * jpConvertReading + * jpDistributeFuriganaInflected + * jpKatakanaToHiragana + * optionsLoad + * optionsSave + * profileConditionsDescriptor + * requestJson + * requestText + * utilIsolate + */  class Backend {      constructor() {          this.translator = new Translator();          this.anki = new AnkiNull();          this.mecab = new Mecab(); -        this.clipboardMonitor = new ClipboardMonitor(); +        this.clipboardMonitor = new ClipboardMonitor({getClipboard: this._onApiClipboardGet.bind(this)}); +        this.ankiNoteBuilder = new AnkiNoteBuilder({renderTemplate: this._renderTemplate.bind(this)});          this.options = null;          this.optionsSchema = null; +        this.defaultAnkiFieldTemplates = null; +        this.audioSystem = new AudioSystem({getAudioUri: this._getAudioUri.bind(this)}); +        this.audioUriBuilder = new AudioUriBuilder();          this.optionsContext = {              depth: 0,              url: window.location.href          }; -        this.isPreparedResolve = null; -        this.isPreparedPromise = new Promise((resolve) => (this.isPreparedResolve = resolve)); +        this.isPrepared = false;          this.clipboardPasteTarget = document.querySelector('#clipboard-paste-target'); @@ -48,12 +69,50 @@ class Backend {          this.apiForwarder = new BackendApiForwarder();          this.messageToken = yomichan.generateId(16); + +        this._messageHandlers = new Map([ +            ['yomichanCoreReady', this._onApiYomichanCoreReady.bind(this)], +            ['optionsSchemaGet', this._onApiOptionsSchemaGet.bind(this)], +            ['optionsGet', this._onApiOptionsGet.bind(this)], +            ['optionsGetFull', this._onApiOptionsGetFull.bind(this)], +            ['optionsSet', this._onApiOptionsSet.bind(this)], +            ['optionsSave', this._onApiOptionsSave.bind(this)], +            ['kanjiFind', this._onApiKanjiFind.bind(this)], +            ['termsFind', this._onApiTermsFind.bind(this)], +            ['textParse', this._onApiTextParse.bind(this)], +            ['textParseMecab', this._onApiTextParseMecab.bind(this)], +            ['definitionAdd', this._onApiDefinitionAdd.bind(this)], +            ['definitionsAddable', this._onApiDefinitionsAddable.bind(this)], +            ['noteView', this._onApiNoteView.bind(this)], +            ['templateRender', this._onApiTemplateRender.bind(this)], +            ['commandExec', this._onApiCommandExec.bind(this)], +            ['audioGetUri', this._onApiAudioGetUri.bind(this)], +            ['screenshotGet', this._onApiScreenshotGet.bind(this)], +            ['forward', this._onApiForward.bind(this)], +            ['frameInformationGet', this._onApiFrameInformationGet.bind(this)], +            ['injectStylesheet', this._onApiInjectStylesheet.bind(this)], +            ['getEnvironmentInfo', this._onApiGetEnvironmentInfo.bind(this)], +            ['clipboardGet', this._onApiClipboardGet.bind(this)], +            ['getDisplayTemplatesHtml', this._onApiGetDisplayTemplatesHtml.bind(this)], +            ['getQueryParserTemplatesHtml', this._onApiGetQueryParserTemplatesHtml.bind(this)], +            ['getZoom', this._onApiGetZoom.bind(this)], +            ['getMessageToken', this._onApiGetMessageToken.bind(this)], +            ['getDefaultAnkiFieldTemplates', this._onApiGetDefaultAnkiFieldTemplates.bind(this)] +        ]); + +        this._commandHandlers = new Map([ +            ['search', this._onCommandSearch.bind(this)], +            ['help', this._onCommandHelp.bind(this)], +            ['options', this._onCommandOptions.bind(this)], +            ['toggle', this._onCommandToggle.bind(this)] +        ]);      }      async prepare() {          await this.translator.prepare();          this.optionsSchema = await requestJson(chrome.runtime.getURL('/bg/data/options-schema.json'), 'GET'); +        this.defaultAnkiFieldTemplates = await requestText(chrome.runtime.getURL('/bg/data/default-anki-field-templates.handlebars'), 'GET');          this.options = await optionsLoad();          try {              this.options = JsonSchema.getValidValueOrDefault(this.optionsSchema, this.options); @@ -65,42 +124,47 @@ class Backend {          this.onOptionsUpdated('background');          if (isObject(chrome.commands) && isObject(chrome.commands.onCommand)) { -            chrome.commands.onCommand.addListener((command) => this._runCommand(command)); +            chrome.commands.onCommand.addListener(this._runCommand.bind(this));          }          if (isObject(chrome.tabs) && isObject(chrome.tabs.onZoomChange)) { -            chrome.tabs.onZoomChange.addListener((info) => this._onZoomChange(info)); +            chrome.tabs.onZoomChange.addListener(this._onZoomChange.bind(this));          }          chrome.runtime.onMessage.addListener(this.onMessage.bind(this)); -        const options = this.getOptionsSync(this.optionsContext); +        this.isPrepared = true; + +        const options = this.getOptions(this.optionsContext);          if (options.general.showGuide) {              chrome.tabs.create({url: chrome.runtime.getURL('/bg/guide.html')});          } -        this.isPreparedResolve(); -        this.isPreparedResolve = null; -        this.isPreparedPromise = null; +        this.clipboardMonitor.on('change', this._onClipboardText.bind(this)); -        this.clipboardMonitor.onClipboardText = (text) => this._onClipboardText(text); +        this._sendMessageAllTabs('backendPrepared'); +        const callback = () => this.checkLastError(chrome.runtime.lastError); +        chrome.runtime.sendMessage({action: 'backendPrepared'}, callback);      } -    onOptionsUpdated(source) { -        this.applyOptions(); - +    _sendMessageAllTabs(action, params={}) {          const callback = () => this.checkLastError(chrome.runtime.lastError);          chrome.tabs.query({}, (tabs) => {              for (const tab of tabs) { -                chrome.tabs.sendMessage(tab.id, {action: 'optionsUpdated', params: {source}}, callback); +                chrome.tabs.sendMessage(tab.id, {action, params}, callback);              }          });      } +    onOptionsUpdated(source) { +        this.applyOptions(); +        this._sendMessageAllTabs('optionsUpdated', {source}); +    } +      onMessage({action, params}, sender, callback) { -        const handler = Backend._messageHandlers.get(action); +        const handler = this._messageHandlers.get(action);          if (typeof handler !== 'function') { return false; }          try { -            const promise = handler(this, params, sender); +            const promise = handler(params, sender);              promise.then(                  (result) => callback({result}),                  (error) => callback({error: errorToJson(error)}) @@ -112,7 +176,7 @@ class Backend {          }      } -    _onClipboardText(text) { +    _onClipboardText({text}) {          this._onCommandSearch({mode: 'popup', query: text});      } @@ -122,7 +186,7 @@ class Backend {      }      applyOptions() { -        const options = this.getOptionsSync(this.optionsContext); +        const options = this.getOptions(this.optionsContext);          if (!options.general.enable) {              this.setExtensionBadgeBackgroundColor('#555555');              this.setExtensionBadgeText('off'); @@ -148,24 +212,15 @@ class Backend {          }      } -    async getOptionsSchema() { -        if (this.isPreparedPromise !== null) { -            await this.isPreparedPromise; -        } +    getOptionsSchema() {          return this.optionsSchema;      } -    async getFullOptions() { -        if (this.isPreparedPromise !== null) { -            await this.isPreparedPromise; -        } +    getFullOptions() {          return this.options;      } -    async setFullOptions(options) { -        if (this.isPreparedPromise !== null) { -            await this.isPreparedPromise; -        } +    setFullOptions(options) {          try {              this.options = JsonSchema.getValidValueOrDefault(this.optionsSchema, utilIsolate(options));          } catch (e) { @@ -174,18 +229,11 @@ class Backend {          }      } -    async getOptions(optionsContext) { -        if (this.isPreparedPromise !== null) { -            await this.isPreparedPromise; -        } -        return this.getOptionsSync(optionsContext); -    } - -    getOptionsSync(optionsContext) { -        return this.getProfileSync(optionsContext).options; +    getOptions(optionsContext) { +        return this.getProfile(optionsContext).options;      } -    getProfileSync(optionsContext) { +    getProfile(optionsContext) {          const profiles = this.options.profiles;          if (typeof optionsContext.index === 'number') {              return profiles[optionsContext.index]; @@ -243,29 +291,43 @@ class Backend {      }      _runCommand(command, params) { -        const handler = Backend._commandHandlers.get(command); +        const handler = this._commandHandlers.get(command);          if (typeof handler !== 'function') { return false; } -        handler(this, params); +        handler(params);          return true;      }      // Message handlers -    _onApiOptionsSchemaGet() { +    _onApiYomichanCoreReady(_params, sender) { +        // tab ID isn't set in background (e.g. browser_action) +        if (typeof sender.tab === 'undefined') { +            const callback = () => this.checkLastError(chrome.runtime.lastError); +            chrome.runtime.sendMessage({action: 'backendPrepared'}, callback); +            return Promise.resolve(); +        } + +        const tabId = sender.tab.id; +        return new Promise((resolve) => { +            chrome.tabs.sendMessage(tabId, {action: 'backendPrepared'}, resolve); +        }); +    } + +    async _onApiOptionsSchemaGet() {          return this.getOptionsSchema();      } -    _onApiOptionsGet({optionsContext}) { +    async _onApiOptionsGet({optionsContext}) {          return this.getOptions(optionsContext);      } -    _onApiOptionsGetFull() { +    async _onApiOptionsGetFull() {          return this.getFullOptions();      }      async _onApiOptionsSet({changedOptions, optionsContext, source}) { -        const options = await this.getOptions(optionsContext); +        const options = this.getOptions(optionsContext);          function getValuePaths(obj) {              const valuePaths = []; @@ -305,20 +367,20 @@ class Backend {      }      async _onApiOptionsSave({source}) { -        const options = await this.getFullOptions(); +        const options = this.getFullOptions();          await optionsSave(options);          this.onOptionsUpdated(source);      }      async _onApiKanjiFind({text, optionsContext}) { -        const options = await this.getOptions(optionsContext); +        const options = this.getOptions(optionsContext);          const definitions = await this.translator.findKanji(text, options);          definitions.splice(options.general.maxResults);          return definitions;      }      async _onApiTermsFind({text, details, optionsContext}) { -        const options = await this.getOptions(optionsContext); +        const options = this.getOptions(optionsContext);          const mode = options.general.resultOutputMode;          const [definitions, length] = await this.translator.findTerms(mode, text, details, options);          definitions.splice(options.general.maxResults); @@ -326,7 +388,7 @@ class Backend {      }      async _onApiTextParse({text, optionsContext}) { -        const options = await this.getOptions(optionsContext); +        const options = this.getOptions(optionsContext);          const results = [];          while (text.length > 0) {              const term = []; @@ -356,12 +418,12 @@ class Backend {      }      async _onApiTextParseMecab({text, optionsContext}) { -        const options = await this.getOptions(optionsContext); -        const results = {}; +        const options = this.getOptions(optionsContext); +        const results = [];          const rawResults = await this.mecab.parseText(text); -        for (const mecabName in rawResults) { +        for (const [mecabName, parsedLines] of Object.entries(rawResults)) {              const result = []; -            for (const parsedLine of rawResults[mecabName]) { +            for (const parsedLine of parsedLines) {                  for (const {expression, reading, source} of parsedLine) {                      const term = [];                      if (expression !== null && reading !== null) { @@ -381,17 +443,17 @@ class Backend {                  }                  result.push([{text: '\n'}]);              } -            results[mecabName] = result; +            results.push([mecabName, result]);          }          return results;      }      async _onApiDefinitionAdd({definition, mode, context, optionsContext}) { -        const options = await this.getOptions(optionsContext); -        const templates = Backend._getTemplates(options); +        const options = this.getOptions(optionsContext); +        const templates = this.defaultAnkiFieldTemplates;          if (mode !== 'kanji') { -            await audioInject( +            await this._audioInject(                  definition,                  options.anki.terms.fields,                  options.audio.sources, @@ -407,20 +469,20 @@ class Backend {              );          } -        const note = await dictNoteFormat(definition, mode, options, templates); +        const note = await this.ankiNoteBuilder.createNote(definition, mode, options, templates);          return this.anki.addNote(note);      }      async _onApiDefinitionsAddable({definitions, modes, optionsContext}) { -        const options = await this.getOptions(optionsContext); -        const templates = Backend._getTemplates(options); +        const options = this.getOptions(optionsContext); +        const templates = this.defaultAnkiFieldTemplates;          const states = [];          try {              const notes = [];              for (const definition of definitions) {                  for (const mode of modes) { -                    const note = await dictNoteFormat(definition, mode, options, templates); +                    const note = await this.ankiNoteBuilder.createNote(definition, mode, options, templates);                      notes.push(note);                  }              } @@ -459,20 +521,20 @@ class Backend {      }      async _onApiNoteView({noteId}) { -        return this.anki.guiBrowse(`nid:${noteId}`); +        return await this.anki.guiBrowse(`nid:${noteId}`);      }      async _onApiTemplateRender({template, data}) { -        return handlebarsRenderDynamic(template, data); +        return this._renderTemplate(template, data);      }      async _onApiCommandExec({command, params}) {          return this._runCommand(command, params);      } -    async _onApiAudioGetUrl({definition, source, optionsContext}) { -        const options = await this.getOptions(optionsContext); -        return await audioGetUrl(definition, source, options); +    async _onApiAudioGetUri({definition, source, optionsContext}) { +        const options = this.getOptions(optionsContext); +        return await this.audioUriBuilder.getUri(definition, source, options);      }      _onApiScreenshotGet({options}, sender) { @@ -621,12 +683,16 @@ class Backend {          return this.messageToken;      } +    async _onApiGetDefaultAnkiFieldTemplates() { +        return this.defaultAnkiFieldTemplates; +    } +      // Command handlers      async _onCommandSearch(params) {          const {mode='existingOrNewTab', query} = params || {}; -        const options = await this.getOptions(this.optionsContext); +        const options = this.getOptions(this.optionsContext);          const {popupWidth, popupHeight} = options.general;          const baseUrl = chrome.runtime.getURL('/bg/search.html'); @@ -647,7 +713,7 @@ class Backend {                  await Backend._focusTab(tab);                  if (queryParams.query) {                      await new Promise((resolve) => chrome.tabs.sendMessage( -                        tab.id, {action: 'searchQueryUpdate', params: {query: queryParams.query}}, resolve +                        tab.id, {action: 'searchQueryUpdate', params: {text: queryParams.query}}, resolve                      ));                  }                  return true; @@ -693,9 +759,10 @@ class Backend {      }      _onCommandOptions(params) { -        if (!(params && params.newTab)) { +        const {mode='existingOrNewTab'} = params || {}; +        if (mode === 'existingOrNewTab') {              chrome.runtime.openOptionsPage(); -        } else { +        } else if (mode === 'newTab') {              const manifest = chrome.runtime.getManifest();              const url = chrome.runtime.getURL(manifest.options_ui.page);              chrome.tabs.create({url}); @@ -709,17 +776,56 @@ class Backend {          };          const source = 'popup'; -        const options = await this.getOptions(optionsContext); +        const options = this.getOptions(optionsContext);          options.general.enable = !options.general.enable;          await this._onApiOptionsSave({source});      }      // Utilities +    async _getAudioUri(definition, source, details) { +        let optionsContext = (typeof details === 'object' && details !== null ? details.optionsContext : null); +        if (!(typeof optionsContext === 'object' && optionsContext !== null)) { +            optionsContext = this.optionsContext; +        } + +        const options = this.getOptions(optionsContext); +        return await this.audioUriBuilder.getUri(definition, source, options); +    } + +    async _audioInject(definition, fields, sources, optionsContext) { +        let usesAudio = false; +        for (const fieldValue of Object.values(fields)) { +            if (fieldValue.includes('{audio}')) { +                usesAudio = true; +                break; +            } +        } + +        if (!usesAudio) { +            return true; +        } + +        try { +            const expressions = definition.expressions; +            const audioSourceDefinition = Array.isArray(expressions) ? expressions[0] : definition; + +            const {uri} = await this.audioSystem.getDefinitionAudio(audioSourceDefinition, sources, {tts: false, optionsContext}); +            const filename = this._createInjectedAudioFileName(audioSourceDefinition); +            if (filename !== null) { +                definition.audio = {url: uri, filename}; +            } + +            return true; +        } catch (e) { +            return false; +        } +    } +      async _injectScreenshot(definition, fields, screenshot) {          let usesScreenshot = false; -        for (const name in fields) { -            if (fields[name].includes('{screenshot}')) { +        for (const fieldValue of Object.values(fields)) { +            if (fieldValue.includes('{screenshot}')) {                  usesScreenshot = true;                  break;              } @@ -752,6 +858,21 @@ class Backend {          definition.screenshotFileName = filename;      } +    async _renderTemplate(template, data) { +        return handlebarsRenderDynamic(template, data); +    } + +    _createInjectedAudioFileName(definition) { +        const {reading, expression} = definition; +        if (!reading && !expression) { return null; } + +        let filename = 'yomichan'; +        if (reading) { filename += `_${reading}`; } +        if (expression) { filename += `_${expression}`; } +        filename += '.mp3'; +        return filename; +    } +      static _getTabUrl(tab) {          return new Promise((resolve) => {              chrome.tabs.sendMessage(tab.id, {action: 'getUrl'}, {frameId: 0}, (response) => { @@ -860,47 +981,7 @@ class Backend {              return 'chrome';          }      } - -    static _getTemplates(options) { -        const templates = options.anki.fieldTemplates; -        return typeof templates === 'string' ? templates : profileOptionsGetDefaultFieldTemplates(); -    }  } -Backend._messageHandlers = new Map([ -    ['optionsSchemaGet', (self, ...args) => self._onApiOptionsSchemaGet(...args)], -    ['optionsGet', (self, ...args) => self._onApiOptionsGet(...args)], -    ['optionsGetFull', (self, ...args) => self._onApiOptionsGetFull(...args)], -    ['optionsSet', (self, ...args) => self._onApiOptionsSet(...args)], -    ['optionsSave', (self, ...args) => self._onApiOptionsSave(...args)], -    ['kanjiFind', (self, ...args) => self._onApiKanjiFind(...args)], -    ['termsFind', (self, ...args) => self._onApiTermsFind(...args)], -    ['textParse', (self, ...args) => self._onApiTextParse(...args)], -    ['textParseMecab', (self, ...args) => self._onApiTextParseMecab(...args)], -    ['definitionAdd', (self, ...args) => self._onApiDefinitionAdd(...args)], -    ['definitionsAddable', (self, ...args) => self._onApiDefinitionsAddable(...args)], -    ['noteView', (self, ...args) => self._onApiNoteView(...args)], -    ['templateRender', (self, ...args) => self._onApiTemplateRender(...args)], -    ['commandExec', (self, ...args) => self._onApiCommandExec(...args)], -    ['audioGetUrl', (self, ...args) => self._onApiAudioGetUrl(...args)], -    ['screenshotGet', (self, ...args) => self._onApiScreenshotGet(...args)], -    ['forward', (self, ...args) => self._onApiForward(...args)], -    ['frameInformationGet', (self, ...args) => self._onApiFrameInformationGet(...args)], -    ['injectStylesheet', (self, ...args) => self._onApiInjectStylesheet(...args)], -    ['getEnvironmentInfo', (self, ...args) => self._onApiGetEnvironmentInfo(...args)], -    ['clipboardGet', (self, ...args) => self._onApiClipboardGet(...args)], -    ['getDisplayTemplatesHtml', (self, ...args) => self._onApiGetDisplayTemplatesHtml(...args)], -    ['getQueryParserTemplatesHtml', (self, ...args) => self._onApiGetQueryParserTemplatesHtml(...args)], -    ['getZoom', (self, ...args) => self._onApiGetZoom(...args)], -    ['getMessageToken', (self, ...args) => self._onApiGetMessageToken(...args)] -]); - -Backend._commandHandlers = new Map([ -    ['search', (self, ...args) => self._onCommandSearch(...args)], -    ['help', (self, ...args) => self._onCommandHelp(...args)], -    ['options', (self, ...args) => self._onCommandOptions(...args)], -    ['toggle', (self, ...args) => self._onCommandToggle(...args)] -]); -  window.yomichanBackend = new Backend();  window.yomichanBackend.prepare(); diff --git a/ext/bg/js/clipboard-monitor.js b/ext/bg/js/clipboard-monitor.js index c2f41385..9a881f57 100644 --- a/ext/bg/js/clipboard-monitor.js +++ b/ext/bg/js/clipboard-monitor.js @@ -16,66 +16,66 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiClipboardGet, jpIsStringPartiallyJapanese*/ - -class ClipboardMonitor { -    constructor() { -        this.timerId = null; -        this.timerToken = null; -        this.interval = 250; -        this.previousText = null; -    } +/* global + * jpIsStringPartiallyJapanese + */ -    onClipboardText(_text) { -        throw new Error('Override me'); +class ClipboardMonitor extends EventDispatcher { +    constructor({getClipboard}) { +        super(); +        this._timerId = null; +        this._timerToken = null; +        this._interval = 250; +        this._previousText = null; +        this._getClipboard = getClipboard;      }      start() {          this.stop();          // The token below is used as a unique identifier to ensure that a new clipboard monitor -        // hasn't been started during the await call. The check below the await apiClipboardGet() +        // hasn't been started during the await call. The check below the await this._getClipboard()          // call will exit early if the reference has changed.          const token = {};          const intervalCallback = async () => { -            this.timerId = null; +            this._timerId = null;              let text = null;              try { -                text = await apiClipboardGet(); +                text = await this._getClipboard();              } catch (e) {                  // NOP              } -            if (this.timerToken !== token) { return; } +            if (this._timerToken !== token) { return; }              if (                  typeof text === 'string' &&                  (text = text.trim()).length > 0 && -                text !== this.previousText +                text !== this._previousText              ) { -                this.previousText = text; +                this._previousText = text;                  if (jpIsStringPartiallyJapanese(text)) { -                    this.onClipboardText(text); +                    this.trigger('change', {text});                  }              } -            this.timerId = setTimeout(intervalCallback, this.interval); +            this._timerId = setTimeout(intervalCallback, this._interval);          }; -        this.timerToken = token; +        this._timerToken = token;          intervalCallback();      }      stop() { -        this.timerToken = null; -        if (this.timerId !== null) { -            clearTimeout(this.timerId); -            this.timerId = null; +        this._timerToken = null; +        if (this._timerId !== null) { +            clearTimeout(this._timerId); +            this._timerId = null;          }      }      setPreviousText(text) { -        this.previousText = text; +        this._previousText = text;      }  } diff --git a/ext/bg/js/context.js b/ext/bg/js/context.js index bec964fb..c3e74656 100644 --- a/ext/bg/js/context.js +++ b/ext/bg/js/context.js @@ -16,7 +16,11 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiCommandExec, apiGetEnvironmentInfo, apiOptionsGet*/ +/* global + * apiCommandExec + * apiGetEnvironmentInfo + * apiOptionsGet + */  function showExtensionInfo() {      const node = document.getElementById('extension-info'); @@ -48,7 +52,9 @@ function setupButtonEvents(selector, command, url) {      }  } -window.addEventListener('DOMContentLoaded', () => { +window.addEventListener('DOMContentLoaded', async () => { +    await yomichan.prepare(); +      showExtensionInfo();      apiGetEnvironmentInfo().then(({browser}) => { diff --git a/ext/bg/js/database.js b/ext/bg/js/database.js index 558f3ceb..08a2a39f 100644 --- a/ext/bg/js/database.js +++ b/ext/bg/js/database.js @@ -16,7 +16,12 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global dictFieldSplit, requestJson, JsonSchema, JSZip*/ +/* global + * JSZip + * JsonSchema + * dictFieldSplit + * requestJson + */  class Database {      constructor() { diff --git a/ext/bg/js/deinflector.js b/ext/bg/js/deinflector.js index e2ced965..d548d271 100644 --- a/ext/bg/js/deinflector.js +++ b/ext/bg/js/deinflector.js @@ -57,9 +57,9 @@ class Deinflector {      static normalizeReasons(reasons) {          const normalizedReasons = []; -        for (const reason in reasons) { +        for (const [reason, reasonInfo] of Object.entries(reasons)) {              const variants = []; -            for (const {kanaIn, kanaOut, rulesIn, rulesOut} of reasons[reason]) { +            for (const {kanaIn, kanaOut, rulesIn, rulesOut} of reasonInfo) {                  variants.push([                      kanaIn,                      kanaOut, diff --git a/ext/bg/js/dictionary.js b/ext/bg/js/dictionary.js index f5c5b21b..3dd1d0c1 100644 --- a/ext/bg/js/dictionary.js +++ b/ext/bg/js/dictionary.js @@ -16,26 +16,18 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiTemplateRender*/ -  function dictEnabledSet(options) {      const enabledDictionaryMap = new Map(); -    const optionsDictionaries = options.dictionaries; -    for (const title in optionsDictionaries) { -        if (!hasOwn(optionsDictionaries, title)) { continue; } -        const dictionary = optionsDictionaries[title]; -        if (!dictionary.enabled) { continue; } -        enabledDictionaryMap.set(title, { -            priority: dictionary.priority || 0, -            allowSecondarySearches: !!dictionary.allowSecondarySearches -        }); +    for (const [title, {enabled, priority, allowSecondarySearches}] of Object.entries(options.dictionaries)) { +        if (!enabled) { continue; } +        enabledDictionaryMap.set(title, {priority, allowSecondarySearches});      }      return enabledDictionaryMap;  }  function dictConfigured(options) { -    for (const title in options.dictionaries) { -        if (options.dictionaries[title].enabled) { +    for (const {enabled} of Object.values(options.dictionaries)) { +        if (enabled) {              return true;          }      } @@ -339,90 +331,3 @@ function dictTagsSort(tags) {  function dictFieldSplit(field) {      return field.length === 0 ? [] : field.split(' ');  } - -async function dictFieldFormat(field, definition, mode, options, templates, exceptions) { -    const data = { -        marker: null, -        definition, -        group: options.general.resultOutputMode === 'group', -        merge: options.general.resultOutputMode === 'merge', -        modeTermKanji: mode === 'term-kanji', -        modeTermKana: mode === 'term-kana', -        modeKanji: mode === 'kanji', -        compactGlossaries: options.general.compactGlossaries -    }; -    const markers = dictFieldFormat.markers; -    const pattern = /\{([\w-]+)\}/g; -    return await stringReplaceAsync(field, pattern, async (g0, marker) => { -        if (!markers.has(marker)) { -            return g0; -        } -        data.marker = marker; -        try { -            return await apiTemplateRender(templates, data); -        } catch (e) { -            if (exceptions) { exceptions.push(e); } -            return `{${marker}-render-error}`; -        } -    }); -} -dictFieldFormat.markers = new Set([ -    'audio', -    'character', -    'cloze-body', -    'cloze-prefix', -    'cloze-suffix', -    'dictionary', -    'expression', -    'furigana', -    'furigana-plain', -    'glossary', -    'glossary-brief', -    'kunyomi', -    'onyomi', -    'reading', -    'screenshot', -    'sentence', -    'tags', -    'url' -]); - -async function dictNoteFormat(definition, mode, options, templates) { -    const note = {fields: {}, tags: options.anki.tags}; -    let fields = []; - -    if (mode === 'kanji') { -        fields = options.anki.kanji.fields; -        note.deckName = options.anki.kanji.deck; -        note.modelName = options.anki.kanji.model; -    } else { -        fields = options.anki.terms.fields; -        note.deckName = options.anki.terms.deck; -        note.modelName = options.anki.terms.model; - -        if (definition.audio) { -            const audio = { -                url: definition.audio.url, -                filename: definition.audio.filename, -                skipHash: '7e2c2f954ef6051373ba916f000168dc', -                fields: [] -            }; - -            for (const name in fields) { -                if (fields[name].includes('{audio}')) { -                    audio.fields.push(name); -                } -            } - -            if (audio.fields.length > 0) { -                note.audio = audio; -            } -        } -    } - -    for (const name in fields) { -        note.fields[name] = await dictFieldFormat(fields[name], definition, mode, options, templates); -    } - -    return note; -} diff --git a/ext/bg/js/handlebars.js b/ext/bg/js/handlebars.js index b1443447..e3ce6bd0 100644 --- a/ext/bg/js/handlebars.js +++ b/ext/bg/js/handlebars.js @@ -16,7 +16,11 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global jpIsCharCodeKanji, jpDistributeFurigana, Handlebars*/ +/* global + * Handlebars + * jpDistributeFurigana + * jpIsCodePointKanji + */  function handlebarsEscape(text) {      return Handlebars.Utils.escapeExpression(text); @@ -62,7 +66,7 @@ function handlebarsFuriganaPlain(options) {  function handlebarsKanjiLinks(options) {      let result = '';      for (const c of options.fn(this)) { -        if (jpIsCharCodeKanji(c.charCodeAt(0))) { +        if (jpIsCodePointKanji(c.codePointAt(0))) {              result += `<a href="#" class="kanji-link">${c}</a>`;          } else {              result += c; diff --git a/ext/bg/js/japanese.js b/ext/bg/js/japanese.js index abb32da4..3b37754d 100644 --- a/ext/bg/js/japanese.js +++ b/ext/bg/js/japanese.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global wanakana*/ +/* global + * wanakana + */  const JP_HALFWIDTH_KATAKANA_MAPPING = new Map([      ['ヲ', 'ヲヺ-'], @@ -115,9 +117,9 @@ const JP_JAPANESE_RANGES = [  // Helper functions -function _jpIsCharCodeInRanges(charCode, ranges) { +function _jpIsCodePointInRanges(codePoint, ranges) {      for (const [min, max] of ranges) { -        if (charCode >= min && charCode <= max) { +        if (codePoint >= min && codePoint <= max) {              return true;          }      } @@ -127,16 +129,16 @@ function _jpIsCharCodeInRanges(charCode, ranges) {  // Character code testing functions -function jpIsCharCodeKanji(charCode) { -    return _jpIsCharCodeInRanges(charCode, JP_CJK_RANGES); +function jpIsCodePointKanji(codePoint) { +    return _jpIsCodePointInRanges(codePoint, JP_CJK_RANGES);  } -function jpIsCharCodeKana(charCode) { -    return _jpIsCharCodeInRanges(charCode, JP_KANA_RANGES); +function jpIsCodePointKana(codePoint) { +    return _jpIsCodePointInRanges(codePoint, JP_KANA_RANGES);  } -function jpIsCharCodeJapanese(charCode) { -    return _jpIsCharCodeInRanges(charCode, JP_JAPANESE_RANGES); +function jpIsCodePointJapanese(codePoint) { +    return _jpIsCodePointInRanges(codePoint, JP_JAPANESE_RANGES);  } @@ -144,8 +146,8 @@ function jpIsCharCodeJapanese(charCode) {  function jpIsStringEntirelyKana(str) {      if (str.length === 0) { return false; } -    for (let i = 0, ii = str.length; i < ii; ++i) { -        if (!jpIsCharCodeKana(str.charCodeAt(i))) { +    for (const c of str) { +        if (!jpIsCodePointKana(c.codePointAt(0))) {              return false;          }      } @@ -154,8 +156,8 @@ function jpIsStringEntirelyKana(str) {  function jpIsStringPartiallyJapanese(str) {      if (str.length === 0) { return false; } -    for (let i = 0, ii = str.length; i < ii; ++i) { -        if (jpIsCharCodeJapanese(str.charCodeAt(i))) { +    for (const c of str) { +        if (jpIsCodePointJapanese(c.codePointAt(0))) {              return true;          }      } @@ -264,8 +266,8 @@ function jpDistributeFurigana(expression, reading) {      const groups = [];      let modePrev = null;      for (const c of expression) { -        const charCode = c.charCodeAt(0); -        const modeCurr = jpIsCharCodeKanji(charCode) || charCode === JP_ITERATION_MARK_CHAR_CODE ? 'kanji' : 'kana'; +        const codePoint = c.codePointAt(0); +        const modeCurr = jpIsCodePointKanji(codePoint) || codePoint === JP_ITERATION_MARK_CHAR_CODE ? 'kanji' : 'kana';          if (modeCurr === modePrev) {              groups[groups.length - 1].text += c;          } else { @@ -311,10 +313,11 @@ function jpDistributeFuriganaInflected(expression, reading, source) {  function jpConvertHalfWidthKanaToFullWidth(text, sourceMapping) {      let result = ''; -    const ii = text.length;      const hasSourceMapping = Array.isArray(sourceMapping); -    for (let i = 0; i < ii; ++i) { +    // This function is safe to use charCodeAt instead of codePointAt, since all +    // the relevant characters are represented with a single UTF-16 character code. +    for (let i = 0, ii = text.length; i < ii; ++i) {          const c = text[i];          const mapping = JP_HALFWIDTH_KATAKANA_MAPPING.get(c);          if (typeof mapping !== 'string') { @@ -355,13 +358,13 @@ function jpConvertHalfWidthKanaToFullWidth(text, sourceMapping) {  function jpConvertNumericTofullWidth(text) {      let result = ''; -    for (let i = 0, ii = text.length; i < ii; ++i) { -        let c = text.charCodeAt(i); +    for (const char of text) { +        let c = char.codePointAt(0);          if (c >= 0x30 && c <= 0x39) { // ['0', '9']              c += 0xff10 - 0x30; // 0xff10 = '0' full width -            result += String.fromCharCode(c); +            result += String.fromCodePoint(c);          } else { -            result += text[i]; +            result += char;          }      }      return result; @@ -377,9 +380,9 @@ function jpConvertAlphabeticToKana(text, sourceMapping) {          sourceMapping.fill(1);      } -    for (let i = 0; i < ii; ++i) { +    for (const char of text) {          // Note: 0x61 is the character code for 'a' -        let c = text.charCodeAt(i); +        let c = char.codePointAt(0);          if (c >= 0x41 && c <= 0x5a) { // ['A', 'Z']              c += (0x61 - 0x41);          } else if (c >= 0x61 && c <= 0x7a) { // ['a', 'z'] @@ -395,10 +398,10 @@ function jpConvertAlphabeticToKana(text, sourceMapping) {                  result += jpToHiragana(part, sourceMapping, result.length);                  part = '';              } -            result += text[i]; +            result += char;              continue;          } -        part += String.fromCharCode(c); +        part += String.fromCodePoint(c);      }      if (part.length > 0) { diff --git a/ext/bg/js/options.js b/ext/bg/js/options.js index f9db99a2..bd0bbe0e 100644 --- a/ext/bg/js/options.js +++ b/ext/bg/js/options.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global utilStringHashCode*/ +/* global + * utilStringHashCode + */  /*   * Generic options functions @@ -58,22 +60,17 @@ const profileOptionsVersionUpdates = [          options.scanning.modifier = options.scanning.requireShift ? 'shift' : 'none';      },      (options) => { -        const fieldTemplatesDefault = profileOptionsGetDefaultFieldTemplates();          options.general.resultOutputMode = options.general.groupResults ? 'group' : 'split'; -        options.anki.fieldTemplates = ( -            (utilStringHashCode(options.anki.fieldTemplates) !== -805327496) ? -            `{{#if merge}}${fieldTemplatesDefault}{{else}}${options.anki.fieldTemplates}{{/if}}` : -            fieldTemplatesDefault -        ); +        options.anki.fieldTemplates = null;      },      (options) => {          if (utilStringHashCode(options.anki.fieldTemplates) === 1285806040) { -            options.anki.fieldTemplates = profileOptionsGetDefaultFieldTemplates(); +            options.anki.fieldTemplates = null;          }      },      (options) => {          if (utilStringHashCode(options.anki.fieldTemplates) === -250091611) { -            options.anki.fieldTemplates = profileOptionsGetDefaultFieldTemplates(); +            options.anki.fieldTemplates = null;          }      },      (options) => { @@ -97,172 +94,6 @@ const profileOptionsVersionUpdates = [      }  ]; -function profileOptionsGetDefaultFieldTemplates() { -    return ` -{{#*inline "glossary-single"}} -    {{~#unless brief~}} -        {{~#if definitionTags~}}<i>({{#each definitionTags}}{{name}}{{#unless @last}}, {{/unless}}{{/each}})</i> {{/if~}} -        {{~#if only~}}({{#each only}}{{{.}}}{{#unless @last}}, {{/unless}}{{/each}} only) {{/if~}} -    {{~/unless~}} -    {{~#if glossary.[1]~}} -        {{~#if compactGlossaries~}} -            {{#each glossary}}{{#multiLine}}{{.}}{{/multiLine}}{{#unless @last}} | {{/unless}}{{/each}} -        {{~else~}} -            <ul>{{#each glossary}}<li>{{#multiLine}}{{.}}{{/multiLine}}</li>{{/each}}</ul> -        {{~/if~}} -    {{~else~}} -        {{~#multiLine}}{{glossary.[0]}}{{/multiLine~}} -    {{~/if~}} -{{/inline}} - -{{#*inline "audio"}}{{/inline}} - -{{#*inline "character"}} -    {{~definition.character~}} -{{/inline}} - -{{#*inline "dictionary"}} -    {{~definition.dictionary~}} -{{/inline}} - -{{#*inline "expression"}} -    {{~#if merge~}} -        {{~#if modeTermKana~}} -            {{~#each definition.reading~}} -                {{{.}}} -                {{~#unless @last}}、{{/unless~}} -            {{~else~}} -                {{~#each definition.expression~}} -                    {{{.}}} -                    {{~#unless @last}}、{{/unless~}} -                {{~/each~}} -            {{~/each~}} -        {{~else~}} -            {{~#each definition.expression~}} -                {{{.}}} -                {{~#unless @last}}、{{/unless~}} -            {{~/each~}} -        {{~/if~}} -    {{~else~}} -        {{~#if modeTermKana~}} -            {{~#if definition.reading~}} -                {{definition.reading}} -            {{~else~}} -                {{definition.expression}} -            {{~/if~}} -        {{~else~}} -            {{definition.expression}} -        {{~/if~}} -    {{~/if~}} -{{/inline}} - -{{#*inline "furigana"}} -    {{~#if merge~}} -        {{~#each definition.expressions~}} -            <span class="expression-{{termFrequency}}">{{~#furigana}}{{{.}}}{{/furigana~}}</span> -            {{~#unless @last}}、{{/unless~}} -        {{~/each~}} -    {{~else~}} -        {{#furigana}}{{{definition}}}{{/furigana}} -    {{~/if~}} -{{/inline}} - -{{#*inline "furigana-plain"}} -    {{~#if merge~}} -        {{~#each definition.expressions~}} -            <span class="expression-{{termFrequency}}">{{~#furiganaPlain}}{{{.}}}{{/furiganaPlain~}}</span> -            {{~#unless @last}}、{{/unless~}} -        {{~/each~}} -    {{~else~}} -        {{#furiganaPlain}}{{{definition}}}{{/furiganaPlain}} -    {{~/if~}} -{{/inline}} - -{{#*inline "glossary"}} -    <div style="text-align: left;"> -    {{~#if modeKanji~}} -        {{~#if definition.glossary.[1]~}} -            <ol>{{#each definition.glossary}}<li>{{.}}</li>{{/each}}</ol> -        {{~else~}} -            {{definition.glossary.[0]}} -        {{~/if~}} -    {{~else~}} -        {{~#if group~}} -            {{~#if definition.definitions.[1]~}} -                <ol>{{#each definition.definitions}}<li>{{> glossary-single brief=../brief compactGlossaries=../compactGlossaries}}</li>{{/each}}</ol> -            {{~else~}} -                {{~> glossary-single definition.definitions.[0] brief=brief compactGlossaries=compactGlossaries~}} -            {{~/if~}} -        {{~else if merge~}} -            {{~#if definition.definitions.[1]~}} -                <ol>{{#each definition.definitions}}<li>{{> glossary-single brief=../brief compactGlossaries=../compactGlossaries}}</li>{{/each}}</ol> -            {{~else~}} -                {{~> glossary-single definition.definitions.[0] brief=brief compactGlossaries=compactGlossaries~}} -            {{~/if~}} -        {{~else~}} -            {{~> glossary-single definition brief=brief compactGlossaries=compactGlossaries~}} -        {{~/if~}} -    {{~/if~}} -    </div> -{{/inline}} - -{{#*inline "glossary-brief"}} -    {{~> glossary brief=true ~}} -{{/inline}} - -{{#*inline "kunyomi"}} -    {{~#each definition.kunyomi}}{{.}}{{#unless @last}}, {{/unless}}{{/each~}} -{{/inline}} - -{{#*inline "onyomi"}} -    {{~#each definition.onyomi}}{{.}}{{#unless @last}}, {{/unless}}{{/each~}} -{{/inline}} - -{{#*inline "reading"}} -    {{~#unless modeTermKana~}} -        {{~#if merge~}} -            {{~#each definition.reading~}} -                {{{.}}} -                {{~#unless @last}}、{{/unless~}} -            {{~/each~}} -        {{~else~}} -            {{~definition.reading~}} -        {{~/if~}} -    {{~/unless~}} -{{/inline}} - -{{#*inline "sentence"}} -    {{~#if definition.cloze}}{{definition.cloze.sentence}}{{/if~}} -{{/inline}} - -{{#*inline "cloze-prefix"}} -    {{~#if definition.cloze}}{{definition.cloze.prefix}}{{/if~}} -{{/inline}} - -{{#*inline "cloze-body"}} -    {{~#if definition.cloze}}{{definition.cloze.body}}{{/if~}} -{{/inline}} - -{{#*inline "cloze-suffix"}} -    {{~#if definition.cloze}}{{definition.cloze.suffix}}{{/if~}} -{{/inline}} - -{{#*inline "tags"}} -    {{~#each definition.definitionTags}}{{name}}{{#unless @last}}, {{/unless}}{{/each~}} -{{/inline}} - -{{#*inline "url"}} -    <a href="{{definition.url}}">{{definition.url}}</a> -{{/inline}} - -{{#*inline "screenshot"}} -    <img src="{{definition.screenshotFileName}}" /> -{{/inline}} - -{{~> (lookup . "marker") ~}} -`.trim(); -} -  function profileOptionsCreateDefaults() {      return {          general: { diff --git a/ext/bg/js/search-frontend.js b/ext/bg/js/search-frontend.js index 509c4009..a470e873 100644 --- a/ext/bg/js/search-frontend.js +++ b/ext/bg/js/search-frontend.js @@ -16,9 +16,13 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiOptionsGet*/ +/* global + * apiOptionsGet + */  async function searchFrontendSetup() { +    await yomichan.prepare(); +      const optionsContext = {          depth: 0,          url: window.location.href diff --git a/ext/bg/js/search-query-parser-generator.js b/ext/bg/js/search-query-parser-generator.js index 1ab23a82..664858a4 100644 --- a/ext/bg/js/search-query-parser-generator.js +++ b/ext/bg/js/search-query-parser-generator.js @@ -16,7 +16,10 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiGetQueryParserTemplatesHtml, TemplateHandler*/ +/* global + * TemplateHandler + * apiGetQueryParserTemplatesHtml + */  class QueryParserGenerator {      constructor() { diff --git a/ext/bg/js/search-query-parser.js b/ext/bg/js/search-query-parser.js index 0d4aaa50..06316ce2 100644 --- a/ext/bg/js/search-query-parser.js +++ b/ext/bg/js/search-query-parser.js @@ -16,7 +16,15 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiTermsFind, apiOptionsSet, apiTextParse, apiTextParseMecab, TextScanner, QueryParserGenerator*/ +/* global + * QueryParserGenerator + * TextScanner + * apiOptionsSet + * apiTermsFind + * apiTextParse + * apiTextParseMecab + * docSentenceExtract + */  class QueryParser extends TextScanner {      constructor(search) { @@ -55,12 +63,14 @@ class QueryParser extends TextScanner {          const {definitions, length} = await apiTermsFind(searchText, {}, this.search.getOptionsContext());          if (definitions.length === 0) { return null; } +        const sentence = docSentenceExtract(textSource, this.search.options.anki.sentenceExt); +          textSource.setEndOffset(length);          this.search.setContent('terms', {definitions, context: {              focus: false,              disableHistory: cause === 'mouse', -            sentence: {text: searchText, offset: 0}, +            sentence,              url: window.location.href          }}); @@ -142,11 +152,11 @@ class QueryParser extends TextScanner {          }          if (this.search.options.parsing.enableMecabParser) {              const mecabResults = await apiTextParseMecab(text, this.search.getOptionsContext()); -            for (const mecabDictName in mecabResults) { +            for (const [mecabDictName, mecabDictResults] of mecabResults) {                  results.push({                      name: `MeCab: ${mecabDictName}`,                      id: `mecab-${mecabDictName}`, -                    parsedText: mecabResults[mecabDictName] +                    parsedText: mecabDictResults                  });              }          } diff --git a/ext/bg/js/search.js b/ext/bg/js/search.js index 98e167ad..e2bdff73 100644 --- a/ext/bg/js/search.js +++ b/ext/bg/js/search.js @@ -16,7 +16,14 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiOptionsSet, apiTermsFind, Display, QueryParser, ClipboardMonitor*/ +/* global + * ClipboardMonitor + * Display + * QueryParser + * apiClipboardGet + * apiOptionsSet + * apiTermsFind + */  class DisplaySearch extends Display {      constructor() { @@ -38,7 +45,26 @@ class DisplaySearch extends Display {          this.introVisible = true;          this.introAnimationTimer = null; -        this.clipboardMonitor = new ClipboardMonitor(); +        this.clipboardMonitor = new ClipboardMonitor({getClipboard: apiClipboardGet}); + +        this._onKeyDownIgnoreKeys = new Map([ +            ['ANY_MOD', new Set([ +                'Tab', 'ArrowUp', 'ArrowDown', 'ArrowLeft', 'ArrowRight', 'PageDown', 'PageUp', 'Home', 'End', +                'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', +                'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', +                'F21', 'F22', 'F23', 'F24' +            ])], +            ['Control', new Set(['C', 'A', 'Z', 'Y', 'X', 'F', 'G'])], +            ['Meta', new Set(['C', 'A', 'Z', 'Y', 'X', 'F', 'G'])], +            ['OS', new Set()], +            ['Alt', new Set()], +            ['AltGraph', new Set()], +            ['Shift', new Set()] +        ]); + +        this._runtimeMessageHandlers = new Map([ +            ['searchQueryUpdate', this.onExternalSearchUpdate.bind(this)] +        ]);      }      static create() { @@ -49,76 +75,41 @@ class DisplaySearch extends Display {      async prepare() {          try { -            const superPromise = super.prepare(); -            const queryParserPromise = this.queryParser.prepare(); -            await Promise.all([superPromise, queryParserPromise]); +            await super.prepare(); +            await this.queryParser.prepare();              const {queryParams: {query='', mode=''}} = parseUrl(window.location.href); -            if (this.search !== null) { -                this.search.addEventListener('click', (e) => this.onSearch(e), false); -            } -            if (this.query !== null) { -                document.documentElement.dataset.searchMode = mode; -                this.query.addEventListener('input', () => this.onSearchInput(), false); - -                if (this.wanakanaEnable !== null) { -                    if (this.options.general.enableWanakana === true) { -                        this.wanakanaEnable.checked = true; -                        window.wanakana.bind(this.query); -                    } else { -                        this.wanakanaEnable.checked = false; -                    } -                    this.wanakanaEnable.addEventListener('change', (e) => { -                        const {queryParams: {query: query2=''}} = parseUrl(window.location.href); -                        if (e.target.checked) { -                            window.wanakana.bind(this.query); -                            apiOptionsSet({general: {enableWanakana: true}}, this.getOptionsContext()); -                        } else { -                            window.wanakana.unbind(this.query); -                            apiOptionsSet({general: {enableWanakana: false}}, this.getOptionsContext()); -                        } -                        this.setQuery(query2); -                        this.onSearchQueryUpdated(this.query.value, false); -                    }); -                } +            document.documentElement.dataset.searchMode = mode; -                this.setQuery(query); -                this.onSearchQueryUpdated(this.query.value, false); +            if (this.options.general.enableWanakana === true) { +                this.wanakanaEnable.checked = true; +                window.wanakana.bind(this.query); +            } else { +                this.wanakanaEnable.checked = false;              } -            if (this.clipboardMonitorEnable !== null && mode !== 'popup') { + +            this.setQuery(query); +            this.onSearchQueryUpdated(this.query.value, false); + +            if (mode !== 'popup') {                  if (this.options.general.enableClipboardMonitor === true) {                      this.clipboardMonitorEnable.checked = true;                      this.clipboardMonitor.start();                  } else {                      this.clipboardMonitorEnable.checked = false;                  } -                this.clipboardMonitorEnable.addEventListener('change', (e) => { -                    if (e.target.checked) { -                        chrome.permissions.request( -                            {permissions: ['clipboardRead']}, -                            (granted) => { -                                if (granted) { -                                    this.clipboardMonitor.start(); -                                    apiOptionsSet({general: {enableClipboardMonitor: true}}, this.getOptionsContext()); -                                } else { -                                    e.target.checked = false; -                                } -                            } -                        ); -                    } else { -                        this.clipboardMonitor.stop(); -                        apiOptionsSet({general: {enableClipboardMonitor: false}}, this.getOptionsContext()); -                    } -                }); +                this.clipboardMonitorEnable.addEventListener('change', this.onClipboardMonitorEnableChange.bind(this));              }              chrome.runtime.onMessage.addListener(this.onRuntimeMessage.bind(this)); -            window.addEventListener('popstate', (e) => this.onPopState(e)); -            window.addEventListener('copy', (e) => this.onCopy(e)); - -            this.clipboardMonitor.onClipboardText = (text) => this.onExternalSearchUpdate(text); +            this.search.addEventListener('click', this.onSearch.bind(this), false); +            this.query.addEventListener('input', this.onSearchInput.bind(this), false); +            this.wanakanaEnable.addEventListener('change', this.onWanakanaEnableChange.bind(this)); +            window.addEventListener('popstate', this.onPopState.bind(this)); +            window.addEventListener('copy', this.onCopy.bind(this)); +            this.clipboardMonitor.on('change', this.onExternalSearchUpdate.bind(this));              this.updateSearchButton();          } catch (e) { @@ -174,28 +165,30 @@ class DisplaySearch extends Display {      }      onRuntimeMessage({action, params}, sender, callback) { -        const handler = DisplaySearch._runtimeMessageHandlers.get(action); +        const handler = this._runtimeMessageHandlers.get(action);          if (typeof handler !== 'function') { return false; } -        const result = handler(this, params, sender); +        const result = handler(params, sender);          callback(result);          return false;      }      onKeyDown(e) {          const key = Display.getKeyFromEvent(e); -        const ignoreKeys = DisplaySearch.onKeyDownIgnoreKeys; +        const ignoreKeys = this._onKeyDownIgnoreKeys; -        const activeModifierMap = { -            'Control': e.ctrlKey, -            'Meta': e.metaKey, -            'ANY_MOD': true -        }; +        const activeModifierMap = new Map([ +            ['Control', e.ctrlKey], +            ['Meta', e.metaKey], +            ['Shift', e.shiftKey], +            ['Alt', e.altKey], +            ['ANY_MOD', true] +        ]);          let preventFocus = false; -        for (const [modifier, keys] of Object.entries(ignoreKeys)) { -            const modifierActive = activeModifierMap[modifier]; -            if (key === modifier || (modifierActive && keys.includes(key))) { +        for (const [modifier, keys] of ignoreKeys.entries()) { +            const modifierActive = activeModifierMap.get(modifier); +            if (key === modifier || (modifierActive && keys.has(key))) {                  preventFocus = true;                  break;              } @@ -211,7 +204,7 @@ class DisplaySearch extends Display {          this.clipboardMonitor.setPreviousText(document.getSelection().toString().trim());      } -    onExternalSearchUpdate(text) { +    onExternalSearchUpdate({text}) {          this.setQuery(text);          const url = new URL(window.location.href);          url.searchParams.set('query', text); @@ -253,6 +246,38 @@ class DisplaySearch extends Display {          }      } +    onWanakanaEnableChange(e) { +        const {queryParams: {query=''}} = parseUrl(window.location.href); +        const enableWanakana = e.target.checked; +        if (enableWanakana) { +            window.wanakana.bind(this.query); +        } else { +            window.wanakana.unbind(this.query); +        } +        this.setQuery(query); +        this.onSearchQueryUpdated(this.query.value, false); +        apiOptionsSet({general: {enableWanakana}}, this.getOptionsContext()); +    } + +    onClipboardMonitorEnableChange(e) { +        if (e.target.checked) { +            chrome.permissions.request( +                {permissions: ['clipboardRead']}, +                (granted) => { +                    if (granted) { +                        this.clipboardMonitor.start(); +                        apiOptionsSet({general: {enableClipboardMonitor: true}}, this.getOptionsContext()); +                    } else { +                        e.target.checked = false; +                    } +                } +            ); +        } else { +            this.clipboardMonitor.stop(); +            apiOptionsSet({general: {enableClipboardMonitor: false}}, this.getOptionsContext()); +        } +    } +      async updateOptions(options) {          await super.updateOptions(options);          this.queryParser.setOptions(this.options); @@ -346,23 +371,4 @@ class DisplaySearch extends Display {      }  } -DisplaySearch.onKeyDownIgnoreKeys = { -    'ANY_MOD': [ -        'Tab', 'ArrowUp', 'ArrowDown', 'ArrowLeft', 'ArrowRight', 'PageDown', 'PageUp', 'Home', 'End', -        'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', -        'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', -        'F21', 'F22', 'F23', 'F24' -    ], -    'Control': ['C', 'A', 'Z', 'Y', 'X', 'F', 'G'], -    'Meta': ['C', 'A', 'Z', 'Y', 'X', 'F', 'G'], -    'OS': [], -    'Alt': [], -    'AltGraph': [], -    'Shift': [] -}; - -DisplaySearch._runtimeMessageHandlers = new Map([ -    ['searchQueryUpdate', (self, {query}) => { self.onExternalSearchUpdate(query); }] -]); -  DisplaySearch.instance = DisplaySearch.create(); diff --git a/ext/bg/js/settings/anki-templates.js b/ext/bg/js/settings/anki-templates.js index 2e80e334..c5222d30 100644 --- a/ext/bg/js/settings/anki-templates.js +++ b/ext/bg/js/settings/anki-templates.js @@ -16,22 +16,33 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global getOptionsContext, getOptionsMutable, settingsSaveOptions -profileOptionsGetDefaultFieldTemplates, ankiGetFieldMarkers, ankiGetFieldMarkersHtml, dictFieldFormat -apiOptionsGet, apiTermsFind*/ +/* global + * AnkiNoteBuilder + * ankiGetFieldMarkers + * ankiGetFieldMarkersHtml + * apiGetDefaultAnkiFieldTemplates + * apiOptionsGet + * apiTemplateRender + * apiTermsFind + * getOptionsContext + * getOptionsMutable + * settingsSaveOptions + */  function onAnkiFieldTemplatesReset(e) {      e.preventDefault();      $('#field-template-reset-modal').modal('show');  } -function onAnkiFieldTemplatesResetConfirm(e) { +async function onAnkiFieldTemplatesResetConfirm(e) {      e.preventDefault();      $('#field-template-reset-modal').modal('hide'); +    const value = await apiGetDefaultAnkiFieldTemplates(); +      const element = document.querySelector('#field-templates'); -    element.value = profileOptionsGetDefaultFieldTemplates(); +    element.value = value;      element.dispatchEvent(new Event('change'));  } @@ -45,10 +56,10 @@ function ankiTemplatesInitialize() {          node.addEventListener('click', onAnkiTemplateMarkerClicked, false);      } -    $('#field-templates').on('change', (e) => onAnkiFieldTemplatesChanged(e)); -    $('#field-template-render').on('click', (e) => onAnkiTemplateRender(e)); -    $('#field-templates-reset').on('click', (e) => onAnkiFieldTemplatesReset(e)); -    $('#field-templates-reset-confirm').on('click', (e) => onAnkiFieldTemplatesResetConfirm(e)); +    $('#field-templates').on('change', onAnkiFieldTemplatesChanged); +    $('#field-template-render').on('click', onAnkiTemplateRender); +    $('#field-templates-reset').on('click', onAnkiFieldTemplatesReset); +    $('#field-templates-reset-confirm').on('click', onAnkiFieldTemplatesResetConfirm);      ankiTemplatesUpdateValue();  } @@ -57,7 +68,7 @@ async function ankiTemplatesUpdateValue() {      const optionsContext = getOptionsContext();      const options = await apiOptionsGet(optionsContext);      let templates = options.anki.fieldTemplates; -    if (typeof templates !== 'string') { templates = profileOptionsGetDefaultFieldTemplates(); } +    if (typeof templates !== 'string') { templates = await apiGetDefaultAnkiFieldTemplates(); }      $('#field-templates').val(templates);      onAnkiTemplatesValidateCompile(); @@ -89,8 +100,9 @@ async function ankiTemplatesValidate(infoNode, field, mode, showSuccessResult, i          if (definition !== null) {              const options = await apiOptionsGet(optionsContext);              let templates = options.anki.fieldTemplates; -            if (typeof templates !== 'string') { templates = profileOptionsGetDefaultFieldTemplates(); } -            result = await dictFieldFormat(field, definition, mode, options, templates, exceptions); +            if (typeof templates !== 'string') { templates = await apiGetDefaultAnkiFieldTemplates(); } +            const ankiNoteBuilder = new AnkiNoteBuilder({renderTemplate: apiTemplateRender}); +            result = await ankiNoteBuilder.formatField(field, definition, mode, options, templates, exceptions);          }      } catch (e) {          exceptions.push(e); @@ -109,7 +121,7 @@ async function ankiTemplatesValidate(infoNode, field, mode, showSuccessResult, i  async function onAnkiFieldTemplatesChanged(e) {      // Get value      let templates = e.currentTarget.value; -    if (templates === profileOptionsGetDefaultFieldTemplates()) { +    if (templates === await apiGetDefaultAnkiFieldTemplates()) {          // Default          templates = null;      } diff --git a/ext/bg/js/settings/anki.js b/ext/bg/js/settings/anki.js index 4263fc51..b706cd1b 100644 --- a/ext/bg/js/settings/anki.js +++ b/ext/bg/js/settings/anki.js @@ -16,9 +16,16 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global getOptionsContext, getOptionsMutable, settingsSaveOptions -utilBackgroundIsolate, utilAnkiGetDeckNames, utilAnkiGetModelNames, utilAnkiGetModelFieldNames -onFormOptionsChanged*/ +/* global + * getOptionsContext + * getOptionsMutable + * onFormOptionsChanged + * settingsSaveOptions + * utilAnkiGetDeckNames + * utilAnkiGetModelFieldNames + * utilAnkiGetModelNames + * utilBackgroundIsolate + */  // Private @@ -154,10 +161,10 @@ async function _ankiFieldsPopulate(tabId, options) {      container.appendChild(fragment);      for (const node of container.querySelectorAll('.anki-field-value')) { -        node.addEventListener('change', (e) => onFormOptionsChanged(e), false); +        node.addEventListener('change', onFormOptionsChanged, false);      }      for (const node of container.querySelectorAll('.marker-link')) { -        node.addEventListener('click', (e) => _onAnkiMarkerClicked(e), false); +        node.addEventListener('click', _onAnkiMarkerClicked, false);      }  } @@ -267,7 +274,7 @@ function ankiGetFieldMarkers(type) {  function ankiInitialize() {      for (const node of document.querySelectorAll('#anki-terms-model,#anki-kanji-model')) { -        node.addEventListener('change', (e) => _onAnkiModelChanged(e), false); +        node.addEventListener('change', _onAnkiModelChanged, false);      }  } diff --git a/ext/bg/js/settings/audio-ui.js b/ext/bg/js/settings/audio-ui.js index 555380b4..206539a4 100644 --- a/ext/bg/js/settings/audio-ui.js +++ b/ext/bg/js/settings/audio-ui.js @@ -37,7 +37,7 @@ AudioSourceUI.Container = class Container {              this.children.push(new AudioSourceUI.AudioSource(this, audioSource, this.children.length));          } -        this._clickListener = () => this.onAddAudioSource(); +        this._clickListener = this.onAddAudioSource.bind(this);          this.addButton.addEventListener('click', this._clickListener, false);      } @@ -105,8 +105,8 @@ AudioSourceUI.AudioSource = class AudioSource {          this.select.value = audioSource; -        this._selectChangeListener = () => this.onSelectChanged(); -        this._removeClickListener = () => this.onRemoveClicked(); +        this._selectChangeListener = this.onSelectChanged.bind(this); +        this._removeClickListener = this.onRemoveClicked.bind(this);          this.select.addEventListener('change', this._selectChangeListener, false);          this.removeButton.addEventListener('click', this._removeClickListener, false); diff --git a/ext/bg/js/settings/audio.js b/ext/bg/js/settings/audio.js index 588d9a11..38dd6349 100644 --- a/ext/bg/js/settings/audio.js +++ b/ext/bg/js/settings/audio.js @@ -16,12 +16,26 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global getOptionsContext, getOptionsMutable, settingsSaveOptions -AudioSourceUI, audioGetTextToSpeechVoice*/ +/* global + * AudioSourceUI + * AudioSystem + * apiAudioGetUri + * getOptionsContext + * getOptionsMutable + * settingsSaveOptions + */  let audioSourceUI = null; +let audioSystem = null;  async function audioSettingsInitialize() { +    audioSystem = new AudioSystem({ +        getAudioUri: async (definition, source) => { +            const optionsContext = getOptionsContext(); +            return await apiAudioGetUri(definition, source, optionsContext); +        } +    }); +      const optionsContext = getOptionsContext();      const options = await getOptionsMutable(optionsContext);      audioSourceUI = new AudioSourceUI.Container( @@ -29,7 +43,7 @@ async function audioSettingsInitialize() {          document.querySelector('.audio-source-list'),          document.querySelector('.audio-source-add')      ); -    audioSourceUI.save = () => settingsSaveOptions(); +    audioSourceUI.save = settingsSaveOptions;      textToSpeechInitialize();  } @@ -37,11 +51,11 @@ async function audioSettingsInitialize() {  function textToSpeechInitialize() {      if (typeof speechSynthesis === 'undefined') { return; } -    speechSynthesis.addEventListener('voiceschanged', () => updateTextToSpeechVoices(), false); +    speechSynthesis.addEventListener('voiceschanged', updateTextToSpeechVoices, false);      updateTextToSpeechVoices(); -    document.querySelector('#text-to-speech-voice').addEventListener('change', (e) => onTextToSpeechVoiceChange(e), false); -    document.querySelector('#text-to-speech-voice-test').addEventListener('click', () => textToSpeechTest(), false); +    document.querySelector('#text-to-speech-voice').addEventListener('change', onTextToSpeechVoiceChange, false); +    document.querySelector('#text-to-speech-voice-test').addEventListener('click', textToSpeechTest, false);  }  function updateTextToSpeechVoices() { @@ -100,16 +114,11 @@ function textToSpeechVoiceCompare(a, b) {  function textToSpeechTest() {      try {          const text = document.querySelector('#text-to-speech-voice-test').dataset.speechText || ''; -        const voiceURI = document.querySelector('#text-to-speech-voice').value; -        const voice = audioGetTextToSpeechVoice(voiceURI); -        if (voice === null) { return; } - -        const utterance = new SpeechSynthesisUtterance(text); -        utterance.lang = 'ja-JP'; -        utterance.voice = voice; -        utterance.volume = 1.0; +        const voiceUri = document.querySelector('#text-to-speech-voice').value; -        speechSynthesis.speak(utterance); +        const audio = audioSystem.createTextToSpeechAudio({text, voiceUri}); +        audio.volume = 1.0; +        audio.play();      } catch (e) {          // NOP      } diff --git a/ext/bg/js/settings/backup.js b/ext/bg/js/settings/backup.js index f4d622a4..21417dfb 100644 --- a/ext/bg/js/settings/backup.js +++ b/ext/bg/js/settings/backup.js @@ -16,10 +16,17 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiOptionsGetFull, apiGetEnvironmentInfo -utilBackend, utilIsolate, utilBackgroundIsolate, utilReadFileArrayBuffer -optionsGetDefault, optionsUpdateVersion -profileOptionsGetDefaultFieldTemplates*/ +/* global + * apiGetDefaultAnkiFieldTemplates + * apiGetEnvironmentInfo + * apiOptionsGetFull + * optionsGetDefault + * optionsUpdateVersion + * utilBackend + * utilBackgroundIsolate + * utilIsolate + * utilReadFileArrayBuffer + */  // Exporting @@ -47,8 +54,7 @@ function _getSettingsExportDateString(date, dateSeparator, dateTimeSeparator, ti  async function _getSettingsExportData(date) {      const optionsFull = await apiOptionsGetFull();      const environment = await apiGetEnvironmentInfo(); - -    const fieldTemplatesDefault = profileOptionsGetDefaultFieldTemplates(); +    const fieldTemplatesDefault = await apiGetDefaultAnkiFieldTemplates();      // Format options      for (const {options} of optionsFull.profiles) { @@ -122,7 +128,7 @@ async function _onSettingsExportClick() {  // Importing  async function _settingsImportSetOptionsFull(optionsFull) { -    return utilIsolate(await utilBackend().setFullOptions( +    return utilIsolate(utilBackend().setFullOptions(          utilBackgroundIsolate(optionsFull)      ));  } @@ -364,10 +370,10 @@ async function _onSettingsResetConfirmClick() {  // Setup -window.addEventListener('DOMContentLoaded', () => { +function backupInitialize() {      document.querySelector('#settings-export').addEventListener('click', _onSettingsExportClick, false);      document.querySelector('#settings-import').addEventListener('click', _onSettingsImportClick, false);      document.querySelector('#settings-import-file').addEventListener('change', _onSettingsImportFileChange, false);      document.querySelector('#settings-reset').addEventListener('click', _onSettingsResetClick, false);      document.querySelector('#settings-reset-modal-confirm').addEventListener('click', _onSettingsResetConfirmClick, false); -}, false); +} diff --git a/ext/bg/js/settings/conditions-ui.js b/ext/bg/js/settings/conditions-ui.js index 5a271321..9d61d25e 100644 --- a/ext/bg/js/settings/conditions-ui.js +++ b/ext/bg/js/settings/conditions-ui.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global conditionsNormalizeOptionValue*/ +/* global + * conditionsNormalizeOptionValue + */  class ConditionsUI {      static instantiateTemplate(templateSelector) { @@ -41,7 +43,7 @@ ConditionsUI.Container = class Container {              this.children.push(new ConditionsUI.ConditionGroup(this, conditionGroup));          } -        this.addButton.on('click', () => this.onAddConditionGroup()); +        this.addButton.on('click', this.onAddConditionGroup.bind(this));      }      cleanup() { @@ -127,7 +129,7 @@ ConditionsUI.ConditionGroup = class ConditionGroup {              this.children.push(new ConditionsUI.Condition(this, condition));          } -        this.addButton.on('click', () => this.onAddCondition()); +        this.addButton.on('click', this.onAddCondition.bind(this));      }      cleanup() { @@ -185,10 +187,10 @@ ConditionsUI.Condition = class Condition {          this.updateOperators();          this.updateInput(); -        this.input.on('change', () => this.onInputChanged()); -        this.typeSelect.on('change', () => this.onConditionTypeChanged()); -        this.operatorSelect.on('change', () => this.onConditionOperatorChanged()); -        this.removeButton.on('click', () => this.onRemoveClicked()); +        this.input.on('change', this.onInputChanged.bind(this)); +        this.typeSelect.on('change', this.onConditionTypeChanged.bind(this)); +        this.operatorSelect.on('change', this.onConditionOperatorChanged.bind(this)); +        this.removeButton.on('click', this.onRemoveClicked.bind(this));      }      cleanup() { @@ -235,10 +237,10 @@ ConditionsUI.Condition = class Condition {      updateInput() {          const conditionDescriptors = this.parent.parent.conditionDescriptors;          const {type, operator} = this.condition; -        const props = { -            placeholder: '', -            type: 'text' -        }; +        const props = new Map([ +            ['placeholder', ''], +            ['type', 'text'] +        ]);          const objects = [];          if (hasOwn(conditionDescriptors, type)) { @@ -252,20 +254,20 @@ ConditionsUI.Condition = class Condition {          for (const object of objects) {              if (hasOwn(object, 'placeholder')) { -                props.placeholder = object.placeholder; +                props.set('placeholder', object.placeholder);              }              if (object.type === 'number') { -                props.type = 'number'; +                props.set('type', 'number');                  for (const prop of ['step', 'min', 'max']) {                      if (hasOwn(object, prop)) { -                        props[prop] = object[prop]; +                        props.set(prop, object[prop]);                      }                  }              }          } -        for (const prop in props) { -            this.input.prop(prop, props[prop]); +        for (const [prop, value] of props.entries()) { +            this.input.prop(prop, value);          }          const {valid} = this.validateValue(this.condition.value); diff --git a/ext/bg/js/settings/dictionaries.js b/ext/bg/js/settings/dictionaries.js index 70a22a16..5e59cc3d 100644 --- a/ext/bg/js/settings/dictionaries.js +++ b/ext/bg/js/settings/dictionaries.js @@ -16,11 +16,23 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global getOptionsContext, getOptionsMutable, getOptionsFullMutable, settingsSaveOptions, apiOptionsGetFull, apiOptionsGet -utilBackgroundIsolate, utilDatabaseDeleteDictionary, utilDatabaseGetDictionaryInfo, utilDatabaseGetDictionaryCounts -utilDatabasePurge, utilDatabaseImport -storageUpdateStats, storageEstimate -PageExitPrevention*/ +/* global + * PageExitPrevention + * apiOptionsGet + * apiOptionsGetFull + * getOptionsContext + * getOptionsFullMutable + * getOptionsMutable + * settingsSaveOptions + * storageEstimate + * storageUpdateStats + * utilBackgroundIsolate + * utilDatabaseDeleteDictionary + * utilDatabaseGetDictionaryCounts + * utilDatabaseGetDictionaryInfo + * utilDatabaseImport + * utilDatabasePurge + */  let dictionaryUI = null; @@ -36,7 +48,7 @@ class SettingsDictionaryListUI {          this.dictionaryEntries = [];          this.extra = null; -        document.querySelector('#dict-delete-confirm').addEventListener('click', (e) => this.onDictionaryConfirmDelete(e), false); +        document.querySelector('#dict-delete-confirm').addEventListener('click', this.onDictionaryConfirmDelete.bind(this), false);      }      setOptionsDictionaries(optionsDictionaries) { @@ -198,10 +210,10 @@ class SettingsDictionaryEntryUI {          this.applyValues(); -        this.eventListeners.addEventListener(this.enabledCheckbox, 'change', (e) => this.onEnabledChanged(e), false); -        this.eventListeners.addEventListener(this.allowSecondarySearchesCheckbox, 'change', (e) => this.onAllowSecondarySearchesChanged(e), false); -        this.eventListeners.addEventListener(this.priorityInput, 'change', (e) => this.onPriorityChanged(e), false); -        this.eventListeners.addEventListener(this.deleteButton, 'click', (e) => this.onDeleteButtonClicked(e), false); +        this.eventListeners.addEventListener(this.enabledCheckbox, 'change', this.onEnabledChanged.bind(this), false); +        this.eventListeners.addEventListener(this.allowSecondarySearchesCheckbox, 'change', this.onAllowSecondarySearchesChanged.bind(this), false); +        this.eventListeners.addEventListener(this.priorityInput, 'change', this.onPriorityChanged.bind(this), false); +        this.eventListeners.addEventListener(this.deleteButton, 'click', this.onDeleteButtonClicked.bind(this), false);      }      cleanup() { @@ -341,14 +353,14 @@ async function dictSettingsInitialize() {          document.querySelector('#dict-groups-extra'),          document.querySelector('#dict-extra-template')      ); -    dictionaryUI.save = () => settingsSaveOptions(); - -    document.querySelector('#dict-purge-button').addEventListener('click', (e) => onDictionaryPurgeButtonClick(e), false); -    document.querySelector('#dict-purge-confirm').addEventListener('click', (e) => onDictionaryPurge(e), false); -    document.querySelector('#dict-file-button').addEventListener('click', (e) => onDictionaryImportButtonClick(e), false); -    document.querySelector('#dict-file').addEventListener('change', (e) => onDictionaryImport(e), false); -    document.querySelector('#dict-main').addEventListener('change', (e) => onDictionaryMainChanged(e), false); -    document.querySelector('#database-enable-prefix-wildcard-searches').addEventListener('change', (e) => onDatabaseEnablePrefixWildcardSearchesChanged(e), false); +    dictionaryUI.save = settingsSaveOptions; + +    document.querySelector('#dict-purge-button').addEventListener('click', onDictionaryPurgeButtonClick, false); +    document.querySelector('#dict-purge-confirm').addEventListener('click', onDictionaryPurge, false); +    document.querySelector('#dict-file-button').addEventListener('click', onDictionaryImportButtonClick, false); +    document.querySelector('#dict-file').addEventListener('change', onDictionaryImport, false); +    document.querySelector('#dict-main').addEventListener('change', onDictionaryMainChanged, false); +    document.querySelector('#database-enable-prefix-wildcard-searches').addEventListener('change', onDatabaseEnablePrefixWildcardSearchesChanged, false);      await onDictionaryOptionsChanged();      await onDatabaseUpdated(); diff --git a/ext/bg/js/settings/main.js b/ext/bg/js/settings/main.js index d1ad2c6b..ebc443df 100644 --- a/ext/bg/js/settings/main.js +++ b/ext/bg/js/settings/main.js @@ -16,13 +16,26 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global getOptionsContext, apiOptionsSave -utilBackend, utilIsolate, utilBackgroundIsolate -ankiErrorShown, ankiFieldsToDict -ankiTemplatesUpdateValue, onAnkiOptionsChanged, onDictionaryOptionsChanged -appearanceInitialize, audioSettingsInitialize, profileOptionsSetup, dictSettingsInitialize -ankiInitialize, ankiTemplatesInitialize, storageInfoInitialize -*/ +/* global + * ankiErrorShown + * ankiFieldsToDict + * ankiInitialize + * ankiTemplatesInitialize + * ankiTemplatesUpdateValue + * apiOptionsSave + * appearanceInitialize + * audioSettingsInitialize + * backupInitialize + * dictSettingsInitialize + * getOptionsContext + * onAnkiOptionsChanged + * onDictionaryOptionsChanged + * profileOptionsSetup + * storageInfoInitialize + * utilBackend + * utilBackgroundIsolate + * utilIsolate + */  function getOptionsMutable(optionsContext) {      return utilBackend().getOptions( @@ -200,7 +213,7 @@ async function formWrite(options) {  }  function formSetupEventListeners() { -    $('input, select, textarea').not('.anki-model').not('.ignore-form-changes *').change((e) => onFormOptionsChanged(e)); +    $('input, select, textarea').not('.anki-model').not('.ignore-form-changes *').change(onFormOptionsChanged);  }  function formUpdateVisibility(options) { @@ -262,6 +275,8 @@ function showExtensionInformation() {  async function onReady() { +    await yomichan.prepare(); +      showExtensionInformation();      formSetupEventListeners(); @@ -271,6 +286,7 @@ async function onReady() {      await dictSettingsInitialize();      ankiInitialize();      ankiTemplatesInitialize(); +    backupInitialize();      storageInfoInitialize(); diff --git a/ext/bg/js/settings/popup-preview-frame.js b/ext/bg/js/settings/popup-preview-frame.js index aa2b6100..6a149841 100644 --- a/ext/bg/js/settings/popup-preview-frame.js +++ b/ext/bg/js/settings/popup-preview-frame.js @@ -16,7 +16,13 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiOptionsGet, Popup, PopupProxyHost, Frontend, TextSourceRange*/ +/* global + * Frontend + * Popup + * PopupProxyHost + * TextSourceRange + * apiOptionsGet + */  class SettingsPopupPreview {      constructor() { @@ -28,6 +34,12 @@ class SettingsPopupPreview {          this.themeChangeTimeout = null;          this.textSource = null;          this._targetOrigin = chrome.runtime.getURL('/').replace(/\/$/, ''); + +        this._windowMessageHandlers = new Map([ +            ['setText', ({text}) => this.setText(text)], +            ['setCustomCss', ({css}) => this.setCustomCss(css)], +            ['setCustomOuterCss', ({css}) => this.setCustomOuterCss(css)] +        ]);      }      static create() { @@ -38,15 +50,12 @@ class SettingsPopupPreview {      async prepare() {          // Setup events -        window.addEventListener('message', (e) => this.onMessage(e), false); +        window.addEventListener('message', this.onMessage.bind(this), false); -        const themeDarkCheckbox = document.querySelector('#theme-dark-checkbox'); -        if (themeDarkCheckbox !== null) { -            themeDarkCheckbox.addEventListener('change', () => this.onThemeDarkCheckboxChanged(themeDarkCheckbox), false); -        } +        document.querySelector('#theme-dark-checkbox').addEventListener('change', this.onThemeDarkCheckboxChanged.bind(this), false);          // Overwrite API functions -        window.apiOptionsGet = (...args) => this.apiOptionsGet(...args); +        window.apiOptionsGet = this.apiOptionsGet.bind(this);          // Overwrite frontend          const popupHost = new PopupProxyHost(); @@ -56,7 +65,7 @@ class SettingsPopupPreview {          this.popup.setChildrenSupported(false);          this.popupSetCustomOuterCssOld = this.popup.setCustomOuterCss; -        this.popup.setCustomOuterCss = (...args) => this.popupSetCustomOuterCss(...args); +        this.popup.setCustomOuterCss = this.popupSetCustomOuterCss.bind(this);          this.frontend = new Frontend(this.popup); @@ -101,14 +110,14 @@ class SettingsPopupPreview {          if (e.origin !== this._targetOrigin) { return; }          const {action, params} = e.data; -        const handler = SettingsPopupPreview._messageHandlers.get(action); +        const handler = this._windowMessageHandlers.get(action);          if (typeof handler !== 'function') { return; } -        handler(this, params); +        handler(params);      } -    onThemeDarkCheckboxChanged(node) { -        document.documentElement.classList.toggle('dark', node.checked); +    onThemeDarkCheckboxChanged(e) { +        document.documentElement.classList.toggle('dark', e.target.checked);          if (this.themeChangeTimeout !== null) {              clearTimeout(this.themeChangeTimeout);          } @@ -171,12 +180,6 @@ class SettingsPopupPreview {      }  } -SettingsPopupPreview._messageHandlers = new Map([ -    ['setText', (self, {text}) => self.setText(text)], -    ['setCustomCss', (self, {css}) => self.setCustomCss(css)], -    ['setCustomOuterCss', (self, {css}) => self.setCustomOuterCss(css)] -]); -  SettingsPopupPreview.instance = SettingsPopupPreview.create(); diff --git a/ext/bg/js/settings/profiles.js b/ext/bg/js/settings/profiles.js index 3e589809..b35b6309 100644 --- a/ext/bg/js/settings/profiles.js +++ b/ext/bg/js/settings/profiles.js @@ -16,9 +16,17 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global getOptionsMutable, getOptionsFullMutable, settingsSaveOptions, apiOptionsGetFull -utilBackgroundIsolate, formWrite -conditionsClearCaches, ConditionsUI, profileConditionsDescriptor*/ +/* global + * ConditionsUI + * apiOptionsGetFull + * conditionsClearCaches + * formWrite + * getOptionsFullMutable + * getOptionsMutable + * profileConditionsDescriptor + * settingsSaveOptions + * utilBackgroundIsolate + */  let currentProfileIndex = 0;  let profileConditionsContainer = null; @@ -39,16 +47,16 @@ async function profileOptionsSetup() {  }  function profileOptionsSetupEventListeners() { -    $('#profile-target').change((e) => onTargetProfileChanged(e)); -    $('#profile-name').change((e) => onProfileNameChanged(e)); -    $('#profile-add').click((e) => onProfileAdd(e)); -    $('#profile-remove').click((e) => onProfileRemove(e)); -    $('#profile-remove-confirm').click((e) => onProfileRemoveConfirm(e)); -    $('#profile-copy').click((e) => onProfileCopy(e)); -    $('#profile-copy-confirm').click((e) => onProfileCopyConfirm(e)); +    $('#profile-target').change(onTargetProfileChanged); +    $('#profile-name').change(onProfileNameChanged); +    $('#profile-add').click(onProfileAdd); +    $('#profile-remove').click(onProfileRemove); +    $('#profile-remove-confirm').click(onProfileRemoveConfirm); +    $('#profile-copy').click(onProfileCopy); +    $('#profile-copy-confirm').click(onProfileCopyConfirm);      $('#profile-move-up').click(() => onProfileMove(-1));      $('#profile-move-down').click(() => onProfileMove(1)); -    $('.profile-form').find('input, select, textarea').not('.profile-form-manual').change((e) => onProfileOptionsChanged(e)); +    $('.profile-form').find('input, select, textarea').not('.profile-form-manual').change(onProfileOptionsChanged);  }  function tryGetIntegerValue(selector, min, max) { diff --git a/ext/bg/js/settings/storage.js b/ext/bg/js/settings/storage.js index cbe1bb4d..ae305e22 100644 --- a/ext/bg/js/settings/storage.js +++ b/ext/bg/js/settings/storage.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiGetEnvironmentInfo*/ +/* global + * apiGetEnvironmentInfo + */  function storageBytesToLabeledString(size) {      const base = 1000; @@ -57,7 +59,7 @@ async function storageInfoInitialize() {      await storageShowInfo(); -    document.querySelector('#storage-refresh').addEventListener('click', () => storageShowInfo(), false); +    document.querySelector('#storage-refresh').addEventListener('click', storageShowInfo, false);  }  async function storageUpdateStats() { diff --git a/ext/bg/js/translator.js b/ext/bg/js/translator.js index a675a9f7..25da9bf0 100644 --- a/ext/bg/js/translator.js +++ b/ext/bg/js/translator.js @@ -16,12 +16,28 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global requestJson -dictTermsMergeBySequence, dictTagBuildSource, dictTermsMergeByGloss, dictTermsSort, dictTagsSort -dictEnabledSet, dictTermsGroup, dictTermsCompressTags, dictTermsUndupe, dictTagSanitize -jpDistributeFurigana, jpConvertHalfWidthKanaToFullWidth, jpConvertNumericTofullWidth -jpConvertAlphabeticToKana, jpHiraganaToKatakana, jpKatakanaToHiragana, jpIsCharCodeJapanese -Database, Deinflector*/ +/* global + * Database + * Deinflector + * dictEnabledSet + * dictTagBuildSource + * dictTagSanitize + * dictTagsSort + * dictTermsCompressTags + * dictTermsGroup + * dictTermsMergeByGloss + * dictTermsMergeBySequence + * dictTermsSort + * dictTermsUndupe + * jpConvertAlphabeticToKana + * jpConvertHalfWidthKanaToFullWidth + * jpConvertNumericTofullWidth + * jpDistributeFurigana + * jpHiraganaToKatakana + * jpIsCodePointJapanese + * jpKatakanaToHiragana + * requestJson + */  class Translator {      constructor() { @@ -199,8 +215,19 @@ class Translator {          const strayDefinitions = defaultDefinitions.filter((definition, index) => !mergedByTermIndices.has(index));          for (const groupedDefinition of dictTermsGroup(strayDefinitions, dictionaries)) { -            groupedDefinition.expressions = [Translator.createExpression(groupedDefinition.expression, groupedDefinition.reading)]; -            definitionsMerged.push(groupedDefinition); +            // from dictTermsMergeBySequence +            const {reasons, score, expression, reading, source, dictionary} = groupedDefinition; +            const compatibilityDefinition = { +                reasons, +                score, +                expression: [expression], +                reading: [reading], +                expressions: [Translator.createExpression(groupedDefinition.expression, groupedDefinition.reading)], +                source, +                dictionary, +                definitions: groupedDefinition.definitions +            }; +            definitionsMerged.push(compatibilityDefinition);          }          await this.buildTermMeta(definitionsMerged, dictionaries); @@ -610,13 +637,14 @@ class Translator {      static getSearchableText(text, options) {          if (!options.scanning.alphanumeric) { -            const ii = text.length; -            for (let i = 0; i < ii; ++i) { -                if (!jpIsCharCodeJapanese(text.charCodeAt(i))) { -                    text = text.substring(0, i); +            let newText = ''; +            for (const c of text) { +                if (!jpIsCodePointJapanese(c.codePointAt(0))) {                      break;                  } +                newText += c;              } +            text = newText;          }          return text; diff --git a/ext/bg/js/util.js b/ext/bg/js/util.js index 5ce4b08c..79c6af06 100644 --- a/ext/bg/js/util.js +++ b/ext/bg/js/util.js @@ -73,7 +73,11 @@ function utilStringHashCode(string) {  }  function utilBackend() { -    return chrome.extension.getBackgroundPage().yomichanBackend; +    const backend = chrome.extension.getBackgroundPage().yomichanBackend; +    if (!backend.isPrepared) { +        throw new Error('Backend not ready yet'); +    } +    return backend;  }  async function utilAnkiGetModelNames() { diff --git a/ext/bg/search.html b/ext/bg/search.html index d6336826..f4c1a737 100644 --- a/ext/bg/search.html +++ b/ext/bg/search.html @@ -80,7 +80,7 @@          <script src="/bg/js/japanese.js"></script>          <script src="/fg/js/document.js"></script>          <script src="/fg/js/source.js"></script> -        <script src="/mixed/js/audio.js"></script> +        <script src="/mixed/js/audio-system.js"></script>          <script src="/mixed/js/display-context.js"></script>          <script src="/mixed/js/display.js"></script>          <script src="/mixed/js/display-generator.js"></script> diff --git a/ext/bg/settings.html b/ext/bg/settings.html index b048a36c..0db76d71 100644 --- a/ext/bg/settings.html +++ b/ext/bg/settings.html @@ -1090,6 +1090,7 @@          <script src="/mixed/js/api.js"></script>          <script src="/bg/js/anki.js"></script> +        <script src="/bg/js/anki-note-builder.js"></script>          <script src="/bg/js/conditions.js"></script>          <script src="/bg/js/dictionary.js"></script>          <script src="/bg/js/handlebars.js"></script> @@ -1098,7 +1099,7 @@          <script src="/bg/js/page-exit-prevention.js"></script>          <script src="/bg/js/profile-conditions.js"></script>          <script src="/bg/js/util.js"></script> -        <script src="/mixed/js/audio.js"></script> +        <script src="/mixed/js/audio-system.js"></script>          <script src="/bg/js/settings/anki.js"></script>          <script src="/bg/js/settings/anki-templates.js"></script> diff --git a/ext/fg/float.html b/ext/fg/float.html index 352a866a..7bbed565 100644 --- a/ext/fg/float.html +++ b/ext/fg/float.html @@ -46,7 +46,7 @@          <script src="/fg/js/document.js"></script>          <script src="/fg/js/source.js"></script> -        <script src="/mixed/js/audio.js"></script> +        <script src="/mixed/js/audio-system.js"></script>          <script src="/mixed/js/display-context.js"></script>          <script src="/mixed/js/display.js"></script>          <script src="/mixed/js/display-generator.js"></script> diff --git a/ext/fg/js/document.js b/ext/fg/js/document.js index 35861475..490f61bb 100644 --- a/ext/fg/js/document.js +++ b/ext/fg/js/document.js @@ -16,7 +16,11 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global TextSourceElement, TextSourceRange, DOM*/ +/* global + * DOM + * TextSourceElement + * TextSourceRange + */  const REGEX_TRANSPARENT_COLOR = /rgba\s*\([^)]*,\s*0(?:\.0+)?\s*\)/; diff --git a/ext/fg/js/float.js b/ext/fg/js/float.js index 8f21a9c5..393c2719 100644 --- a/ext/fg/js/float.js +++ b/ext/fg/js/float.js @@ -16,7 +16,12 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global popupNestedInitialize, apiForward, apiGetMessageToken, Display*/ +/* global + * Display + * apiForward + * apiGetMessageToken + * popupNestedInitialize + */  class DisplayFloat extends Display {      constructor() { @@ -33,8 +38,27 @@ class DisplayFloat extends Display {          this._messageToken = null;          this._messageTokenPromise = null; -        yomichan.on('orphaned', () => this.onOrphaned()); -        window.addEventListener('message', (e) => this.onMessage(e), false); +        this._onKeyDownHandlers = new Map([ +            ['C', (e) => { +                if (e.ctrlKey && !window.getSelection().toString()) { +                    this.onSelectionCopy(); +                    return true; +                } +                return false; +            }], +            ...this._onKeyDownHandlers +        ]); + +        this._windowMessageHandlers = new Map([ +            ['setContent', ({type, details}) => this.setContent(type, details)], +            ['clearAutoPlayTimer', () => this.clearAutoPlayTimer()], +            ['setCustomCss', ({css}) => this.setCustomCss(css)], +            ['prepare', ({options, popupInfo, url, childrenSupported, scale, uniqueId}) => this.prepare(options, popupInfo, url, childrenSupported, scale, uniqueId)], +            ['setContentScale', ({scale}) => this.setContentScale(scale)] +        ]); + +        yomichan.on('orphaned', this.onOrphaned.bind(this)); +        window.addEventListener('message', this.onMessage.bind(this), false);      }      async prepare(options, popupInfo, url, childrenSupported, scale, uniqueId) { @@ -96,18 +120,6 @@ class DisplayFloat extends Display {          }      } -    onKeyDown(e) { -        const key = Display.getKeyFromEvent(e); -        const handler = DisplayFloat._onKeyDownHandlers.get(key); -        if (typeof handler === 'function') { -            if (handler(this, e)) { -                e.preventDefault(); -                return true; -            } -        } -        return super.onKeyDown(e); -    } -      async getMessageToken() {          // this._messageTokenPromise is used to ensure that only one call to apiGetMessageToken is made.          if (this._messageTokenPromise === null) { @@ -126,10 +138,10 @@ class DisplayFloat extends Display {              return;          } -        const handler = DisplayFloat._messageHandlers.get(action); +        const handler = this._windowMessageHandlers.get(action);          if (typeof handler !== 'function') { return; } -        handler(this, params); +        handler(params);      }      getOptionsContext() { @@ -153,22 +165,4 @@ class DisplayFloat extends Display {      }  } -DisplayFloat._onKeyDownHandlers = new Map([ -    ['C', (self, e) => { -        if (e.ctrlKey && !window.getSelection().toString()) { -            self.onSelectionCopy(); -            return true; -        } -        return false; -    }] -]); - -DisplayFloat._messageHandlers = new Map([ -    ['setContent', (self, {type, details}) => self.setContent(type, details)], -    ['clearAutoPlayTimer', (self) => self.clearAutoPlayTimer()], -    ['setCustomCss', (self, {css}) => self.setCustomCss(css)], -    ['prepare', (self, {options, popupInfo, url, childrenSupported, scale, uniqueId}) => self.prepare(options, popupInfo, url, childrenSupported, scale, uniqueId)], -    ['setContentScale', (self, {scale}) => self.setContentScale(scale)] -]); -  DisplayFloat.instance = new DisplayFloat(); diff --git a/ext/fg/js/frontend-api-sender.js b/ext/fg/js/frontend-api-sender.js index 8dc6aaf3..4431df61 100644 --- a/ext/fg/js/frontend-api-sender.js +++ b/ext/fg/js/frontend-api-sender.js @@ -31,6 +31,8 @@ class FrontendApiSender {      invoke(action, params, target) {          if (this.disconnected) { +            // attempt to reconnect the next time +            this.disconnected = false;              return Promise.reject(new Error('Disconnected'));          } @@ -70,6 +72,7 @@ class FrontendApiSender {      onDisconnect() {          this.disconnected = true; +        this.port = null;          for (const id of this.callbacks.keys()) {              this.onError(id, 'Disconnected'); diff --git a/ext/fg/js/frontend-initialize.js b/ext/fg/js/frontend-initialize.js index 54b874f2..8424b21d 100644 --- a/ext/fg/js/frontend-initialize.js +++ b/ext/fg/js/frontend-initialize.js @@ -16,9 +16,15 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global PopupProxyHost, PopupProxy, Frontend*/ +/* global + * Frontend + * PopupProxy + * PopupProxyHost + */  async function main() { +    await yomichan.prepare(); +      const data = window.frontendInitializationData || {};      const {id, depth=0, parentFrameId, ignoreNodes, url, proxy=false} = data; @@ -29,7 +35,7 @@ async function main() {          const popupHost = new PopupProxyHost();          await popupHost.prepare(); -        popup = popupHost.getOrCreatePopup(); +        popup = popupHost.getOrCreatePopup(null, null, depth);      }      const frontend = new Frontend(popup, ignoreNodes); diff --git a/ext/fg/js/frontend.js b/ext/fg/js/frontend.js index 67045241..768b9326 100644 --- a/ext/fg/js/frontend.js +++ b/ext/fg/js/frontend.js @@ -16,7 +16,14 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiGetZoom, apiOptionsGet, apiTermsFind, apiKanjiFind, docSentenceExtract, TextScanner*/ +/* global + * TextScanner + * apiGetZoom + * apiKanjiFind + * apiOptionsGet + * apiTermsFind + * docSentenceExtract + */  class Frontend extends TextScanner {      constructor(popup, ignoreNodes) { @@ -39,6 +46,15 @@ class Frontend extends TextScanner {          this._contentScale = 1.0;          this._orphaned = true;          this._lastShowPromise = Promise.resolve(); + +        this._windowMessageHandlers = new Map([ +            ['popupClose', () => this.onSearchClear(true)], +            ['selectionCopy', () => document.execCommand('copy')] +        ]); + +        this._runtimeMessageHandlers = new Map([ +            ['popupSetVisibleOverride', ({visible}) => { this.popup.setVisibleOverride(visible); }] +        ]);      }      async prepare() { @@ -55,9 +71,9 @@ class Frontend extends TextScanner {                  window.visualViewport.addEventListener('resize', this.onVisualViewportResize.bind(this));              } -            yomichan.on('orphaned', () => this.onOrphaned()); -            yomichan.on('optionsUpdated', () => this.updateOptions()); -            yomichan.on('zoomChanged', (e) => this.onZoomChanged(e)); +            yomichan.on('orphaned', this.onOrphaned.bind(this)); +            yomichan.on('optionsUpdated', this.updateOptions.bind(this)); +            yomichan.on('zoomChanged', this.onZoomChanged.bind(this));              chrome.runtime.onMessage.addListener(this.onRuntimeMessage.bind(this));              this._updateContentScale(); @@ -72,17 +88,17 @@ class Frontend extends TextScanner {      onWindowMessage(e) {          const action = e.data; -        const handler = Frontend._windowMessageHandlers.get(action); +        const handler = this._windowMessageHandlers.get(action);          if (typeof handler !== 'function') { return false; } -        handler(this); +        handler();      }      onRuntimeMessage({action, params}, sender, callback) { -        const handler = Frontend._runtimeMessageHandlers.get(action); +        const handler = this._runtimeMessageHandlers.get(action);          if (typeof handler !== 'function') { return false; } -        const result = handler(this, params, sender); +        const result = handler(params, sender);          callback(result);          return false;      } @@ -237,12 +253,3 @@ class Frontend extends TextScanner {          return visualViewport !== null && typeof visualViewport === 'object' ? visualViewport.scale : 1.0;      }  } - -Frontend._windowMessageHandlers = new Map([ -    ['popupClose', (self) => self.onSearchClear(true)], -    ['selectionCopy', () => document.execCommand('copy')] -]); - -Frontend._runtimeMessageHandlers = new Map([ -    ['popupSetVisibleOverride', (self, {visible}) => { self.popup.setVisibleOverride(visible); }] -]); diff --git a/ext/fg/js/popup-nested.js b/ext/fg/js/popup-nested.js index 3e5f5b80..06f8fc4b 100644 --- a/ext/fg/js/popup-nested.js +++ b/ext/fg/js/popup-nested.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiOptionsGet*/ +/* global + * apiOptionsGet + */  let popupNestedInitialized = false; diff --git a/ext/fg/js/popup-proxy-host.js b/ext/fg/js/popup-proxy-host.js index e55801ff..793d3949 100644 --- a/ext/fg/js/popup-proxy-host.js +++ b/ext/fg/js/popup-proxy-host.js @@ -16,7 +16,11 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiFrameInformationGet, FrontendApiReceiver, Popup*/ +/* global + * FrontendApiReceiver + * Popup + * apiFrameInformationGet + */  class PopupProxyHost {      constructor() { @@ -34,20 +38,20 @@ class PopupProxyHost {          if (typeof frameId !== 'number') { return; }          this._apiReceiver = new FrontendApiReceiver(`popup-proxy-host#${frameId}`, new Map([ -            ['getOrCreatePopup', ({id, parentId}) => this._onApiGetOrCreatePopup(id, parentId)], -            ['setOptions', ({id, options}) => this._onApiSetOptions(id, options)], -            ['hide', ({id, changeFocus}) => this._onApiHide(id, changeFocus)], -            ['isVisible', ({id}) => this._onApiIsVisibleAsync(id)], -            ['setVisibleOverride', ({id, visible}) => this._onApiSetVisibleOverride(id, visible)], -            ['containsPoint', ({id, x, y}) => this._onApiContainsPoint(id, x, y)], -            ['showContent', ({id, elementRect, writingMode, type, details}) => this._onApiShowContent(id, elementRect, writingMode, type, details)], -            ['setCustomCss', ({id, css}) => this._onApiSetCustomCss(id, css)], -            ['clearAutoPlayTimer', ({id}) => this._onApiClearAutoPlayTimer(id)], -            ['setContentScale', ({id, scale}) => this._onApiSetContentScale(id, scale)] +            ['getOrCreatePopup', this._onApiGetOrCreatePopup.bind(this)], +            ['setOptions', this._onApiSetOptions.bind(this)], +            ['hide', this._onApiHide.bind(this)], +            ['isVisible', this._onApiIsVisibleAsync.bind(this)], +            ['setVisibleOverride', this._onApiSetVisibleOverride.bind(this)], +            ['containsPoint', this._onApiContainsPoint.bind(this)], +            ['showContent', this._onApiShowContent.bind(this)], +            ['setCustomCss', this._onApiSetCustomCss.bind(this)], +            ['clearAutoPlayTimer', this._onApiClearAutoPlayTimer.bind(this)], +            ['setContentScale', this._onApiSetContentScale.bind(this)]          ]));      } -    getOrCreatePopup(id=null, parentId=null) { +    getOrCreatePopup(id=null, parentId=null, depth=null) {          // Find by existing id          if (id !== null) {              const popup = this._popups.get(id); @@ -76,7 +80,14 @@ class PopupProxyHost {          }          // Create new popup -        const depth = (parent !== null ? parent.depth + 1 : 0); +        if (parent !== null) { +            if (depth !== null) { +                throw new Error('Depth cannot be set when parent exists'); +            } +            depth = parent.depth + 1; +        } else if (depth === null) { +            depth = 0; +        }          const popup = new Popup(id, depth, this._frameIdPromise);          if (parent !== null) {              popup.setParent(parent); @@ -87,56 +98,57 @@ class PopupProxyHost {      // Message handlers -    async _onApiGetOrCreatePopup(id, parentId) { +    async _onApiGetOrCreatePopup({id, parentId}) {          const popup = this.getOrCreatePopup(id, parentId);          return {              id: popup.id          };      } -    async _onApiSetOptions(id, options) { +    async _onApiSetOptions({id, options}) {          const popup = this._getPopup(id);          return await popup.setOptions(options);      } -    async _onApiHide(id, changeFocus) { +    async _onApiHide({id, changeFocus}) {          const popup = this._getPopup(id);          return popup.hide(changeFocus);      } -    async _onApiIsVisibleAsync(id) { +    async _onApiIsVisibleAsync({id}) {          const popup = this._getPopup(id);          return await popup.isVisible();      } -    async _onApiSetVisibleOverride(id, visible) { +    async _onApiSetVisibleOverride({id, visible}) {          const popup = this._getPopup(id);          return await popup.setVisibleOverride(visible);      } -    async _onApiContainsPoint(id, x, y) { +    async _onApiContainsPoint({id, x, y}) {          const popup = this._getPopup(id); +        [x, y] = PopupProxyHost._convertPopupPointToRootPagePoint(popup, x, y);          return await popup.containsPoint(x, y);      } -    async _onApiShowContent(id, elementRect, writingMode, type, details) { +    async _onApiShowContent({id, elementRect, writingMode, type, details}) {          const popup = this._getPopup(id);          elementRect = PopupProxyHost._convertJsonRectToDOMRect(popup, elementRect);          if (!PopupProxyHost._popupCanShow(popup)) { return; }          return await popup.showContent(elementRect, writingMode, type, details);      } -    async _onApiSetCustomCss(id, css) { +    async _onApiSetCustomCss({id, css}) {          const popup = this._getPopup(id);          return popup.setCustomCss(css);      } -    async _onApiClearAutoPlayTimer(id) { +    async _onApiClearAutoPlayTimer({id}) {          const popup = this._getPopup(id);          return popup.clearAutoPlayTimer();      } -    async _onApiSetContentScale(id, scale) { +    async _onApiSetContentScale({id, scale}) {          const popup = this._getPopup(id);          return popup.setContentScale(scale);      } @@ -152,14 +164,17 @@ class PopupProxyHost {      }      static _convertJsonRectToDOMRect(popup, jsonRect) { -        let x = jsonRect.x; -        let y = jsonRect.y; +        const [x, y] = PopupProxyHost._convertPopupPointToRootPagePoint(popup, jsonRect.x, jsonRect.y); +        return new DOMRect(x, y, jsonRect.width, jsonRect.height); +    } + +    static _convertPopupPointToRootPagePoint(popup, x, y) {          if (popup.parent !== null) {              const popupRect = popup.parent.getContainerRect();              x += popupRect.x;              y += popupRect.y;          } -        return new DOMRect(x, y, jsonRect.width, jsonRect.height); +        return [x, y];      }      static _popupCanShow(popup) { diff --git a/ext/fg/js/popup-proxy.js b/ext/fg/js/popup-proxy.js index 093cdd2e..f7cef214 100644 --- a/ext/fg/js/popup-proxy.js +++ b/ext/fg/js/popup-proxy.js @@ -16,7 +16,9 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global FrontendApiSender*/ +/* global + * FrontendApiSender + */  class PopupProxy {      constructor(id, depth, parentId, parentFrameId, url) { diff --git a/ext/fg/js/popup.js b/ext/fg/js/popup.js index 4927f4bd..d752812e 100644 --- a/ext/fg/js/popup.js +++ b/ext/fg/js/popup.js @@ -16,7 +16,10 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global apiInjectStylesheet, apiGetMessageToken*/ +/* global + * apiGetMessageToken + * apiInjectStylesheet + */  class Popup {      constructor(id, depth, frameIdPromise) { @@ -260,7 +263,7 @@ class Popup {              'mozfullscreenchange',              'webkitfullscreenchange'          ]; -        const onFullscreenChanged = () => this._onFullscreenChanged(); +        const onFullscreenChanged = this._onFullscreenChanged.bind(this);          for (const eventName of fullscreenEvents) {              this._fullscreenEventListeners.addEventListener(document, eventName, onFullscreenChanged, false);          } diff --git a/ext/manifest.json b/ext/manifest.json index fd9b6fec..2a602e3b 100644 --- a/ext/manifest.json +++ b/ext/manifest.json @@ -1,7 +1,7 @@  {      "manifest_version": 2,      "name": "Yomichan (testing)", -    "version": "20.2.24.0", +    "version": "20.3.14.0",      "description": "Japanese dictionary with Anki integration (testing)",      "icons": {"16": "mixed/img/icon16.png", "48": "mixed/img/icon48.png", "128": "mixed/img/icon128.png"}, diff --git a/ext/mixed/css/display.css b/ext/mixed/css/display.css index 6a5383bc..688a357c 100644 --- a/ext/mixed/css/display.css +++ b/ext/mixed/css/display.css @@ -30,8 +30,8 @@   * General   */ -html:root[data-yomichan-page=float]:not([data-yomichan-theme]), -html:root[data-yomichan-page=float]:not([data-yomichan-theme]) body { +:root[data-yomichan-page=float]:not([data-yomichan-theme]), +:root[data-yomichan-page=float]:not([data-yomichan-theme]) body {      background-color: transparent;  } @@ -65,10 +65,6 @@ ol, ul {      height: 2.28571428em; /* 14px => 32px */  } -.invisible { -    visibility: hidden; -} -  /*   * Navigation   */ @@ -82,17 +78,18 @@ ol, ul {      padding: 0.25em 0.5em;      border-bottom-width: 0.07142857em; /* 14px => 1px */      border-bottom-style: solid; +    z-index: 10;  } -html:root[data-yomichan-page=search] .navigation-header { +:root[data-yomichan-page=search] .navigation-header {      position: sticky;  } -html:root[data-yomichan-page=float] .navigation-header { +:root[data-yomichan-page=float] .navigation-header {      position: fixed;  } -html:root[data-yomichan-page=float] .navigation-header:not([hidden])~.navigation-header-spacer { +:root[data-yomichan-page=float] .navigation-header:not([hidden])~.navigation-header-spacer {      height: 2.1em;  } @@ -136,7 +133,7 @@ html:root[data-yomichan-page=float] .navigation-header:not([hidden])~.navigation      margin-right: 0.2em;  } -html:root[data-yomichan-page=search][data-search-mode=popup] .search-input { +:root[data-yomichan-page=search][data-search-mode=popup] .search-input {      display: none;  } @@ -150,7 +147,7 @@ html:root[data-yomichan-page=search][data-search-mode=popup] .search-input {      padding-bottom: 0.72em;  } -html:root[data-yomichan-page=float] .entry { +:root[data-yomichan-page=float] .entry {      padding-left: 0.72em;      padding-right: 0.72em;  } @@ -231,7 +228,7 @@ button.action-button {      margin-right: 0.375em;  } -html:root:not([data-enable-search-tags=true]) .tag[data-category=search] { +:root:not([data-enable-search-tags=true]) .tag[data-category=search] {      display: none;  } @@ -280,10 +277,6 @@ html:root:not([data-enable-search-tags=true]) .tag[data-category=search] {      display: inline;  } -.term-expression-details>.action-play-audio { -    display: none; -} -  .term-expression-details>.tags {      display: inline;  } @@ -321,8 +314,8 @@ html:root:not([data-enable-search-tags=true]) .tag[data-category=search] {      bottom: 0.5em;  } -.term-expression-list[data-multi=true] .term-expression-details>.action-play-audio { -    display: block; +.term-expression-list:not([data-multi=true]) .term-expression-details>.action-play-audio { +    display: none;  }  .term-expression-list[data-multi=true] .term-expression-details>.tags { diff --git a/ext/mixed/js/api.js b/ext/mixed/js/api.js index 7ea68d59..0ab07039 100644 --- a/ext/mixed/js/api.js +++ b/ext/mixed/js/api.js @@ -69,8 +69,8 @@ function apiTemplateRender(template, data) {      return _apiInvoke('templateRender', {data, template});  } -function apiAudioGetUrl(definition, source, optionsContext) { -    return _apiInvoke('audioGetUrl', {definition, source, optionsContext}); +function apiAudioGetUri(definition, source, optionsContext) { +    return _apiInvoke('audioGetUri', {definition, source, optionsContext});  }  function apiCommandExec(command, params) { @@ -117,6 +117,10 @@ function apiGetMessageToken() {      return _apiInvoke('getMessageToken');  } +function apiGetDefaultAnkiFieldTemplates() { +    return _apiInvoke('getDefaultAnkiFieldTemplates'); +} +  function _apiInvoke(action, params={}) {      const data = {action, params};      return new Promise((resolve, reject) => { diff --git a/ext/mixed/js/audio-system.js b/ext/mixed/js/audio-system.js new file mode 100644 index 00000000..31c476b1 --- /dev/null +++ b/ext/mixed/js/audio-system.js @@ -0,0 +1,185 @@ +/* + * Copyright (C) 2019-2020  Alex Yatskov <alex@foosoft.net> + * Author: Alex Yatskov <alex@foosoft.net> + * + * 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 <https://www.gnu.org/licenses/>. + */ + +class TextToSpeechAudio { +    constructor(text, voice) { +        this.text = text; +        this.voice = voice; +        this._utterance = null; +        this._volume = 1; +    } + +    get currentTime() { +        return 0; +    } +    set currentTime(value) { +        // NOP +    } + +    get volume() { +        return this._volume; +    } +    set volume(value) { +        this._volume = value; +        if (this._utterance !== null) { +            this._utterance.volume = value; +        } +    } + +    play() { +        try { +            if (this._utterance === null) { +                this._utterance = new SpeechSynthesisUtterance(this.text || ''); +                this._utterance.lang = 'ja-JP'; +                this._utterance.volume = this._volume; +                this._utterance.voice = this.voice; +            } + +            speechSynthesis.cancel(); +            speechSynthesis.speak(this._utterance); +        } catch (e) { +            // NOP +        } +    } + +    pause() { +        try { +            speechSynthesis.cancel(); +        } catch (e) { +            // NOP +        } +    } +} + +class AudioSystem { +    constructor({getAudioUri}) { +        this._cache = new Map(); +        this._cacheSizeMaximum = 32; +        this._getAudioUri = getAudioUri; + +        if (typeof speechSynthesis !== 'undefined') { +            // speechSynthesis.getVoices() will not be populated unless some API call is made. +            speechSynthesis.addEventListener('voiceschanged', this._onVoicesChanged.bind(this)); +        } +    } + +    async getDefinitionAudio(definition, sources, details) { +        const key = `${definition.expression}:${definition.reading}`; +        const cacheValue = this._cache.get(definition); +        if (typeof cacheValue !== 'undefined') { +            const {audio, uri, source} = cacheValue; +            return {audio, uri, source}; +        } + +        for (const source of sources) { +            const uri = await this._getAudioUri(definition, source, details); +            if (uri === null) { continue; } + +            try { +                const audio = await this._createAudio(uri, details); +                this._cacheCheck(); +                this._cache.set(key, {audio, uri, source}); +                return {audio, uri, source}; +            } catch (e) { +                // NOP +            } +        } + +        throw new Error('Could not create audio'); +    } + +    createTextToSpeechAudio({text, voiceUri}) { +        const voice = this._getTextToSpeechVoiceFromVoiceUri(voiceUri); +        if (voice === null) { +            throw new Error('Invalid text-to-speech voice'); +        } +        return new TextToSpeechAudio(text, voice); +    } + +    _onVoicesChanged() { +        // NOP +    } + +    async _createAudio(uri, details) { +        const ttsParameters = this._getTextToSpeechParameters(uri); +        if (ttsParameters !== null) { +            if (typeof details === 'object' && details !== null) { +                if (details.tts === false) { +                    throw new Error('Text-to-speech not permitted'); +                } +            } +            return this.createTextToSpeechAudio(ttsParameters); +        } + +        return await this._createAudioFromUrl(uri); +    } + +    _createAudioFromUrl(url) { +        return new Promise((resolve, reject) => { +            const audio = new Audio(url); +            audio.addEventListener('loadeddata', () => { +                const duration = audio.duration; +                if (duration === 5.694694 || duration === 5.720718) { +                    // Hardcoded values for invalid audio +                    reject(new Error('Could not retrieve audio')); +                } else { +                    resolve(audio); +                } +            }); +            audio.addEventListener('error', () => reject(audio.error)); +        }); +    } + +    _getTextToSpeechVoiceFromVoiceUri(voiceUri) { +        try { +            for (const voice of speechSynthesis.getVoices()) { +                if (voice.voiceURI === voiceUri) { +                    return voice; +                } +            } +        } catch (e) { +            // NOP +        } +        return null; +    } + +    _getTextToSpeechParameters(uri) { +        const m = /^tts:[^#?]*\?([^#]*)/.exec(uri); +        if (m === null) { return null; } + +        const searchParameters = new URLSearchParams(m[1]); +        const text = searchParameters.get('text'); +        const voiceUri = searchParameters.get('voice'); +        return (text !== null && voiceUri !== null ? {text, voiceUri} : null); +    } + +    _cacheCheck() { +        const removeCount = this._cache.size - this._cacheSizeMaximum; +        if (removeCount <= 0) { return; } + +        const removeKeys = []; +        for (const key of this._cache.keys()) { +            removeKeys.push(key); +            if (removeKeys.length >= removeCount) { break; } +        } + +        for (const key of removeKeys) { +            this._cache.delete(key); +        } +    } +} diff --git a/ext/mixed/js/audio.js b/ext/mixed/js/audio.js deleted file mode 100644 index b5a025be..00000000 --- a/ext/mixed/js/audio.js +++ /dev/null @@ -1,178 +0,0 @@ -/* - * Copyright (C) 2019-2020  Alex Yatskov <alex@foosoft.net> - * Author: Alex Yatskov <alex@foosoft.net> - * - * 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 <https://www.gnu.org/licenses/>. - */ - -/*global apiAudioGetUrl*/ - -class TextToSpeechAudio { -    constructor(text, voice) { -        this.text = text; -        this.voice = voice; -        this._utterance = null; -        this._volume = 1; -    } - -    get currentTime() { -        return 0; -    } -    set currentTime(value) { -        // NOP -    } - -    get volume() { -        return this._volume; -    } -    set volume(value) { -        this._volume = value; -        if (this._utterance !== null) { -            this._utterance.volume = value; -        } -    } - -    play() { -        try { -            if (this._utterance === null) { -                this._utterance = new SpeechSynthesisUtterance(this.text || ''); -                this._utterance.lang = 'ja-JP'; -                this._utterance.volume = this._volume; -                this._utterance.voice = this.voice; -            } - -            speechSynthesis.cancel(); -            speechSynthesis.speak(this._utterance); -        } catch (e) { -            // NOP -        } -    } - -    pause() { -        try { -            speechSynthesis.cancel(); -        } catch (e) { -            // NOP -        } -    } - -    static createFromUri(ttsUri) { -        const m = /^tts:[^#?]*\?([^#]*)/.exec(ttsUri); -        if (m === null) { return null; } - -        const searchParameters = new URLSearchParams(m[1]); -        const text = searchParameters.get('text'); -        let voice = searchParameters.get('voice'); -        if (text === null || voice === null) { return null; } - -        voice = audioGetTextToSpeechVoice(voice); -        if (voice === null) { return null; } - -        return new TextToSpeechAudio(text, voice); -    } -} - -function audioGetFromUrl(url, willDownload) { -    const tts = TextToSpeechAudio.createFromUri(url); -    if (tts !== null) { -        if (willDownload) { -            throw new Error('AnkiConnect does not support downloading text-to-speech audio.'); -        } -        return Promise.resolve(tts); -    } - -    return new Promise((resolve, reject) => { -        const audio = new Audio(url); -        audio.addEventListener('loadeddata', () => { -            if (audio.duration === 5.694694 || audio.duration === 5.720718) { -                // Hardcoded values for invalid audio -                reject(new Error('Could not retrieve audio')); -            } else { -                resolve(audio); -            } -        }); -        audio.addEventListener('error', () => reject(audio.error)); -    }); -} - -async function audioGetFromSources(expression, sources, optionsContext, willDownload, cache=null) { -    const key = `${expression.expression}:${expression.reading}`; -    if (cache !== null) { -        const cacheValue = cache.get(expression); -        if (typeof cacheValue !== 'undefined') { -            return cacheValue; -        } -    } - -    for (let i = 0, ii = sources.length; i < ii; ++i) { -        const source = sources[i]; -        const url = await apiAudioGetUrl(expression, source, optionsContext); -        if (url === null) { -            continue; -        } - -        try { -            let audio = await audioGetFromUrl(url, willDownload); -            if (willDownload) { -                // AnkiConnect handles downloading URLs into cards -                audio = null; -            } -            const result = {audio, url, source}; -            if (cache !== null) { -                cache.set(key, result); -            } -            return result; -        } catch (e) { -            // NOP -        } -    } -    return {audio: null, url: null, source: null}; -} - -function audioGetTextToSpeechVoice(voiceURI) { -    try { -        for (const voice of speechSynthesis.getVoices()) { -            if (voice.voiceURI === voiceURI) { -                return voice; -            } -        } -    } catch (e) { -        // NOP -    } -    return null; -} - -function audioPrepareTextToSpeech(options) { -    if ( -        audioPrepareTextToSpeech.state || -        !options.audio.textToSpeechVoice || -        !( -            options.audio.sources.includes('text-to-speech') || -            options.audio.sources.includes('text-to-speech-reading') -        ) -    ) { -        // Text-to-speech not in use. -        return; -    } - -    // Chrome needs this value called once before it will become populated. -    // The first call will return an empty list. -    audioPrepareTextToSpeech.state = true; -    try { -        speechSynthesis.getVoices(); -    } catch (e) { -        // NOP -    } -} -audioPrepareTextToSpeech.state = false; diff --git a/ext/mixed/js/core.js b/ext/mixed/js/core.js index 83813796..0d50e915 100644 --- a/ext/mixed/js/core.js +++ b/ext/mixed/js/core.js @@ -175,21 +175,6 @@ function promiseTimeout(delay, resolveValue) {      return promise;  } -function stringReplaceAsync(str, regex, replacer) { -    let match; -    let index = 0; -    const parts = []; -    while ((match = regex.exec(str)) !== null) { -        parts.push(str.substring(index, match.index), replacer(...match, match.index, str)); -        index = regex.lastIndex; -    } -    if (parts.length === 0) { -        return Promise.resolve(str); -    } -    parts.push(str.substring(index)); -    return Promise.all(parts).then((v) => v.join('')); -} -  /*   * Common events @@ -269,7 +254,11 @@ const yomichan = (() => {          constructor() {              super(); +            this._isBackendPreparedResolve = null; +            this._isBackendPreparedPromise = new Promise((resolve) => (this._isBackendPreparedResolve = resolve)); +              this._messageHandlers = new Map([ +                ['backendPrepared', this._onBackendPrepared.bind(this)],                  ['getUrl', this._onMessageGetUrl.bind(this)],                  ['optionsUpdated', this._onMessageOptionsUpdated.bind(this)],                  ['zoomChanged', this._onMessageZoomChanged.bind(this)] @@ -280,6 +269,11 @@ const yomichan = (() => {          // Public +        prepare() { +            chrome.runtime.sendMessage({action: 'yomichanCoreReady'}); +            return this._isBackendPreparedPromise; +        } +          generateId(length) {              const array = new Uint8Array(length);              window.crypto.getRandomValues(array); @@ -305,6 +299,10 @@ const yomichan = (() => {              return false;          } +        _onBackendPrepared() { +            this._isBackendPreparedResolve(); +        } +          _onMessageGetUrl() {              return {url: window.location.href};          } diff --git a/ext/mixed/js/display-generator.js b/ext/mixed/js/display-generator.js index d7e77cc0..49afc44b 100644 --- a/ext/mixed/js/display-generator.js +++ b/ext/mixed/js/display-generator.js @@ -16,7 +16,10 @@   * along with this program.  If not, see <http://www.gnu.org/licenses/>.   */ -/*global apiGetDisplayTemplatesHtml, TemplateHandler*/ +/* global + * TemplateHandler + * apiGetDisplayTemplatesHtml + */  class DisplayGenerator {      constructor() { @@ -298,7 +301,7 @@ class DisplayGenerator {      }      static _isCharacterKanji(c) { -        const code = c.charCodeAt(0); +        const code = c.codePointAt(0);          return (              code >= 0x4e00 && code < 0x9fb0 ||              code >= 0x3400 && code < 0x4dc0 diff --git a/ext/mixed/js/display.js b/ext/mixed/js/display.js index 5d3076ee..515e28a7 100644 --- a/ext/mixed/js/display.js +++ b/ext/mixed/js/display.js @@ -16,11 +16,24 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global docRangeFromPoint, docSentenceExtract -apiKanjiFind, apiTermsFind, apiNoteView, apiOptionsGet, apiDefinitionsAddable, apiDefinitionAdd -apiScreenshotGet, apiForward -audioPrepareTextToSpeech, audioGetFromSources -DisplayGenerator, WindowScroll, DisplayContext, DOM*/ +/* global + * AudioSystem + * DOM + * DisplayContext + * DisplayGenerator + * WindowScroll + * apiAudioGetUri + * apiDefinitionAdd + * apiDefinitionsAddable + * apiForward + * apiKanjiFind + * apiNoteView + * apiOptionsGet + * apiScreenshotGet + * apiTermsFind + * docRangeFromPoint + * docSentenceExtract + */  class Display {      constructor(spinner, container) { @@ -32,7 +45,7 @@ class Display {          this.index = 0;          this.audioPlaying = null;          this.audioFallback = null; -        this.audioCache = new Map(); +        this.audioSystem = new AudioSystem({getAudioUri: this._getAudioUri.bind(this)});          this.styleNode = null;          this.eventListeners = new EventListenerCollection(); @@ -45,10 +58,115 @@ class Display {          this.displayGenerator = new DisplayGenerator();          this.windowScroll = new WindowScroll(); +        this._onKeyDownHandlers = new Map([ +            ['Escape', () => { +                this.onSearchClear(); +                return true; +            }], +            ['PageUp', (e) => { +                if (e.altKey) { +                    this.entryScrollIntoView(this.index - 3, null, true); +                    return true; +                } +                return false; +            }], +            ['PageDown', (e) => { +                if (e.altKey) { +                    this.entryScrollIntoView(this.index + 3, null, true); +                    return true; +                } +                return false; +            }], +            ['End', (e) => { +                if (e.altKey) { +                    this.entryScrollIntoView(this.definitions.length - 1, null, true); +                    return true; +                } +                return false; +            }], +            ['Home', (e) => { +                if (e.altKey) { +                    this.entryScrollIntoView(0, null, true); +                    return true; +                } +                return false; +            }], +            ['ArrowUp', (e) => { +                if (e.altKey) { +                    this.entryScrollIntoView(this.index - 1, null, true); +                    return true; +                } +                return false; +            }], +            ['ArrowDown', (e) => { +                if (e.altKey) { +                    this.entryScrollIntoView(this.index + 1, null, true); +                    return true; +                } +                return false; +            }], +            ['B', (e) => { +                if (e.altKey) { +                    this.sourceTermView(); +                    return true; +                } +                return false; +            }], +            ['F', (e) => { +                if (e.altKey) { +                    this.nextTermView(); +                    return true; +                } +                return false; +            }], +            ['E', (e) => { +                if (e.altKey) { +                    this.noteTryAdd('term-kanji'); +                    return true; +                } +                return false; +            }], +            ['K', (e) => { +                if (e.altKey) { +                    this.noteTryAdd('kanji'); +                    return true; +                } +                return false; +            }], +            ['R', (e) => { +                if (e.altKey) { +                    this.noteTryAdd('term-kana'); +                    return true; +                } +                return false; +            }], +            ['P', (e) => { +                if (e.altKey) { +                    const index = this.index; +                    if (index < 0 || index >= this.definitions.length) { return; } + +                    const entry = this.getEntry(index); +                    if (entry !== null && entry.dataset.type === 'term') { +                        this.audioPlay(this.definitions[index], this.firstExpressionIndex, index); +                    } +                    return true; +                } +                return false; +            }], +            ['V', (e) => { +                if (e.altKey) { +                    this.noteTryView(); +                    return true; +                } +                return false; +            }] +        ]); +          this.setInteractive(true);      }      async prepare(options=null) { +        await yomichan.prepare();          const displayGeneratorPromise = this.displayGenerator.prepare();          const updateOptionsPromise = this.updateOptions(options);          await Promise.all([displayGeneratorPromise, updateOptionsPromise]); @@ -215,9 +333,9 @@ class Display {      onKeyDown(e) {          const key = Display.getKeyFromEvent(e); -        const handler = Display._onKeyDownHandlers.get(key); +        const handler = this._onKeyDownHandlers.get(key);          if (typeof handler === 'function') { -            if (handler(this, e)) { +            if (handler(e)) {                  e.preventDefault();                  return true;              } @@ -259,13 +377,12 @@ class Display {          this.updateDocumentOptions(this.options);          this.updateTheme(this.options.general.popupTheme);          this.setCustomCss(this.options.general.customPopupCss); -        audioPrepareTextToSpeech(this.options);      }      updateDocumentOptions(options) {          const data = document.documentElement.dataset;          data.ankiEnabled = `${options.anki.enable}`; -        data.audioEnabled = `${options.audio.enable}`; +        data.audioEnabled = `${options.audio.enabled}`;          data.compactGlossaries = `${options.general.compactGlossaries}`;          data.enableSearchTags = `${options.scanning.enableSearchTags}`;          data.debug = `${options.general.debugInfo}`; @@ -520,15 +637,13 @@ class Display {      updateAdderButtons(states) {          for (let i = 0; i < states.length; ++i) { -            const state = states[i];              let noteId = null; -            for (const mode in state) { +            for (const [mode, info] of Object.entries(states[i])) {                  const button = this.adderButtonFind(i, mode);                  if (button === null) {                      continue;                  } -                const info = state[mode];                  if (!info.canAdd && noteId === null && info.noteId) {                      noteId = info.noteId;                  } @@ -635,7 +750,7 @@ class Display {              this.setSpinnerVisible(true);              const context = {}; -            if (this.noteUsesScreenshot()) { +            if (this.noteUsesScreenshot(mode)) {                  const screenshot = await this.getScreenshot();                  if (screenshot) {                      context.screenshot = screenshot; @@ -672,16 +787,16 @@ class Display {              }              const sources = this.options.audio.sources; -            let {audio, source} = await audioGetFromSources(expression, sources, this.getOptionsContext(), false, this.audioCache); -            let info; -            if (audio === null) { +            let audio, source, info; +            try { +                ({audio, source} = await this.audioSystem.getDefinitionAudio(expression, sources)); +                info = `From source ${1 + sources.indexOf(source)}: ${source}`; +            } catch (e) {                  if (this.audioFallback === null) {                      this.audioFallback = new Audio('/mixed/mp3/button.mp3');                  }                  audio = this.audioFallback;                  info = 'Could not find audio'; -            } else { -                info = `From source ${1 + sources.indexOf(source)}: ${source}`;              }              const button = this.audioButtonFindImage(entryIndex); @@ -705,10 +820,11 @@ class Display {          }      } -    noteUsesScreenshot() { -        const fields = this.options.anki.terms.fields; -        for (const name in fields) { -            if (fields[name].includes('{screenshot}')) { +    noteUsesScreenshot(mode) { +        const optionsAnki = this.options.anki; +        const fields = (mode === 'kanji' ? optionsAnki.kanji : optionsAnki.terms).fields; +        for (const fieldValue of Object.values(fields)) { +            if (fieldValue.includes('{screenshot}')) {                  return true;              }          } @@ -814,121 +930,9 @@ class Display {          const key = event.key;          return (typeof key === 'string' ? (key.length === 1 ? key.toUpperCase() : key) : '');      } -} - -Display._onKeyDownHandlers = new Map([ -    ['Escape', (self) => { -        self.onSearchClear(); -        return true; -    }], - -    ['PageUp', (self, e) => { -        if (e.altKey) { -            self.entryScrollIntoView(self.index - 3, null, true); -            return true; -        } -        return false; -    }], - -    ['PageDown', (self, e) => { -        if (e.altKey) { -            self.entryScrollIntoView(self.index + 3, null, true); -            return true; -        } -        return false; -    }], - -    ['End', (self, e) => { -        if (e.altKey) { -            self.entryScrollIntoView(self.definitions.length - 1, null, true); -            return true; -        } -        return false; -    }], - -    ['Home', (self, e) => { -        if (e.altKey) { -            self.entryScrollIntoView(0, null, true); -            return true; -        } -        return false; -    }], -    ['ArrowUp', (self, e) => { -        if (e.altKey) { -            self.entryScrollIntoView(self.index - 1, null, true); -            return true; -        } -        return false; -    }], - -    ['ArrowDown', (self, e) => { -        if (e.altKey) { -            self.entryScrollIntoView(self.index + 1, null, true); -            return true; -        } -        return false; -    }], - -    ['B', (self, e) => { -        if (e.altKey) { -            self.sourceTermView(); -            return true; -        } -        return false; -    }], - -    ['F', (self, e) => { -        if (e.altKey) { -            self.nextTermView(); -            return true; -        } -        return false; -    }], - -    ['E', (self, e) => { -        if (e.altKey) { -            self.noteTryAdd('term-kanji'); -            return true; -        } -        return false; -    }], - -    ['K', (self, e) => { -        if (e.altKey) { -            self.noteTryAdd('kanji'); -            return true; -        } -        return false; -    }], - -    ['R', (self, e) => { -        if (e.altKey) { -            self.noteTryAdd('term-kana'); -            return true; -        } -        return false; -    }], - -    ['P', (self, e) => { -        if (e.altKey) { -            const index = self.index; -            if (index < 0 || index >= self.definitions.length) { return; } - -            const entry = self.getEntry(index); -            if (entry !== null && entry.dataset.type === 'term') { -                self.audioPlay(self.definitions[index], self.firstExpressionIndex, index); -            } -            return true; -        } -        return false; -    }], - -    ['V', (self, e) => { -        if (e.altKey) { -            self.noteTryView(); -            return true; -        } -        return false; -    }] -]); +    async _getAudioUri(definition, source) { +        const optionsContext = this.getOptionsContext(); +        return await apiAudioGetUri(definition, source, optionsContext); +    } +} diff --git a/ext/mixed/js/scroll.js b/ext/mixed/js/scroll.js index 5829d294..72da8b65 100644 --- a/ext/mixed/js/scroll.js +++ b/ext/mixed/js/scroll.js @@ -26,7 +26,7 @@ class WindowScroll {          this.animationEndTime = 0;          this.animationEndX = 0;          this.animationEndY = 0; -        this.requestAnimationFrameCallback = (t) => this.onAnimationFrame(t); +        this.requestAnimationFrameCallback = this.onAnimationFrame.bind(this);      }      toY(y) { diff --git a/ext/mixed/js/text-scanner.js b/ext/mixed/js/text-scanner.js index ff0eac8b..a08e09fb 100644 --- a/ext/mixed/js/text-scanner.js +++ b/ext/mixed/js/text-scanner.js @@ -16,7 +16,11 @@   * along with this program.  If not, see <https://www.gnu.org/licenses/>.   */ -/*global docRangeFromPoint, TextSourceRange, DOM*/ +/* global + * DOM + * TextSourceRange + * docRangeFromPoint + */  class TextScanner {      constructor(node, ignoreNodes, ignoreElements, ignorePoints) { |