import { doExtrasFetch, getApiUrl, modules } from '../../extensions.js';
import { saveTtsProviderSettings } from './index.js';
export { XTTSTtsProvider };
class XTTSTtsProvider {
    //########//
    // Config //
    //########//
    settings;
    ready = false;
    voices = [];
    separator = '. ';
    /**
     * Perform any text processing before passing to TTS engine.
     * @param {string} text Input text
     * @returns {string} Processed text
     */
    processText(text) {
        // Replace fancy ellipsis with "..."
        text = text.replace(/…/g, '...');
        // Remove quotes
        text = text.replace(/["“”‘’]/g, '');
        // Replace multiple "." with single "."
        text = text.replace(/\.+/g, '.');
        return text;
    }
    languageLabels = {
        'Arabic': 'ar',
        'Brazilian Portuguese': 'pt',
        'Chinese': 'zh-cn',
        'Czech': 'cs',
        'Dutch': 'nl',
        'English': 'en',
        'French': 'fr',
        'German': 'de',
        'Italian': 'it',
        'Polish': 'pl',
        'Russian': 'ru',
        'Spanish': 'es',
        'Turkish': 'tr',
        'Japanese': 'ja',
        'Korean': 'ko',
        'Hungarian': 'hu',
        'Hindi': 'hi',
    };
    defaultSettings = {
        provider_endpoint: 'http://localhost:8020',
        language: 'en',
        temperature: 0.75,
        length_penalty: 1.0,
        repetition_penalty: 5.0,
        top_k: 50,
        top_p: 0.85,
        speed: 1,
        enable_text_splitting: true,
        stream_chunk_size: 100,
        voiceMap: {},
        streaming: false,
    };
    get settingsHtml() {
        let html = `
        
        
        
        
        
        Use XTTSv2 TTS Server.
        
        
        
        
        
        
        
        
        
        
        
        
        
        
        
        
        `;
        return html;
    }
    onSettingsChange() {
        // Used when provider settings are updated from UI
        this.settings.provider_endpoint = $('#xtts_tts_endpoint').val();
        this.settings.language = $('#xtts_api_language').val();
        // Update the default TTS settings based on input fields
        this.settings.speed = $('#xtts_speed').val();
        this.settings.temperature = $('#xtts_temperature').val();
        this.settings.length_penalty = $('#xtts_length_penalty').val();
        this.settings.repetition_penalty = $('#xtts_repetition_penalty').val();
        this.settings.top_k = $('#xtts_top_k').val();
        this.settings.top_p = $('#xtts_top_p').val();
        this.settings.stream_chunk_size = $('#xtts_stream_chunk_size').val();
        this.settings.enable_text_splitting = $('#xtts_enable_text_splitting').is(':checked');
        this.settings.streaming = $('#xtts_tts_streaming').is(':checked');
        // Update the UI to reflect changes
        $('#xtts_tts_speed_output').text(this.settings.speed);
        $('#xtts_tts_temperature_output').text(this.settings.temperature);
        $('#xtts_length_penalty_output').text(this.settings.length_penalty);
        $('#xtts_repetition_penalty_output').text(this.settings.repetition_penalty);
        $('#xtts_top_k_output').text(this.settings.top_k);
        $('#xtts_top_p_output').text(this.settings.top_p);
        $('#xtts_stream_chunk_size_output').text(this.settings.stream_chunk_size);
        saveTtsProviderSettings();
        this.changeTTSSettings();
    }
    async loadSettings(settings) {
        // Pupulate Provider UI given input settings
        if (Object.keys(settings).length == 0) {
            console.info('Using default TTS Provider settings');
        }
        // Only accept keys defined in defaultSettings
        this.settings = this.defaultSettings;
        for (const key in settings) {
            if (key in this.settings) {
                this.settings[key] = settings[key];
            } else {
                throw `Invalid setting passed to TTS Provider: ${key}`;
            }
        }
        const apiCheckInterval = setInterval(() => {
            // Use Extras API if TTS support is enabled
            if (modules.includes('tts') || modules.includes('xtts-tts')) {
                const baseUrl = new URL(getApiUrl());
                baseUrl.pathname = '/api/tts';
                this.settings.provider_endpoint = baseUrl.toString();
                $('#xtts_tts_endpoint').val(this.settings.provider_endpoint);
                clearInterval(apiCheckInterval);
            }
        }, 2000);
        // Set initial values from the settings
        $('#xtts_tts_endpoint').val(this.settings.provider_endpoint);
        $('#xtts_api_language').val(this.settings.language);
        $('#xtts_speed').val(this.settings.speed);
        $('#xtts_temperature').val(this.settings.temperature);
        $('#xtts_length_penalty').val(this.settings.length_penalty);
        $('#xtts_repetition_penalty').val(this.settings.repetition_penalty);
        $('#xtts_top_k').val(this.settings.top_k);
        $('#xtts_top_p').val(this.settings.top_p);
        $('#xtts_enable_text_splitting').prop('checked', this.settings.enable_text_splitting);
        $('#xtts_stream_chunk_size').val(this.settings.stream_chunk_size);
        $('#xtts_tts_streaming').prop('checked', this.settings.streaming);
        // Update the UI to reflect changes
        $('#xtts_tts_speed_output').text(this.settings.speed);
        $('#xtts_tts_temperature_output').text(this.settings.temperature);
        $('#xtts_length_penalty_output').text(this.settings.length_penalty);
        $('#xtts_repetition_penalty_output').text(this.settings.repetition_penalty);
        $('#xtts_top_k_output').text(this.settings.top_k);
        $('#xtts_top_p_output').text(this.settings.top_p);
        $('#xtts_stream_chunk_size_output').text(this.settings.stream_chunk_size);
        // Register input/change event listeners to update settings on user interaction
        $('#xtts_tts_endpoint').on('input', () => { this.onSettingsChange(); });
        $('#xtts_api_language').on('change', () => { this.onSettingsChange(); });
        $('#xtts_speed').on('input', () => { this.onSettingsChange(); });
        $('#xtts_temperature').on('input', () => { this.onSettingsChange(); });
        $('#xtts_length_penalty').on('input', () => { this.onSettingsChange(); });
        $('#xtts_repetition_penalty').on('input', () => { this.onSettingsChange(); });
        $('#xtts_top_k').on('input', () => { this.onSettingsChange(); });
        $('#xtts_top_p').on('input', () => { this.onSettingsChange(); });
        $('#xtts_enable_text_splitting').on('change', () => { this.onSettingsChange(); });
        $('#xtts_stream_chunk_size').on('input', () => { this.onSettingsChange(); });
        $('#xtts_tts_streaming').on('change', () => { this.onSettingsChange(); });
        await this.checkReady();
        console.debug('XTTS: Settings loaded');
    }
    // Perform a simple readiness check by trying to fetch voiceIds
    async checkReady() {
        await Promise.allSettled([this.fetchTtsVoiceObjects(), this.changeTTSSettings()]);
    }
    async onRefreshClick() {
        return;
    }
    //#################//
    //  TTS Interfaces //
    //#################//
    async getVoice(voiceName) {
        if (this.voices.length == 0) {
            this.voices = await this.fetchTtsVoiceObjects();
        }
        const match = this.voices.filter(
            XTTSVoice => XTTSVoice.name == voiceName,
        )[0];
        if (!match) {
            throw `TTS Voice name ${voiceName} not found`;
        }
        return match;
    }
    async generateTts(text, voiceId) {
        const response = await this.fetchTtsGeneration(text, voiceId);
        return response;
    }
    //###########//
    // API CALLS //
    //###########//
    async fetchTtsVoiceObjects() {
        const response = await doExtrasFetch(`${this.settings.provider_endpoint}/speakers`);
        if (!response.ok) {
            throw new Error(`HTTP ${response.status}: ${await response.json()}`);
        }
        const responseJson = await response.json();
        return responseJson;
    }
    // Each time a parameter is changed, we change the configuration
    async changeTTSSettings() {
        if (!this.settings.provider_endpoint) {
            return;
        }
        const response = await doExtrasFetch(
            `${this.settings.provider_endpoint}/set_tts_settings`,
            {
                method: 'POST',
                headers: {
                    'Content-Type': 'application/json',
                    'Cache-Control': 'no-cache',
                },
                body: JSON.stringify({
                    'temperature': this.settings.temperature,
                    'speed': this.settings.speed,
                    'length_penalty': this.settings.length_penalty,
                    'repetition_penalty': this.settings.repetition_penalty,
                    'top_p': this.settings.top_p,
                    'top_k': this.settings.top_k,
                    'enable_text_splitting': this.settings.enable_text_splitting,
                    'stream_chunk_size': this.settings.stream_chunk_size,
                }),
            },
        );
        return response;
    }
    async fetchTtsGeneration(inputText, voiceId) {
        console.info(`Generating new TTS for voice_id ${voiceId}`);
        if (this.settings.streaming) {
            const params = new URLSearchParams();
            params.append('text', inputText);
            params.append('speaker_wav', voiceId);
            params.append('language', this.settings.language);
            return `${this.settings.provider_endpoint}/tts_stream/?${params.toString()}`;
        }
        const response = await doExtrasFetch(
            `${this.settings.provider_endpoint}/tts_to_audio/`,
            {
                method: 'POST',
                headers: {
                    'Content-Type': 'application/json',
                    'Cache-Control': 'no-cache',  // Added this line to disable caching of file so new files are always played - Rolyat 7/7/23
                },
                body: JSON.stringify({
                    'text': inputText,
                    'speaker_wav': voiceId,
                    'language': this.settings.language,
                }),
            },
        );
        if (!response.ok) {
            toastr.error(response.statusText, 'TTS Generation Failed');
            throw new Error(`HTTP ${response.status}: ${await response.text()}`);
        }
        return response;
    }
    // Interface not used by XTTS TTS
    async fetchTtsFromHistory(history_item_id) {
        return Promise.resolve(history_item_id);
    }
}