/*
 * Copyright (C) 2023-2024  Yomitan Authors
 * Copyright (C) 2019-2022  Yomichan Authors
 *
 * 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/>.
 */

import {AnkiConnect} from '../../comm/anki-connect.js';
import {EventListenerCollection} from '../../core/event-listener-collection.js';
import {ExtensionError} from '../../core/extension-error.js';
import {log} from '../../core/log.js';
import {toError} from '../../core/to-error.js';
import {getDynamicFieldMarkers, getStandardFieldMarkers} from '../../data/anki-template-util.js';
import {stringContainsAnyFieldMarker} from '../../data/anki-util.js';
import {getRequiredPermissionsForAnkiFieldValue, hasPermissions, setPermissionsGranted} from '../../data/permissions-util.js';
import {querySelectorNotNull} from '../../dom/query-selector.js';
import {SelectorObserver} from '../../dom/selector-observer.js';
import {ObjectPropertyAccessor} from '../../general/object-property-accessor.js';

export class AnkiController {
    /**
     * @param {import('./settings-controller.js').SettingsController} settingsController
     */
    constructor(settingsController) {
        /** @type {import('./settings-controller.js').SettingsController} */
        this._settingsController = settingsController;
        /** @type {AnkiConnect} */
        this._ankiConnect = new AnkiConnect();
        /** @type {SelectorObserver<AnkiCardController>} */
        this._selectorObserver = new SelectorObserver({
            selector: '.anki-card',
            ignoreSelector: null,
            onAdded: this._createCardController.bind(this),
            onRemoved: this._removeCardController.bind(this),
            isStale: this._isCardControllerStale.bind(this)
        });
        /** @type {Intl.Collator} */
        this._stringComparer = new Intl.Collator(); // Locale does not matter
        /** @type {?Promise<import('anki-controller').AnkiData>} */
        this._getAnkiDataPromise = null;
        /** @type {HTMLElement} */
        this._ankiErrorMessageNode = querySelectorNotNull(document, '#anki-error-message');
        const ankiErrorMessageNodeDefaultContent = this._ankiErrorMessageNode.textContent;
        /** @type {string} */
        this._ankiErrorMessageNodeDefaultContent = typeof ankiErrorMessageNodeDefaultContent === 'string' ? ankiErrorMessageNodeDefaultContent : '';
        /** @type {HTMLElement} */
        this._ankiErrorMessageDetailsNode = querySelectorNotNull(document, '#anki-error-message-details');
        /** @type {HTMLElement} */
        this._ankiErrorMessageDetailsContainer = querySelectorNotNull(document, '#anki-error-message-details-container');
        /** @type {HTMLElement} */
        this._ankiErrorMessageDetailsToggle = querySelectorNotNull(document, '#anki-error-message-details-toggle');
        /** @type {HTMLElement} */
        this._ankiErrorInvalidResponseInfo = querySelectorNotNull(document, '#anki-error-invalid-response-info');
        /** @type {HTMLElement} */
        this._ankiCardPrimary = querySelectorNotNull(document, '#anki-card-primary');
        /** @type {?Error} */
        this._ankiError = null;
        /** @type {?import('core').TokenObject} */
        this._validateFieldsToken = null;
        /** @type {?HTMLInputElement} */
        this._ankiEnableCheckbox = document.querySelector('[data-setting="anki.enable"]');
    }

    /** @type {import('./settings-controller.js').SettingsController} */
    get settingsController() {
        return this._settingsController;
    }

    /** */
    async prepare() {
        /** @type {HTMLElement} */
        const ankiApiKeyInput = querySelectorNotNull(document, '#anki-api-key-input');
        const ankiCardPrimaryTypeRadios = /** @type {NodeListOf<HTMLInputElement>} */ (document.querySelectorAll('input[type=radio][name=anki-card-primary-type]'));
        /** @type {HTMLElement} */
        const ankiErrorLog = querySelectorNotNull(document, '#anki-error-log');


        this._ankiErrorMessageDetailsToggle.addEventListener('click', this._onAnkiErrorMessageDetailsToggleClick.bind(this), false);
        if (this._ankiEnableCheckbox !== null) {
            this._ankiEnableCheckbox.addEventListener(
                /** @type {string} */ ('settingChanged'),
                /** @type {EventListener} */ (this._onAnkiEnableChanged.bind(this)),
                false
            );
        }
        for (const input of ankiCardPrimaryTypeRadios) {
            input.addEventListener('change', this._onAnkiCardPrimaryTypeRadioChange.bind(this), false);
        }

        const testAnkiNoteViewerButtons = /** @type {NodeListOf<HTMLButtonElement>} */ (document.querySelectorAll('.test-anki-note-viewer-button'));
        const onTestAnkiNoteViewerButtonClick = this._onTestAnkiNoteViewerButtonClick.bind(this);
        for (const button of testAnkiNoteViewerButtons) {
            button.addEventListener('click', onTestAnkiNoteViewerButtonClick, false);
        }

        ankiErrorLog.addEventListener('click', this._onAnkiErrorLogLinkClick.bind(this));

        ankiApiKeyInput.addEventListener('focus', this._onApiKeyInputFocus.bind(this));
        ankiApiKeyInput.addEventListener('blur', this._onApiKeyInputBlur.bind(this));

        await this._updateOptions();
        this._settingsController.on('optionsChanged', this._onOptionsChanged.bind(this));

        const onAnkiSettingChanged = () => { void this._updateOptions(); };
        const nodes = [ankiApiKeyInput, ...document.querySelectorAll('[data-setting="anki.enable"]')];
        for (const node of nodes) {
            node.addEventListener('settingChanged', onAnkiSettingChanged);
        }
    }

    /**
     * @returns {Promise<import('anki-controller').AnkiData>}
     */
    async getAnkiData() {
        let promise = this._getAnkiDataPromise;
        if (promise === null) {
            promise = this._getAnkiData();
            this._getAnkiDataPromise = promise;
            void promise.finally(() => { this._getAnkiDataPromise = null; });
        }
        return promise;
    }

    /**
     * @param {string} model
     * @returns {Promise<string[]>}
     */
    async getModelFieldNames(model) {
        return await this._ankiConnect.getModelFieldNames(model);
    }

    /**
     * @param {string} fieldValue
     * @returns {string[]}
     */
    getRequiredPermissions(fieldValue) {
        return getRequiredPermissionsForAnkiFieldValue(fieldValue);
    }

    // Private

    /** */
    async _updateOptions() {
        const options = await this._settingsController.getOptions();
        const optionsContext = this._settingsController.getOptionsContext();
        this._onOptionsChanged({options, optionsContext});
    }

    /**
     * @param {import('settings-controller').EventArgument<'optionsChanged'>} details
     */
    _onOptionsChanged({options: {anki, dictionaries}}) {
        /** @type {?string} */
        let apiKey = anki.apiKey;
        if (apiKey === '') { apiKey = null; }
        this._ankiConnect.server = anki.server;
        this._ankiConnect.enabled = anki.enable;
        this._ankiConnect.apiKey = apiKey;

        this._selectorObserver.disconnect();
        this._selectorObserver.observe(document.documentElement, true);

        this._setupFieldMenus(dictionaries);
    }

    /** */
    _onAnkiErrorMessageDetailsToggleClick() {
        const node = /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsContainer);
        node.hidden = !node.hidden;
    }

    /**
     * @param {import('dom-data-binder').SettingChangedEvent} event
     */
    _onAnkiEnableChanged({detail: {value}}) {
        if (this._ankiConnect.server === null) { return; }
        this._ankiConnect.enabled = typeof value === 'boolean' && value;

        for (const cardController of this._selectorObserver.datas()) {
            void cardController.updateAnkiState();
        }
    }

    /**
     * @param {Event} e
     */
    _onAnkiCardPrimaryTypeRadioChange(e) {
        const node = /** @type {HTMLInputElement} */ (e.currentTarget);
        if (!node.checked) { return; }
        const {value, ankiCardMenu} = node.dataset;
        if (typeof value !== 'string') { return; }
        this._setAnkiCardPrimaryType(value, ankiCardMenu);
    }

    /** */
    _onAnkiErrorLogLinkClick() {
        if (this._ankiError === null) { return; }
        log.log({error: this._ankiError});
    }

    /**
     * @param {MouseEvent} e
     */
    _onTestAnkiNoteViewerButtonClick(e) {
        const element = /** @type {HTMLElement} */ (e.currentTarget);
        // Anki note GUI mode
        const {mode} = element.dataset;
        if (typeof mode !== 'string') { return; }

        const normalizedMode = this._normalizeAnkiNoteGuiMode(mode);
        if (normalizedMode === null) { return; }
        void this._testAnkiNoteViewerSafe(normalizedMode);
    }

    /**
     * @param {Event} e
     */
    _onApiKeyInputFocus(e) {
        const element = /** @type {HTMLInputElement} */ (e.currentTarget);
        element.type = 'text';
    }

    /**
     * @param {Event} e
     */
    _onApiKeyInputBlur(e) {
        const element = /** @type {HTMLInputElement} */ (e.currentTarget);
        element.type = 'password';
    }

    /**
     * @param {string} ankiCardType
     * @param {string} [ankiCardMenu]
     */
    _setAnkiCardPrimaryType(ankiCardType, ankiCardMenu) {
        if (this._ankiCardPrimary === null) { return; }
        this._ankiCardPrimary.dataset.ankiCardType = ankiCardType;
        if (typeof ankiCardMenu !== 'undefined') {
            this._ankiCardPrimary.dataset.ankiCardMenu = ankiCardMenu;
        } else {
            delete this._ankiCardPrimary.dataset.ankiCardMenu;
        }
    }

    /**
     * @param {Element} node
     * @returns {AnkiCardController}
     */
    _createCardController(node) {
        const cardController = new AnkiCardController(this._settingsController, this, /** @type {HTMLElement} */ (node));
        void cardController.prepare();
        return cardController;
    }

    /**
     * @param {Element} _node
     * @param {AnkiCardController} cardController
     */
    _removeCardController(_node, cardController) {
        cardController.cleanup();
    }

    /**
     * @param {Element} _node
     * @param {AnkiCardController} cardController
     * @returns {boolean}
     */
    _isCardControllerStale(_node, cardController) {
        return cardController.isStale();
    }

    /**
     * @param {import('settings').DictionariesOptions} dictionaries
     */
    _setupFieldMenus(dictionaries) {
        /** @type {[types: import('dictionary').DictionaryEntryType[], templateName: string][]} */
        const fieldMenuTargets = [
            [['term'], 'anki-card-terms-field-menu'],
            [['kanji'], 'anki-card-kanji-field-menu'],
            [['term', 'kanji'], 'anki-card-all-field-menu']
        ];
        const {templates} = this._settingsController;
        for (const [types, templateName] of fieldMenuTargets) {
            const templateContent = templates.getTemplateContent(templateName);
            if (templateContent === null) {
                log.warn(new Error(`Failed to set up menu "${templateName}": element not found`));
                continue;
            }

            const container = templateContent.querySelector('.popup-menu-body');
            if (container === null) {
                log.warn(new Error(`Failed to set up menu "${templateName}": body not found`));
                return;
            }

            while (container.firstChild) {
                container.removeChild(container.firstChild);
            }

            let markers = [];
            for (const type of types) {
                markers.push(...getStandardFieldMarkers(type));
            }
            if (types.includes('term')) {
                markers.push(...getDynamicFieldMarkers(dictionaries));
            }
            markers = [...new Set(markers.sort())];

            const fragment = document.createDocumentFragment();
            for (const marker of markers) {
                const option = document.createElement('button');
                option.textContent = marker;
                option.className = 'popup-menu-item popup-menu-item-thin';
                option.dataset.menuAction = 'setFieldMarker';
                option.dataset.marker = marker;
                fragment.appendChild(option);
            }
            container.appendChild(fragment);
        }
    }

    /**
     * @returns {Promise<import('anki-controller').AnkiData>}
     */
    async _getAnkiData() {
        this._setAnkiStatusChanging();
        const [
            [deckNames, getDeckNamesError],
            [modelNames, getModelNamesError]
        ] = await Promise.all([
            this._getDeckNames(),
            this._getModelNames()
        ]);

        if (getDeckNamesError !== null) {
            this._showAnkiError(getDeckNamesError);
        } else if (getModelNamesError !== null) {
            this._showAnkiError(getModelNamesError);
        } else {
            this._hideAnkiError();
        }

        return {deckNames, modelNames};
    }

    /**
     * @returns {Promise<[deckNames: string[], error: ?Error]>}
     */
    async _getDeckNames() {
        try {
            const result = await this._ankiConnect.getDeckNames();
            this._sortStringArray(result);
            return [result, null];
        } catch (e) {
            return [[], toError(e)];
        }
    }

    /**
     * @returns {Promise<[modelNames: string[], error: ?Error]>}
     */
    async _getModelNames() {
        try {
            const result = await this._ankiConnect.getModelNames();
            this._sortStringArray(result);
            return [result, null];
        } catch (e) {
            return [[], toError(e)];
        }
    }

    /** */
    _setAnkiStatusChanging() {
        const ankiErrorMessageNode = /** @type {HTMLElement} */ (this._ankiErrorMessageNode);
        ankiErrorMessageNode.textContent = this._ankiErrorMessageNodeDefaultContent;
        ankiErrorMessageNode.classList.remove('danger-text');
    }

    /** */
    _hideAnkiError() {
        const ankiErrorMessageNode = /** @type {HTMLElement} */ (this._ankiErrorMessageNode);
        /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsContainer).hidden = true;
        /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsToggle).hidden = true;
        /** @type {HTMLElement} */ (this._ankiErrorInvalidResponseInfo).hidden = true;
        ankiErrorMessageNode.textContent = (this._ankiConnect.enabled ? 'Connected' : 'Not enabled');
        ankiErrorMessageNode.classList.remove('danger-text');
        /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsNode).textContent = '';
        this._ankiError = null;
    }

    /**
     * @param {Error} error
     */
    _showAnkiError(error) {
        const ankiErrorMessageNode = /** @type {HTMLElement} */ (this._ankiErrorMessageNode);
        this._ankiError = error;

        let errorString = typeof error === 'object' && error !== null ? error.message : null;
        if (!errorString) { errorString = `${error}`; }
        if (!/[.!?]$/.test(errorString)) { errorString += '.'; }
        ankiErrorMessageNode.textContent = errorString;
        ankiErrorMessageNode.classList.add('danger-text');

        const data = error instanceof ExtensionError ? error.data : void 0;
        let details = '';
        if (typeof data !== 'undefined') {
            details += `${JSON.stringify(data, null, 4)}\n\n`;
        }
        details += `${error.stack}`.trimEnd();
        /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsNode).textContent = details;

        /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsContainer).hidden = true;
        /** @type {HTMLElement} */ (this._ankiErrorInvalidResponseInfo).hidden = !errorString.includes('Invalid response');
        /** @type {HTMLElement} */ (this._ankiErrorMessageDetailsToggle).hidden = false;
    }

    /**
     * @param {string[]} array
     */
    _sortStringArray(array) {
        const stringComparer = this._stringComparer;
        array.sort((a, b) => stringComparer.compare(a, b));
    }

    /**
     * @param {import('settings').AnkiNoteGuiMode} mode
     */
    async _testAnkiNoteViewerSafe(mode) {
        this._setAnkiNoteViewerStatus(false, null);
        try {
            await this._testAnkiNoteViewer(mode);
        } catch (e) {
            this._setAnkiNoteViewerStatus(true, toError(e));
            return;
        }
        this._setAnkiNoteViewerStatus(true, null);
    }

    /**
     * @param {import('settings').AnkiNoteGuiMode} mode
     */
    async _testAnkiNoteViewer(mode) {
        const queries = [
            '"よむ" deck:current',
            '"よむ"',
            'deck:current',
            ''
        ];

        let noteId = null;
        for (const query of queries) {
            const notes = await this._settingsController.application.api.findAnkiNotes(query);
            if (notes.length > 0) {
                noteId = notes[0];
                break;
            }
        }

        if (noteId === null) {
            throw new Error('Could not find a note to test with');
        }

        await this._settingsController.application.api.viewNotes([noteId], mode, false);
    }

    /**
     * @param {boolean} visible
     * @param {?Error} error
     */
    _setAnkiNoteViewerStatus(visible, error) {
        /** @type {HTMLElement} */
        const node = querySelectorNotNull(document, '#test-anki-note-viewer-results');
        if (visible) {
            const success = (error === null);
            node.textContent = success ? 'Success!' : error.message;
            node.dataset.success = `${success}`;
        } else {
            node.textContent = '';
            delete node.dataset.success;
        }
        node.hidden = !visible;
    }

    /**
     * @param {string} value
     * @returns {?import('settings').AnkiNoteGuiMode}
     */
    _normalizeAnkiNoteGuiMode(value) {
        switch (value) {
            case 'browse':
            case 'edit':
                return value;
            default:
                return null;
        }
    }

    /**
     * @param {import('anki').Note[]} notes
     * @returns {Promise<?((number | null)[] | null)>}
     */
    async addNotes(notes) {
        return await this._ankiConnect.addNotes(notes);
    }

    /**
     * @param {import('anki').Note[]} notes
     * @returns {Promise<boolean[]>}
     */
    async canAddNotes(notes) {
        return await this._ankiConnect.canAddNotes(notes);
    }
}

class AnkiCardController {
    /**
     * @param {import('./settings-controller.js').SettingsController} settingsController
     * @param {AnkiController} ankiController
     * @param {HTMLElement} node
     */
    constructor(settingsController, ankiController, node) {
        /** @type {import('./settings-controller.js').SettingsController} */
        this._settingsController = settingsController;
        /** @type {AnkiController} */
        this._ankiController = ankiController;
        /** @type {HTMLElement} */
        this._node = node;
        const {ankiCardType} = node.dataset;
        /** @type {string} */
        this._optionsType = typeof ankiCardType === 'string' ? ankiCardType : 'terms';
        /** @type {import('dictionary').DictionaryEntryType} */
        this._dictionaryEntryType = ankiCardType === 'kanji' ? 'kanji' : 'term';
        /** @type {string|undefined} */
        this._cardMenu = node.dataset.ankiCardMenu;
        /** @type {EventListenerCollection} */
        this._eventListeners = new EventListenerCollection();
        /** @type {EventListenerCollection} */
        this._fieldEventListeners = new EventListenerCollection();
        /** @type {import('settings').AnkiNoteFields} */
        this._fields = {};
        /** @type {?string} */
        this._modelChangingTo = null;
        /** @type {?Element} */
        this._ankiCardFieldsContainer = null;
        /** @type {boolean} */
        this._cleaned = false;
        /** @type {import('anki-controller').FieldEntry[]} */
        this._fieldEntries = [];
        /** @type {AnkiCardSelectController} */
        this._deckController = new AnkiCardSelectController();
        /** @type {AnkiCardSelectController} */
        this._modelController = new AnkiCardSelectController();
    }

    /** */
    async prepare() {
        const options = await this._settingsController.getOptions();
        const ankiOptions = options.anki;
        if (this._cleaned) { return; }

        const cardOptions = this._getCardOptions(ankiOptions, this._optionsType);
        if (cardOptions === null) { return; }
        const {deck, model, fields} = cardOptions;
        /** @type {HTMLSelectElement} */
        const deckControllerSelect = querySelectorNotNull(this._node, '.anki-card-deck');
        /** @type {HTMLSelectElement} */
        const modelControllerSelect = querySelectorNotNull(this._node, '.anki-card-model');
        this._deckController.prepare(deckControllerSelect, deck);
        this._modelController.prepare(modelControllerSelect, model);
        this._fields = fields;

        this._ankiCardFieldsContainer = this._node.querySelector('.anki-card-fields');

        /** @type {HTMLTextAreaElement} */
        const mainSettingsEntry = querySelectorNotNull(document, '[data-modal-action="show,anki-cards"]');
        mainSettingsEntry.addEventListener('click', (() => {
            const updatedCardOptions = this._getCardOptions(ankiOptions, this._optionsType);
            if (updatedCardOptions === null) { return; }
            this._deckController.prepare(deckControllerSelect, updatedCardOptions.deck);
            this._modelController.prepare(modelControllerSelect, updatedCardOptions.model);
            this._fields = updatedCardOptions.fields;
            void this.updateAnkiState();
        }).bind(this), false);

        this._setupFields();

        this._eventListeners.addEventListener(this._deckController.select, 'change', this._onCardDeckChange.bind(this), false);
        this._eventListeners.addEventListener(this._modelController.select, 'change', this._onCardModelChange.bind(this), false);
        this._eventListeners.on(this._settingsController, 'permissionsChanged', this._onPermissionsChanged.bind(this));

        await this.updateAnkiState();
    }

    /** */
    cleanup() {
        this._cleaned = true;
        this._fieldEntries = [];
        this._eventListeners.removeAllEventListeners();
    }

    /** */
    async updateAnkiState() {
        if (this._fields === null) { return; }
        const {deckNames, modelNames} = await this._ankiController.getAnkiData();
        if (this._cleaned) { return; }
        this._deckController.setOptionValues(deckNames);
        this._modelController.setOptionValues(modelNames);
    }

    /**
     * @returns {boolean}
     */
    isStale() {
        return (this._optionsType !== this._node.dataset.ankiCardType);
    }

    // Private

    /**
     * @param {Event} e
     */
    _onCardDeckChange(e) {
        const node = /** @type {HTMLSelectElement} */ (e.currentTarget);
        void this._setDeck(node.value);
    }

    /**
     * @param {Event} e
     */
    _onCardModelChange(e) {
        const node = /** @type {HTMLSelectElement} */ (e.currentTarget);
        void this._setModel(node.value);
    }

    /**
     * @param {number} index
     * @param {Event} e
     */
    _onFieldChange(index, e) {
        const node = /** @type {HTMLInputElement} */ (e.currentTarget);
        void this._validateFieldPermissions(node, index, true);
        this._validateField(node, index);
    }

    /**
     * @param {number} index
     * @param {Event} e
     */
    _onFieldInput(index, e) {
        const node = /** @type {HTMLInputElement} */ (e.currentTarget);
        this._validateField(node, index);
    }

    /**
     * @param {number} index
     * @param {import('dom-data-binder').SettingChangedEvent} e
     */
    _onFieldSettingChanged(index, e) {
        const node = /** @type {HTMLInputElement} */ (e.currentTarget);
        void this._validateFieldPermissions(node, index, false);
    }

    /**
     * @param {import('popup-menu').MenuOpenEvent} event
     */
    _onFieldMenuOpen(event) {
        const button = /** @type {HTMLElement} */ (event.currentTarget);
        const {menu} = event.detail;
        const {index, fieldName} = button.dataset;
        const indexNumber = typeof index === 'string' ? Number.parseInt(index, 10) : 0;
        if (typeof fieldName !== 'string') { return; }

        const defaultValue = this._getDefaultFieldValue(fieldName, indexNumber, this._dictionaryEntryType, null);
        if (defaultValue === '') { return; }

        const match = /^\{([\w\W]+)\}$/.exec(defaultValue);
        if (match === null) { return; }

        const defaultMarker = match[1];
        const item = menu.bodyNode.querySelector(`.popup-menu-item[data-marker="${defaultMarker}"]`);
        if (item === null) { return; }

        item.classList.add('popup-menu-item-bold');
    }

    /**
     * @param {import('popup-menu').MenuCloseEvent} event
     */
    _onFieldMenuClose(event) {
        const button = /** @type {HTMLElement} */ (event.currentTarget);
        const {action, item} = event.detail;
        switch (action) {
            case 'setFieldMarker':
                if (item !== null) {
                    const {marker} = item.dataset;
                    if (typeof marker === 'string') {
                        this._setFieldMarker(button, marker);
                    }
                }
                break;
        }
    }

    /**
     * @param {HTMLInputElement} node
     * @param {number} index
     */
    _validateField(node, index) {
        let valid = (node.dataset.hasPermissions !== 'false');
        if (valid && index === 0 && !stringContainsAnyFieldMarker(node.value)) {
            valid = false;
        }
        node.dataset.invalid = `${!valid}`;
    }

    /**
     * @param {Element} element
     * @param {string} marker
     */
    _setFieldMarker(element, marker) {
        const container = element.closest('.anki-card-field-value-container');
        if (container === null) { return; }
        /** @type {HTMLInputElement} */
        const input = querySelectorNotNull(container, '.anki-card-field-value');
        input.value = `{${marker}}`;
        input.dispatchEvent(new Event('change'));
    }

    /**
     * @param {import('settings').AnkiOptions} ankiOptions
     * @param {string} optionsType
     * @returns {?import('settings').AnkiNoteOptions}
     */
    _getCardOptions(ankiOptions, optionsType) {
        switch (optionsType) {
            case 'terms': return ankiOptions.terms;
            case 'kanji': return ankiOptions.kanji;
            default: return null;
        }
    }

    /** */
    _setupFields() {
        this._fieldEventListeners.removeAllEventListeners();

        const totalFragment = document.createDocumentFragment();
        this._fieldEntries = [];
        let index = 0;
        for (const [fieldName, fieldValue] of Object.entries(this._fields)) {
            const content = this._settingsController.instantiateTemplateFragment('anki-card-field');

            /** @type {HTMLElement} */
            const fieldNameContainerNode = querySelectorNotNull(content, '.anki-card-field-name-container');
            fieldNameContainerNode.dataset.index = `${index}`;
            /** @type {HTMLElement} */
            const fieldNameNode = querySelectorNotNull(content, '.anki-card-field-name');
            fieldNameNode.textContent = fieldName;

            /** @type {HTMLElement} */
            const valueContainer = querySelectorNotNull(content, '.anki-card-field-value-container');
            valueContainer.dataset.index = `${index}`;

            /** @type {HTMLInputElement} */
            const inputField = querySelectorNotNull(content, '.anki-card-field-value');
            inputField.value = fieldValue;
            inputField.dataset.setting = ObjectPropertyAccessor.getPathString(['anki', this._optionsType, 'fields', fieldName]);
            void this._validateFieldPermissions(inputField, index, false);

            this._fieldEventListeners.addEventListener(inputField, 'change', this._onFieldChange.bind(this, index), false);
            this._fieldEventListeners.addEventListener(inputField, 'input', this._onFieldInput.bind(this, index), false);
            this._fieldEventListeners.addEventListener(inputField, 'settingChanged', this._onFieldSettingChanged.bind(this, index), false);
            this._validateField(inputField, index);

            /** @type {?HTMLElement} */
            const menuButton = content.querySelector('.anki-card-field-value-menu-button');
            if (menuButton !== null) {
                if (typeof this._cardMenu !== 'undefined') {
                    menuButton.dataset.menu = this._cardMenu;
                } else {
                    delete menuButton.dataset.menu;
                }
                menuButton.dataset.index = `${index}`;
                menuButton.dataset.fieldName = fieldName;
                this._fieldEventListeners.addEventListener(menuButton, 'menuOpen', this._onFieldMenuOpen.bind(this), false);
                this._fieldEventListeners.addEventListener(menuButton, 'menuClose', this._onFieldMenuClose.bind(this), false);
            }

            totalFragment.appendChild(content);
            this._fieldEntries.push({fieldName, inputField, fieldNameContainerNode});

            ++index;
        }

        const ELEMENT_NODE = Node.ELEMENT_NODE;
        const container = this._ankiCardFieldsContainer;
        if (container !== null) {
            const childNodesFrozen = [...container.childNodes];
            for (const node of childNodesFrozen) {
                if (node.nodeType === ELEMENT_NODE && node instanceof HTMLElement && node.dataset.persistent === 'true') { continue; }
                container.removeChild(node);
            }
            container.appendChild(totalFragment);
        }

        void this._validateFields();
    }

    /** */
    async _validateFields() {
        const token = {};
        this._validateFieldsToken = token;

        let fieldNames;
        try {
            fieldNames = await this._ankiController.getModelFieldNames(this._modelController.value);
        } catch (e) {
            return;
        }

        if (token !== this._validateFieldsToken) { return; }

        const fieldNamesSet = new Set(fieldNames);
        let index = 0;
        for (const {fieldName, fieldNameContainerNode} of this._fieldEntries) {
            fieldNameContainerNode.dataset.invalid = `${!fieldNamesSet.has(fieldName)}`;
            fieldNameContainerNode.dataset.orderMatches = `${index < fieldNames.length && fieldName === fieldNames[index]}`;
            ++index;
        }
    }

    /**
     * @param {string} value
     */
    async _setDeck(value) {
        if (this._deckController.value === value) { return; }
        this._deckController.value = value;

        await this._settingsController.modifyProfileSettings([{
            action: 'set',
            path: ObjectPropertyAccessor.getPathString(['anki', this._optionsType, 'deck']),
            value
        }]);
    }

    /**
     * @param {string} value
     */
    async _setModel(value) {
        const select = this._modelController.select;
        if (this._modelChangingTo !== null) {
            // Revert
            select.value = this._modelChangingTo;
            return;
        }
        if (this._modelController.value === value) { return; }

        let fieldNames;
        let options;
        try {
            this._modelChangingTo = value;
            fieldNames = await this._ankiController.getModelFieldNames(value);
            options = await this._ankiController.settingsController.getOptions();
        } catch (e) {
            // Revert
            select.value = this._modelController.value;
            return;
        } finally {
            this._modelChangingTo = null;
        }

        const cardOptions = this._getCardOptions(options.anki, this._optionsType);
        const oldFields = cardOptions !== null ? cardOptions.fields : null;

        /** @type {import('settings').AnkiNoteFields} */
        const fields = {};
        for (let i = 0, ii = fieldNames.length; i < ii; ++i) {
            const fieldName = fieldNames[i];
            fields[fieldName] = this._getDefaultFieldValue(fieldName, i, this._dictionaryEntryType, oldFields);
        }

        /** @type {import('settings-modifications').Modification[]} */
        const targets = [
            {
                action: 'set',
                path: ObjectPropertyAccessor.getPathString(['anki', this._optionsType, 'model']),
                value
            },
            {
                action: 'set',
                path: ObjectPropertyAccessor.getPathString(['anki', this._optionsType, 'fields']),
                value: fields
            }
        ];

        this._modelController.value = value;
        this._fields = fields;

        await this._settingsController.modifyProfileSettings(targets);

        this._setupFields();
    }

    /**
     * @param {string[]} permissions
     */
    async _requestPermissions(permissions) {
        try {
            await setPermissionsGranted({permissions}, true);
        } catch (e) {
            log.error(e);
        }
    }

    /**
     * @param {HTMLInputElement} node
     * @param {number} index
     * @param {boolean} request
     */
    async _validateFieldPermissions(node, index, request) {
        const fieldValue = node.value;
        const permissions = this._ankiController.getRequiredPermissions(fieldValue);
        if (permissions.length > 0) {
            node.dataset.requiredPermission = permissions.join(' ');
            const hasPermissions2 = await (
                request ?
                setPermissionsGranted({permissions}, true) :
                hasPermissions({permissions})
            );
            node.dataset.hasPermissions = `${hasPermissions2}`;
        } else {
            delete node.dataset.requiredPermission;
            delete node.dataset.hasPermissions;
        }

        this._validateField(node, index);
    }

    /**
     * @param {import('settings-controller').EventArgument<'permissionsChanged'>} details
     */
    _onPermissionsChanged({permissions: {permissions}}) {
        const permissionsSet = new Set(permissions);
        for (let i = 0, ii = this._fieldEntries.length; i < ii; ++i) {
            const {inputField} = this._fieldEntries[i];
            const {requiredPermission} = inputField.dataset;
            if (typeof requiredPermission !== 'string') { continue; }
            const requiredPermissionArray = (requiredPermission.length === 0 ? [] : requiredPermission.split(' '));

            let hasPermissions2 = true;
            for (const permission of requiredPermissionArray) {
                if (!permissionsSet.has(permission)) {
                    hasPermissions2 = false;
                    break;
                }
            }

            inputField.dataset.hasPermissions = `${hasPermissions2}`;
            this._validateField(inputField, i);
        }
    }

    /**
     * @param {string} fieldName
     * @param {number} index
     * @param {import('dictionary').DictionaryEntryType} dictionaryEntryType
     * @param {?import('settings').AnkiNoteFields} oldFields
     * @returns {string}
     */
    _getDefaultFieldValue(fieldName, index, dictionaryEntryType, oldFields) {
        if (
            typeof oldFields === 'object' &&
            oldFields !== null &&
            Object.prototype.hasOwnProperty.call(oldFields, fieldName)
        ) {
            return oldFields[fieldName];
        }

        if (index === 0) {
            return (dictionaryEntryType === 'kanji' ? '{character}' : '{expression}');
        }

        const markers = getStandardFieldMarkers(dictionaryEntryType);
        const markerAliases = new Map([
            ['expression', ['phrase', 'term', 'word']],
            ['glossary', ['definition', 'meaning']],
            ['audio', ['sound']],
            ['dictionary', ['dict']],
            ['pitch-accents', ['pitch']]
        ]);

        const hyphenPattern = /-/g;
        for (const marker of markers) {
            const names = [marker];
            const aliases = markerAliases.get(marker);
            if (typeof aliases !== 'undefined') {
                names.push(...aliases);
            }

            let pattern = '^(?:';
            for (let i = 0, ii = names.length; i < ii; ++i) {
                const name = names[i];
                if (i > 0) { pattern += '|'; }
                pattern += name.replace(hyphenPattern, '[-_ ]*');
            }
            pattern += ')$';
            const patternRegExp = new RegExp(pattern, 'i');

            if (patternRegExp.test(fieldName)) {
                return `{${marker}}`;
            }
        }

        return '';
    }
}

class AnkiCardSelectController {
    constructor() {
        /** @type {?string} */
        this._value = null;
        /** @type {?HTMLSelectElement} */
        this._select = null;
        /** @type {string[]} */
        this._optionValues = [];
        /** @type {boolean} */
        this._hasExtraOption = false;
        /** @type {boolean} */
        this._selectNeedsUpdate = false;
    }

    /** @type {string} */
    get value() {
        if (this._value === null) { throw new Error('Invalid value'); }
        return this._value;
    }

    set value(value) {
        this._value = value;
        this._updateSelect();
    }

    /** @type {HTMLSelectElement} */
    get select() {
        if (this._select === null) { throw new Error('Invalid value'); }
        return this._select;
    }

    /**
     * @param {HTMLSelectElement} select
     * @param {string} value
     */
    prepare(select, value) {
        this._select = select;
        this._value = value;
        this._updateSelect();
    }

    /**
     * @param {string[]} optionValues
     */
    setOptionValues(optionValues) {
        this._optionValues = optionValues;
        this._selectNeedsUpdate = true;
        this._updateSelect();
    }

    // Private

    /** */
    _updateSelect() {
        const select = this._select;
        const value = this._value;
        if (select === null || value === null) { return; }
        let optionValues = this._optionValues;
        const hasOptionValues = Array.isArray(optionValues) && optionValues.length > 0;

        if (!hasOptionValues) {
            optionValues = [];
        }

        const hasExtraOption = !optionValues.includes(value);
        if (hasExtraOption) {
            optionValues = [...optionValues, value];
        }

        if (this._selectNeedsUpdate || hasExtraOption !== this._hasExtraOption) {
            this._setSelectOptions(select, optionValues);
            select.value = value;
            this._hasExtraOption = hasExtraOption;
            this._selectNeedsUpdate = false;
        }

        if (hasOptionValues) {
            select.dataset.invalid = `${hasExtraOption}`;
        } else {
            delete select.dataset.invalid;
        }
    }

    /**
     * @param {HTMLSelectElement} select
     * @param {string[]} optionValues
     */
    _setSelectOptions(select, optionValues) {
        const fragment = document.createDocumentFragment();
        for (const optionValue of optionValues) {
            const option = document.createElement('option');
            option.value = optionValue;
            option.textContent = optionValue;
            fragment.appendChild(option);
        }
        select.textContent = '';
        select.appendChild(fragment);
    }
}