import 'dart:ui'; import 'package:aves/app_mode.dart'; import 'package:aves/model/settings/accessibility_animations.dart'; import 'package:aves/model/settings/screen_on.dart'; import 'package:aves/model/settings/settings.dart'; import 'package:aves/model/source/collection_source.dart'; import 'package:aves/model/source/media_store_source.dart'; import 'package:aves/services/common/services.dart'; import 'package:aves/theme/durations.dart'; import 'package:aves/theme/icons.dart'; import 'package:aves/theme/themes.dart'; import 'package:aves/utils/debouncer.dart'; import 'package:aves/widgets/common/behaviour/route_tracker.dart'; import 'package:aves/widgets/common/behaviour/routes.dart'; import 'package:aves/widgets/common/extensions/build_context.dart'; import 'package:aves/widgets/common/providers/highlight_info_provider.dart'; import 'package:aves/widgets/home_page.dart'; import 'package:aves/widgets/welcome_page.dart'; import 'package:equatable/equatable.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_gen/gen_l10n/app_localizations.dart'; import 'package:overlay_support/overlay_support.dart'; import 'package:provider/provider.dart'; import 'package:tuple/tuple.dart'; class AvesApp extends StatefulWidget { const AvesApp({Key? key}) : super(key: key); @override _AvesAppState createState() => _AvesAppState(); } class _AvesAppState extends State { final ValueNotifier appModeNotifier = ValueNotifier(AppMode.main); late Future _appSetup; final _mediaStoreSource = MediaStoreSource(); final Debouncer _mediaStoreChangeDebouncer = Debouncer(delay: Durations.contentChangeDebounceDelay); final Set changedUris = {}; // observers are not registered when using the same list object with different items // the list itself needs to be reassigned List _navigatorObservers = []; final EventChannel _mediaStoreChangeChannel = const EventChannel('deckers.thibault/aves/media_store_change'); final EventChannel _newIntentChannel = const EventChannel('deckers.thibault/aves/intent'); final EventChannel _errorChannel = const EventChannel('deckers.thibault/aves/error'); final GlobalKey _navigatorKey = GlobalKey(debugLabel: 'app-navigator'); Widget getFirstPage({Map? intentData}) => settings.hasAcceptedTerms ? HomePage(intentData: intentData) : const WelcomePage(); @override void initState() { super.initState(); EquatableConfig.stringify = true; _appSetup = _setup(); _mediaStoreChangeChannel.receiveBroadcastStream().listen((event) => _onMediaStoreChange(event as String?)); _newIntentChannel.receiveBroadcastStream().listen((event) => _onNewIntent(event as Map?)); _errorChannel.receiveBroadcastStream().listen((event) => _onError(event as String?)); } @override Widget build(BuildContext context) { // place the settings provider above `MaterialApp` // so it can be used during navigation transitions return ChangeNotifierProvider.value( value: settings, child: ListenableProvider>.value( value: appModeNotifier, child: Provider.value( value: _mediaStoreSource, child: DurationsProvider( child: HighlightInfoProvider( child: OverlaySupport( child: FutureBuilder( future: _appSetup, builder: (context, snapshot) { final initialized = !snapshot.hasError && snapshot.connectionState == ConnectionState.done; final home = initialized ? getFirstPage() : Scaffold( body: snapshot.hasError ? _buildError(snapshot.error!) : const SizedBox(), ); return Selector>( selector: (context, s) => Tuple2(s.locale, s.initialized ? s.accessibilityAnimations.enabled : true), builder: (context, s, child) { final settingsLocale = s.item1; final areAnimationsEnabled = s.item2; return MaterialApp( navigatorKey: _navigatorKey, home: home, navigatorObservers: _navigatorObservers, builder: (context, child) { if (!areAnimationsEnabled) { child = Theme( data: Theme.of(context).copyWith( // strip page transitions used by `MaterialPageRoute` pageTransitionsTheme: DirectPageTransitionsTheme(), ), child: child!, ); } return child!; }, onGenerateTitle: (context) => context.l10n.appName, darkTheme: Themes.darkTheme, themeMode: ThemeMode.dark, locale: settingsLocale, localizationsDelegates: const [ ...AppLocalizations.localizationsDelegates, ], supportedLocales: AppLocalizations.supportedLocales, // checkerboardRasterCacheImages: true, // checkerboardOffscreenLayers: true, ); }, ); }, ), ), ), ), ), ), ); } Widget _buildError(Object error) { return Container( alignment: Alignment.center, padding: const EdgeInsets.all(16), child: Column( mainAxisSize: MainAxisSize.min, children: [ const Icon(AIcons.error), const SizedBox(height: 16), Text(error.toString()), ], ), ); } Future _setup() async { await settings.init( isRotationLocked: await windowService.isRotationLocked(), ); // keep screen on settings.updateStream.where((key) => key == Settings.keepScreenOnKey).listen( (_) => settings.keepScreenOn.apply(), ); settings.keepScreenOn.apply(); // error reporting await reportService.init(); settings.updateStream.where((key) => key == Settings.isErrorReportingEnabledKey).listen( (_) => reportService.setCollectionEnabled(settings.isErrorReportingEnabled), ); await reportService.setCollectionEnabled(settings.isErrorReportingEnabled); FlutterError.onError = reportService.recordFlutterError; final now = DateTime.now(); final hasPlayServices = await availability.hasPlayServices; await reportService.setCustomKeys({ 'build_mode': kReleaseMode ? 'release' : kProfileMode ? 'profile' : 'debug', 'has_play_services': hasPlayServices, 'locales': window.locales.join(', '), 'time_zone': '${now.timeZoneName} (${now.timeZoneOffset})', }); _navigatorObservers = [ ReportingRouteTracker(), ]; } void _onNewIntent(Map? intentData) { debugPrint('$runtimeType onNewIntent with intentData=$intentData'); // do not reset when relaunching the app if (appModeNotifier.value == AppMode.main && (intentData == null || intentData.isEmpty == true)) return; reportService.log('New intent'); _navigatorKey.currentState!.pushReplacement(DirectMaterialPageRoute( settings: const RouteSettings(name: HomePage.routeName), builder: (_) => getFirstPage(intentData: intentData), )); } void _onMediaStoreChange(String? uri) { if (uri != null) changedUris.add(uri); if (changedUris.isNotEmpty) { _mediaStoreChangeDebouncer(() async { final todo = changedUris.toSet(); changedUris.clear(); final tempUris = await _mediaStoreSource.refreshUris(todo); if (tempUris.isNotEmpty) { changedUris.addAll(tempUris); _onMediaStoreChange(null); } }); } } void _onError(String? error) => reportService.recordError(error, null); }