etm-powersync-app/lib/theme/etm_tokens.dart
Patrick Schurig ETM-Schurig d0a475a5d9 feat: refonte UI complète — design system EtmTokens + 4 écrans
Design system
- lib/theme/etm_tokens.dart : source de vérité couleurs + typo (IBM Plex Sans/Mono)
- lib/models/nymea_user.dart : modèle utilisateur nymea avec permissions EtmRole
- app_theme.dart : ThemeData migré vers IBM Plex Sans + couleurs EtmTokens

Navigation & drawer
- DrawerMenuButton : logo vert gradient + ombre
- Bottom nav : EtmTokens.green actif, EtmTokens.muted inactif
- DrawerPanel 320 px, restyled navy + gradient header + badges rôle

Dashboard (01_dashboard.html)
- Hero système : status pill + 3 métriques mono + illustration maison CustomPainter
- EnergyFlowWidget : 4 nœuds animés CustomPainter (flèches directionnelles)
  · gridPower > 0 = soutirage → flèche Grid→Home (amber)
  · gridPower < 0 = injection → flèche Home→Grid (bleu)
- EVChargingCard restyled : badge En charge + puissance mono 38px + 3 modes + SOC bar
- KPI 2×2 : spark bars, trend line, progress bar
- Consommateurs principaux + Décisions d'Héos (chips motifs)
- Prévisions placeholder explicite

Énergie
- KPI 2×2 avec icônes + fond soft + IBM Plex Mono
- Sélecteur période vert pill
- LineChart double axe : kW (gauche) / SOC % (droite, normalisé)
- BarChart bilan énergétique Wh (amber/bleu)
- Section Météo & prévision placeholder

Things
- Grille 2 col à hauteur intrinsèque (pas de childAspectRatio)
- Bandeau statut global (simulation / connecté / hors-ligne)
- _CategoryCard : header icon+label+count, séparateur coloré, liste tous items
- thing_category.dart : couleurs migrées vers EtmTokens

A/C — Climatisation / Chauffage
- Thermostats pièces EN HAUT : actives expandées, éteintes compactes
- Températures actuelle → cible ± avec EtmTokens.mono
- Sélecteur mode 4 boutons (Chauf/Clim/Auto/Vent)
- Chip "Chauffe au solaire en ce moment" (Héos)
- Sources pilotées par Héos EN BAS :
  · PAC SG-Ready : 4 états (Bloqué grisé / Normal / Recommandé / Forcé) + toggle Auto
  · Chauffe-eau : Surplus/Éco/Boost + temp eau 52°→60°C
  · Climatiseur : pré-refroidissement anticipé + info solaire

Packages ajoutés : google_fonts, flutter_staggered_grid_view, flutter_secure_storage
Asset : assets/house.svg (illustration maison CustomPainter)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-29 21:51:51 +02:00

86 lines
2.4 KiB
Dart

import 'package:flutter/material.dart';
import 'package:google_fonts/google_fonts.dart';
/// Tokens de marque ETM PowerSync.
/// Source de vérité unique pour les couleurs et la typographie.
class EtmTokens {
EtmTokens._();
// ---- Couleurs de marque ----
static const Color navy = Color(0xFF0D2B3B);
static const Color navy2 = Color(0xFF10384C);
static const Color blue = Color(0xFF31A3DD);
static const Color amber = Color(0xFFFEC113);
static const Color green = Color(0xFF1DB86A);
static const Color greenDark = Color(0xFF159C58);
static const Color orange = Color(0xFFE8923A);
static const Color danger = Color(0xFFE8423F);
// ---- Neutres ----
static const Color ink = navy;
static const Color muted = Color(0xFF6B7D88);
static const Color faint = Color(0xFF9AABB4);
static const Color line = Color(0xFFE6ECF0);
static const Color bg = Color(0xFFF3F6F8);
static const Color card = Colors.white;
// ---- Soft ----
static const Color greenSoft = Color(0xFFE7F8EF);
static const Color blueSoft = Color(0xFFE9F5FB);
static const Color amberSoft = Color(0xFFFFF6DD);
// ---- Rayons ----
static const double radius = 16;
static const double radiusLg = 22;
// ---- Typographie ----
static TextStyle sans({
double size = 14,
FontWeight weight = FontWeight.w400,
Color color = ink,
double? height,
double? letterSpacing,
}) =>
GoogleFonts.ibmPlexSans(
fontSize: size,
fontWeight: weight,
color: color,
height: height,
letterSpacing: letterSpacing,
);
/// IBM Plex Mono pour TOUS les chiffres (alignement tabulaire).
static TextStyle mono({
double size = 14,
FontWeight weight = FontWeight.w600,
Color color = ink,
}) =>
GoogleFonts.ibmPlexMono(
fontSize: size,
fontWeight: weight,
color: color,
fontFeatures: const [FontFeature.tabularFigures()],
);
static TextStyle sectionLabel() => sans(
size: 12,
weight: FontWeight.w600,
color: faint,
letterSpacing: 1.2,
);
/// Décoration shadow standard pour les cartes.
static List<BoxShadow> get cardShadow => [
BoxShadow(
color: navy.withValues(alpha: 0.04),
blurRadius: 2,
offset: const Offset(0, 1),
),
BoxShadow(
color: navy.withValues(alpha: 0.06),
blurRadius: 30,
offset: const Offset(0, 10),
),
];
}