Files
Holzleitner-Lieferservice-App/lib/feature/authentication/presentation/login_enforcer.dart
Dennis Nemec 467f4b4ed2 feat(auth): Login-Timeout (10s) mit Hinweisbanner
Haengt der interaktive Login (Browser-Tab/Token-Exchange) bei Verbindungsabbruch/Issuer-Hang, bricht er nach 10s ab; LoginPage zeigt 'Einloggen nicht moeglich. Spaeter erneut versuchen.' (Unauthenticated.loginTimedOut).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-18 13:08:18 +02:00

65 lines
1.9 KiB
Dart

import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/authentication/presentation/login_page.dart';
import '../bloc/auth_bloc.dart';
import '../bloc/auth_state.dart';
/// Routet die App zwischen Bootstrap-Splash, Login-Page und der
/// eigentlichen UI:
/// * `AuthBootstrapping` → Splash (verhindert Login-Page-Flash beim
/// Cold-Start, während der Refresh-Token gegen Keycloak gegengeprüft
/// wird).
/// * `Authenticated` → `child` (= reguläre App).
/// * sonst → LoginPage; `sessionExpired`-Banner wenn ein Refresh
/// serverseitig abgewiesen wurde.
class LoginEnforcer extends StatelessWidget {
final Widget child;
const LoginEnforcer({super.key, required this.child});
@override
Widget build(BuildContext context) {
return BlocBuilder<AuthBloc, AuthState>(
builder: (context, state) {
if (state is Authenticated) {
return child;
}
if (state is AuthBootstrapping) {
return const _AuthBootstrapSplash();
}
final unauth = state is Unauthenticated ? state : null;
return LoginPage(
sessionExpired: unauth?.sessionExpired ?? false,
loginTimedOut: unauth?.loginTimedOut ?? false,
);
},
);
}
}
class _AuthBootstrapSplash extends StatelessWidget {
const _AuthBootstrapSplash();
@override
Widget build(BuildContext context) {
return Scaffold(
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Padding(
padding: const EdgeInsets.symmetric(horizontal: 60),
child: Image.asset(
'assets/holzleitner_Logo_2017_RZ_transparent.png',
),
),
const SizedBox(height: 32),
const CircularProgressIndicator(),
],
),
),
);
}
}