37 KiB
VotianLT – UI Theme & Gestaltungsrichtlinien
Gilt für alle Views unter
src/main/java/de/assecutor/votianlt/pages/view/Theme-Datei:src/main/frontend/themes/votian-modern/styles.cssStand: UI-Änderungen bis 23.03.2026 berücksichtigt (Landing-Hero-CTA/Demo-Button,ViewToolbar-Migrationen, Landing-/Dashboard-Updates, TabSheet-Dialoge, Message-/Statistics-Layouts)
1. Design-Prinzipien
- Konsistenz: Alle Views nutzen dieselben Klassen, Abstände und Farben.
- Kein Overflow: Kein View darf horizontal scrollen. Alle Container haben
box-sizing: border-box,max-width: 100%undmin-width: 0. - Kein externes Margin/Padding am View-Root: Views füllen ihren Container vollständig aus. Abstände zum Bildschirmrand entstehen ausschließlich durch den
view-container. - Lumo-Overrides: Vaadin-Lumo-Standardstile werden durch gezielte CSS-Regeln mit höherer Spezifizität überschrieben – nicht mit inline Styles in Java-Code lösen.
2. Farben & Custom Properties
Alle Farben sind als CSS-Custom-Properties auf html definiert und müssen statt hartcodierter Hex-Werte verwendet werden.
Hauptfarben
| Variable | Wert | Verwendung |
|---|---|---|
--app-accent |
#2563eb |
Primäraktionen, Links, Highlights |
--app-accent-strong |
#1d4ed8 |
Hover-Zustand von Primärbuttons |
--app-accent-soft |
rgba(37,99,235,0.12) |
Selektierter Zeilenhintergrund |
--app-success |
#059669 |
Erfolgsstatus, abgeschlossene Aufgaben |
--app-warning |
#d97706 |
Warnungen |
--app-danger |
#dc2626 |
Fehler, kritische Zustände |
Oberflächen & Hintergründe
| Variable | Wert | Verwendung |
|---|---|---|
--app-surface |
rgba(255,255,255,0.78) |
Karten, Panels (Glasmorphismus) |
--app-surface-solid |
#ffffff |
Volldeckende Flächen |
--app-surface-muted |
rgba(247,250,255,0.88) |
Gedämpfte Panels |
--app-shell-background |
Radial+Linear-Gradient | Seitenhintergrund (Body) |
--app-sidebar-background |
Linear-Gradient dunkelblau | Drawer/Sidebar |
Text
| Variable | Wert | Verwendung |
|---|---|---|
--app-text-strong |
#0f172a |
Überschriften, wichtige Labels |
--app-text |
#1e293b |
Fließtext |
--app-text-muted |
#64748b |
Sekundärtext, Beschreibungen |
Ränder
| Variable | Wert | Verwendung |
|---|---|---|
--app-border |
rgba(148,163,184,0.18) |
Subtile Trennlinien |
--app-border-strong |
rgba(148,163,184,0.28) |
Sichtbare Kartenränder |
--app-field-border |
#d6dde7 |
Formularfeld-Rand |
--app-field-border-hover |
#c6d0dd |
Formularfeld-Rand (Hover) |
Schatten
Alle globalen Schatten-Variablen sind auf none gesetzt – das Design ist bewusst schattenlos.
| Variable | Wert | Verwendung |
|---|---|---|
--app-shadow-sm |
none |
– |
--app-shadow-md |
none |
– |
--app-shadow-lg |
none |
– |
--app-shadow-xl |
none |
– |
Auch die Lumo-Schatten-Overrides sind none:
--lumo-box-shadow-xs: none;
--lumo-box-shadow-s: var(--app-shadow-sm); /* → none */
--lumo-box-shadow-m: var(--app-shadow-md); /* → none */
--lumo-box-shadow-l: var(--app-shadow-lg); /* → none */
Ausnahme: Buttons behalten eigene Schatten (siehe Abschnitt 24).
3. Typografie
Schriftart: Manrope (Google Fonts, Gewichte 400–800)
Fallbacks: Avenir Next, Segoe UI, sans-serif
Lumo-Überschreibungen:
--lumo-font-family: "Manrope", "Avenir Next", "Segoe UI", sans-serif;
--lumo-base-color: #f5f7fb;
Alle h1–h6 haben letter-spacing: -0.03em und color: var(--app-text-strong).
Responsive Schriftgrößen (clamp)
| Element | Wert |
|---|---|
.form-title |
clamp(1.55rem, 3vw, 2.2rem), font-weight 800 |
.section-title |
clamp(1.4rem, 3.4vw, 2rem), font-weight 800 |
.hero-panel-title |
clamp(2rem, 5vw, 3.4rem), font-weight 800 |
.dashboard-title |
clamp(1.7rem, 3.2vw, 2.5rem), font-weight 800 |
.login-card-title |
clamp(1.8rem, 4vw, 2.4rem), font-weight 800 |
Regel: Überschriften mit font-weight: 800 und letter-spacing: -0.05em bis -0.06em für Großtitel.
4. Layout-Struktur (App Shell)
vaadin-app-layout (336px Drawer)
├── [part=drawer] ← Sidebar (Dunkelblau-Gradient)
│ ├── .app-drawer-header
│ ├── .app-drawer-scroll (Scroller, flex: 1)
│ │ └── .app-nav-container
│ │ └── .app-nav-tree (TreeGrid)
│ └── .app-user-menu
└── [Light DOM default slot]
└── .view-container ← Routed View Container
└── <geroutete View>
view-container
Der Container, in dem alle gerouteten Views dargestellt werden:
.view-container {
display: flex;
flex-direction: column;
background: transparent;
margin: 20px; /* Abstand zum Browserrand */
width: calc(100% - 40px);
height: calc(100dvh - 40px);
overflow: hidden;
border-radius: 12px;
box-sizing: border-box;
}
Wichtig: Views, die in .view-container landen, erhalten automatisch:
flex: 1 1 0; min-height: 0; min-width: 0; max-width: 100%; box-sizing: border-box;
5. View-Klassen
Jede View muss genau eine der folgenden Root-Klassen tragen:
| Klasse | Verwendung | Extends |
|---|---|---|
data-view |
Listen, Tabellen, Grids | Main oder VerticalLayout |
form-page |
Formulare mit zentriertem Inhalt | VerticalLayout |
admin-form-view |
Admin-Formulare | VerticalLayout |
message-hub-view |
Nachrichten-Übersicht | Main |
statistics-chat-view |
Statistik/Chat | VerticalLayout |
dashboard-view |
Dashboard-Seiten | VerticalLayout |
dashboard-home-view |
Haupt-Dashboard (transparenter Hintergrund) | VerticalLayout |
landing-view |
Startseite (Shell-Gradient im Body, weiße Inhalts-Panels) | Main |
login-view |
Login-Seite (Shell-Gradient, einspaltig, fullscreen) | Main |
Kritische Java-Regeln
// ✅ Richtig für Daten-Views:
addClassName("data-view");
// KEIN setJustifyContentMode(CENTER) — wird per CSS auf START erzwungen
// ✅ Richtig für Formular-Views (zentrierter Inhalt):
addClassName("form-page");
setJustifyContentMode(JustifyContentMode.CENTER);
setDefaultHorizontalComponentAlignment(Alignment.CENTER);
// ❌ Falsch: setPadding(true) auf View-Root oder form-shell ohne form-card
// → wird per CSS auf 0 zurückgesetzt, erzeugt keine visuelle Wirkung
6. Interne Layout-Klassen
form-shell – Vollbreite-Wrapper
VerticalLayout layout = new VerticalLayout();
layout.setPadding(false);
layout.setSpacing(true);
layout.setWidthFull();
layout.addClassName("form-shell");
Regeln:
- Kein
setPadding(true)→ hat keine Wirkung, da via CSS überschrieben - Kein horizontales Margin
box-sizing: border-box,max-width: 100%,min-width: 0
form-card – Schmales zentriertes Karten-Formular
container.addClassNames("form-shell", "narrow", "form-card");
container.setWidth("400px");
container.setMaxWidth("100%");
Bekommt weißen Hintergrund (0.9 Alpha), border-radius: 28px, backdrop-filter: blur(18px).
surface-panel – Datenpanel / Tabellenkarte
Div panel = new Div(grid);
panel.addClassNames("surface-panel", "data-grid-panel");
panel.setWidthFull();
.surface-panel {
border: 1px solid var(--app-border-strong);
background: var(--app-surface);
backdrop-filter: blur(18px);
box-shadow: var(--app-shadow-lg);
border-radius: 28px;
box-sizing: border-box;
max-width: 100%;
}
.data-grid-panel {
padding: 0.8rem;
min-height: 420px;
box-sizing: border-box;
max-width: 100%;
}
7. ViewToolbar
Einheitliche Toolbar für alle Seiten-Überschriften:
add(new ViewToolbar(getTranslation("my.view.title")));
// Mit Aktionsbutton:
add(new ViewToolbar(getTranslation("my.view.title"), myButton));
// Mit gruppierten Sekundäraktionen:
add(new ViewToolbar(
getTranslation("my.view.title"),
ViewToolbar.group(backButton, exportButton),
addButton));
.view-toolbar {
width: 100%;
padding: 0.75rem 1rem;
box-sizing: border-box;
}
.view-toolbar-title-row {
gap: 0.8rem;
}
.view-toolbar-title {
color: var(--app-text-strong);
font-weight: 800;
letter-spacing: -0.05em;
}
.view-toolbar-actions,
.view-toolbar-group {
width: 100%;
justify-content: flex-end;
}
Regeln:
ViewToolbarist Standard für neue Header und ersetzt ad-hocHorizontalLayout+H2/H1-Konstrukte.- Bereits migriert:
EditProfileView,ImprintView,StatisticsView,UserMessagesView,MessageDetailsView. - Titel im Toolbar-Header bleiben einzeilig (
white-space: nowrapwird im Component-Code gesetzt). - Ausnahme: Für komplexe Header-Zeilen wie in
MessageDetailsViewdarf der Inhalt einerViewToolbarersetzt werden, die Basiskomponente bleibt aberViewToolbar.
8. Daten-Grids
Standard-Muster für alle Tabellen-Views:
// View-Root
setSizeFull();
addClassName("data-view");
// ViewToolbar
add(new ViewToolbar(getTranslation("title"), addButton));
// Grid
Grid<T> grid = new Grid<>(T.class, false);
grid.setWidthFull();
grid.setHeightFull();
grid.addClassName("data-grid");
// Panel-Wrapper
Div panel = new Div(grid);
panel.addClassNames("surface-panel", "data-grid-panel");
panel.setWidthFull();
add(panel);
Grid-Styling:
- Hintergrund:
rgba(255,255,255,0.88),border-radius: 24px - Header: Gedämpftes Grau, Uppercase,
font-weight: 800,font-size: 0.76rem - Selektierte Zeile:
--app-accent-softHintergrund - Auch eingebettete Grids in Formular-/Summary-Bereichen werden so gekapselt; kein nacktes
Griddirekt in einen Content-Abschnitt hängen.
9. Formulare (Form-Views)
// View-Root
setSizeFull();
setJustifyContentMode(JustifyContentMode.CENTER);
setDefaultHorizontalComponentAlignment(Alignment.CENTER);
setPadding(true);
addClassName("form-page");
// Formular-Container
VerticalLayout container = new VerticalLayout();
container.setWidth("600px");
container.setMaxWidth("100%");
container.setPadding(true);
container.setSpacing(true);
container.addClassNames("form-shell", "form-card");
// Formular-Felder
FormLayout form = new FormLayout();
form.setWidthFull();
form.setResponsiveSteps(new FormLayout.ResponsiveStep("0", 2));
Formularfelder haben automatisch:
border-radius: 20pxbackground: #ffffffborder: 1px solid var(--app-field-border)- Hover: Blaue Highlight-Farbe
- Fokus: Blauer Ring
rgba(37,99,235,0.18)
10. Filter-Panels
HorizontalLayout filterBar = new HorizontalLayout();
filterBar.addClassName("filter-panel");
filterBar.setWidthFull();
.filter-panel {
padding: 1rem 1.15rem;
border-radius: 26px;
flex-wrap: wrap;
gap: 1rem;
min-width: 0;
max-width: 100%;
box-sizing: border-box;
}
Wichtig: Filter-Felder niemals mit fixen Pixel-Breiten versehen → flex-wrap: wrap und min-width: 0 sorgen für korrektes Responsive-Verhalten.
11. Dashboard Home View
Die dashboard-home-view hat einen transparenten Hintergrund, der den Shell-Gradient durchscheinen lässt.
.dashboard-home-view {
background: transparent !important;
}
.view-container:has(.dashboard-home-view) {
border-radius: 12px;
background: transparent;
overflow: auto;
}
body:has(.dashboard-home-view) {
background: var(--app-shell-background);
}
/* Surface-Panels und Hero-Panels im Dashboard: kein Border, kein Schatten */
.dashboard-home-view .surface-panel {
border: 0;
background: transparent;
backdrop-filter: none;
box-shadow: none;
}
.dashboard-home-view .hero-panel {
box-shadow: none;
}
Aktuelles Strukturmuster:
- Das eingeloggte Dashboard besteht aus Hero + Systemsektion; App-Overview und Footer bleiben auf der öffentlichen Landing Page.
- Dashboard-Feature-Karten dürfen vollständig klickbar sein, indem
feature-cardinRouterLink.feature-card-linkgewrappt wird.
RouterLink link = new RouterLink();
link.setRoute(ShowJobsView.class);
link.add(card);
link.addClassName("feature-card-link");
12. Sidebar / Navigation
Drawer Header (klickbar)
Der Drawer-Header (Logo + App-Name) ist klickbar und navigiert zum Dashboard:
header.getStyle().set("cursor", "pointer");
header.getElement().setProperty("title", "Zum Dashboard");
header.addClickListener(event -> UI.getCurrent().navigate("dashboard"));
// AdminLayout navigiert zu "admin-dashboard"
Nav-Rows
.app-nav-row {
border-radius: 18px;
border: 1px solid rgba(255,255,255,0.06);
background: rgba(255,255,255,0.08);
max-width: calc(100% - 4px);
transition: transform 0.18s ease, background-color 0.18s ease;
}
.app-nav-row:hover {
transform: translateX(4px);
background: rgba(255,255,255,0.14);
}
.app-nav-row-root {
width: 20.5rem;
}
.app-nav-row-management-child {
width: 18.25rem;
}
.app-nav-row-user-child {
width: 17rem;
}
Zusatzregeln:
- Root-Einträge und Kind-Einträge bekommen unterschiedliche Breitenklassen entsprechend ihrer Ebene im
TreeGrid. - Drawer-Labels werden im Renderer mit
white-space: nowrapversehen; Zeilenumbrüche in der Navigation sind nicht vorgesehen.
User Menu
.app-user-menu {
margin: auto 0.5rem 0.5rem;
padding: 0.45rem 0.65rem;
border-radius: 22px;
border: 1px solid rgba(255,255,255,0.1);
background: rgba(255,255,255,0.08);
backdrop-filter: blur(14px);
}
13. Karten-Hierarchie
| Klasse | Radius | Hintergrund | Einsatz |
|---|---|---|---|
surface-panel |
28px | rgba(255,255,255,0.78) + Blur | Hauptdatenpanels |
form-card |
28px | rgba(255,255,255,0.90) + Blur | Formular-Cards |
section-card, content-card |
26px | rgba(255,255,255,0.88) | Abschnittspanels |
simple-card |
26px | rgba(255,255,255,0.88) | Einfache Karten |
detail-card |
24px | rgba(255,255,255,0.88) | Detail-Informationen |
hero-panel |
34px | Dunkelblau-Gradient | Hero-Bereiche |
feature-card |
24px | Linear-Gradient weiß/blau | Feature-Kacheln, optional klickbar via feature-card-link |
message-card |
22px | Linear-Gradient weiß | Nachrichten |
station-tile |
24px | Linear-Gradient weiß | Station-Kacheln |
job-task-card |
22px | Linear-Gradient weiß | Aufgaben-Karten |
14. Station Tiles & Kacheln
Station Tile
.station-tile {
border-radius: 24px;
border: 1px solid var(--app-border-strong);
background: linear-gradient(180deg, rgba(255,255,255,0.98), rgba(244,248,255,0.9));
box-shadow: var(--app-shadow-sm);
transition: transform 0.2s ease, box-shadow 0.2s ease, border-color 0.2s ease;
}
.station-tile:hover {
transform: translateY(-3px);
box-shadow: var(--app-shadow-md);
border-color: rgba(37,99,235,0.24);
}
.station-tile.validated {
background: linear-gradient(180deg, rgba(236,253,245,0.98), rgba(209,250,229,0.88));
border-color: rgba(5,150,105,0.26);
}
Add Station Tile (gestrichelt)
.add-station-tile {
background: linear-gradient(180deg, rgba(241,245,249,0.9), rgba(248,250,252,0.98));
border-style: dashed;
}
Stations Grid
.stations-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));
gap: 1rem;
width: 100%;
}
15. Job Task Cards
.job-task-card {
display: flex;
align-items: center;
gap: 1rem;
padding: 1rem;
border: 1px solid var(--app-border-strong);
border-radius: 22px;
background: linear-gradient(180deg, rgba(255,255,255,0.98), rgba(246,249,255,0.9));
box-shadow: var(--app-shadow-sm);
transition: transform 0.2s ease, box-shadow 0.2s ease;
cursor: pointer;
}
.job-task-card:hover {
transform: translateY(-2px);
box-shadow: var(--app-shadow-md);
border-color: rgba(37,99,235,0.24);
}
.job-task-card.completed {
border-color: rgba(5,150,105,0.22);
background: linear-gradient(180deg, rgba(236,253,245,0.98), rgba(220,252,231,0.88));
}
Task Status Badges
.job-task-status.open {
background: rgba(220,38,38,0.12);
color: var(--lumo-error-text-color);
}
.job-task-status.completed {
background: rgba(5,150,105,0.12);
color: var(--lumo-success-text-color);
}
16. Dashboard Stat Cards
.dashboard-stat-card {
position: relative;
border-radius: 24px;
padding: 1.15rem;
border: 1px solid var(--app-border-strong);
background: linear-gradient(180deg, rgba(255,255,255,0.98), rgba(246,249,255,0.92));
box-shadow: var(--app-shadow-md);
}
.dashboard-stat-card::before {
content: "";
position: absolute;
inset: 0 auto 0 0;
width: 4px;
background: var(--dashboard-accent, var(--app-accent));
}
Akzent-Farben
.accent-blue { --dashboard-accent: #2563eb; }
.accent-green { --dashboard-accent: #059669; }
.accent-purple { --dashboard-accent: #7c3aed; }
.accent-orange { --dashboard-accent: #d97706; }
.accent-gray { --dashboard-accent: #64748b; }
.accent-red { --dashboard-accent: #dc2626; }
17. Message & Chat Components
Header & Thread Layout
Message-Views verwenden ViewToolbar als erstes sichtbares Header-Element. Eigene .message-thread-header-Layouts sollen für neue Implementierungen nicht mehr aufgebaut werden.
addClassName("message-hub-view");
VerticalLayout contentLayout = new VerticalLayout();
contentLayout.setPadding(true);
contentLayout.setSpacing(true);
contentLayout.setWidthFull();
contentLayout.setHeightFull();
contentLayout.addClassName("message-thread-layout");
contentLayout.add(new ViewToolbar(getTranslation("usermessages.title.with", clientName)));
.message-thread-layout {
width: 100%;
margin: 0;
padding: 0;
background: transparent;
border: none;
box-shadow: none;
}
.message-thread-scroller {
border: 1px solid var(--app-border-strong);
background: linear-gradient(180deg, rgba(249,250,251,0.95), rgba(243,244,246,0.88));
border-radius: 24px;
overflow-y: auto !important;
}
.message-thread {
padding: 1rem !important;
}
.message-thread-input {
border: 1px solid var(--app-border-strong);
background: rgba(255,255,255,0.82);
border-radius: 22px;
padding: 0.85rem 1rem;
}
Message Card
.message-card {
padding: 1rem;
border: 1px solid var(--app-border-strong);
border-radius: 22px;
background: linear-gradient(180deg, rgba(255,255,255,0.98), rgba(246,249,255,0.9));
box-shadow: var(--app-shadow-sm);
transition: transform 0.22s ease, box-shadow 0.22s ease;
}
.message-card:hover {
transform: translateY(-4px);
box-shadow: 0 24px 48px rgba(15,23,42,0.14);
}
Chat Bubbles
.message-bubble {
max-width: min(78%, 720px);
padding: 0.85rem 1rem;
border-radius: 22px;
box-shadow: var(--app-shadow-sm);
}
.message-bubble.client {
background: linear-gradient(135deg, rgba(255,255,255,0.98), rgba(241,245,249,0.92));
border: 1px solid rgba(148,163,184,0.25);
}
.message-bubble.server {
background: linear-gradient(135deg, rgba(191,219,254,0.8), rgba(167,243,208,0.74));
border: 1px solid rgba(59,130,246,0.22);
}
Chat AI/User Bubbles
.chat-bubble--user {
background: linear-gradient(135deg, #2563eb, #1d4ed8);
color: #eff6ff;
padding: 0.85rem 1rem;
border-radius: 24px;
}
.chat-bubble--ai {
background: rgba(255,255,255,0.94);
border: 1px solid var(--app-border-strong);
padding: 1rem;
border-radius: 24px;
}
18. Dialoge
vaadin-dialog-overlay::part(content),
vaadin-confirm-dialog-overlay::part(content) {
border-radius: 28px;
border: 1px solid var(--app-border-strong);
background: rgba(255,255,255,0.94);
backdrop-filter: blur(20px);
box-shadow: var(--app-shadow-xl);
}
Dialog Panels
| Klasse | Verwendung |
|---|---|
.dialog-form-panel |
Formulare im Dialog |
.dialog-content-panel |
Inhalte im Dialog |
.dialog-task-card |
Aufgaben-Karten im Dialog |
.dialog-cargo-card |
Fracht-Karten im Dialog |
.dialog-form-panel,
.dialog-content-panel,
.dialog-cargo-card {
backdrop-filter: blur(12px);
padding: 1rem;
border-radius: 24px;
}
.dialog-task-card {
transition: transform 0.18s ease, box-shadow 0.18s ease, border-color 0.18s ease;
}
Station Dialoge mit TabSheet und innerer Karte
PickupStationDialog und DeliveryStationDialog verwenden nicht mehr die Standard-Overlay-Karte, sondern ein transparentes Overlay mit eigener weißer Innenkarte.
getElement().setAttribute("theme", "no-inner-card");
TabSheet tabSheet = new TabSheet();
tabSheet.setWidthFull();
tabSheet.setSizeFull();
Div frame = new Div();
frame.getStyle().set("border", "10px solid transparent");
frame.getStyle().set("display", "flex");
frame.getStyle().set("flex-direction", "column");
frame.getStyle().set("flex", "1");
frame.setSizeFull();
Div whiteCard = new Div();
whiteCard.getStyle().set("background", "white");
whiteCard.getStyle().set("border-radius", "24px");
whiteCard.getStyle().set("flex", "1");
whiteCard.getStyle().set("overflow", "auto");
whiteCard.setSizeFull();
whiteCard.add(tabSheet);
frame.add(whiteCard);
add(frame);
vaadin-dialog-overlay[theme~="no-inner-card"]::part(content) {
border-radius: 0;
border: none;
background: none;
backdrop-filter: none;
box-shadow: none;
padding: 0;
margin: 0;
}
vaadin-tabsheet::part(content) {
border-radius: 0 0 24px 24px;
background: rgba(255,255,255,0.86);
}
[part="tabs-container"] {
background: white;
border-radius: 24px 24px 0 0;
border-bottom: 1px solid #e0e0e0;
}
Tab-Konventionen:
PickupStationDialog: Adresse, Termine, FrachtDeliveryStationDialog: Adresse, Aufgaben- Fehlerindikatoren sitzen direkt am jeweiligen
Tab
19. Detail Cards
.detail-card {
border: 1px solid var(--app-border-strong);
border-radius: 24px;
background: rgba(255,255,255,0.9);
box-shadow: var(--app-shadow-sm);
padding: 1rem;
}
.detail-card--accent {
border-color: rgba(37,99,235,0.22);
background: linear-gradient(180deg, rgba(219,234,254,0.62), rgba(239,246,255,0.9));
}
.detail-card--code,
.detail-card--comment {
font-family: ui-monospace, "SFMono-Regular", monospace;
}
20. Route & Summary Cards
.route-card,
.summary-card {
border: 1px solid var(--app-border-strong);
border-radius: 24px;
background: rgba(255,255,255,0.84);
box-shadow: var(--app-shadow-sm);
padding: 1rem;
}
.services-panel,
.notes-panel {
border: 1px solid var(--app-border-strong);
border-radius: 24px;
background: rgba(255,255,255,0.84);
box-shadow: var(--app-shadow-sm);
padding: 0.75rem 1rem 1rem;
}
21. Login Seite
Die Login-Seite verwendet ein einspaltiges, zentriertes Layout ohne Highlight-Panel.
.login-view {
padding-inline: 0;
background: var(--app-shell-background) !important;
min-height: 100vh;
min-height: 100dvh;
}
.login-shell {
width: min(750px, 100%);
display: grid;
grid-template-columns: 1fr;
gap: 1.5rem;
}
.login-card {
width: 100%;
padding: clamp(1.35rem, 3vw, 2rem);
border-radius: 32px;
border: 1px solid var(--app-border-strong);
background: #ffffff;
box-shadow: var(--app-shadow-lg);
}
.login-card vaadin-login-form-wrapper {
background: #ffffff !important;
}
Java-Aufbau (LoginView)
// Login-Card enthält direkt: flashBox, loginForm, 2FA-Felder, registerButton, versionSpan
// KEIN login-highlight Panel, KEIN H1-Titel (login-card-title entfernt)
loginLayout.add(flashBox, loginForm, twoFaField, verify2faButton, registerButton, versionSpan);
Div loginShell = new Div(loginLayout);
loginShell.addClassName("login-shell");
Eyebrow Chip (Landing Page)
.eyebrow-chip {
display: inline-flex;
align-items: center;
padding: 0.45rem 0.8rem;
border-radius: 999px;
background: rgba(255,255,255,0.14);
color: rgba(248,250,252,0.92);
font-size: 0.78rem;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
}
22. Landing Page
.landing-view {
padding: 20px;
gap: 20px;
background: transparent !important;
min-height: 100vh;
min-height: 100dvh;
}
body:has(.landing-view) {
background: var(--app-shell-background);
}
.landing-view .surface-panel {
border: 1px solid var(--app-border-strong);
background: transparent;
backdrop-filter: none;
box-shadow: none;
}
.landing-view .landing-header,
.landing-view .section-panel,
.landing-view .app-overview-panel,
.landing-view .footer-panel {
background: var(--app-surface-solid);
}
.landing-header {
display: flex;
align-items: center;
justify-content: space-between;
gap: 1rem;
flex-wrap: wrap;
}
.landing-logo {
font-size: clamp(1.6rem, 4vw, 2.2rem);
font-weight: 800;
letter-spacing: -0.05em;
}
.landing-nav-button,
.landing-language-button {
border-radius: 999px;
}
Anonymer Header
- Im anonymen Zustand enthält die Header-Navigation nur
Login,Registrierenund die Sprachwahl. - Öffentliche Primäraktionen gehören nicht in die Header-Leiste, sondern in das Hero-Panel.
- Der
Demo-Einstieg wird deshalb nicht als zusätzlicher Header-Button geführt.
private Component createAnonymousNavigation() {
HorizontalLayout navButtons = new HorizontalLayout();
navButtons.setSpacing(true);
navButtons.setDefaultVerticalComponentAlignment(FlexComponent.Alignment.CENTER);
navButtons.addClassName("landing-nav");
Button loginBtn = new Button(getTranslation("start.button.login"), event -> login());
loginBtn.addThemeVariants(ButtonVariant.LUMO_CONTRAST);
loginBtn.addClassName("landing-nav-button");
Button registerBtn = new Button(getTranslation("start.button.register"), event -> register());
registerBtn.addThemeVariants(ButtonVariant.LUMO_PRIMARY);
registerBtn.addClassName("landing-nav-button");
Button languageBtn = createLanguageSelector();
navButtons.add(loginBtn, registerBtn, languageBtn);
return navButtons;
}
App-Section vs. Footer
CTA-Text und Slogan liegen auf der Landing Page jetzt im app-overview-panel, nicht mehr im Footer.
.app-cta {
margin-top: 0.75rem;
color: var(--app-accent-strong);
font-weight: 700;
max-width: 820px;
}
.app-slogan {
color: var(--app-accent-strong);
font-style: italic;
}
Hero Panel
.hero-panel {
position: relative;
overflow: hidden;
min-height: 520px;
justify-content: center;
text-align: center;
border-radius: 34px;
border: 1px solid rgba(255,255,255,0.14);
background: linear-gradient(135deg, #081224 0%, #1d4ed8 54%, #0f766e 100%);
box-shadow: var(--app-shadow-xl);
}
.hero-panel-title {
color: #f8fafc;
font-size: clamp(2rem, 5vw, 3.4rem);
font-weight: 800;
letter-spacing: -0.06em;
}
.hero-panel-text {
color: rgba(226,232,240,0.92);
font-size: clamp(1rem, 2vw, 1.15rem);
}
.hero-actions {
gap: 1rem;
align-items: center;
}
.hero-action-group {
gap: 0.45rem;
align-items: center;
}
.hero-choice-hint {
max-width: 420px;
margin: 0;
color: rgba(226,232,240,0.82);
font-size: 0.95rem;
line-height: 1.55;
text-align: center;
}
Hero-CTA-Regeln
- Öffentliche CTAs werden in der Hero-Kachel vertikal gestapelt und nicht über Header und Hero verteilt.
- Reihenfolge auf der Landing Page: zuerst
Demo, danachJetzt kostenlos testen. - Beide CTA-Buttons verwenden dieselbe Primärdarstellung:
LUMO_PRIMARY,LUMO_LARGE,setWidthFull(), Pill-Optik. - Jeder CTA bekommt direkt darunter einen eigenen Erklärungstext.
- Keine kombinierten Erklärungstexte für mehrere Buttons verwenden; Hinweise immer 1:1 an den jeweiligen CTA binden.
Button demoButton = new Button(getTranslation("start.button.demo"), event -> loginDemo());
demoButton.addThemeVariants(ButtonVariant.LUMO_PRIMARY, ButtonVariant.LUMO_LARGE);
demoButton.addClassNames("hero-cta", "hero-demo-cta");
demoButton.setWidthFull();
Button ctaButton = new Button(getTranslation("cta.freetest"), event -> register());
ctaButton.addThemeVariants(ButtonVariant.LUMO_PRIMARY, ButtonVariant.LUMO_LARGE);
ctaButton.addClassName("hero-cta");
ctaButton.setWidthFull();
Paragraph demoHint = new Paragraph(getTranslation("start.hero.demo.hint"));
demoHint.addClassName("hero-choice-hint");
Paragraph trialHint = new Paragraph(getTranslation("start.hero.trial.hint"));
trialHint.addClassName("hero-choice-hint");
VerticalLayout demoAction = new VerticalLayout(demoButton, demoHint);
demoAction.addClassName("hero-action-group");
VerticalLayout trialAction = new VerticalLayout(ctaButton, trialHint);
trialAction.addClassName("hero-action-group");
VerticalLayout heroActions = new VerticalLayout();
heroActions.setPadding(false);
heroActions.setSpacing(false);
heroActions.setWidthFull();
heroActions.setMaxWidth("420px");
heroActions.setDefaultHorizontalComponentAlignment(FlexComponent.Alignment.CENTER);
heroActions.addClassName("hero-actions");
heroActions.add(demoAction, trialAction);
start.hero.demo.hint=Demo startet sofort mit vorbereiteten Beispieldaten.
start.hero.trial.hint="Jetzt kostenlos testen" erstellt Ihren eigenen Account für den kostenlosen Probemonat.
23. Timeline Components
.timeline-entry-card {
margin-bottom: 0.75rem;
padding: 1rem;
border: 1px solid var(--app-border-strong);
border-radius: 24px;
background: rgba(255,255,255,0.9);
box-shadow: var(--app-shadow-sm);
border-left: 4px solid var(--timeline-accent, var(--app-accent));
}
.timeline-reason {
font-weight: 700;
color: var(--app-text-strong);
}
.timeline-timestamp {
color: var(--app-text-muted);
font-size: 0.78rem;
}
24. Invoice Components
Invoice Generator
.invoice-generator-panel {
backdrop-filter: blur(16px);
border-radius: 24px;
padding: 1rem;
border: 1px solid var(--app-border-strong);
background: rgba(255,255,255,0.9);
}
.invoice-generator-template {
display: flex;
align-items: center;
gap: 0.75rem;
padding: 0.95rem 1rem;
border: 1px solid var(--app-border-strong);
border-radius: 18px;
background: linear-gradient(180deg, rgba(255,255,255,0.98), rgba(246,249,255,0.92));
cursor: grab;
transition: transform 0.2s ease, box-shadow 0.2s ease;
}
.invoice-generator-template:hover {
transform: translateY(-2px);
box-shadow: var(--app-shadow-sm);
}
.invoice-generator-canvas {
position: relative;
overflow: hidden;
border: 2px dashed rgba(148,163,184,0.5);
border-radius: 24px;
background:
linear-gradient(180deg, rgba(255,255,255,0.95), rgba(241,245,249,0.9)),
linear-gradient(90deg, rgba(148,163,184,0.08) 1px, transparent 1px),
linear-gradient(rgba(148,163,184,0.08) 1px, transparent 1px);
background-size: auto, 28px 28px, 28px 28px;
}
Price Table
.price-table {
width: 100%;
}
.price-row {
display: flex;
justify-content: space-between;
gap: 1rem;
padding: 0.35rem 0;
}
.price-row--strong .price-row-label,
.price-row--strong .price-row-value {
font-weight: 800;
color: var(--app-text-strong);
}
25. Buttons
vaadin-button {
font-weight: 700;
border-radius: var(--lumo-border-radius-m);
transition: transform 0.18s ease, box-shadow 0.18s ease, opacity 0.18s ease;
}
vaadin-button:not([disabled]):hover {
transform: translateY(-1px);
}
/* Primary Buttons – eigene Schatten (Ausnahme vom globalen none) */
vaadin-button[theme~="primary"] {
box-shadow: 0 8px 20px rgba(37,99,235,0.22);
}
vaadin-button[theme~="primary"]:not([disabled]):hover {
box-shadow: 0 12px 28px rgba(37,99,235,0.32);
}
/* Success Primary */
vaadin-button[theme~="primary"][theme~="success"] {
box-shadow: 0 8px 20px rgba(5,150,105,0.24);
}
vaadin-button[theme~="primary"][theme~="success"]:not([disabled]):hover {
box-shadow: 0 12px 28px rgba(5,150,105,0.34);
}
/* Error Primary */
vaadin-button[theme~="primary"][theme~="error"] {
box-shadow: 0 8px 20px rgba(220,38,38,0.24);
}
/* Tertiary Error Hover */
vaadin-button[theme~="tertiary"][theme~="error"]:not([disabled]):hover {
background: rgba(220,38,38,0.08);
transform: translateY(-1px);
}
/* Disabled */
vaadin-button[disabled] {
opacity: 0.46;
transform: none !important;
}
Button-Varianten-Konvention (Java)
| Rolle | Theme-Variante | Beispiel |
|---|---|---|
| Hauptaktion (Erstellen, Speichern, Anwenden) | LUMO_PRIMARY |
addButton, applyFilter |
| Sekundäraktion (Abbrechen, Zurück, Export, Paginierung) | LUMO_TERTIARY |
cancelButton, backButton, exportButton |
| Gefährliche Aktion (Löschen) | LUMO_ERROR |
deleteButton |
// ✅ Sekundäre Buttons immer mit LUMO_TERTIARY:
cancelButton.addThemeVariants(ButtonVariant.LUMO_TERTIARY);
backButton.addThemeVariants(ButtonVariant.LUMO_TERTIARY);
exportButton.addThemeVariants(ButtonVariant.LUMO_TERTIARY);
// ✅ Primäre Buttons mit LUMO_PRIMARY:
addButton.addThemeVariants(ButtonVariant.LUMO_PRIMARY);
Pill-Buttons (Nav, Landing): border-radius: 999px
26. Overflow & Box-Model Regeln
Pflicht bei allen Container-Elementen
component.getStyle()
.set("box-sizing", "border-box")
.set("max-width", "100%")
.set("min-width", "0");
Warum box-sizing: border-box kritisch ist
Ohne box-sizing: border-box gilt:
Gesamtbreite = width + padding-left + padding-right + border-left + border-right
surface-panelhatborder: 1px→ ohne box-sizing ist es100% + 2px→ Overflowdata-grid-panelhatpadding: 0.8rem→ ohne box-sizing ist es100% + 1.6rem→ Overflow
Warum min-width: 0 kritisch ist
In Flex-Containern verhindert die Default-Eigenschaft min-width: auto, dass Flex-Items kleiner werden als ihr Inhalt. min-width: 0 erlaubt das Schrumpfen unter die Inhaltsgröße.
27. Lumo-Override-Strategie
Padding-Override
/* Lumo: vaadin-vertical-layout[theme~="padding"] → Spezifizität 0,0,1,1 */
/* Override braucht mindestens 0,0,2,0: */
vaadin-vertical-layout.form-shell:not(.form-card) { padding: 0; }
Inline-Style-Override
setJustifyContentMode() setzt Inline-Styles → nur !important überschreibt:
vaadin-vertical-layout.data-view {
justify-content: flex-start !important;
align-items: stretch !important;
}
Regel: !important nur für Inline-Style-Overrides verwenden.
28. Responsive Verhalten
Breakpoints
| Breakpoint | Änderungen |
|---|---|
max-width: 720px |
view-container Margin reduziert auf 10px, border-radius auf 8px |
max-height: 820px |
Drawer-Header und Nav-Rows bekommen kompakteres Padding |
Hinweis: Login-Shell ist jetzt immer einspaltig – der alte 980px-Breakpoint entfällt.
Mobile View-Container
@media (max-width: 720px) {
.view-container {
margin: 10px;
width: calc(100% - 20px);
height: calc(100dvh - 20px);
border-radius: 8px;
}
}
Flex-Wrap-Pflicht
Alle horizontalen Layouts, die Felder oder Buttons enthalten, müssen flex-wrap: wrap haben:
layout.getStyle().set("flex-wrap", "wrap");
29. Photo Gallery
.job-photo-gallery {
max-width: 600px;
min-height: 500px;
height: 500px;
position: relative;
display: flex;
align-items: center;
justify-content: center;
padding: 1rem;
background: rgba(255,255,255,0.96);
border-radius: 24px;
}
.job-photo-counter {
position: absolute;
top: var(--lumo-space-s);
right: var(--lumo-space-s);
padding: var(--lumo-space-xs) var(--lumo-space-s);
border-radius: 999px;
background: rgba(15,23,42,0.72);
color: #f8fafc;
font-weight: 700;
}
.job-photo-nav {
position: absolute;
top: 50%;
transform: translateY(-50%);
border-radius: 999px;
background: rgba(255,255,255,0.9);
box-shadow: var(--app-shadow-sm);
}
30. Empty State
.empty-state-card {
text-align: center;
color: var(--app-text-muted);
padding: 1rem;
border: 1px dashed var(--app-border-strong);
border-radius: 24px;
background: rgba(248,250,252,0.74);
}
31. Inline Flash (Fehlermeldungen)
.inline-flash {
width: 100%;
padding: 0.95rem 1rem;
border-radius: 18px;
border: 1px solid rgba(220,38,38,0.22);
background: rgba(254,242,242,0.92);
color: var(--lumo-error-text-color);
box-sizing: border-box;
}
32. Checkliste für neue Views
Beim Erstellen einer neuen View folgende Punkte prüfen:
- Root-Element hat genau eine View-Klasse (
data-view,form-page, etc.) - Kein
setPadding(true)auf dem View-Root oderform-shell-Wrappern (ohneform-card) - Kein
setJustifyContentMode(CENTER)auf Daten-Views (data-view,admin-form-view) - Alle Container haben
max-width: 100%undbox-sizing: border-box - Alle Flex-Container mit wechselnden Inhalten haben
flex-wrap: wrap - Formularfelder haben keine fixen Pixel-Breiten (
setWidth("200px")verboten) ViewToolbarist das erste Kind-Element- Grid-Panel verwendet
surface-panel data-grid-panelmitsetWidthFull() - Schmale Formular-Container haben
setWidth("Xpx")undsetMaxWidth("100%") - Inline-Styles nur für dynamische Werte verwenden, alles andere per CSS-Klassen
- Sekundäre Buttons (
Abbrechen,Zurück,Export) habenLUMO_TERTIARY, Hauptaktionen habenLUMO_PRIMARY - Tabbed Dialoge verwenden das
no-inner-card-Pattern mit innerer weißer Karte statt zusätzlicher Overlay-Dekoration