style changes

media-pesata
Antonio De Lucreziis 11 months ago
parent 4c19ffdd9f
commit bbad670baa

@ -186,7 +186,7 @@ const CORSI_DISPONIBILI: Corso[] = [
{ nome: 'Ultrafiltri e metodi non-standard', anno: 'M', cfu: 6 }, { nome: 'Ultrafiltri e metodi non-standard', anno: 'M', cfu: 6 },
] ]
function MediaPesataApp() { export function MediaPesataApp() {
// Funzioni per localStorage // Funzioni per localStorage
const loadFromStorage = () => { const loadFromStorage = () => {
try { try {
@ -220,13 +220,22 @@ function MediaPesataApp() {
} }
// Inizializzazione con dati salvati // Inizializzazione con dati salvati
const initialData = loadFromStorage() // const initialData = loadFromStorage()
const [tipoStudente, setTipoStudente] = useState<TipoStudente>(initialData.tipoStudente) const [tipoStudente, setTipoStudente] = useState<TipoStudente>('triennale')
const [corsiSelezionati, setCorsiSelezionati] = useState<CorsoSelezionato[]>(initialData.corsiSelezionati) const [corsiSelezionati, setCorsiSelezionati] = useState<CorsoSelezionato[]>([])
const [showCustomForm, setShowCustomForm] = useState(false) const [showCustomForm, setShowCustomForm] = useState(false)
const [customCorso, setCustomCorso] = useState<CorsoCustom>({ nome: '', cfu: 0 }) const [customCorso, setCustomCorso] = useState<CorsoCustom>({ nome: '', cfu: 0 })
const [sezioniAperte, setSezioniAperte] = useState<Record<string, boolean>>(initialData.sezioniAperte) const [sezioniAperte, setSezioniAperte] = useState<Record<string, boolean>>({})
const [mostraRisultati, setMostraRisultati] = useState(initialData.mostraRisultati) 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 // Salva automaticamente quando cambiano i dati importanti
useEffect(() => { useEffect(() => {
@ -482,15 +491,6 @@ function MediaPesataApp() {
} }
const cambiaTipoStudente = (nuovoTipo: TipoStudente) => { const cambiaTipoStudente = (nuovoTipo: TipoStudente) => {
if (corsiSelezionati.length > 0 && nuovoTipo !== tipoStudente) {
if (
!confirm(
'Cambiando il tipo di studente, alcune materie potrebbero non essere più disponibili. Continuare?',
)
) {
return
}
}
setTipoStudente(nuovoTipo) setTipoStudente(nuovoTipo)
} }
@ -501,64 +501,64 @@ function MediaPesataApp() {
const cfuError = totaleCfu > maxCfu const cfuError = totaleCfu > maxCfu
return ( return (
<div className="media-pesata-app"> <div class="media-pesata-app">
{/* Selezione tipo studente */} {/* Selezione tipo studente */}
<div className="student-type-selector"> <div class="card student-type-switcher wide">
<div class="grid-center">
<h2>Corso di Laurea</h2> <h2>Corso di Laurea</h2>
<div className="radio-group"> <div class="compound-button">
<label> <button
<input class={tipoStudente === 'triennale' ? 'active' : ''}
type="radio" onClick={() => cambiaTipoStudente('triennale')}
value="triennale" >
checked={tipoStudente === 'triennale'}
onChange={e => cambiaTipoStudente((e.target as HTMLInputElement).value as TipoStudente)}
/>
Triennale Triennale
</label> </button>
<label> <button
<input class={tipoStudente === 'magistrale' ? 'active' : ''}
type="radio" onClick={() => cambiaTipoStudente('magistrale')}
value="magistrale" >
checked={tipoStudente === 'magistrale'}
onChange={e => cambiaTipoStudente((e.target as HTMLInputElement).value as TipoStudente)}
/>
Magistrale Magistrale
</label> </button>
</div>
</div> </div>
</div> </div>
{/* Counter CFU */} {/* Counter CFU */}
<div className={`cfu-counter ${cfuError ? 'error' : ''}`}> <div class={`cfu-counter wide ${cfuError ? 'error' : ''}`}>
<h3> <h3>
CFU Totali: {totaleCfu}/{maxCfu} CFU Totali: {totaleCfu}/{maxCfu}
{tipoStudente === 'triennale' ? ' (+9 tesi)' : ' (+27 tesi)'} {tipoStudente === 'triennale' ? ' (+9 tesi)' : ' (+27 tesi)'}
</h3> </h3>
{cfuError && <p className="error-text"> Hai superato il limite di CFU consentiti!</p>} {cfuError && <p class="error-text"> Hai superato il limite di CFU consentiti!</p>}
</div> </div>
<div className="main-content">
{/* Sezione selezione corsi */} {/* Sezione selezione corsi */}
<div className="course-selection"> <div class="card">
<div class="title">
<h2>Seleziona Materie</h2> <h2>Seleziona Materie</h2>
</div>
{Object.entries(gruppiCorsi).map(([categoria, corsi]) => ( {Object.entries(gruppiCorsi).map(([categoria, corsi]) => (
<div key={categoria} className="course-category"> <div key={categoria} class="course-category">
<button className="category-header" onClick={() => toggleSezione(categoria)}> <button onClick={() => toggleSezione(categoria)}>
<h3>{categoria}</h3> <div class="h-flex">
<span className={`toggle-icon ${sezioniAperte[categoria] ? 'expanded' : ''}`}></span> {categoria}
<div class="spacer"></div>
<span class={`toggle-icon ${sezioniAperte[categoria] ? 'expanded' : ''}`}></span>
</div>
</button> </button>
{sezioniAperte[categoria] && ( {sezioniAperte[categoria] && (
<div className="course-grid"> <div class="course-grid">
{corsi.map((corso, index) => ( {corsi.map((corso, index) => (
<button <button
key={index} key={index}
className="course-button" class="course-button"
onClick={() => aggiungiCorso(corso)} onClick={() => aggiungiCorso(corso)}
disabled={corsiSelezionati.some(c => c.nome === corso.nome)} disabled={corsiSelezionati.some(c => c.nome === corso.nome)}
> >
<span className="course-name">{corso.nome}</span> <span class="course-name">{corso.nome}</span>
<span className="course-cfu">{corso.cfu} CFU</span> <span class="course-cfu">{corso.cfu} CFU</span>
</button> </button>
))} ))}
</div> </div>
@ -567,14 +567,12 @@ function MediaPesataApp() {
))} ))}
{/* Form per materia custom */} {/* Form per materia custom */}
<div className="custom-course"> <div class="custom-course">
<h3>Materia Personalizzata</h3> <h3>Materia Personalizzata</h3>
{!showCustomForm ? ( {!showCustomForm ? (
<button onClick={() => setShowCustomForm(true)} className="add-custom-btn"> <button onClick={() => setShowCustomForm(true)}>+ Aggiungi Materia Personalizzata</button>
+ Aggiungi Materia Personalizzata
</button>
) : ( ) : (
<div className="custom-form"> <div class="custom-form">
<input <input
type="text" type="text"
placeholder="Nome materia" placeholder="Nome materia"
@ -597,42 +595,34 @@ function MediaPesataApp() {
}) })
} }
/> />
<button onClick={aggiungiCorsoCustom} className="confirm-btn"> <button onClick={aggiungiCorsoCustom}>Aggiungi</button>
Aggiungi <button onClick={() => setShowCustomForm(false)}>Annulla</button>
</button>
<button onClick={() => setShowCustomForm(false)} className="cancel-btn">
Annulla
</button>
</div> </div>
)} )}
</div> </div>
</div> </div>
{/* Sezione lista corsi selezionati */} {/* Sezione lista corsi selezionati */}
<div className="selected-courses"> <div class="card">
<div className="section-header"> <div class="h-flex">
<div class="title">
<h2>Materie Selezionate</h2> <h2>Materie Selezionate</h2>
{corsiSelezionati.length > 0 && ( </div>
<button onClick={resetTutto} className="reset-btn"> <div class="spacer"></div>
🗑 Cancella Tutto {corsiSelezionati.length > 0 && <button onClick={resetTutto}>🗑 Cancella Tutto</button>}
</button>
)}
</div> </div>
{corsiSelezionati.length === 0 ? ( {corsiSelezionati.length === 0 ? (
<p className="no-courses">Nessuna materia selezionata</p> <p>Nessuna materia selezionata</p>
) : ( ) : (
<div className="courses-list"> <div class="courses-list">
{corsiSelezionati.map(corso => ( {corsiSelezionati.map(corso => (
<div key={corso.id} className="course-item"> <div key={corso.id} class="course-item">
<div className="course-info"> <span class="course-name">{corso.nome}</span>
<span className="course-name">{corso.nome}</span> <span class="course-cfu">{corso.cfu} CFU</span>
<span className="course-cfu">{corso.cfu} CFU</span>
{corso.passFailOnly && <span className="pass-fail-badge">Pass/Fail</span>}
</div>
{!corso.passFailOnly && ( {!corso.passFailOnly && (
<div className="course-grade"> <div class="course-grade tall">
<input <input
type="number" type="number"
placeholder="Voto" placeholder="Voto"
@ -648,8 +638,9 @@ function MediaPesataApp() {
} }
/> />
<label className={`lode-checkbox ${corso.voto !== 30 ? 'disabled' : ''}`}> <label class={`lode-checkbox ${corso.voto !== 30 ? 'disabled' : ''}`}>
<input <input
class="star"
type="checkbox" type="checkbox"
checked={corso.lode} checked={corso.lode}
disabled={corso.voto !== 30} disabled={corso.voto !== 30}
@ -662,56 +653,55 @@ function MediaPesataApp() {
</div> </div>
)} )}
<div class="actions tall">
<button <button
class="icon remove-btn"
onClick={() => rimuoviCorso(corso.id)} onClick={() => rimuoviCorso(corso.id)}
className="remove-btn"
title="Rimuovi materia" title="Rimuovi materia"
> >
× ×
</button> </button>
</div> </div>
</div>
))} ))}
</div> </div>
)} )}
</div> </div>
</div>
{/* Pulsante Calcola */} {/* Pulsante Calcola */}
{corsiSelezionati.length > 0 && ( {corsiSelezionati.length > 0 && (
<div className="calculate-section"> <div class="calculate-section wide">
<button onClick={calcolaMedia} className="calculate-btn"> <button onClick={calcolaMedia}>🧮 Calcola Media e Voto di Laurea</button>
🧮 Calcola Media e Voto di Laurea
</button>
</div> </div>
)} )}
{/* Risultati */} {/* Risultati */}
{risultati && mostraRisultati && ( {risultati && mostraRisultati && (
<div className="results"> <div class="results wide">
<h2>Risultati</h2> <h2>Risultati</h2>
{risultati.errore ? ( {risultati.errore ? (
<div className="error-message"> <div class="error-message">
<p> {risultati.errore}</p> <p> {risultati.errore}</p>
</div> </div>
) : ( ) : (
<div className="results-grid"> <div class="results-grid">
<div className="result-item"> <div class="result-item">
<span className="label">Media Pesata:</span> <span class="label">Media Pesata:</span>
<span className="value">{risultati.mediaPesata}</span> <span class="value">{risultati.mediaPesata}</span>
</div> </div>
<div className="result-item"> <div class="result-item">
<span className="label">Bonus Lodi:</span> <span class="label">Bonus Lodi:</span>
<span className="value">+{risultati.bonusLodi}</span> <span class="value">+{risultati.bonusLodi}</span>
</div> </div>
<div className="result-item highlight"> <div class="result-item highlight">
<span className="label">Voto di Ammissione:</span> <span class="label">Voto di Ammissione:</span>
<span className="value">{risultati.votoAmmissione}</span> <span class="value">{risultati.votoAmmissione}</span>
</div> </div>
<div className="result-item highlight"> <div class="result-item highlight">
<span className="label">Massimo Voto Di Laurea Possibile:</span> <span class="label">Massimo Voto Di Laurea Possibile:</span>
<span className="value"> <span class="value">
{risultati.massimoVotoLaurea} {risultati.massimoVotoLaurea}
{risultati.conLode && <span className="lode-badge">(+lode)</span>} {risultati.conLode && <span class="lode-badge">(+lode)</span>}
</span> </span>
</div> </div>
</div> </div>
@ -720,9 +710,9 @@ function MediaPesataApp() {
)} )}
{/* Nota informativa */} {/* Nota informativa */}
<div className="info-section"> <div class="card wide">
<h3>📋 Come viene calcolata la media</h3> <h3>📋 Come viene calcolata la media</h3>
<div className="info-content"> <div class="info-content">
<h4>Regole di esclusione CFU:</h4> <h4>Regole di esclusione CFU:</h4>
<ul> <ul>
<li> <li>
@ -763,11 +753,11 @@ function MediaPesataApp() {
} }
// Funzione per inizializzare l'app // Funzione per inizializzare l'app
export function initMediaPesataApp() { // export function initMediaPesataApp() {
const container = document.getElementById('media-pesata-app') // const container = document.getElementById('media-pesata-app')
if (container) { // if (container) {
render(<MediaPesataApp />, container) // render(<MediaPesataApp />, container)
} // }
} // }
export default MediaPesataApp // export default MediaPesataApp

@ -1,42 +1,21 @@
--- ---
import '@/styles/pages/media-pesata.css' import '@/styles/pages/media-pesata.css'
import PageLayout from '../layouts/PageLayout.astro' import PageLayout from '../layouts/PageLayout.astro'
import { MediaPesataApp } from '@/client/MediaPesataApp'
--- ---
<PageLayout title="Voto Laurea" description="Calcola la tua media pesata e il voto di laurea seguendo le regole del dipartimento"> <PageLayout
title="Voto Laurea"
description="Calcola la tua media pesata e il voto di laurea seguendo le regole del dipartimento"
>
<div class="media-pesata-container"> <div class="media-pesata-container">
<h1>Calcolo Media e Voto di Laurea</h1> <h1>Calcolo Media e Voto di Laurea</h1>
<p>Calcola la tua media pesata e il voto con cui ti siederai alla discussione di laurea, seguendo le regole del dipartimento di Matematica.</p> <p>
Calcola la tua media pesata e il voto con cui ti siederai alla discussione di laurea, seguendo le regole del
dipartimento di Matematica.
</p>
<div id="media-pesata-app"></div> <MediaPesataApp client:load />
</div> </div>
</PageLayout> </PageLayout>
<script>
// Importiamo e inizializziamo l'applicazione
import('../client/MediaPesataApp')
.then(module => {
const { initMediaPesataApp } = module
initMediaPesataApp()
})
</script>
<style>
.media-pesata-container {
max-width: 1200px;
margin: 0 auto;
padding: 2rem;
}
h1 {
text-align: center;
margin-bottom: 1rem;
color: var(--color-primary);
}
p {
text-align: center;
margin-bottom: 2rem;
color: var(--color-text-secondary);
}
</style>

@ -75,7 +75,7 @@ Controls - for things like buttons, input, select
input[type='text'], input[type='text'],
input[type='password'] { input[type='password'] {
width: 100%; width: 100%;
height: 2.5rem; min-height: 1.75rem;
/* @include neo-brutalist-card; */ /* @include neo-brutalist-card; */
border: 3px solid #222; 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 { form {
display: grid; display: grid;
gap: 1rem; gap: 1rem;

@ -325,6 +325,38 @@ Custom Page Styles
opacity: 0 !important; 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) { @media screen and (min-width: 1024px) {
.mobile-only { .mobile-only {
display: none !important; display: none !important;

@ -1,10 +1,116 @@
@layer page { @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 { .media-pesata-app {
max-width: 1200px; max-width: 1200px;
margin: 0 auto; margin: 0 auto;
padding: 2rem; padding: 2rem;
min-height: 100vh; min-height: 100vh;
background: var(--homepage-whatsphc-bg);
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 { .media-pesata-title {
@ -24,9 +130,11 @@
box-shadow: 4px 4px 0 0 var(--palette-black); box-shadow: 4px 4px 0 0 var(--palette-black);
padding: 1.5rem; padding: 1.5rem;
margin-bottom: 2rem; margin-bottom: 2rem;
}
.student-type-selector h2 { display: grid;
place-content: center;
h2 {
margin-bottom: 1rem; margin-bottom: 1rem;
color: var(--palette-black); color: var(--palette-black);
font-family: var(--font-display); font-family: var(--font-display);
@ -34,14 +142,14 @@
font-size: 1.5rem; font-size: 1.5rem;
text-align: center; text-align: center;
} }
}
.radio-group { /* .radio-group {
display: flex; display: flex;
gap: 1.5rem; gap: 1.5rem;
justify-content: center; justify-content: center;
}
.radio-group label { label {
display: flex; display: flex;
align-items: center; align-items: center;
gap: 0.5rem; gap: 0.5rem;
@ -57,38 +165,86 @@
box-shadow: 2px 2px 0 0 var(--palette-black); box-shadow: 2px 2px 0 0 var(--palette-black);
} }
.radio-group label:hover { label:hover {
transform: translate(-1px, -1px); transform: translate(-1px, -1px);
box-shadow: 3px 3px 0 0 var(--palette-black); box-shadow: 3px 3px 0 0 var(--palette-black);
} }
.radio-group label:active { label:active {
transform: translate(1px, 1px); transform: translate(1px, 1px);
box-shadow: 1px 1px 0 0 var(--palette-black); box-shadow: 1px 1px 0 0 var(--palette-black);
} }
.radio-group input[type='radio'] { input[type='radio'] {
width: 18px; width: 18px;
height: 18px; height: 18px;
margin: 0; margin: 0;
accent-color: #007bff; 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 */ /* Counter CFU */
.cfu-counter { .cfu-counter {
background: var(--project-card-bg); background: hsl(from var(--media-pesata-accent-2) h s 75%);
border: var(--border-large); border: var(--border-large);
border-radius: 6px; border-radius: 6px;
box-shadow: 4px 4px 0 0 var(--palette-black); box-shadow: 4px 4px 0 0 var(--palette-black);
padding: 1rem; padding: 1rem;
margin-bottom: 2rem;
text-align: center; text-align: center;
}
.cfu-counter.error { &.error {
background: #fee; background: #ffd8d8;
border-color: #dc3545; border-color: #dc3545;
} }
}
.cfu-counter h3 { .cfu-counter h3 {
margin: 0; margin: 0;
@ -119,14 +275,6 @@
text-align: center; text-align: center;
} }
/* Layout principale */
.main-content {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 2rem;
margin-bottom: 2rem;
}
@media (max-width: 768px) { @media (max-width: 768px) {
.main-content { .main-content {
grid-template-columns: 1fr; grid-template-columns: 1fr;
@ -151,24 +299,51 @@
} }
.course-category { .course-category {
margin-bottom: 2rem; display: grid;
} gap: 0.5rem;
> button {
padding: 0.25rem 0.75rem 0.25rem 0.5rem;
font-size: 20px;
/* Rimozione del vecchio stile per h3 che causava il rettangolo blu */ place-content: stretch;
place-items: stretch;
}
.course-grid { .course-grid {
display: grid; display: grid;
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
gap: 0.5rem; gap: 0.5rem;
margin-top: 1rem;
padding-top: 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 { .course-button {
display: flex; display: grid;
flex-direction: column; grid-template-rows: auto auto;
align-items: flex-start; padding: 0.5rem;
padding: 0.75rem; gap: 0.25rem;
justify-content: stretch;
justify-items: start;
border: 2px solid var(--palette-black); border: 2px solid var(--palette-black);
border-radius: 4px; border-radius: 4px;
background: #f8f9fa; background: #f8f9fa;
@ -177,49 +352,44 @@
text-align: left; text-align: left;
font-family: var(--font-secondary); font-family: var(--font-secondary);
box-shadow: 2px 2px 0 0 var(--palette-black); box-shadow: 2px 2px 0 0 var(--palette-black);
}
.course-button:hover:not(:disabled) { &:hover:not(:disabled) {
background: var(--homepage-projects-bg); background: var(--homepage-projects-bg);
transform: translate(-1px, -1px); transform: translate(-1px, -1px);
box-shadow: 3px 3px 0 0 var(--palette-black); box-shadow: 3px 3px 0 0 var(--palette-black);
} }
.course-button:active:not(:disabled) { &:active:not(:disabled) {
transform: translate(1px, 1px); transform: translate(1px, 1px);
box-shadow: 1px 1px 0 0 var(--palette-black); box-shadow: 1px 1px 0 0 var(--palette-black);
} }
.course-button:disabled { &:disabled {
opacity: 0.5; opacity: 0.5;
cursor: not-allowed; cursor: not-allowed;
background: #e0e0e0; background: #e0e0e0;
} }
.course-button.selected { > .course-name {
background: var(--guide-base);
transform: translate(1px, 1px);
box-shadow: 1px 1px 0 0 var(--palette-black);
}
.course-name {
font-weight: 600; font-weight: 600;
font-size: 0.9rem; font-size: 0.9rem;
line-height: 1.3; line-height: 1.3;
margin-bottom: 0.25rem;
color: var(--palette-black); color: var(--palette-black);
} }
.course-cfu { > .course-cfu {
font-size: 0.8rem; font-size: 0.8rem;
color: #666; color: #666;
font-weight: bold; font-weight: bold;
font-family: var(--font-mono); /* font-family: var(--font-mono); */
}
}
}
} }
/* Materia personalizzata */ /* Materia personalizzata */
.custom-course { .custom-course {
margin-top: 2rem; margin-top: 1rem;
padding-top: 1.5rem; padding-top: 1.5rem;
border-top: 3px solid var(--palette-black); border-top: 3px solid var(--palette-black);
} }
@ -252,30 +422,32 @@
box-shadow: 5px 5px 0 0 var(--palette-black); box-shadow: 5px 5px 0 0 var(--palette-black);
} }
.custom-form { input {
display: flex; padding: 0.25rem 0.5rem;
gap: 0.5rem; font-size: 14px;
flex-wrap: wrap;
margin-top: 1rem;
}
.custom-form input { border: 2px solid #222;
padding: 0.5rem;
border: 2px solid var(--palette-black);
border-radius: 4px; border-radius: 4px;
font-size: 0.9rem; box-shadow: 2px 2px 0 0 #222;
background: #fff; background: #fff;
font-family: var(--font-primary); font-family: var(--font-primary);
} }
.custom-form input[type='text'] { .custom-form {
display: flex;
gap: 0.5rem;
flex-wrap: wrap;
input[type='text'] {
flex: 2; flex: 2;
min-width: 200px; min-width: 200px;
} }
.custom-form input[type='number'] { input[type='number'] {
flex: 0 0 80px; flex: 0 0 80px;
} }
}
.confirm-btn, .confirm-btn,
.cancel-btn { .cancel-btn {
@ -351,61 +523,79 @@
.courses-list { .courses-list {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 1rem; 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 { .course-item {
display: flex; display: grid;
grid-template-rows: auto 1fr;
grid-template-columns: 1fr auto auto;
align-items: center; align-items: center;
gap: 1rem;
padding: 1rem; gap: 0.5rem 1rem;
background: var(--guide-base); padding: 0.5rem 1rem 0.5rem 0.5rem;
background: #fff;
border: 2px solid var(--palette-black); border: 2px solid var(--palette-black);
border-radius: 4px; border-radius: 4px;
transition: all 64ms linear; transition: all 64ms linear;
}
.course-item:hover {
transform: translate(-1px, -1px);
box-shadow: 2px 2px 0 0 var(--palette-black);
}
.course-info { > .course-name {
flex: 1; grid-column: 1 / 2;
display: flex; grid-row: 1 / 2;
flex-direction: column;
gap: 0.25rem;
}
.course-info .course-name {
font-weight: 600; font-weight: 600;
font-size: 0.9rem; font-size: 0.9rem;
color: var(--palette-black); color: var(--palette-black);
font-family: var(--font-secondary); font-family: var(--font-secondary);
} }
.course-info .course-cfu { > .course-cfu {
grid-column: 1 / 2;
grid-row: 2 / 3;
font-size: 0.8rem; font-size: 0.8rem;
font-weight: bold;
color: #666; color: #666;
font-family: var(--font-mono); /* font-family: var(--font-mono); */
}
> .tall {
grid-row: 1 / -1;
} }
.course-grade { .course-grade {
display: flex; display: grid;
align-items: center; grid-auto-flow: column;
gap: 0.75rem; place-content: center;
gap: 0.5rem;
} }
.course-grade input[type='number'] { > .actions {
width: 90px; grid-column: -2 / -1;
padding: 0.5rem;
border: 2px solid var(--palette-black);
border-radius: 4px;
font-size: 0.9rem;
text-align: center;
background: #fff;
font-weight: 600;
font-family: var(--font-primary);
} }
.lode-checkbox { .lode-checkbox {
@ -416,57 +606,67 @@
cursor: pointer; cursor: pointer;
font-weight: 600; font-weight: 600;
font-family: var(--font-secondary); font-family: var(--font-secondary);
}
.lode-checkbox.disabled { &.disabled {
opacity: 0.5; opacity: 0.5;
cursor: not-allowed; cursor: not-allowed;
} }
.lode-checkbox input[type='checkbox'] {
width: 16px;
height: 16px;
margin: 0;
} }
.remove-btn { .remove-btn {
background: #d63031; background: #d63031;
color: white; color: white;
border: 2px solid var(--palette-black); width: 1.75rem;
width: 30px; height: 1.75rem;
height: 30px; font-size: 20px;
border-radius: 50%; padding: 0;
aspect-ratio: 1;
cursor: pointer; margin-top: 4px;
font-size: 1.2rem;
display: flex; &:hover {
background: #b71c1c;
}
}
@media screen and (max-width: 1024px) {
justify-items: start;
align-items: center; align-items: center;
justify-content: center; gap: 0.5rem;
transition: all 64ms linear;
flex-shrink: 0; grid-template-columns: 1fr auto;
box-shadow: 2px 2px 0 0 var(--palette-black); 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;
} }
.remove-btn:hover { .course-cfu {
background: #b71c1c; display: grid;
transform: translate(-1px, -1px); justify-self: center;
box-shadow: 3px 3px 0 0 var(--palette-black); align-self: start;
grid-column: 2 / 3;
grid-row: 1 / 2;
} }
.remove-btn:active { .course-grade {
transform: translate(1px, 1px); grid-row: 2 / 3;
box-shadow: 1px 1px 0 0 var(--palette-black);
} }
/* Calcolo e risultati */ .actions {
.calculation-section { grid-column: 2 / 3;
background: var(--homepage-principal-bg); grid-row: 2 / 3;
border: var(--border-large);
border-radius: 6px; justify-self: end;
box-shadow: 4px 4px 0 0 var(--palette-black); }
padding: 2rem; }
text-align: center;
margin-top: 2rem;
} }
.calculate-button { .calculate-button {
@ -482,18 +682,18 @@
cursor: pointer; cursor: pointer;
transition: all 64ms linear; transition: all 64ms linear;
margin-bottom: 2rem; margin-bottom: 2rem;
}
.calculate-button:hover { &:hover {
background: #2b8b47; background: #2b8b47;
transform: translate(-1px, -1px); transform: translate(-1px, -1px);
box-shadow: 5px 5px 0 0 var(--palette-black); box-shadow: 5px 5px 0 0 var(--palette-black);
} }
.calculate-button:active { &:active {
transform: translate(2px, 2px); transform: translate(2px, 2px);
box-shadow: 2px 2px 0 0 var(--palette-black); box-shadow: 2px 2px 0 0 var(--palette-black);
} }
}
.results { .results {
background: #fff; background: #fff;
@ -502,9 +702,8 @@
box-shadow: 4px 4px 0 0 var(--palette-black); box-shadow: 4px 4px 0 0 var(--palette-black);
padding: 2rem; padding: 2rem;
margin-top: 1rem; margin-top: 1rem;
}
.results h2 { > h2 {
margin-bottom: 1.5rem; margin-bottom: 1.5rem;
color: var(--palette-black); color: var(--palette-black);
font-family: var(--font-display); font-family: var(--font-display);
@ -519,6 +718,11 @@
gap: 1rem; gap: 1rem;
} }
@media screen and (max-width: 1024px) {
padding: 1rem;
}
}
/* Ordine degli elementi: il massimo voto di laurea al centro */ /* Ordine degli elementi: il massimo voto di laurea al centro */
.result-item:nth-child(4) { .result-item:nth-child(4) {
order: -1; order: -1;
@ -533,7 +737,7 @@
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
padding: 1rem; padding: 1rem;
background: var(--guide-base); background: hsl(from var(--media-pesata-accent-3) h s 75%);
border: 2px solid var(--palette-black); border: 2px solid var(--palette-black);
border-radius: 4px; border-radius: 4px;
font-family: var(--font-secondary); font-family: var(--font-secondary);
@ -542,7 +746,7 @@
} }
.result-item.highlight { .result-item.highlight {
background: var(--homepage-projects-bg); background: hsl(from var(--media-pesata-accent-2) h s 75%);
font-weight: bold; font-weight: bold;
font-size: 1.2rem; font-size: 1.2rem;
} }
@ -579,11 +783,7 @@
} }
/* Responsive */ /* Responsive */
@media (max-width: 768px) { /* @media (max-width: 768px) {
.media-pesata-app {
padding: 1rem;
}
.course-grid { .course-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
@ -629,17 +829,16 @@
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
/* Mantieni il massimo voto centrato anche su mobile */
.result-item:nth-child(4) { .result-item:nth-child(4) {
order: -1; order: -1;
grid-column: 1; grid-column: 1;
justify-self: center; justify-self: center;
max-width: 100%; max-width: 100%;
} }
} } */
/* Layout ultra-compatto per schermi molto piccoli */ /* Layout ultra-compatto per schermi molto piccoli */
@media (max-width: 480px) { /* @media (max-width: 480px) {
.course-item { .course-item {
padding: 0.5rem; padding: 0.5rem;
gap: 0.25rem; gap: 0.25rem;
@ -678,7 +877,7 @@
height: 26px; height: 26px;
font-size: 0.9rem; font-size: 0.9rem;
} }
} } */
.section-header { .section-header {
display: flex; display: flex;
@ -773,8 +972,31 @@
/* Calculate button */ /* Calculate button */
.calculate-section { .calculate-section {
text-align: center; display: grid;
margin: 2rem 0; 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 { .calculate-btn {
@ -855,3 +1077,18 @@
} }
} }
} }
@layer utility {
@media screen and (max-width: 1024px) {
.card {
> .h-flex {
flex-direction: column;
align-items: start;
> .spacer {
display: none;
}
}
}
}
}

Loading…
Cancel
Save