import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:hl_lieferservice/bloc/app_bloc.dart'; import 'package:hl_lieferservice/data/cache/attachment_cache.dart'; import 'package:hl_lieferservice/data/network/keycloak_oidc_token_provider.dart'; import 'package:hl_lieferservice/feature/authentication/bloc/auth_bloc.dart'; import 'package:hl_lieferservice/feature/authentication/bloc/auth_event.dart'; import 'package:hl_lieferservice/feature/authentication/presentation/login_enforcer.dart'; import 'package:hl_lieferservice/main.dart' show locator; import 'package:hl_lieferservice/feature/car_selection/bloc/bloc.dart'; import 'package:hl_lieferservice/feature/car_selection/presentation/car_selection_enforcer.dart'; import 'package:hl_lieferservice/feature/car_selection/repository/car_selection_repository.dart'; import 'package:hl_lieferservice/data/repository/cars_repository_impl.dart'; import 'package:hl_lieferservice/feature/cars/bloc/cars_bloc.dart'; import 'package:hl_lieferservice/feature/cars/bloc/cars_state.dart'; import 'package:hl_lieferservice/feature/cars/presentation/car_management_page.dart'; import 'package:hl_lieferservice/data/repository/payment_methods_repository_impl.dart'; import 'package:hl_lieferservice/feature/payment_methods/bloc/payment_methods_cubit.dart'; import 'package:holzleitner_api/holzleitner_api.dart' show HolzleitnerApi; import 'package:hl_lieferservice/data/repository/tour_repository_impl.dart'; import 'package:hl_lieferservice/feature/delivery/bloc/phase_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/bloc/tour_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/bloc/tour_state.dart'; import 'package:hl_lieferservice/widget/home/bloc/navigation_bloc.dart'; import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart'; import 'package:hl_lieferservice/widget/operations/presentation/operation_view_enforcer.dart'; import 'package:hl_lieferservice/bloc/app_states.dart'; import 'home/presentation/home.dart'; class DeliveryApp extends StatefulWidget { const DeliveryApp({super.key}); @override State createState() => _DeliveryAppState(); } class _DeliveryAppState extends State { @override Widget build(BuildContext context) { return BlocBuilder( builder: (context, state) { if (state is AppConfigLoaded) { return MultiBlocProvider( providers: [ BlocProvider(create: (context) => NavigationBloc()), BlocProvider(create: (context) => OperationBloc()), BlocProvider( create: (context) => AuthBloc( tokenProvider: locator(), operationBloc: context.read(), ) // Beim ersten Build: prüfen, ob ein // Refresh-Token aus der Secure Storage da ist, // und ggf. direkt einloggen. ..add(const RestoreSessionRequested()), ), BlocProvider( // Phase-C+D-2-Migration: produktive TourRepository-Impl // gegen das generierte Rust-Backend-API. Account-Filter // serverseitig aus dem JWT, deshalb braucht der Bloc // keinen AuthBloc-Bezug mehr. create: (context) => TourBloc( tourRepository: TourRepositoryImpl(locator()), opBloc: context.read(), attachmentCache: locator(), ), ), BlocProvider( create: (context) => CarSelectBloc(repository: CarSelectionRepository()), ), BlocProvider( // Phase-D-Migration: produktive CarsRepository-Impl // gegen das generierte Rust-Backend-API. Account-Filter // serverseitig aus dem JWT, deshalb braucht der Bloc // keinen AuthBloc-Bezug mehr. create: (context) => CarsBloc( repository: CarsRepositoryImpl(locator()), opBloc: context.read(), ), ), BlocProvider( // PhaseBloc liest die Team-Fahrzeug-Anzahl jetzt direkt // aus dem CarsBloc — der ist die alleinige Quelle der // Fahrzeug-Stammdaten. Beim ersten Load eines Fahrzeugs // bestimmt das die Eintrittsphase (Auswählen vs. Sortieren). create: (context) => PhaseBloc( carCountResolver: () { final carsState = context.read().state; return carsState is CarsLoaded ? carsState.cars.length : null; }, // Bindet die persistierten Phasen-Häkchen an die aktuelle // Tour-Version (Tour.syncedAt). Ein erneuter Sync/Seed // schreibt eine neue syncedAt → neuer Token → frische // Phasen, ohne dass alte lokale Häkchen hängen bleiben. tourTokenResolver: () { final tourState = context.read().state; return tourState is TourLoaded ? tourState.details.tour.syncedAt .millisecondsSinceEpoch .toString() : null; }, ), ), BlocProvider( // Zahlungsmethoden sind firmenweite Stammdaten — wir laden // sie einmal beim App-Start und cachen sie im Cubit. Der // Detail-Screen einer Lieferung greift darauf zu, um den // `paymentMethodId`-FK auf einen lesbaren Namen aufzulösen. create: (context) => PaymentMethodsCubit( repository: PaymentMethodsRepositoryImpl(locator()), )..load(), ), ], child: MaterialApp( title: 'Holzleitner Auslieferung', // Wrap the Navigator (not just the home route) so the loading // overlay covers every pushed route — DeliveryDetail, Cars, // dialogs, etc. — not only the initial home tree. builder: (context, child) => OperationViewEnforcer(child: child ?? const SizedBox.shrink()), home: BlocBuilder( builder: (context, state) { if (state is AppConfigLoading) { return Scaffold( body: Center(child: CircularProgressIndicator()), ); } if (state is AppConfigLoadingFailed) { return Scaffold(body: Center(child: Text(state.message))); } if (state is AppConfigLoaded) { return LoginEnforcer( child: CarSelectionEnforcer(child: Home()), ); } return Container(); }, ), routes: {"/cars": (context) => CarManagementPage()}, ), ); } if (state is AppConfigLoadingFailed) { return MaterialApp( title: 'Holzleitner Auslieferung', home: Scaffold( body: Center(child: Text("Fehler beim Laden der Konfiguration")), ), ); } return MaterialApp( title: 'Holzleitner Auslieferung', home: Scaffold( body: Center(child: const CircularProgressIndicator()), ), ); }, ); } }