import 'dart:async'; import 'package:bloc/bloc.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/widgets.dart'; import 'package:freezed_annotation/freezed_annotation.dart'; import 'package:go_router/go_router.dart'; import 'package:stream_transform/stream_transform.dart'; import '../../../account_manager/account_manager.dart'; import '../../init.dart'; import '../../layout/layout.dart'; import '../../settings/settings.dart'; import '../../tools/tools.dart'; import '../../veilid_processor/views/developer.dart'; part 'router_cubit.freezed.dart'; part 'router_cubit.g.dart'; part 'router_state.dart'; final _rootNavKey = GlobalKey(debugLabel: 'rootNavKey'); final _homeNavKey = GlobalKey(debugLabel: 'homeNavKey'); class RouterCubit extends Cubit { RouterCubit(AccountRepository accountRepository) : super(const RouterState( isInitialized: false, hasAnyAccount: false, hasActiveChat: false, )) { // Watch for changes that the router will care about Future.delayed(Duration.zero, () async { await eventualInitialized.future; emit(state.copyWith(isInitialized: true)); }); // Subscribe to repository streams _accountRepositorySubscription = accountRepository.stream.listen((event) { switch (event) { case AccountRepositoryChange.localAccounts: emit(state.copyWith( hasAnyAccount: accountRepository.getLocalAccounts().isNotEmpty)); break; case AccountRepositoryChange.userLogins: case AccountRepositoryChange.activeLocalAccount: break; } }); } void setHasActiveChat(bool active) { emit(state.copyWith(hasActiveChat: active)); } @override Future close() async { await _accountRepositorySubscription.cancel(); await super.close(); } /// Our application routes List get routes => [ GoRoute( path: '/', builder: (context, state) => const IndexPage(), ), ShellRoute( navigatorKey: _homeNavKey, builder: (context, state, child) => HomeShell(child: HomeAccountReadyShell(child: child)), routes: [ GoRoute( path: '/home', builder: (context, state) => const HomeAccountReadyMain(), ), GoRoute( path: '/home/chat', builder: (context, state) => const HomeAccountReadyChat(), ), ], ), GoRoute( path: '/new_account', builder: (context, state) => const NewAccountPage(), ), GoRoute( path: '/settings', builder: (context, state) => const SettingsPage(), ), GoRoute( path: '/developer', builder: (context, state) => const DeveloperPage(), ) ]; /// Redirects when our state changes String? redirect(BuildContext context, GoRouterState goRouterState) { // No matter where we are, if there's not switch (goRouterState.matchedLocation) { case '/': // Wait for initialization to complete if (!eventualInitialized.isCompleted) { return null; } return state.hasAnyAccount ? '/home' : '/new_account'; case '/new_account': return state.hasAnyAccount ? '/home' : null; case '/home': if (!state.hasAnyAccount) { return '/new_account'; } if (responsiveVisibility( context: context, tablet: false, tabletLandscape: false, desktop: false)) { if (state.hasActiveChat) { return '/home/chat'; } } return null; case '/home/chat': if (!state.hasAnyAccount) { return '/new_account'; } if (responsiveVisibility( context: context, tablet: false, tabletLandscape: false, desktop: false)) { if (!state.hasActiveChat) { return '/home'; } } else { return '/home'; } return null; case '/settings': return null; case '/developer': return null; default: return state.hasAnyAccount ? null : '/new_account'; } } /// Make a GoRouter instance that uses this cubit GoRouter router() { final r = _router; if (r != null) { return r; } return _router = GoRouter( navigatorKey: _rootNavKey, refreshListenable: StreamListenable(stream.startWith(state).distinct()), debugLogDiagnostics: kDebugMode, initialLocation: '/', routes: routes, redirect: redirect, ); } //////////////// late final StreamSubscription _accountRepositorySubscription; GoRouter? _router; }