ZAVI - UniAxa

Zero Attention Video Interface: an invisible interface to skip, close and anvance trough courses. Less distractions, more time for you.

// ==UserScript==
// @name            ZAVI - UniAxa
// @namespace       https://tampermonkey.net/
// @license         GPL-3.0-or-later
// @version         1.8
// @description:it  Zero Attention Video Interface: un'interfaccia invisibile per saltare, chiudere e avanzare nei corsi. Meno interazione, più tempo per te.
// @description     Zero Attention Video Interface: an invisible interface to skip, close and anvance trough courses. Less distractions, more time for you.
// @icon            
// @author          GP
// @match           https://uniaxa.it/*
// @grant           none
// ==/UserScript==

/**
 * ---------------------------
 * Time: 09/07/2025 14:30.
 * Author: GP
 * ---------------------------
 */

/**
 * -==THIS SCRIPT REQUIRES TIMEHOOKER TO WORK PROPERLY==-
 * Otherwise it will act just like a bot without the speeding capability.
 * I left a lot of comments to make it easier for you to understand how it works.
 * NOTICE: You may need to tweek the script a bit and translate some parts if you are using this program outside italy. Ask to ChatGPT for advice if unsure.
 *
 * -==QUESTO SCRIPT NECESSITA DI TIMEHOOKER PER FUNZIONARE CORRETTAMENTE==-
 * Altrimenti agirà solamente come un semplice bot senza la funzionalità di velocizzare la pagina.
 * Ho lasciato un sacco di commenti per rendere più facile la comprensione.
 * N.B.: E possibile che, nel caso tu stia utilizzando questo script fuori dall'italia, dovrai apportare alcune modifiche e tradurre alcune sezioni. In caso di incertezza, chiedi a una mano a ChatGPT.
 *
 * Download Timehooker (ENG): https://greasyfork.org/en/scripts/438894-timerhooker-english-version
 */

(function () {
    'use strict';

    // TUTTI I COMMENTI (le stringhe che iniziano con '//') POSSONO ESSERE RIMOSSI O MODIFICATI A PIACIMENTO SENZA CHE IL CODICE NE RISENTA IN ALCUN MODO.

    // Intervallo di 1 secondo per evitare che lo script sovraccarichi il browser.
    const CHECK_INTERVAL_MS = 1000;

    // Queste sono semplicissime variabili. Tradurre il loro nome è irrilevante.
    let lastVideoState = null;
    let lastAvantiState = null;
    let lastPlayButtonFound = true;
    let videoVelocizzato = false;
    let inGestionePopup = false;
    let popupQueue = [];

    // -== FUNZIONE PER VELOCIZZARE LA PAGINA ==-
    // Rinnovo il mio invito a installare Timehooker, per non rendere questa parte dek codice inutile.
    function velocizza() {
        if (!videoVelocizzato && typeof timer !== 'undefined') {
            timer.change(1 / 99999999999); // Nel caso volessi modificare la velocità, non rimmuovere il '1 / '. Il codice non si rompe ma, al posto di velocizzare rallenta.
            videoVelocizzato = true;
            console.log("⏩ Velocizzo il video."); // 'console.log(...)' serve solamente ad inviare dei messaggi nella console (accessibile utilizzando gli 'Strumenti per Sviluppatori' su Chrome).
        }
    }

    // -==FUNZIONE PER RIPRISTINARE LA PAGINA==-
    // Terminato il video, in automatico il programma ripristina la velocità a 'x1', per evitare che la sessione rischi espirare.
    function ripristinaVelocità() {
        if (typeof timer !== 'undefined') {
            timer.change(1);
            videoVelocizzato = false;
            console.log("🔁 Ripristino la velocità normale.");
        }
    }

    // -==FUZIONE PER PASSARE ALLA PROSSIMA LEZIONE==-
    // Controlla se il pulsante 'Avanti' sia attivo e, nel caso, lo clicca.
    function clickPulsanteAvanti() {
        const avantiButton = document.querySelector('#puls_avanti'); // Potrebbe essere necessario sistituire '#puls_avanti'.

        if (!avantiButton) {
            if (lastAvantiState !== 'non-trovato') {
                console.log("⚪ Pulsante 'Avanti' non trovato.");
                lastAvantiState = 'non-trovato';
            }
            return false;
        }

        if (!avantiButton.classList.contains('disabled') && !avantiButton.disabled) {
            if (lastAvantiState !== 'attivo') {
                console.log("➡️ Pulsante 'Avanti' attivo. Lo clicco.");
                lastAvantiState = 'attivo';
            }
            ripristinaVelocità();
            avantiButton.click();
            return true;
        } else {
            if (lastAvantiState !== 'non-attivo') {
                console.log("⏳ Pulsante 'Avanti' non ancora attivo.");
                lastAvantiState = 'non-attivo';
            }
            return false;
        }
    }

    // -==FUNZIONE PER CHIUDERE IL POPUP AL TERMINE DELLA LEZIONE ==-
    // Alla fine di ogni lezione, generalmente compare un popup che chiede all'utente di tornare alla schermata 'Home' per proseguire con il corso.
    // Questa funzione ha l'obbietivo di controllare se quest'ultimo esista e, nel caso sia visibile a schermo, chiuderlo e tornare automaticamente alla schermata 'Home'.
    function chiudiPopupFineSezioneEHome() {
      const chiudiFine = [...document.querySelectorAll('button.btn.pulsCont[data-dismiss="modal"]')] // Questa stringa è difficile da comprendere, nel caso sia necessario adattare qualcosa, basta sostituire 'modal'.
          .find(btn =>
              btn.innerText.trim().toUpperCase() === "CHIUDI" && // Potrebbe essere necessario sostituire 'CHIUDI'.
              (btn.offsetParent !== null || btn.getBoundingClientRect().height > 0)
          );

      if (chiudiFine) {
          console.log("🎯 Trovato popup di fine sezione visibile. Lo chiudo.");
          chiudiFine.click();

          setTimeout(() => {
              const homeBtn = document.querySelector('#puls_menu');
              if (homeBtn && (homeBtn.offsetParent !== null || homeBtn.getBoundingClientRect().height > 0)) {
                  console.log("🏠 Clicco il pulsante 'Home'.");
                  homeBtn.click();
              } else {
                  console.log("❓ Pulsante 'Home' non visibile.");
              }
          }, 100);
          return true;
      }

      return false;
  }


    // -==FUNZIONE PER GESTIRE I POPUP DI FINE LEZIONE==-
    // Al termine di alcune lezioni, è necessario visualizzare alcune risorse che vengono illustrate sotto forma di popup. Questo codice automatizza il processo di apertura e di chiusura.
    function gestisciPopup(callback) {
        if (popupQueue.length === 0) {
            popupQueue = [...document.querySelectorAll('#mioContAree button.areaVideo')] // Potrebbe essere necessario sostituire '#mioContAree" e "button.areaVideo'.
                .filter(btn => !btn.classList.contains('cliccato'));
        }

        if (popupQueue.length === 0) {
            inGestionePopup = false;
            if (callback) callback();
            return;
        }

        inGestionePopup = true;

        const button = popupQueue.shift();
        button.classList.add('cliccato');
        console.log(`📌 Clicco su bottone popup con ID: ${button.id}`);
        button.click();

        // -==SOTTOFUNZIONE PER LA GESTIONE DEI POPUP==-
        function aspettaChiudiEContinua(tentativi = 0) {
            const chiudiBtn = document.querySelector('button[data-dismiss="modal"]'); // Potrebbe essere necessario sostituire 'modal'.

            if (chiudiBtn) {
                console.log("✅ Trovato pulsante 'Chiudi'. Lo clicco.");
                chiudiBtn.click();

                setTimeout(() => {
                    gestisciPopup(callback);
                }, 100);
            } else if (tentativi < 10) {
                setTimeout(() => aspettaChiudiEContinua(tentativi + 1), 300);
            } else {
                console.log("❌ Nessun pulsante 'Chiudi' trovato. Continuo con il prossimo.");
                gestisciPopup(callback);
            }
        }

        aspettaChiudiEContinua();
    }

    // -==FUNZIONE PER GESTIRE IL VIDEO==-
    // Questa è la funzione più importante di tutto il programma. Una funzione per gestirle tutte.
    // Racchiude tutta la logica che gestisce se e quando attivare le varie funzioni (oltre, chiaramente, a gestire i video)
    function gestisciVideo() {
        // 1. Priorità: chiudi popup finale e torna alla home
        if (chiudiPopupFineSezioneEHome()) return;

        // 2. Se stiamo gestendo popup intermedi, pausa qui
        if (inGestionePopup) return;

        // 3. Prova a cliccare "Avanti"
        const avantiCliccato = clickPulsanteAvanti();
        if (avantiCliccato) return;

        // 4. Gestione video
        const playButton = document.querySelector('button.vjs-play-control');

        if (!playButton) {
            if (lastPlayButtonFound === true) {
                console.log("🛑 Pulsante Play scomparso. Ripristino velocità.");
                ripristinaVelocità();
                lastPlayButtonFound = false;
            }
            return;
        } else {
            lastPlayButtonFound = true;
        }

        const classList = playButton.classList;

        // Questo codice ha lo scopo di controllare lo stato del video ed agire di conseguenza. Per farlo, controlla il pulsante play-pausa situato sull'angolo in basso a sinistra.
        if (classList.contains('vjs-ended')) { // Potrebbe essere necessario sostituire 'vjs-ended'.
            if (lastVideoState !== 'ended') {
                console.log("🔴 Video terminato. Controllo popup...");
                ripristinaVelocità();
                lastVideoState = 'ended';

                // Attiva popup handling
                gestisciPopup(() => {
                    console.log("📦 Gestione popup completata.");
                });
            }
            return;
        }

        if (classList.contains('vjs-paused')) { // Potrebbe essere necessario sostituire 'vjs-paused'.
            if (lastVideoState !== 'paused') {
                console.log("🟡 Video in pausa. Provo a cliccare...");
                lastVideoState = 'paused';
            }
            playButton.click();
            velocizza();
            return;
        }

        if (classList.contains('vjs-playing')) { // Potrebbe essere necessario sostituire 'vjs-playing'.
            if (lastVideoState !== 'playing') {
                console.log("🟢 Video in riproduzione.");
                lastVideoState = 'playing';
            }
            velocizza();
            return;
        }

        if (lastVideoState !== 'unknown') {
            console.log("⚠️ Stato del video sconosciuto.");
            lastVideoState = 'unknown';
        }
    }

    setInterval(gestisciVideo, CHECK_INTERVAL_MS); //Questa stringa serve semplicemente ad attivare tutto il programma.
})();