feat(ui): implement full UI layer for all modules

Add interactive island components and server partials for notes,
students, and admin modules, following the Figma prototype design.

- static/styles/ui.css: shared component library (buttons, tables,
  chips, cards, filters, tabs, form inputs)
- notes: NotesView (student grade view with UE cards, promo tabs,
  weighted averages), AdminConsultNotes, AdminUEs islands + partials
- students: ConsultStudents (list/filter/delete), AdminPromotions
  (CRUD) islands + partials
- admin: AdminModules, AdminUsers, AdminRoles islands + partials
- All partials use State type with unknown cast for session access

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-26 22:54:10 +02:00
parent 34b7ac0231
commit 5ba8b8cb68
25 changed files with 2059 additions and 77 deletions
+393
View File
@@ -0,0 +1,393 @@
/* ui.css — Shared UI components for PolyMPR app pages */
/* -------------------------------------------------------
Page layout
------------------------------------------------------- */
.page-content {
padding: 1.5rem;
max-width: 960px;
}
.page-title {
font-size: 1.2rem;
font-weight: var(--font-weight-bold);
margin: 0 0 0.75rem 0;
padding-bottom: 0.75rem;
border-bottom: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
}
/* -------------------------------------------------------
Filters bar
------------------------------------------------------- */
.filters {
display: flex;
gap: 0.5rem;
align-items: center;
flex-wrap: wrap;
margin-bottom: 1.25rem;
}
.filter-input,
.filter-select {
padding: 0.3rem 0.5rem;
background: light-dark(white, #141228);
border: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
border-radius: 3px;
color: light-dark(var(--light-foreground), var(--dark-foreground));
font-size: 0.8rem;
font-family: inherit;
min-width: 8rem;
}
.filter-input:focus,
.filter-select:focus {
outline: none;
border-color: light-dark(
var(--light-accent-color),
var(--dark-accent-color)
);
}
/* -------------------------------------------------------
Buttons
------------------------------------------------------- */
.btn {
padding: 0.3rem 0.75rem;
border-radius: 3px;
font-size: 0.8rem;
font-family: inherit;
font-weight: var(--font-weight-bold);
cursor: pointer;
border: 1px solid;
text-decoration: none;
display: inline-flex;
align-items: center;
gap: 0.3rem;
line-height: 1.4;
background: transparent;
transition: background 100ms, color 100ms;
}
.btn::before {
all: unset;
}
.btn-primary {
border-color: light-dark(var(--light-accent-color), var(--dark-accent-color));
color: light-dark(var(--light-accent-color), var(--dark-accent-color));
}
.btn-primary:hover {
background: light-dark(var(--light-accent-color), var(--dark-accent-color));
color: light-dark(
var(--light-background-color),
var(--dark-background-color)
);
}
.btn-secondary {
border-color: light-dark(
var(--light-foreground-dimmer),
var(--dark-foreground-dimmer)
);
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
}
.btn-secondary:hover {
border-color: light-dark(
var(--light-foreground-dim),
var(--dark-foreground-dim)
);
color: light-dark(var(--light-foreground), var(--dark-foreground));
}
.btn-danger {
border-color: #933;
color: light-dark(var(--light-strong-color), var(--dark-strong-color));
}
.btn-danger:hover {
background: #933;
color: white;
}
.btn-sm {
padding: 0.15rem 0.5rem;
font-size: 0.75rem;
}
/* -------------------------------------------------------
Data table
------------------------------------------------------- */
.data-table-wrap {
background: light-dark(white, #141228);
border: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
border-radius: 4px;
overflow: auto;
}
.data-table {
width: 100%;
border-collapse: collapse;
font-size: 0.8rem;
}
.data-table th {
padding: 0.5rem 1rem;
font-size: 0.7rem;
font-weight: var(--font-weight-bold);
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
text-align: left;
border-bottom: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
}
.data-table td {
padding: 0.55rem 1rem;
border-bottom: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
}
.data-table tr:last-child td {
border-bottom: none;
}
.data-table tbody tr:nth-child(even) td {
background: light-dark(#f5f4ff, #141229);
}
.data-table .col-promo {
color: light-dark(var(--light-accent-color), var(--dark-accent-color));
font-weight: var(--font-weight-bold);
font-size: 0.75rem;
}
.data-table .col-dim {
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
font-size: 0.75rem;
}
.data-table .col-actions {
display: flex;
gap: 0.4rem;
align-items: center;
}
/* -------------------------------------------------------
UE card (student notes view)
------------------------------------------------------- */
.ue-card {
background: light-dark(white, #141228);
border: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
border-radius: 4px;
margin-bottom: 1rem;
overflow: hidden;
position: relative;
padding-left: 3px;
}
.ue-card::before {
content: "";
position: absolute;
left: 0;
top: 0;
bottom: 0;
width: 3px;
background: light-dark(var(--light-accent-color), var(--dark-accent-color));
transform: none;
transition: none;
border-radius: 0;
}
.ue-card-header {
padding: 0.65rem 1rem 0.5rem 1.1rem;
border-bottom: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
}
.ue-card-title {
font-weight: var(--font-weight-bold);
font-size: 0.85rem;
margin: 0;
}
.ue-card-avg {
font-size: 0.7rem;
margin: 0.2rem 0 0;
}
.ue-card-avg.avg-good {
color: light-dark(var(--light-accent-color), var(--dark-accent-color));
}
.ue-card-avg.avg-warn {
color: light-dark(var(--light-strong-color), var(--dark-strong-color));
}
.ue-module-row {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0.45rem 1rem 0.45rem 1.1rem;
border-bottom: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
}
.ue-module-row:last-child {
border-bottom: none;
}
.ue-module-name {
font-size: 0.8rem;
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
}
/* -------------------------------------------------------
Score chip
------------------------------------------------------- */
.score-chip {
display: inline-flex;
align-items: center;
justify-content: center;
min-width: 4.5rem;
padding: 0.15rem 0.5rem;
border-radius: 12px;
border: 1px solid;
font-size: 0.75rem;
font-weight: var(--font-weight-bold);
background: light-dark(white, #1a172d);
}
.score-chip.score-good {
border-color: light-dark(var(--light-accent-color), var(--dark-accent-color));
color: light-dark(var(--light-accent-color), var(--dark-accent-color));
}
.score-chip.score-warn {
border-color: light-dark(
var(--light-strong-color),
var(--dark-strong-color)
);
color: light-dark(var(--light-strong-color), var(--dark-strong-color));
}
.score-chip.score-none {
border-color: light-dark(
var(--light-foreground-dimmer),
var(--dark-foreground-dimmer)
);
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
}
/* -------------------------------------------------------
Tabs
------------------------------------------------------- */
.tabs {
display: flex;
gap: 0.6rem;
margin-bottom: 1.25rem;
flex-wrap: wrap;
}
.tab-btn {
padding: 0.35rem 0.9rem;
border: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
border-radius: 3px;
background: transparent;
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
font-size: 0.82rem;
font-family: inherit;
cursor: pointer;
}
.tab-btn::before {
all: unset;
}
.tab-btn.active {
border-color: light-dark(
var(--light-accent-color),
var(--dark-accent-color)
);
color: light-dark(var(--light-accent-color), var(--dark-accent-color));
font-weight: var(--font-weight-bold);
border-bottom-width: 2px;
}
/* -------------------------------------------------------
Status states
------------------------------------------------------- */
.state-loading,
.state-empty {
padding: 2.5rem;
text-align: center;
color: light-dark(var(--light-foreground-dim), var(--dark-foreground-dim));
font-size: 0.9rem;
}
.state-error {
padding: 1rem;
color: light-dark(var(--light-strong-color), var(--dark-strong-color));
font-size: 0.85rem;
border: 1px solid #933;
border-radius: 4px;
background: light-dark(#fff0f0, #1a1010);
margin-bottom: 1rem;
}
/* -------------------------------------------------------
Inline form row (for inline add/edit)
------------------------------------------------------- */
.form-row {
display: flex;
gap: 0.5rem;
align-items: center;
margin-bottom: 1rem;
flex-wrap: wrap;
}
.form-input {
padding: 0.35rem 0.5rem;
background: light-dark(white, #141228);
border: 1px solid
light-dark(var(--light-foreground-dimmer), var(--dark-foreground-dimmer));
border-radius: 3px;
color: light-dark(var(--light-foreground), var(--dark-foreground));
font-size: 0.82rem;
font-family: inherit;
min-width: 12rem;
}
.form-input:focus {
outline: none;
border-color: light-dark(
var(--light-accent-color),
var(--dark-accent-color)
);
}
/* -------------------------------------------------------
Toolbar (title + action button)
------------------------------------------------------- */
.toolbar {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 1rem;
gap: 1rem;
}