/** * Modul: Dashboard * Zweck: Styles für das Dashboard — Begrüßung, Widget-Grid, alle Widget-Typen, FAB-Speed-Dial * Abhängigkeiten: tokens.css, layout.css */ /* -------------------------------------------------------- * Dashboard-Layout * -------------------------------------------------------- */ .dashboard { padding: var(--space-4); padding-bottom: calc(var(--nav-height-mobile) + var(--safe-area-inset-bottom) + var(--space-16)); max-width: var(--content-max-width); margin: 0 auto; } @media (min-width: 1024px) { .dashboard { padding: var(--space-6) var(--space-8); padding-bottom: var(--space-12); } } @media (min-width: 1440px) { .dashboard { padding: var(--space-6) var(--space-10); } } /* -------------------------------------------------------- * Widget-Grid * * Mobile: 1 Spalte * Tablet: 2 Spalten * Desktop: 3 Spalten * Wide: 4 Spalten (ab 1440px) * -------------------------------------------------------- */ .dashboard__grid { display: grid; gap: var(--space-4); grid-template-columns: 1fr; } @media (min-width: 768px) { .dashboard__grid { grid-template-columns: repeat(2, 1fr); } .widget--wide { grid-column: 1 / -1; } } @media (min-width: 1024px) { .dashboard__grid { grid-template-columns: repeat(3, 1fr); gap: var(--space-5); } .widget--wide { grid-column: span 2; } } @media (min-width: 1440px) { .dashboard__grid { grid-template-columns: repeat(4, 1fr); } /* Wide-Widgets nehmen 2 von 4 Spalten ein */ .widget--wide { grid-column: span 2; } /* Greeting-Widget über alle 4 Spalten */ .widget-greeting { grid-column: 1 / -1; } } /* -------------------------------------------------------- * Begrüßungs-Widget * -------------------------------------------------------- */ .widget-greeting { background: linear-gradient(135deg, var(--color-accent) 0%, #6BA3FF 100%); border-radius: var(--radius-md); padding: var(--space-4) var(--space-5); color: #ffffff; grid-column: 1 / -1; } .widget-greeting__content { display: flex; flex-direction: column; gap: 2px; } .widget-greeting__title { font-size: var(--text-xl); font-weight: var(--font-weight-bold); } @media (min-width: 1024px) { .widget-greeting__title { font-size: var(--text-2xl); } } .widget-greeting__date { font-size: var(--text-sm); opacity: 0.85; } .widget-greeting__chips { display: flex; flex-wrap: wrap; gap: var(--space-2); margin-top: var(--space-2); } .greeting-chip { display: inline-flex; align-items: center; gap: 4px; background: rgba(255, 255, 255, 0.18); color: #ffffff; font-size: var(--text-xs); font-weight: var(--font-weight-medium); padding: 2px var(--space-2); border-radius: var(--radius-full); } .greeting-chip--warn { background: rgba(255, 59, 48, 0.35); } /* -------------------------------------------------------- * Basis-Widget (Card) * * Kompaktes Padding: 12px Header, 12px Body. * Internes Spacing: 8–12px (enger als vorher 16–24px). * -------------------------------------------------------- */ .widget { background-color: var(--color-surface); border-radius: var(--radius-md); box-shadow: var(--shadow-sm); overflow: hidden; display: flex; flex-direction: column; } .widget__header { display: flex; align-items: center; justify-content: space-between; padding: var(--space-3) var(--space-4) var(--space-2); } .widget__title { font-size: var(--text-sm); font-weight: var(--font-weight-semibold); color: var(--color-text-primary); display: flex; align-items: center; gap: var(--space-2); } .widget__title-icon { width: 16px; height: 16px; color: var(--color-accent); } .widget__link { font-size: var(--text-xs); color: var(--color-accent); font-weight: var(--font-weight-medium); display: inline-flex; align-items: center; gap: 2px; transition: opacity var(--transition-fast); } .widget__link:hover { opacity: 0.7; } .widget__badge { display: inline-flex; align-items: center; justify-content: center; background-color: var(--color-accent); color: #ffffff; font-size: var(--text-2xs); font-weight: var(--font-weight-bold); min-width: 16px; height: 16px; padding: 0 4px; border-radius: var(--radius-full); line-height: 1; } .widget__body { flex: 1; padding: var(--space-2) var(--space-4) var(--space-3); } .widget__empty { padding: var(--space-5) var(--space-4); text-align: center; color: var(--color-text-tertiary); font-size: var(--text-sm); display: flex; flex-direction: column; align-items: center; gap: var(--space-1); } .widget__empty .empty-state__icon { width: 28px; height: 28px; color: var(--color-text-disabled); margin-bottom: var(--space-1); } /* Widget hover lift (desktop) — dezent, max 1px */ @media (min-width: 1024px) { .widget { transition: transform var(--transition-fast), box-shadow var(--transition-fast); } .widget:hover { transform: translateY(-1px); box-shadow: var(--shadow-md); } } /* -------------------------------------------------------- * Aufgaben-Widget * Kompakte Items: 8px vertikales Padding, enger Spacing. * -------------------------------------------------------- */ .task-item { display: flex; align-items: flex-start; gap: var(--space-2); padding: var(--space-2) 0; border-bottom: 1px solid var(--color-border-subtle); cursor: pointer; transition: opacity var(--transition-fast); } .task-item:last-child { border-bottom: none; } .task-item:hover { opacity: 0.7; } .task-item__priority { width: 7px; height: 7px; border-radius: var(--radius-full); flex-shrink: 0; margin-top: 5px; } .task-item__priority--urgent { background-color: var(--color-priority-urgent); } .task-item__priority--high { background-color: var(--color-priority-high); } .task-item__priority--medium { background-color: var(--color-priority-medium); } .task-item__priority--low { background-color: var(--color-priority-low); } .task-item__content { flex: 1; min-width: 0; } .task-item__title { font-size: var(--text-sm); font-weight: var(--font-weight-medium); color: var(--color-text-primary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; line-height: var(--line-height-snug); } .task-item__meta { font-size: var(--text-xs); color: var(--color-text-tertiary); margin-top: 1px; } .task-item__meta--overdue { color: var(--color-danger); font-weight: var(--font-weight-medium); } .task-item__avatar { width: 22px; height: 22px; border-radius: var(--radius-full); display: flex; align-items: center; justify-content: center; font-size: 9px; font-weight: var(--font-weight-bold); color: #ffffff; flex-shrink: 0; margin-top: 1px; } /* -------------------------------------------------------- * Termine-Widget * Kompakt: Farbbalken + Inhalt, enge Abstände. * -------------------------------------------------------- */ .event-item { display: flex; align-items: stretch; gap: var(--space-2); padding: var(--space-2) 0; border-bottom: 1px solid var(--color-border-subtle); cursor: pointer; transition: opacity var(--transition-fast); } .event-item:last-child { border-bottom: none; } .event-item:hover { opacity: 0.7; } .event-item__bar { width: 3px; border-radius: var(--radius-full); flex-shrink: 0; background-color: var(--color-accent); } .event-item__content { flex: 1; min-width: 0; } .event-item__title { font-size: var(--text-sm); font-weight: var(--font-weight-medium); color: var(--color-text-primary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; line-height: var(--line-height-snug); } .event-item__time { font-size: var(--text-xs); color: var(--color-text-tertiary); margin-top: 1px; display: flex; align-items: center; gap: var(--space-1); } .event-time-badge { font-size: var(--text-2xs); font-weight: var(--font-weight-semibold); padding: 1px 5px; border-radius: var(--radius-full); background-color: var(--color-surface-3); color: var(--color-text-secondary); } .event-time-badge--today { background-color: var(--color-accent-light); color: var(--color-accent); } /* -------------------------------------------------------- * Essen-Widget (Slot-Grid) * -------------------------------------------------------- */ .widget--meals { overflow: visible; } .meal-slots { display: grid; grid-template-columns: repeat(4, 1fr); gap: 1px; background-color: var(--color-border); } .meal-slot { display: flex; flex-direction: column; align-items: center; justify-content: center; gap: 2px; padding: var(--space-2) var(--space-1); background-color: var(--color-surface); cursor: pointer; transition: background-color var(--transition-fast); text-align: center; min-height: 72px; } .meal-slot:hover { background-color: var(--color-surface-2); } .meal-slot:first-child { border-bottom-left-radius: var(--radius-md); } .meal-slot:last-child { border-bottom-right-radius: var(--radius-md); } .meal-slot__icon { width: 18px; height: 18px; color: var(--color-text-disabled); flex-shrink: 0; } .meal-slot--filled .meal-slot__icon { color: var(--color-accent); } .meal-slot__type { font-size: var(--text-2xs); font-weight: var(--font-weight-semibold); color: var(--color-text-disabled); text-transform: uppercase; letter-spacing: 0.4px; } .meal-slot--filled .meal-slot__type { color: var(--color-text-secondary); } .meal-slot__title { font-size: var(--text-xs); color: var(--color-text-secondary); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; max-width: 100%; padding: 0 var(--space-1); } .meal-slot--filled .meal-slot__title { color: var(--color-text-primary); font-weight: var(--font-weight-medium); } /* -------------------------------------------------------- * Notizen-Grid-Widget * -------------------------------------------------------- */ .notes-grid-widget { display: grid; grid-template-columns: repeat(2, 1fr); gap: var(--space-3); padding: var(--space-3) var(--space-4); } @media (min-width: 1024px) { .notes-grid-widget { grid-template-columns: repeat(3, 1fr); } } .note-item { border-radius: var(--radius-sm); padding: var(--space-3); cursor: pointer; transition: opacity var(--transition-fast), transform var(--transition-fast); border-left: 3px solid var(--note-color, var(--color-accent)); background-color: var(--color-surface-2); } .note-item:hover { opacity: 0.8; transform: translateY(-1px); } .note-item__title { font-size: var(--text-sm); font-weight: var(--font-weight-semibold); color: var(--color-text-primary); margin-bottom: 2px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } .note-item__content { font-size: var(--text-xs); color: var(--color-text-secondary); display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; } /* -------------------------------------------------------- * Wetter-Widget * * Eigener Gradient, überschreibt .widget Hintergrund. * Kompakt: Icon kleiner, Forecast enger. * -------------------------------------------------------- */ .weather-widget { background: linear-gradient(135deg, var(--color-accent) 0%, #1E5CB3 100%); color: #ffffff; } .weather-widget__main { display: flex; align-items: center; justify-content: space-between; padding: var(--space-3) var(--space-4); } .weather-widget__temp { font-size: var(--text-4xl); font-weight: var(--font-weight-bold); line-height: 1; margin-bottom: 2px; } .weather-widget__desc { font-size: var(--text-sm); font-weight: var(--font-weight-medium); text-transform: capitalize; margin-bottom: 1px; } .weather-widget__city { font-size: var(--text-xs); opacity: 0.85; margin-bottom: var(--space-1); } .weather-widget__meta { font-size: var(--text-xs); opacity: 0.7; line-height: 1.3; } .weather-widget__icon { width: 64px; height: 64px; flex-shrink: 0; filter: drop-shadow(0 2px 4px rgba(0, 0, 0, 0.15)); } .weather-forecast { display: flex; border-top: 1px solid rgba(255, 255, 255, 0.15); padding: var(--space-2) var(--space-4); gap: var(--space-3); } .weather-forecast__day { display: flex; flex-direction: column; align-items: center; gap: 2px; flex: 1; } .weather-forecast__label { font-size: var(--text-xs); font-weight: var(--font-weight-semibold); opacity: 0.85; text-transform: capitalize; } .weather-forecast__icon { width: 28px; height: 28px; } .weather-forecast__temps { display: flex; gap: var(--space-1); font-size: var(--text-xs); } .weather-forecast__high { font-weight: var(--font-weight-semibold); } .weather-forecast__low { opacity: 0.6; } /* -------------------------------------------------------- * Skeleton-Zustände (pro Widget) * Kompakt: gleiches Spacing wie echte Widgets. * -------------------------------------------------------- */ .widget-skeleton { background-color: var(--color-surface); border-radius: var(--radius-md); box-shadow: var(--shadow-sm); padding: var(--space-3) var(--space-4); } .skeleton-line { height: 12px; margin-bottom: var(--space-2); border-radius: var(--radius-xs); } .skeleton-line--short { width: 35%; } .skeleton-line--medium { width: 60%; } .skeleton-line--full { width: 100%; } /* -------------------------------------------------------- * FAB Speed-Dial * -------------------------------------------------------- */ .fab-container { position: fixed; bottom: calc(var(--nav-height-mobile) + var(--safe-area-inset-bottom) + var(--space-4)); right: var(--space-4); z-index: calc(var(--z-nav) - 1); display: flex; flex-direction: column-reverse; align-items: flex-end; gap: var(--space-3); } @media (min-width: 1024px) { .fab-container { bottom: var(--space-8); } } .fab-main { width: 52px; height: 52px; border-radius: var(--radius-full); background-color: var(--color-accent); color: #ffffff; box-shadow: var(--shadow-lg); display: flex; align-items: center; justify-content: center; cursor: pointer; transition: transform var(--transition-base), background-color var(--transition-fast); border: none; flex-shrink: 0; } @media (min-width: 1024px) { .fab-main { width: 48px; height: 48px; } } .fab-main:hover { background-color: var(--color-accent-hover); } .fab-main:active { transform: scale(0.95); } .fab-main--open { transform: rotate(45deg); background-color: var(--neutral-600); } .fab-actions { display: flex; flex-direction: column; align-items: flex-end; gap: var(--space-2); opacity: 0; pointer-events: none; transform: translateY(8px); transition: opacity var(--transition-base), transform var(--transition-base); } .fab-actions--visible { opacity: 1; pointer-events: auto; transform: translateY(0); } .fab-action { display: flex; align-items: center; gap: var(--space-3); cursor: pointer; } .fab-action__label { background-color: var(--color-surface); color: var(--color-text-primary); font-size: var(--text-sm); font-weight: var(--font-weight-medium); padding: var(--space-2) var(--space-3); border-radius: var(--radius-sm); box-shadow: var(--shadow-md); white-space: nowrap; } .fab-action__btn { width: 40px; height: 40px; border-radius: var(--radius-full); background-color: var(--color-surface); color: var(--color-accent); box-shadow: var(--shadow-md); display: flex; align-items: center; justify-content: center; border: none; cursor: pointer; flex-shrink: 0; transition: background-color var(--transition-fast); } .fab-action__btn:hover { background-color: var(--color-accent-light); }