diff --git a/src/assets/macchinisti/francesco-baldino.jpg b/src/assets/macchinisti/francesco-baldino.jpg new file mode 100644 index 0000000..438dfbe Binary files /dev/null and b/src/assets/macchinisti/francesco-baldino.jpg differ diff --git a/src/client/MediaPesataApp.tsx b/src/client/MediaPesataApp.tsx new file mode 100644 index 0000000..0e030ec --- /dev/null +++ b/src/client/MediaPesataApp.tsx @@ -0,0 +1,766 @@ +import { useState, useEffect } from 'preact/hooks' +import { render } from 'preact' + +// Tipi per la gestione dei dati +type TipoStudente = 'triennale' | 'magistrale' + +interface Corso { + nome: string + anno: '1' | '2' | '3' | 'M' | 'istituzioni' + cfu: number + passFailOnly?: boolean // Per materie senza voto (pass/fail) +} + +interface CorsoSelezionato { + id: string + nome: string + cfu: number + voto: number | null + lode: boolean + passFailOnly?: boolean // Per materie senza voto (pass/fail) +} + +interface CorsoCustom { + nome: string + cfu: number +} + +// Dati dei corsi aggiornati dalla tabella ufficiale +const CORSI_DISPONIBILI: Corso[] = [ + // Primo Anno + { nome: 'Analisi matematica 1', anno: '1', cfu: 15 }, + { nome: 'Aritmetica', anno: '1', cfu: 9 }, + { nome: 'Fisica I con laboratorio', anno: '1', cfu: 9 }, + { nome: 'Fondamenti di programmazione con laboratorio', anno: '1', cfu: 9 }, + { nome: 'Geometria 1', anno: '1', cfu: 15 }, + { nome: 'Laboratorio di introduzione alla matematica computazionale', anno: '1', cfu: 6, passFailOnly: true }, + { nome: 'Laboratorio di comunicazione mediante calcolatore', anno: '1', cfu: 3, passFailOnly: true }, + + // Secondo Anno + { nome: 'Algebra 1', anno: '2', cfu: 6 }, + { nome: 'Algoritmi e strutture dati', anno: '2', cfu: 6 }, + { nome: 'Analisi matematica 2', anno: '2', cfu: 12 }, + { nome: 'Analisi numerica con laboratorio', anno: '2', cfu: 9 }, + { nome: 'Elementi di probabilità e statistica', anno: '2', cfu: 6 }, + { nome: 'Geometria 2', anno: '2', cfu: 12 }, + { nome: 'Inglese scientifico', anno: '2', cfu: 6, passFailOnly: true }, + { nome: 'Laboratorio didattico di matematica computazionale', anno: '2', cfu: 3, passFailOnly: true }, + + // Terzo Anno + { nome: 'Algebra 2', anno: '3', cfu: 6 }, + { nome: 'Analisi matematica 3', anno: '3', cfu: 6 }, + { nome: 'Analisi reale', anno: '3', cfu: 6 }, + { nome: 'Calcolo scientifico', anno: '3', cfu: 6 }, + { nome: 'Elementi di analisi complessa', anno: '3', cfu: 6 }, + { nome: 'Elementi di calcolo delle variazioni', anno: '3', cfu: 6 }, + { nome: 'Elementi di geometria algebrica', anno: '3', cfu: 6 }, + { nome: 'Elementi di meccanica celeste', anno: '3', cfu: 6 }, + { nome: 'Elementi di teoria degli insiemi', anno: '3', cfu: 6 }, + { nome: 'Elementi di teoria delle rappresentazioni', anno: '3', cfu: 6 }, + { nome: 'Elementi di topologia algebrica', anno: '3', cfu: 6 }, + { nome: 'Equazioni alle derivate parziali', anno: '3', cfu: 6 }, + { nome: 'Fisica II', anno: '3', cfu: 9 }, + { nome: 'Fisica III', anno: '3', cfu: 6 }, + { nome: 'Geometria e topologia differenziale', anno: '3', cfu: 6 }, + { nome: 'Gruppi e rappresentazioni', anno: '3', cfu: 6 }, + { nome: 'Laboratorio computazionale', anno: '3', cfu: 6, passFailOnly: true }, + { nome: 'Laboratorio sperimentale di matematica computazionale', anno: '3', cfu: 6, passFailOnly: true }, + { nome: 'Linguaggi di programmazione con laboratorio', anno: '3', cfu: 9 }, + { nome: 'Logica matematica', anno: '3', cfu: 6 }, + { nome: 'Matematiche elementari da un punto di vista superiore: aritmetica', anno: '3', cfu: 6 }, + { nome: 'Matematiche elementari da un punto di vista superiore: geometria', anno: '3', cfu: 6 }, + { nome: 'Meccanica razionale', anno: '3', cfu: 6 }, + { nome: 'Metodi numerici per equazioni differenziali ordinarie', anno: '3', cfu: 6 }, + { nome: 'Metodi topologici in analisi globale', anno: '3', cfu: 6 }, + { nome: 'Ottimizzazione non lineare', anno: '3', cfu: 6 }, + { nome: 'Probabilità', anno: '3', cfu: 6 }, + { nome: 'Ricerca operativa', anno: '3', cfu: 6 }, + { nome: 'Sistemi dinamici', anno: '3', cfu: 6 }, + { nome: 'Spazi di Sobolev', anno: '3', cfu: 6 }, + { nome: 'Statistica matematica', anno: '3', cfu: 6 }, + { nome: 'Storia della matematica', anno: '3', cfu: 6 }, + { nome: 'Teoria algebrica dei numeri 1', anno: '3', cfu: 6 }, + { nome: 'Teoria dei campi e teoria di Galois', anno: '3', cfu: 6 }, + { nome: 'Teoria dei numeri elementare', anno: '3', cfu: 6 }, + { nome: 'Teoria della misura', anno: '3', cfu: 6 }, + + // Istituzioni (Magistrale) + { nome: 'Istituzioni di algebra', anno: 'istituzioni', cfu: 11 }, + { nome: 'Istituzioni di analisi matematica', anno: 'istituzioni', cfu: 11 }, + { nome: 'Istituzioni di analisi numerica', anno: 'istituzioni', cfu: 11 }, + { nome: 'Istituzioni di didattica della matematica', anno: 'istituzioni', cfu: 11 }, + { nome: 'Istituzioni di fisica matematica', anno: 'istituzioni', cfu: 11 }, + { nome: 'Istituzioni di geometria', anno: 'istituzioni', cfu: 11 }, + { nome: 'Istituzioni di probabilità', anno: 'istituzioni', cfu: 11 }, + + // Materie a scelta (Magistrale) + { nome: '4-varietà', anno: 'M', cfu: 6 }, + { nome: 'Algebra superiore A', anno: 'M', cfu: 6 }, + { nome: 'Algebre e gruppi di Lie', anno: 'M', cfu: 6 }, + { nome: 'Analisi armonica', anno: 'M', cfu: 6 }, + { nome: 'Analisi complessa A', anno: 'M', cfu: 6 }, + { nome: 'Analisi complessa B', anno: 'M', cfu: 6 }, + { nome: 'Analisi convessa', anno: 'M', cfu: 6 }, + { nome: 'Analisi dei dati', anno: 'M', cfu: 6 }, + { nome: 'Analisi non standard', anno: 'M', cfu: 6 }, + { nome: 'Analisi reale', anno: 'M', cfu: 6 }, + { nome: 'Analisi su spazi gaussiani', anno: 'M', cfu: 6 }, + { nome: 'Analisi superiore', anno: 'M', cfu: 6 }, + { nome: 'Analisi superiore A', anno: 'M', cfu: 6 }, + { nome: 'Analisi superiore B', anno: 'M', cfu: 6 }, + { nome: 'Aspetti matematici nella computazione quantistica', anno: 'M', cfu: 6 }, + { nome: 'Calcolo delle variazioni B', anno: 'M', cfu: 6 }, + { nome: 'Calcolo della variazioni A', anno: 'M', cfu: 6 }, // Variante nome + { nome: 'Combinatoria algebrica', anno: 'M', cfu: 6 }, + { nome: 'Complementi di analisi funzionale', anno: 'M', cfu: 6 }, + { nome: 'Complementi di meccanica razionale', anno: 'M', cfu: 6 }, + { nome: 'Crittografia post-quantistica', anno: 'M', cfu: 6 }, + { nome: 'Curve ellittiche', anno: 'M', cfu: 6 }, + { nome: 'Determinazione orbitale', anno: 'M', cfu: 6 }, + { nome: 'Didattica della matematica e nuove tecnologie', anno: 'M', cfu: 6 }, + { nome: 'Dinamica del sistema solare', anno: 'M', cfu: 6 }, + { nome: 'Dinamica iperbolica', anno: 'M', cfu: 6 }, + { nome: 'Dinamica olomorfa', anno: 'M', cfu: 6 }, + { nome: 'Elementi di calcolo in gruppi omogenei', anno: 'M', cfu: 6 }, + { nome: 'Equazioni della fluidodinamica', anno: 'M', cfu: 6 }, + { nome: 'Equazioni differenziali stocastiche e applicazioni', anno: 'M', cfu: 6 }, + { nome: 'Equazioni ellittiche', anno: 'M', cfu: 6 }, + { nome: 'Finanza matematica', anno: 'M', cfu: 6 }, + { nome: 'Fisica matematica', anno: 'M', cfu: 6 }, + { nome: 'Forme modulari', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica B', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica C', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica complessa', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica D', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica E', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica F', anno: 'M', cfu: 6 }, + { nome: 'Geometria algebrica G', anno: 'M', cfu: 6 }, + { nome: 'Geometria e analisi complessa', anno: 'M', cfu: 6 }, + { nome: 'Geometria differenziale complessa', anno: 'M', cfu: 6 }, + { nome: 'Geometria iperbolica', anno: 'M', cfu: 6 }, + { nome: 'Geometria riemanniana', anno: 'M', cfu: 6 }, + { nome: 'Gruppi algebrici lineari', anno: 'M', cfu: 6 }, + { nome: 'Gruppi di Coxeter', anno: 'M', cfu: 6 }, + { nome: 'Gruppi di Galois e gruppi fondamentali', anno: 'M', cfu: 6 }, + { nome: 'Meccanica celeste', anno: 'M', cfu: 6 }, + { nome: 'Meccanica spaziale', anno: 'M', cfu: 6 }, + { nome: 'Meccanica superiore', anno: 'M', cfu: 6 }, + { nome: 'Metodi di analisi armonica in analisi non lineare', anno: 'M', cfu: 6 }, + { nome: 'Metodi di approssimazione', anno: 'M', cfu: 6 }, + { nome: 'Metodi matematici della crittografia', anno: 'M', cfu: 6 }, + { nome: 'Metodi matematici della meccanica quantistica', anno: 'M', cfu: 6 }, + { nome: 'Metodi numerici per catene di Markov', anno: 'M', cfu: 6 }, + { nome: 'Metodi numerici per equazioni alle derivate parziali', anno: 'M', cfu: 6 }, + { nome: 'Metodi numerici per il calcolo tensoriale', anno: 'M', cfu: 6 }, + { nome: 'Metodi numerici per il controllo ottimo', anno: 'M', cfu: 6 }, + { nome: 'Metodi numerici per la grafica', anno: 'M', cfu: 6 }, + { nome: 'Metodi numerici per problemi inversi', anno: 'M', cfu: 6 }, + { nome: "Metodi probabilistici per l'algebra lineare numerica", anno: 'M', cfu: 6 }, + { nome: 'Modelli matematici in biomedicina e fisica matematica', anno: 'M', cfu: 6 }, + { nome: 'Origini e sviluppo delle matematiche moderne', anno: 'M', cfu: 6 }, + { nome: 'Probabilità superiore', anno: 'M', cfu: 6 }, + { nome: 'Problemi e metodi della ricerca in didattica della matematica', anno: 'M', cfu: 6 }, + { nome: 'Problemi e metodi in storia della matematica', anno: 'M', cfu: 6 }, + { nome: 'Sistemi dinamici aleatori', anno: 'M', cfu: 6 }, + { nome: 'Statistica superiore', anno: 'M', cfu: 6 }, + { nome: 'Storia della matematica antica e della sua tradizione', anno: 'M', cfu: 6 }, + { nome: 'Superfici di Riemann e curve algebriche', anno: 'M', cfu: 6 }, + { nome: 'Tecnologie per la didattica', anno: 'M', cfu: 6 }, + { nome: 'Teoria algebrica dei numeri 2', anno: 'M', cfu: 6 }, + { nome: 'Teoria analitica dei numeri A', anno: 'M', cfu: 6 }, + { nome: 'Teoria dei giochi', anno: 'M', cfu: 6 }, + { nome: 'Teoria dei modelli', anno: 'M', cfu: 6 }, + { nome: 'Teoria dei nodi A', anno: 'M', cfu: 6 }, + { nome: 'Teoria degli insiemi', anno: 'M', cfu: 6 }, + { nome: 'Teoria degli insiemi A', anno: 'M', cfu: 6 }, + { nome: 'Teoria degli insiemi B', anno: 'M', cfu: 6 }, + { nome: 'Teoria delle categorie', anno: 'M', cfu: 6 }, + { nome: 'Teoria delle rappresentazioni A', anno: 'M', cfu: 6 }, + { nome: "Teoria e metodi dell'ottimizzazione", anno: 'M', cfu: 6 }, + { nome: 'Teoria ergodica', anno: 'M', cfu: 6 }, + { nome: 'Teoria geometrica della misura', anno: 'M', cfu: 6 }, + { nome: 'Topologia algebrica', anno: 'M', cfu: 6 }, + { nome: 'Topologia algebrica A', anno: 'M', cfu: 6 }, + { nome: 'Topologia algebrica B', anno: 'M', cfu: 6 }, + { nome: 'Topologia differenziale', anno: 'M', cfu: 6 }, + { nome: 'Topologia e geometria in bassa dimensione', anno: 'M', cfu: 6 }, + { nome: 'Ultrafiltri e metodi non-standard', anno: 'M', cfu: 6 }, +] + +export function MediaPesataApp() { + // Funzioni per localStorage + const loadFromStorage = () => { + try { + const savedData = localStorage.getItem('media-pesata-data') + if (savedData) { + const parsed = JSON.parse(savedData) + return { + tipoStudente: parsed.tipoStudente || 'triennale', + corsiSelezionati: parsed.corsiSelezionati || [], + sezioniAperte: parsed.sezioniAperte || {}, + mostraRisultati: parsed.mostraRisultati || false, + } + } + } catch (error) { + console.warn('Errore nel caricamento dei dati salvati:', error) + } + return { + tipoStudente: 'triennale' as TipoStudente, + corsiSelezionati: [], + sezioniAperte: {}, + mostraRisultati: false, + } + } + + const saveToStorage = (data: any) => { + try { + localStorage.setItem('media-pesata-data', JSON.stringify(data)) + } catch (error) { + console.warn('Errore nel salvataggio dei dati:', error) + } + } + + // Inizializzazione con dati salvati + // const initialData = loadFromStorage() + const [tipoStudente, setTipoStudente] = useState('triennale') + const [corsiSelezionati, setCorsiSelezionati] = useState([]) + const [showCustomForm, setShowCustomForm] = useState(false) + const [customCorso, setCustomCorso] = useState({ nome: '', cfu: 0 }) + const [sezioniAperte, setSezioniAperte] = useState>({}) + const [mostraRisultati, setMostraRisultati] = useState(false) + + // Load data from localStorage on mount + useEffect(() => { + const initialData = loadFromStorage() + setTipoStudente(initialData.tipoStudente) + setCorsiSelezionati(initialData.corsiSelezionati) + setSezioniAperte(initialData.sezioniAperte) + setMostraRisultati(initialData.mostraRisultati) + }, []) + + // Salva automaticamente quando cambiano i dati importanti + useEffect(() => { + const dataToSave = { + tipoStudente, + corsiSelezionati, + sezioniAperte, + mostraRisultati, + } + saveToStorage(dataToSave) + }, [tipoStudente, corsiSelezionati, sezioniAperte, mostraRisultati]) + + const toggleSezione = (nomeSezione: string) => { + setSezioniAperte(prev => ({ + ...prev, + [nomeSezione]: !prev[nomeSezione], + })) + } + + const calcolaMedia = () => { + const corsiConVoto = corsiSelezionati.filter(corso => corso.voto !== null && !corso.passFailOnly) + if (corsiConVoto.length === 0) { + alert('Inserisci almeno un voto per calcolare la media!') + return + } + setMostraRisultati(true) + } + + // Funzioni per la gestione dei corsi + const aggiungiCorso = (corso: Corso) => { + // Controlla se la materia esiste già + const materiaEsiste = corsiSelezionati.some(c => c.nome.toLowerCase() === corso.nome.toLowerCase()) + if (materiaEsiste) { + alert('Questa materia è già stata aggiunta') + return + } + + // Controllo per magistrali: massimo 3 istituzioni + if (tipoStudente === 'magistrale' && corso.anno === 'istituzioni') { + const istituzioniAttuali = corsiSelezionati.filter(c => c.nome.toLowerCase().includes('istituzioni')).length + if (istituzioniAttuali >= 3) { + alert('Puoi selezionare al massimo 3 istituzioni per la magistrale') + return + } + } + + const id = Date.now().toString() + const nuovoCorso: CorsoSelezionato = { + id, + nome: corso.nome, + cfu: corso.cfu, + voto: null, + lode: false, + passFailOnly: corso.passFailOnly, + } + setCorsiSelezionati([...corsiSelezionati, nuovoCorso]) + } + + const aggiungiCorsoCustom = () => { + // Validazioni + if (!customCorso.nome.trim()) { + alert('Il nome della materia non può essere vuoto') + return + } + if (customCorso.cfu <= 0 || customCorso.cfu > 30) { + alert('I CFU devono essere tra 1 e 30') + return + } + if (!Number.isInteger(customCorso.cfu)) { + alert('I CFU devono essere un numero intero') + return + } + + // Controlla se la materia esiste già + const materiaEsiste = corsiSelezionati.some( + corso => corso.nome.toLowerCase().trim() === customCorso.nome.toLowerCase().trim(), + ) + if (materiaEsiste) { + alert('Questa materia è già stata aggiunta') + return + } + + const id = Date.now().toString() + const nuovoCorso: CorsoSelezionato = { + id, + nome: customCorso.nome.trim(), + cfu: customCorso.cfu, + voto: null, + lode: false, + } + setCorsiSelezionati([...corsiSelezionati, nuovoCorso]) + setCustomCorso({ nome: '', cfu: 0 }) + setShowCustomForm(false) + } + + const rimuoviCorso = (id: string) => { + setCorsiSelezionati(corsiSelezionati.filter(corso => corso.id !== id)) + } + + const aggiornaVoto = (id: string, voto: number | null) => { + // Validazione voto: deve essere tra 18 e 30 e intero + if (voto !== null && (voto < 18 || voto > 30 || !Number.isInteger(voto))) { + return // Ignora valori non validi + } + + setCorsiSelezionati( + corsiSelezionati.map(corso => + corso.id === id ? { ...corso, voto, lode: voto !== 30 ? false : corso.lode } : corso, + ), + ) + } + + const aggiornaLode = (id: string, lode: boolean) => { + setCorsiSelezionati(corsiSelezionati.map(corso => (corso.id === id ? { ...corso, lode } : corso))) + } + + const resetTutto = () => { + if (corsiSelezionati.length > 0) { + if (confirm('Sei sicuro di voler cancellare tutte le materie selezionate?')) { + setCorsiSelezionati([]) + setCustomCorso({ nome: '', cfu: 0 }) + setShowCustomForm(false) + setSezioniAperte({}) + setMostraRisultati(false) + // Pulisce anche il localStorage + try { + localStorage.removeItem('media-pesata-data') + } catch (error) { + console.warn('Errore nella pulizia del localStorage:', error) + } + } + } + } + + // Calcoli della media pesata + const calcolaMediaPesata = () => { + // Escludi le materie pass/fail dal calcolo + const corsiConVoto = corsiSelezionati.filter(corso => corso.voto !== null && !corso.passFailOnly) + if (corsiConVoto.length === 0) { + return { + mediaPesata: 0, + votoAmmissione: 0, + massimoVotoLaurea: 0, + conLode: false, + bonusLodi: 0, + errore: 'Nessun voto inserito per il calcolo della media', + } + } + + // Calcola CFU totali con voto + const cfuTotaliConVoto = corsiConVoto.reduce((sum, corso) => sum + corso.cfu, 0) + const cfuDaEscludere = tipoStudente === 'triennale' ? 15 : 9 + + // Se i CFU sono insufficienti per l'esclusione, avvisa l'utente + if (cfuTotaliConVoto < cfuDaEscludere) { + return { + mediaPesata: 0, + votoAmmissione: 0, + massimoVotoLaurea: 0, + conLode: false, + bonusLodi: 0, + errore: `Hai inserito solo ${cfuTotaliConVoto} CFU con voto. Servono almeno ${cfuDaEscludere} CFU per applicare le regole di esclusione.`, + } + } + + // Ordina per voto crescente + const corsiOrdinati = [...corsiConVoto].sort((a, b) => a.voto! - b.voto!) + + let cfuEsclusi = 0 + const corsiValidi: CorsoSelezionato[] = [] + + for (const corso of corsiOrdinati) { + if (cfuEsclusi < cfuDaEscludere) { + const cfuRimanentiDaEscludere = cfuDaEscludere - cfuEsclusi + if (corso.cfu <= cfuRimanentiDaEscludere) { + // Escludi tutto il corso + cfuEsclusi += corso.cfu + } else { + // Escludi solo una parte del corso + const cfuValidi = corso.cfu - cfuRimanentiDaEscludere + corsiValidi.push({ ...corso, cfu: cfuValidi }) + cfuEsclusi = cfuDaEscludere + } + } else { + corsiValidi.push(corso) + } + } + + // Calcola media pesata + const sommaPesata = corsiValidi.reduce((sum, corso) => sum + corso.voto! * corso.cfu, 0) + const sommaCfu = corsiValidi.reduce((sum, corso) => sum + corso.cfu, 0) + + const mediaPesata = sommaCfu > 0 ? sommaPesata / sommaCfu : 0 + + // Calcola bonus lodi + const bonusLodi = corsiConVoto.reduce((bonus, corso) => { + if (corso.lode) { + return bonus + (corso.cfu > 6 ? 0.5 : 0.25) + } + return bonus + }, 0) + + // Cap del bonus lodi basato sul tipo di studente + const capBonusLodi = tipoStudente === 'triennale' ? 1.5 : 2 + const bonusLodiFinal = Math.min(bonusLodi, capBonusLodi) + + // Voto di ammissione alla laurea (media pesata * 11/3) + const votoAmmissione = (mediaPesata * 11) / 3 + + // Voto di ammissione finale = voto ammissione + bonus lodi + const votoAmmissioneFinale = votoAmmissione + bonusLodiFinal + + // Massimo voto di laurea possibile = voto ammissione + 10 (cappato a 110) + const massimoVotoLaurea = Math.min(votoAmmissioneFinale + 10, 110) + const conLode = massimoVotoLaurea === 110 + + return { + mediaPesata: Math.round(mediaPesata * 100) / 100, + votoAmmissione: Math.round(votoAmmissioneFinale * 100) / 100, + massimoVotoLaurea: Math.round(massimoVotoLaurea * 100) / 100, + conLode, + bonusLodi: Math.round(bonusLodiFinal * 100) / 100, + errore: null, + } + } + + // Filtra corsi disponibili in base al tipo di studente + const getCorsiDisponibili = () => { + if (tipoStudente === 'triennale') { + return CORSI_DISPONIBILI.filter(corso => corso.anno !== 'istituzioni') + } else { + return CORSI_DISPONIBILI.filter( + corso => corso.anno === 'istituzioni' || corso.anno === '3' || corso.anno === 'M', + ) + } + } + + // Raggruppa corsi per categoria + const raggruppaCorsi = () => { + const corsi = getCorsiDisponibili() + const gruppi: Record = {} + + if (tipoStudente === 'triennale') { + gruppi['Primo Anno'] = corsi.filter(c => c.anno === '1') + gruppi['Secondo Anno'] = corsi.filter(c => c.anno === '2') + gruppi['Terzo Anno'] = corsi.filter(c => c.anno === '3') + gruppi['Materie a Scelta'] = corsi.filter(c => c.anno === 'M') + } else { + // Per magistrali: prima le istituzioni, poi tutto il resto come "Materie a Scelta" + gruppi['Istituzioni'] = corsi.filter(c => c.anno === 'istituzioni') + gruppi['Materie a Scelta'] = corsi.filter(c => c.anno === '3' || c.anno === 'M') + } + + return gruppi + } + + const cambiaTipoStudente = (nuovoTipo: TipoStudente) => { + setTipoStudente(nuovoTipo) + } + + const gruppiCorsi = raggruppaCorsi() + const risultati = calcolaMediaPesata() + const totaleCfu = corsiSelezionati.reduce((sum, corso) => sum + corso.cfu, 0) + const maxCfu = tipoStudente === 'triennale' ? 171 : 93 // 180 − 9 e 120 − 27 per la tesi + const cfuError = totaleCfu > maxCfu + + return ( +
+ {/* Selezione tipo studente */} +
+
+

Corso di Laurea

+
+ + +
+
+
+ + {/* Counter CFU */} +
+

+ CFU Totali: {totaleCfu}/{maxCfu} + {tipoStudente === 'triennale' ? ' (+9 tesi)' : ' (+27 tesi)'} +

+ {cfuError &&

⚠️ Hai superato il limite di CFU consentiti!

} +
+ + {/* Sezione selezione corsi */} +
+
+

Seleziona Materie

+
+ + {Object.entries(gruppiCorsi).map(([categoria, corsi]) => ( +
+ + + {sezioniAperte[categoria] && ( +
+ {corsi.map((corso, index) => ( + + ))} +
+ )} +
+ ))} + + {/* Form per materia custom */} +
+

Materia Personalizzata

+ {!showCustomForm ? ( + + ) : ( +
+ + setCustomCorso({ ...customCorso, nome: (e.target as HTMLInputElement).value }) + } + /> + + setCustomCorso({ + ...customCorso, + cfu: parseInt((e.target as HTMLInputElement).value) || 0, + }) + } + /> + + +
+ )} +
+
+ + {/* Sezione lista corsi selezionati */} +
+
+
+

Materie Selezionate

+
+
+ {corsiSelezionati.length > 0 && } +
+ + {corsiSelezionati.length === 0 ? ( +

Nessuna materia selezionata

+ ) : ( +
+ {corsiSelezionati.map(corso => ( +
+ {corso.nome} + {corso.cfu} CFU + + {!corso.passFailOnly && ( +
+ + aggiornaVoto( + corso.id, + parseInt((e.target as HTMLInputElement).value) || null, + ) + } + /> + + +
+ )} + +
+ +
+
+ ))} +
+ )} +
+ + {/* Pulsante Calcola */} + {corsiSelezionati.length > 0 && ( +
+ +
+ )} + + {/* Risultati */} + {risultati && mostraRisultati && ( +
+

Risultati

+ {risultati.errore ? ( +
+

⚠️ {risultati.errore}

+
+ ) : ( +
+
+ Media Pesata: + {risultati.mediaPesata} +
+
+ Bonus Lodi: + +{risultati.bonusLodi} +
+
+ Voto di Ammissione: + {risultati.votoAmmissione} +
+
+ Massimo Voto Di Laurea Possibile: + + {risultati.massimoVotoLaurea} + {risultati.conLode && (+lode)} + +
+
+ )} +
+ )} + + {/* Nota informativa */} +
+

📋 Come viene calcolata la media

+
+

Regole di esclusione CFU:

+
    +
  • + Triennale: I 15 CFU con i voti più bassi vengono esclusi dalla media +
  • +
  • + Magistrale: I 9 CFU con i voti più bassi vengono esclusi dalla media +
  • +
  • Se un corso ha più CFU di quelli da escludere, viene diviso proporzionalmente
  • +
+ +

Calcolo del voto finale:

+
    +
  • + Media pesata: Somma dei (voto × CFU) diviso per i CFU totali +
  • +
  • + Bonus lodi: +0.5 per lodi in materie > 6 CFU, +0.25 per lodi in materie + ≤ 6 CFU (max +1.5 per triennale, max +2 per magistrale) +
  • +
  • + Voto di laurea: (Voto finale × 11) ÷ 3 +
  • +
+ +

Note:

+
    +
  • + Le materie Pass/Fail non contribuiscono al calcolo della media +
  • +
  • Il voto finale è limitato a 30
  • +
  • Per i magistrali: massimo 3 istituzioni selezionabili
  • +
+
+
+
+ ) +} + +// Funzione per inizializzare l'app +// export function initMediaPesataApp() { +// const container = document.getElementById('media-pesata-app') +// if (container) { +// render(, container) +// } +// } + +// export default MediaPesataApp diff --git a/src/components/Header.astro b/src/components/Header.astro index dcb6da6..db587a4 100644 --- a/src/components/Header.astro +++ b/src/components/Header.astro @@ -6,6 +6,7 @@ const links = [ { href: '/notizie', text: 'Notizie' }, { href: '/guide', text: 'Guide' }, { href: '/domande-esami', text: 'Domande Orali' }, + { href: '/media-pesata', text: 'Calcolo Media' }, // Beta testing - solo URL diretto { href: '/storia', text: 'Storia' }, // { href: '/login', text: 'Login' }, ] diff --git a/src/content/news/2025-06-28-media.md b/src/content/news/2025-06-28-media.md new file mode 100644 index 0000000..9150937 --- /dev/null +++ b/src/content/news/2025-06-28-media.md @@ -0,0 +1,22 @@ +--- +title: Calcola la tua media ed il voto di laurea con il nuovissimo calcolatore del PHC! +description: È ora disponibile uno strumento per calcolare la propria media pesata e il voto di ammissione alla laurea secondo le regole del dipartimento +publishDate: 2025-06-26 +--- + +# Calcola la tua media ed il voto di laurea con il nuovissimo calcolatore del PHC! + +È ora disponibile nella sezione "Calcolo Media" del sito uno strumento per calcolare la propria media pesata e il voto di ammissione alla laurea secondo le regole ufficiali del dipartimento. + +

+phc.dm.unipi.it/media-pesata/ +

+ +Il calcolatore applica automaticamente le regole di esclusione previste dal regolamento: + +- **Triennale**: vengono esclusi i 15 CFU con i voti più bassi +- **Magistrale**: vengono esclusi i 9 CFU con i voti più bassi + +Il sistema calcola anche il bonus per le lodi, che vale +0.5 punti per ogni materia superiore a 6 CFU e +0.25 punti per materie da 6 CFU o meno, con un tetto massimo di +1.5 punti per la triennale e +2 punti per la magistrale. + +Se dovessero esserci bug scriveteci un'email a macchinisti@lists.dm.unipi.it! diff --git a/src/data/macchinisti.yaml b/src/data/macchinisti.yaml index 8b85f76..62c1279 100644 --- a/src/data/macchinisti.yaml +++ b/src/data/macchinisti.yaml @@ -25,7 +25,8 @@ - fullName: Francesco Baldino entranceDate: 2022 - description: Bla bla Star Wars + description: | + Appassionato di Star Wars, NixOS e lunghe camminate in montagna. Pokemon preferito: Latias. social: github: https://github.com/Fran314 website: https://poisson.phc.dm.unipi.it/~baldino diff --git a/src/pages/media-pesata.astro b/src/pages/media-pesata.astro new file mode 100644 index 0000000..6654f55 --- /dev/null +++ b/src/pages/media-pesata.astro @@ -0,0 +1,21 @@ +--- +import '@/styles/pages/media-pesata.css' +import PageLayout from '../layouts/PageLayout.astro' + +import { MediaPesataApp } from '@/client/MediaPesataApp' +--- + + +
+

Calcolo Media e Voto di Laurea

+

+ Calcola la tua media pesata e il voto con cui ti siederai alla discussione di laurea, seguendo le regole del + dipartimento di Matematica. +

+ + +
+
diff --git a/src/styles/controls.css b/src/styles/controls.css index 3bac52b..098273a 100644 --- a/src/styles/controls.css +++ b/src/styles/controls.css @@ -75,7 +75,7 @@ Controls - for things like buttons, input, select input[type='text'], input[type='password'] { width: 100%; - height: 2.5rem; + min-height: 1.75rem; /* @include neo-brutalist-card; */ border: 3px solid #222; @@ -89,6 +89,95 @@ Controls - for things like buttons, input, select } } + input[type='checkbox'] { + appearance: none; + -webkit-appearance: none; + -moz-appearance: none; + + width: calc(1.5rem + 1px); + height: calc(1.5rem + 1px); + + background: #fff; + border: 3px solid #222; + border-radius: 4px; + box-shadow: 3px 3px 0 0 #222; + + position: relative; + cursor: pointer; + + transition: all 64ms linear; + + &:hover { + transform: translate(-1px, -1px); + box-shadow: 4px 4px 0 0 #222; + } + + &:active { + transform: translate(1px, 1px); + box-shadow: 2px 2px 0 0 #222; + } + + &:checked { + background: #1e6733; + + &::after { + content: ''; + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + + width: 0.8rem; + height: 0.8rem; + background: #1e6733; + + clip-path: polygon(10% 55%, 35% 75%, 85% 25%, 90% 35%, 40% 95%, 5% 60%); + } + + &:hover { + background: #2b8b47; + } + } + + &:disabled { + background: #eee; + border-color: #888; + box-shadow: 3px 3px 0 0 #888; + cursor: not-allowed; + + &:hover { + transform: none; + box-shadow: 3px 3px 0 0 #888; + } + + &:checked { + background: #aaa; + + &::after { + background: #666; + } + } + } + + &.star { + &:checked::after { + background: rgb(255, 197, 49); + clip-path: polygon( + 50% 0%, + 61% 35%, + 98% 35%, + 68% 57%, + 79% 91%, + 50% 70%, + 21% 91%, + 32% 57%, + 2% 35%, + 39% 35% + ); + } + } + } + form { display: grid; gap: 1rem; diff --git a/src/styles/main.css b/src/styles/main.css index 3ab0f56..cec7c82 100644 --- a/src/styles/main.css +++ b/src/styles/main.css @@ -325,6 +325,38 @@ Custom Page Styles opacity: 0 !important; } + .grid-center { + display: grid; + place-content: center; + place-items: center; + } + + .h-flex { + display: flex; + gap: 0.5rem; + flex-direction: row; + } + + .v-flex { + display: flex; + gap: 0.5rem; + flex-direction: column; + } + + .h-flex, + .v-flex { + place-self: stretch; + align-items: center; + + > * { + flex-shrink: 0; + } + + > .spacer { + flex-grow: 1; + } + } + @media screen and (min-width: 1024px) { .mobile-only { display: none !important; diff --git a/src/styles/pages.scss b/src/styles/pages.scss new file mode 100644 index 0000000..e69de29 diff --git a/src/styles/pages/media-pesata.css b/src/styles/pages/media-pesata.css new file mode 100644 index 0000000..d618a08 --- /dev/null +++ b/src/styles/pages/media-pesata.css @@ -0,0 +1,1104 @@ +@layer page { + :root { + --media-pesata-accent: #e4c5ff; + --media-pesata-bg: hsl(from var(--media-pesata-accent) h s 95%); + + --media-pesata-accent-2: #ffe4c5; + --media-pesata-accent-3: #c5ffe4; + + --card-base: var(--media-pesata-accent); + } + + main { + justify-self: center; + background: var(--media-pesata-bg); + + display: flex; + flex-direction: column; + align-items: center; + + padding: 4.5rem 3rem; + gap: 4.5rem; + + @media screen and (max-width: 1024px) { + padding: 3rem 1rem; + gap: 3rem; + + .card { + width: 100%; + } + } + } + + .media-pesata-container { + max-width: 72rem; + margin: 0 auto; + padding: 2rem; + + @media screen and (max-width: 768px) { + padding: 0; + } + } + + h1 { + text-align: center; + margin-bottom: 1rem; + color: var(--color-primary); + } + + p { + text-align: center; + margin-bottom: 2rem; + color: var(--color-text-secondary); + } + + .card { + align-content: start; + gap: 1rem; + } + + button { + display: grid; + grid-auto-flow: column; + place-content: center; + place-items: center; + + padding: 0.25rem 1rem; + gap: 0.5rem; + font-size: 16px; + + border: 2px solid #222; + border-radius: 4px; + box-shadow: 2px 2px 0 0 #222; + + transition: all 64ms linear; + text-decoration: none; + color: #222; + + cursor: pointer; + + &:hover { + transform: translate(-1px, -1px); + box-shadow: 3px 3px 0 0 #222; + } + + &:active { + transform: translate(1px, 1px); + box-shadow: 1px 1px 0 0 #222; + } + } + + .media-pesata-app { + max-width: 1200px; + margin: 0 auto; + padding: 2rem; + min-height: 100vh; + + display: grid; + grid-template-columns: 1fr 1fr; + grid-template-rows: repeat(7, auto); + + align-items: start; + gap: 2rem; + + > .wide { + grid-column: 1 / -1; + } + + @media screen and (max-width: 1024px) { + padding: 0; + + grid-template-columns: 1fr; + gap: 1rem; + } + } + + .media-pesata-title { + text-align: center; + color: var(--palette-black); + margin-bottom: 2rem; + font-family: var(--font-display); + font-weight: 700; + font-size: 2.5rem; + } + + /* Selezione tipo studente */ + .student-type-selector { + background: #fff; + border: var(--border-large); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 1.5rem; + margin-bottom: 2rem; + + display: grid; + place-content: center; + + h2 { + margin-bottom: 1rem; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + font-size: 1.5rem; + text-align: center; + } + } + + /* .radio-group { + display: flex; + gap: 1.5rem; + justify-content: center; + + label { + display: flex; + align-items: center; + gap: 0.5rem; + font-size: 1.1rem; + cursor: pointer; + padding: 0.5rem 1rem; + border: 2px solid var(--palette-black); + border-radius: 6px; + background: #fff; + font-family: var(--font-secondary); + font-weight: 600; + transition: all 64ms linear; + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + label:hover { + transform: translate(-1px, -1px); + box-shadow: 3px 3px 0 0 var(--palette-black); + } + + label:active { + transform: translate(1px, 1px); + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + input[type='radio'] { + width: 18px; + height: 18px; + margin: 0; + accent-color: #007bff; + } + } */ + + .compound-button { + display: flex; + border: 3px solid #222; + border-radius: 6px; + box-shadow: 4px 4px 0 0 #222; + overflow: hidden; + width: fit-content; + + button { + border: none; + border-radius: 0; + box-shadow: none; + margin: 0; + flex: 1; + min-width: 120px; + padding: 0.5rem; + + &:not(:last-child) { + border-right: 3px solid #222; + } + + &:hover { + transform: none; + box-shadow: none; + background: #f0f0f0; + } + + &:active { + transform: none; + box-shadow: none; + background: #e0e0e0; + } + + &.active { + background: hsl(from var(--media-pesata-accent) h s 75%); + color: #f4fef7; + + &:hover { + background: hsl(from var(--media-pesata-accent) h s 85%); + } + + &:active { + background: hsl(from var(--media-pesata-accent) h s 95%); + } + } + } + } + + /* Counter CFU */ + .cfu-counter { + background: hsl(from var(--media-pesata-accent-2) h s 75%); + border: var(--border-large); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 1rem; + text-align: center; + + &.error { + background: #ffd8d8; + border-color: #dc3545; + } + } + + .cfu-counter h3 { + margin: 0; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + font-size: 1.3rem; + } + + .error-message { + color: #d63031; + font-weight: bold; + margin: 0; + font-family: var(--font-secondary); + } + + .error-message { + background: #ff6b7a; + border: 2px solid var(--palette-black); + border-radius: 6px; + padding: 1rem; + color: var(--palette-black); + font-weight: 600; + } + + .error-message p { + margin: 0; + text-align: center; + } + + @media (max-width: 768px) { + .main-content { + grid-template-columns: 1fr; + } + } + + /* Sezione selezione corsi */ + .course-selection { + background: #fff; + border: var(--border-large); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 1.5rem; + } + + .course-selection h2 { + margin-bottom: 1.5rem; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + font-size: 1.5rem; + } + + .course-category { + display: grid; + gap: 0.5rem; + + > button { + padding: 0.25rem 0.75rem 0.25rem 0.5rem; + font-size: 20px; + + place-content: stretch; + place-items: stretch; + } + + .course-grid { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); + gap: 0.5rem; + + padding: 0.25rem; + + border-radius: 0.25rem; + overflow-y: auto; + max-height: 50vh; + + position: relative; + + /* bottom linear gradient for scroll */ + &::after { + content: ''; + position: sticky; + bottom: -0.25rem; + left: 0; + right: 0; + height: 1rem; + margin-top: calc(-1rem - 0.25rem); + + background: linear-gradient(to top, #0004, transparent); + } + + .course-button { + display: grid; + grid-template-rows: auto auto; + padding: 0.5rem; + gap: 0.25rem; + justify-content: stretch; + justify-items: start; + + border: 2px solid var(--palette-black); + border-radius: 4px; + background: #f8f9fa; + cursor: pointer; + transition: all 64ms linear; + text-align: left; + font-family: var(--font-secondary); + box-shadow: 2px 2px 0 0 var(--palette-black); + + &:hover:not(:disabled) { + background: var(--homepage-projects-bg); + transform: translate(-1px, -1px); + box-shadow: 3px 3px 0 0 var(--palette-black); + } + + &:active:not(:disabled) { + transform: translate(1px, 1px); + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + &:disabled { + opacity: 0.5; + cursor: not-allowed; + background: #e0e0e0; + } + + > .course-name { + font-weight: 600; + font-size: 0.9rem; + line-height: 1.3; + color: var(--palette-black); + } + + > .course-cfu { + font-size: 0.8rem; + color: #666; + font-weight: bold; + /* font-family: var(--font-mono); */ + } + } + + @media screen and (590px <= width <= 1024px) { + &::after { + display: none; + } + } + } + } + + /* Materia personalizzata */ + .custom-course { + margin-top: 1rem; + padding-top: 1.5rem; + border-top: 3px solid var(--palette-black); + } + + .custom-course h3 { + margin-bottom: 1rem; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + font-size: 1.2rem; + } + + .add-custom-btn { + background: #1e6733; + color: #f4fef7; + border: 3px solid var(--palette-black); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 0.75rem 1.5rem; + cursor: pointer; + font-size: 0.9rem; + font-weight: 600; + font-family: var(--font-secondary); + transition: all 64ms linear; + } + + .add-custom-btn:hover { + background: #2b8b47; + transform: translate(-1px, -1px); + box-shadow: 5px 5px 0 0 var(--palette-black); + } + + input { + padding: 0.25rem 0.5rem; + font-size: 14px; + + border: 2px solid #222; + border-radius: 4px; + box-shadow: 2px 2px 0 0 #222; + + background: #fff; + font-family: var(--font-primary); + } + + .custom-form { + display: flex; + gap: 0.5rem; + flex-wrap: wrap; + + input[type='text'] { + flex: 2; + min-width: 200px; + } + + input[type='number'] { + flex: 0 0 80px; + } + } + + .confirm-btn, + .cancel-btn { + padding: 0.5rem 1rem; + border: 2px solid var(--palette-black); + border-radius: 4px; + cursor: pointer; + font-size: 0.9rem; + font-weight: 600; + font-family: var(--font-secondary); + transition: all 64ms linear; + } + + .confirm-btn { + background: #28a745; + color: white; + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + .confirm-btn:hover { + background: #218838; + transform: translate(-1px, -1px); + box-shadow: 3px 3px 0 0 var(--palette-black); + } + + .confirm-btn:active { + transform: translate(1px, 1px); + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + .cancel-btn { + background: #6c757d; + color: white; + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + .cancel-btn:hover { + background: #5a6268; + transform: translate(-1px, -1px); + box-shadow: 3px 3px 0 0 var(--palette-black); + } + + .cancel-btn:active { + transform: translate(1px, 1px); + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + /* Sezione corsi selezionati */ + .selected-courses { + background: #fff; + border: var(--border-large); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 1.5rem; + } + + .selected-courses h2 { + margin-bottom: 1.5rem; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + font-size: 1.5rem; + } + + .no-courses { + text-align: center; + color: #666; + font-style: italic; + padding: 2rem; + font-family: var(--font-secondary); + } + + .courses-list { + display: flex; + flex-direction: column; + gap: 0.5rem; + + /* @media screen and (max-width: 1024px) { + gap: 0; + + .course-item { + &:first-child:not(:last-child) { + border-bottom-left-radius: 0; + border-bottom-right-radius: 0; + } + + &:last-child:not(:first-child) { + border-top-left-radius: 0; + border-top-right-radius: 0; + + border-top: none; + } + + &:not(:first-child, :last-child) { + border-radius: 0; + border-top: none; + } + } + } */ + } + + .course-item { + display: grid; + grid-template-rows: auto 1fr; + grid-template-columns: 1fr auto auto; + align-items: center; + + gap: 0.5rem 1rem; + padding: 0.5rem 1rem 0.5rem 0.5rem; + background: #fff; + border: 2px solid var(--palette-black); + border-radius: 4px; + transition: all 64ms linear; + + > .course-name { + grid-column: 1 / 2; + grid-row: 1 / 2; + + font-weight: 600; + font-size: 0.9rem; + color: var(--palette-black); + font-family: var(--font-secondary); + } + + > .course-cfu { + grid-column: 1 / 2; + grid-row: 2 / 3; + + font-size: 0.8rem; + font-weight: bold; + color: #666; + /* font-family: var(--font-mono); */ + } + + > .tall { + grid-row: 1 / -1; + } + + .course-grade { + display: grid; + grid-auto-flow: column; + place-content: center; + + gap: 0.5rem; + + input[type='number'] { + min-width: 4.25rem; + } + } + + > .actions { + grid-column: -2 / -1; + } + + .lode-checkbox { + display: flex; + align-items: center; + gap: 0.25rem; + font-size: 0.8rem; + cursor: pointer; + font-weight: 600; + font-family: var(--font-secondary); + + &.disabled { + opacity: 0.5; + cursor: not-allowed; + } + } + + .remove-btn { + background: #d63031; + color: white; + width: 1.75rem; + height: 1.75rem; + font-size: 20px; + padding: 0; + + margin-top: 4px; + + &:hover { + background: #b71c1c; + } + } + + @media screen and (max-width: 1024px) { + justify-items: start; + align-items: center; + gap: 0.5rem; + + grid-template-columns: 1fr auto; + grid-template-rows: auto auto; + + padding: 0.5rem; + + .course-name { + display: grid; + place-content: center; + place-items: center; + + grid-column: 1 / 2; + grid-row: 1 / 2; + } + + .course-cfu { + display: grid; + justify-self: center; + align-self: start; + + grid-column: 2 / 3; + grid-row: 1 / 2; + } + + .course-grade { + grid-row: 2 / 3; + } + + .actions { + grid-column: 2 / 3; + grid-row: 2 / 3; + + justify-self: end; + } + } + } + + .calculate-button { + background: #1e6733; + color: #f4fef7; + border: 3px solid var(--palette-black); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 1rem 2rem; + font-size: 1.2rem; + font-weight: 600; + font-family: var(--font-secondary); + cursor: pointer; + transition: all 64ms linear; + margin-bottom: 2rem; + + &:hover { + background: #2b8b47; + transform: translate(-1px, -1px); + box-shadow: 5px 5px 0 0 var(--palette-black); + } + + &:active { + transform: translate(2px, 2px); + box-shadow: 2px 2px 0 0 var(--palette-black); + } + } + + .results { + background: #fff; + border: 3px solid var(--palette-black); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + padding: 2rem; + margin-top: 1rem; + + > h2 { + margin-bottom: 1.5rem; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + font-size: 1.6rem; + text-align: center; + } + + .results-grid { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(280px, 1fr)); + gap: 1rem; + } + + @media screen and (max-width: 1024px) { + padding: 1rem; + } + } + + /* Ordine degli elementi: il massimo voto di laurea al centro */ + .result-item:nth-child(4) { + order: -1; + grid-column: 1 / -1; + justify-self: center; + max-width: 500px; + width: 100%; + } + + .result-item { + display: flex; + justify-content: space-between; + align-items: center; + padding: 1rem; + background: hsl(from var(--media-pesata-accent-3) h s 75%); + border: 2px solid var(--palette-black); + border-radius: 4px; + font-family: var(--font-secondary); + font-weight: 600; + font-size: 1rem; + } + + .result-item.highlight { + background: hsl(from var(--media-pesata-accent-2) h s 75%); + font-weight: bold; + font-size: 1.2rem; + } + + .result-item .label { + font-weight: 600; + color: var(--palette-black); + font-size: 0.9rem; + line-height: 1.2; + } + + .result-item .value { + font-weight: bold; + font-size: 1.1rem; + font-family: var(--font-mono); + color: var(--palette-black); + } + + .pass-fail-badge { + background: var(--color-accent); + color: white; + padding: 2px 8px; + border-radius: 12px; + font-size: 0.8rem; + font-weight: bold; + text-transform: uppercase; + letter-spacing: 0.5px; + } + + .lode-badge { + color: var(--palette-black); + font-weight: bold; + margin-left: 0.5rem; + } + + /* Responsive */ + /* @media (max-width: 768px) { + .course-grid { + grid-template-columns: 1fr; + } + + .course-item { + flex-direction: row; + flex-wrap: wrap; + align-items: center; + gap: 0.5rem; + padding: 0.75rem; + } + + .course-info { + flex: 1 1 100%; + flex-direction: row; + justify-content: space-between; + align-items: center; + gap: 0.5rem; + } + + .course-grade { + flex: 1 1 auto; + justify-content: flex-start; + gap: 0.5rem; + } + + .course-grade input[type='number'] { + width: 70px; + } + + .remove-btn { + width: 28px; + height: 28px; + font-size: 1rem; + flex-shrink: 0; + } + + .custom-form { + flex-direction: column; + } + + .results-grid { + grid-template-columns: 1fr; + } + + .result-item:nth-child(4) { + order: -1; + grid-column: 1; + justify-self: center; + max-width: 100%; + } + } */ + + /* Layout ultra-compatto per schermi molto piccoli */ + /* @media (max-width: 480px) { + .course-item { + padding: 0.5rem; + gap: 0.25rem; + } + + .course-info { + flex: 1 1 100%; + margin-bottom: 0.25rem; + } + + .course-info .course-name { + font-size: 0.85rem; + } + + .course-info .course-cfu { + font-size: 0.75rem; + } + + .course-grade { + flex: 1 1 auto; + gap: 0.25rem; + } + + .course-grade input[type='number'] { + width: 65px; + padding: 0.375rem; + font-size: 0.85rem; + } + + .lode-checkbox { + font-size: 0.75rem; + } + + .remove-btn { + width: 26px; + height: 26px; + font-size: 0.9rem; + } + } */ + + .section-header { + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; + } + + .reset-btn { + background: #d63031; + color: white; + border: 2px solid var(--palette-black); + padding: 0.5rem 1rem; + border-radius: 4px; + font-size: 0.9rem; + font-weight: 600; + cursor: pointer; + transition: all 64ms linear; + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + .reset-btn:hover { + background: #b71c1c; + transform: translate(-1px, -1px); + box-shadow: 3px 3px 0 0 var(--palette-black); + } + + .reset-btn:active { + transform: translate(1px, 1px); + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + .cfu-info { + font-size: 0.9rem; + color: #666; + margin: 0.25rem 0 0 0; + font-weight: normal; + } + + .error-text { + color: #d63031; + font-weight: 600; + margin: 0.5rem 0 0 0; + } + + /* Category headers collapsible */ + .category-header { + width: 100%; + background: #e4c5ff; + border: 2px solid var(--palette-black); + border-radius: 4px; + padding: 0.5rem 1rem; + cursor: pointer; + display: flex; + justify-content: space-between; + align-items: center; + transition: all 64ms linear; + margin-bottom: 0.5rem; + min-height: 36px; + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + .category-header:hover { + background: #d9b3ff; + transform: translate(-1px, -1px); + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + .category-header:active { + transform: translate(0px, 0px); + box-shadow: 1px 1px 0 0 var(--palette-black); + } + + .category-header h3 { + margin: 0; + font-family: var(--font-display); + font-weight: 600; + color: var(--palette-black); + font-size: 1rem; + } + + .toggle-icon { + font-size: 1rem; + color: var(--palette-black); + transition: transform 150ms ease; + user-select: none; + } + + .toggle-icon.expanded { + transform: rotate(90deg); + } + + /* Calculate button */ + .calculate-section { + display: grid; + place-content: center; + padding: 2rem 0 1rem; + + button { + font-size: 20px; + padding: 1rem; + + /* border 3px */ + border: 3px solid var(--palette-black); + border-radius: 6px; + box-shadow: 4px 4px 0 0 var(--palette-black); + + &:hover { + transform: translate(-1px, -1px); + box-shadow: 5px 5px 0 0 var(--palette-black); + } + + &:active { + transform: translate(2px, 2px); + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + background: hsl(from var(--media-pesata-accent-2) h s 75%); + } + } + + .calculate-btn { + background: #1e6733; + color: #f4fef7; + border: var(--border-large); + border-radius: 6px; + padding: 0.75rem 1.5rem; + font-size: 1rem; + font-weight: 600; + cursor: pointer; + transition: all 64ms linear; + font-family: var(--font-primary); + display: inline-block; + box-shadow: 4px 4px 0 0 var(--palette-black); + } + + .calculate-btn:hover { + background: #2b8b47; + transform: translate(-1px, -1px); + box-shadow: 5px 5px 0 0 var(--palette-black); + } + + .calculate-btn:active { + transform: translate(2px, 2px); + box-shadow: 2px 2px 0 0 var(--palette-black); + } + + /* Info section */ + .info-section { + background: var(--color-accent-light); + border: var(--border-large); + border-radius: 8px; + padding: 1.5rem; + margin: 2rem 0; + } + + .info-section h3 { + margin-top: 0; + margin-bottom: 1rem; + color: var(--palette-black); + font-family: var(--font-display); + font-weight: 600; + } + + .info-content h4 { + color: var(--palette-black); + font-weight: 600; + margin-top: 1rem; + margin-bottom: 0.5rem; + } + + .info-content ul { + margin-bottom: 1rem; + padding-left: 1.5rem; + } + + .info-content li { + margin-bottom: 0.5rem; + line-height: 1.4; + } + + /* Gestione testi lunghi su schermi più piccoli */ + @media (max-width: 480px) { + .result-item { + flex-direction: column; + align-items: flex-start; + gap: 0.5rem; + } + + .result-item .label { + font-size: 0.85rem; + } + + .result-item .value { + align-self: flex-end; + font-size: 1.2rem; + } + } +} + +@layer utility { + @media screen and (max-width: 1024px) { + .card { + > .h-flex { + flex-direction: column; + align-items: start; + + > .spacer { + display: none; + } + } + } + } +}