"use strict";
import { luciToggleListener } from "./toggles.js";
import { makeElementBounce, showAlert } from "./alerts.js";
import Api from "./mqtt/api.js";
import Sensori from "./sensori.js";

const luceTemplate = {
    luogo: "",
    stato: false,
    id: "",
    "sensore-in": "",
    "luce-out": "",
};

class Luci {
    /**
     * the div where everything related to the Luci is stored
     */
    static luciContainer = document.getElementById('luciContainer');
    /**
     * the array where all of the user's luci are stored
     */
    static luciBucket = [];
    /**
     * describes how many lights have already been added
     */
    static index = 0;

    static sensoreMovimento = null;

    constructor(servicesArray) {
        Api.fixedCallbackLuciStato = Luci.switchLuceFromMqttEvent;
        Luci.init(servicesArray);
        Luci.modalListener();
    }

    static init(servicesArray) {
        Luci.fillTable();

        const luciBtn = document.getElementById('mainButtonContainer').children[0];
        luciBtn.addEventListener('click', async() => {
            if(servicesArray[0]){
                try{
                    const antifurto = await import('./antifurto.js');
                    antifurto.default.toggleContainer(false);
                } catch (error) {
                    if(error.message !== "Antifurto.toggleContainer is not a function" && error.message !== "Cannot read properties of null (reading 'classList')")
                        throw error;
                }
            }
            if(servicesArray[1]){
                try{
                    const scenari = await import('./scenari.js');
                    scenari.default.toggleContainer(false);
                } catch (error) {
                    if(error.message !== "Scenari.toggleContainer is not a function" && error.message !== "Cannot read properties of null (reading 'classList')")
                        throw error;
                }
            }
            Luci.toggleContainer(true);
        }, false);

        // listen for the closing modal
        const cancelBtn = document.querySelector('#luci-modal .btn.btn-secondary');
        cancelBtn.addEventListener('click', () => {
            // delay 0.5s to allow the modal to close (otherwise it's not pretty)
            setTimeout(() => {
                Luci.selectReset();
            }, 500);
        }, false);
    }
    /**
     * shows and listens for the modal that allows the user to add a new light
     */
    static modalListener() {

        const selectStanza = document.getElementById('luogoLuce');
        const altroInput = document.getElementById('nuovaStanzaDIV');
        const altroClassList = altroInput.classList;
        const form = document.getElementById('search-form');
        const inputField = altroInput.querySelector('input');

        // show or hide the text input for the new room
        selectStanza.addEventListener('change', event => {
            event.preventDefault();
            const selected = event.target.value;

            if (selected === "altro") {
                altroClassList.remove('invisible');
                // set altroInput to required
                inputField.setAttribute('required', 'required');
            } else {
                altroClassList.add('invisible');
                // remove required from altroInput
                inputField.removeAttribute('required');
            }
        }, false);

        form.addEventListener('submit', event => {
            event.preventDefault();

            const selectedStanza = selectStanza.value;
            const selectIn = document.getElementById('scelta-porta-in');
            const selectedIn = selectIn.value;
            const selectOut = document.getElementById('scelta-porta-out');
            const selectedOut = selectOut.value;
            if (selectedStanza === "unselected" || selectedIn === "unselected" || selectedOut === "unselected") {
                alert("Selezionare i valori validi dai menu a tendina.");
                return;
            }

            let stanza = selectStanza.options[selectStanza.selectedIndex].text.trim(); // get selected option text
            if (selectedStanza === "altro") {
                const nuovaStanza = altroInput.querySelector('#nuovaStanza').value.trim();
                if (nuovaStanza == null || nuovaStanza == "" || nuovaStanza.toLowerCase() === "altro") {
                    alert("Inserire un nome valido per la stanza.");
                    return;
                }
                for (const stanza of Luci.luciBucket)
                    if (stanza.luogo === nuovaStanza) {
                        alert("Stanza giĆ  esistente");
                        return;
                    }
                stanza = nuovaStanza;
                // reset the select
                Luci.selectReset();
            }
            stanza = stanza[0].toUpperCase() + stanza.substring(1, stanza.length);
            // add stanza to table
            const luceJSON = Luci.createLight(stanza, false, selectedIn, selectedOut);
            Luci.sendLightToServer(luceJSON);

            // close modal
            const closeBtn = document.querySelector('#luci-modal .btn.btn-secondary');
            closeBtn.click();
        }, false);
    }
    /**
     * sends the light to the server and adds it to the table if the server responds with success
     * @param {luceTemplate} luce the light to be added to the table and the server
     */
    static sendLightToServer(luce) {
        // HERE server send luce
        try {
            /*const apiRes = await Api.makeNewLuci(luce);
            if (apiRes == null) {
                showAlert(`${luce.luogo} creato`, "La nuova luce e' stata creata, usala subito!", true);
                Luci.showNewLight(luce);
            }*/
            Api.makeNewLuci(luce);
        } catch (error) {
            if (error.error)
                showAlert("Errore di comunicazione.", error.error, false);
        }
    }

    static showCreatedLight(luce) {
        Luci.showNewLight(luce);
        showAlert(`${luce.luogo} creato`, "La nuova luce e' stata creata, usala subito!", true);
    }
    /**
     * creates a new light and adds it to the bucket
     * @param {String} luogo the name of the room
     * @param {Boolean} stato the status of the light
     * @returns {luceTemplate} the new light
     */
    static createLight(luogo, stato, valIn, valOut) {
        if(luogo == null)
            throw new Error("luogo non impostato");
        if (valIn == null || valOut == null || valIn.trim() == "" || valOut.trim() == "")
            throw new Error("i sensori in/out devono essere impostati");
        const luce = Object.assign({}, luceTemplate);
        luce.luogo = luogo[0].toUpperCase() + luogo.substring(1, luogo.length).toLowerCase();
        luce.stato = stato != null ? stato : false;
        luce.id = `luce-${Luci.index}`;
        luce["sensore-in"] = valIn.trim();
        luce["luce-out"] = valOut.trim();
        Luci.luciBucket.push(luce);
        Luci.index++;
        return luce;
    }
    /**
     * shows a new light in the table after creating it
     * @param {luceTemplate} luceJSON the light to show
     */
    static showNewLight(luceJSON) {
        const selectIn = document.getElementById('scelta-porta-in');
        const sensoreIn = selectIn.value !== "unselected" ? selectIn.value : luceJSON["sensore-in"];
        const selectOut = document.getElementById('scelta-porta-out');
        const sensoreOut = selectOut.value !== "unselected" ? selectOut.value : luceJSON["luce-out"];
        let sensoreInJSON = null;
        let sensoreOutJSON = null;
        try {
            if (selectIn.children.length > 1){
                sensoreInJSON = Sensori.claimSensore(sensoreIn);
                // if(sensoreInJSON != null)
                //     Luci.removeSensoreOption(selectIn, sensoreInJSON);
            }
            if (sensoreInJSON != null && selectOut.children.length > 1){
                sensoreOutJSON = Sensori.claimSensore(sensoreOut);
                // if(sensoreOutJSON != null)
                //     Luci.removeSensoreOption(selectOut, sensoreOutJSON);
            }
            if(sensoreInJSON == null || sensoreOutJSON == null)
                return;
        } catch (error) {
            if (error.message === "Errore impossibile")
                showAlert("Errore", "Impossibile rimuovere il sensore dai menu a tendina.", false);
            else showAlert("Errore", error.message, false);
            // se ho un errore, reimposto tutto
            //Luci.resetBothSelectForSensors();
            Sensori.liberaSensore(sensoreInJSON);
            Sensori.liberaSensore(sensoreOutJSON);
            return;
        }
        let utilizzo = '';
        if(sensoreInJSON?.utilizzo == null)
            utilizzo += sensoreInJSON.utilizzo = `Interruttore [${sensoreInJSON.nome}]`;
        utilizzo = utilizzo !== '' ? utilizzo + '<br />' : utilizzo;
        if(sensoreOutJSON?.utilizzo == null)
            utilizzo += sensoreOutJSON.utilizzo = `Luce [${sensoreOutJSON.nome}]`;
        utilizzo += `<br />per ${luceJSON.luogo}`;
        const luce = luceJSON.id == null ? (  Luci.createLight(luceJSON.luogo, luceJSON.stato, luceJSON["sensore-in"], luceJSON["luce-out"])  ) : luceJSON;
        const table = document.getElementById('table-row-luci');
        const row = document.createElement('tr');

        row.innerHTML = `
        <th scope="row" id="${luce.id}">${luce.luogo}</th>
        <td>
            <div class="switch-container no-box-sizing">
                <div class="toggle-button no-box-sizing ${luce.stato ? 'active' : ''}" data-bs-toggle="tooltip" data-bs-placement="bottom" data-bs-html="true" title="${utilizzo}" data-sensore-in="${JSON.stringify(sensoreInJSON)}" data-sensore-out="${JSON.stringify(sensoreOutJSON)}">
                    <div class="inner-circle no-box-sizing" />
                </div>
            </div>
        </td>`;
        const lastRow = table.lastElementChild;
        table.insertBefore(row, lastRow);
        Sensori.initializaTooltips();
        // add listener for switch
        const toggle = row.querySelector('.toggle-button');
        luciToggleListener(luce,toggle);
        // remove options from the select of the place
        const select = document.getElementById('luogoLuce');
        const options = select.options;
        const luogoLowercase = luce.luogo.toLowerCase();
        for (const option of options) {
            if (option.text.toLowerCase() === luogoLowercase) {
                select.removeChild(option);
                break;
            }
        }
    }
    /**
     * rimuove il sensore dalle opzioni per l'utente (gia' fatto dalla classe Sensori)
     * @param {*} selectNode
     * @param {*} sensore
     * @returns
     */
     /*static removeSensoreOption(selectNode,sensore){
        if(sensore == null){
            throw new Error("Nome sensore non impostato");
        }
        const options = selectNode.querySelectorAll("option");
        let removed = false;
        for(const option of options){
            if(option.value === sensore.nome){
                selectNode.removeChild(option);
                removed = true;
                break;
            }
        }
        if(!removed){
            // Sensori.liberaSensore(nome);
            console.log('sensore non rimosso')
            throw new Error("Errore impossibile");
        }
    }*/

    static switchLuceFromMqttEvent(mqttObject){
        // mqttObject == {output:outVal,stato};
        // TODO from the OUT and the sensors active, find the light and change the status
        mqttObject.output = mqttObject.output.toUpperCase();
        const luce = Luci.luciBucket.find(luce => luce["luce-out"].toUpperCase() === mqttObject.output);
        if(luce == null){
            console.log("luce non trovata");
            return;
        }
        luce.stato = mqttObject.stato;
        const toggle = document.getElementById(luce.id).nextElementSibling.querySelector('.toggle-button');
        luce.stato ? toggle.classList.add('active') : toggle.classList.remove('active');
    }
    /**
     * resets the select menu used to add a new light
     */
    static selectReset() {
        const select = document.getElementById('luogoLuce');
        const altroInput = document.getElementById('nuovaStanzaDIV');
        const inputField = altroInput.querySelector('input');

        select.selectedIndex = 0;
        altroInput.classList.add('invisible');
        inputField.removeAttribute('required');
        inputField.value = "";
    }
    /**
     * fills the table with the user's luci from the server
     */
    static fillTable() {
        // HERE server get all lights
        try {
            if(!Api.isConnected){
                setTimeout(() => Luci.fillTable(), 2500);
                return;
            }
            /*const luci = await Api.getLuci(Luci.luciFromMqtt);
            // for each, add to table and to luciBucket
            if (await luci != null) {
                Luci.resetBothSelectForSensors();*/
                //for (const luce of luci)
                 //   Luci.showNewLight(luce);
            //}
            // Api.getLuci(Luci.luciFromMqtt);
	        Sensori.showAvailableOptions();
            Api.getAllLuci();
        } catch (error) {
            if (error.error)
                showAlert("Errore di comunicazione", error.error, false);
            else if (error.message)
                showAlert("Errore", error.message, false);
        }
    }

    static luciFromMqtt(datiVari){
        /*
        {
        "lamp0":
            {
            "nome": "Garage",
            "input": "IN6",
            "output": "OUT4",
            "stato":false
            },
        "lamp1":
            {
            "nome": "Garage",
            "input": "IN6",
            "output": "OUT4",
            "stato":false
            },
        "sensM":"IN7"
        }
        */

        let index = 0;
        for(const luce in datiVari){
            if(luce === "sensM"){
                const sensore = Sensori.claimSensore(datiVari["sensM"]);
                sensore.utilizzo = `Sensore di presenza per le luci [${sensore.nome}]`;
                // Sensori.showSensore(sensore);//FIXME non e bello in questo modo, ma sballa dirottamente
                Luci.sensoreMovimento = sensore;
                continue;
            }
            // qui mi occupo delle luci
            const luceObj = datiVari[`lamp${index}`];
            // const luceInput = Sensori.claimSensore(luceObj.input);
            // const luceOutput = Sensori.claimSensore(luceObj.output);
            // questi 2 sono gia' fatti da showNewLight
            Luci.showNewLight({
                luogo:luceObj.nome,
                "sensore-in":luceObj.input,
                "luce-out":luceObj.output,
                stato:luceObj.stato
            });
            index++;
        }
    }

    static updateStatusAfterMqtt(luci){
        for (const luce of luci) {
            const table = document.getElementById('table-row-luci');
            const luceJSON = Luci.luciBucket.find(old_luce => old_luce.out === luce.out);
            const th = table.querySelector(`#${luceJSON.id}`);
            const td = th.nextElementSibling;
            if(luce.stato !== luceJSON.stato){
                // click the toggle
                const toggle = td.querySelector('.toggle-button');
                toggle.click();
            }
        }
    }
    /**
     * toggle the visibility of the luci container
     * @param {Boolean} visible
     */
    static toggleContainer(visible) {
        if (visible) {
            Luci.luciContainer.classList.remove('invisible');
            if (Luci.luciBucket.length === 0) {
                // make the + button bounce
                const plusBtn = document.querySelector('.fa-circle-plus');
                makeElementBounce(plusBtn);
            }
        } else {
            Luci.luciContainer.classList.add('invisible');
        }
    }

    /*static showCreatedSensore(sensore) {
        if(Luci.sensoreMovimento != null)
            Sensori.liberaSensore(Luci.sensoreMovimento.nome);
        const claimedSensore = Sensori.claimSensore(sensore.sensM);
        claimedSensore.utilizzo = `Sensore di presenza per le luci [${claimedSensore.nome}]`;
        Sensori.showSensore(claimedSensore);
        Luci.sensoreMovimento = claimedSensore;
    }*/
}

export default Luci;