main.dart 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. import 'dart:io';
  2. import 'package:easy_localization/easy_localization.dart';
  3. import 'package:flutter/foundation.dart';
  4. import 'package:flutter/material.dart';
  5. import 'package:flutter/services.dart';
  6. import 'package:flutter_displaymode/flutter_displaymode.dart';
  7. import 'package:hive_flutter/hive_flutter.dart';
  8. import 'package:hooks_riverpod/hooks_riverpod.dart';
  9. import 'package:immich_mobile/constants/locales.dart';
  10. import 'package:immich_mobile/modules/backup/background_service/background.service.dart';
  11. import 'package:immich_mobile/modules/backup/models/hive_backup_albums.model.dart';
  12. import 'package:immich_mobile/modules/backup/models/hive_duplicated_assets.model.dart';
  13. import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
  14. import 'package:immich_mobile/modules/login/models/hive_saved_login_info.model.dart';
  15. import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
  16. import 'package:immich_mobile/routing/router.dart';
  17. import 'package:immich_mobile/routing/tab_navigation_observer.dart';
  18. import 'package:immich_mobile/shared/providers/app_state.provider.dart';
  19. import 'package:immich_mobile/shared/providers/asset.provider.dart';
  20. import 'package:immich_mobile/shared/providers/release_info.provider.dart';
  21. import 'package:immich_mobile/shared/providers/server_info.provider.dart';
  22. import 'package:immich_mobile/shared/providers/websocket.provider.dart';
  23. import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
  24. import 'package:immich_mobile/shared/views/version_announcement_overlay.dart';
  25. import 'package:immich_mobile/utils/immich_app_theme.dart';
  26. import 'constants/hive_box.dart';
  27. void main() async {
  28. await Hive.initFlutter();
  29. Hive.registerAdapter(HiveSavedLoginInfoAdapter());
  30. Hive.registerAdapter(HiveBackupAlbumsAdapter());
  31. Hive.registerAdapter(HiveDuplicatedAssetsAdapter());
  32. await Future.wait([
  33. Hive.openBox(userInfoBox),
  34. Hive.openBox<HiveSavedLoginInfo>(hiveLoginInfoBox),
  35. Hive.openBox<HiveBackupAlbums>(hiveBackupInfoBox),
  36. Hive.openBox(hiveGithubReleaseInfoBox),
  37. Hive.openBox(userSettingInfoBox),
  38. Hive.openBox<HiveDuplicatedAssets>(duplicatedAssetsBox),
  39. EasyLocalization.ensureInitialized(),
  40. ]);
  41. SystemChrome.setSystemUIOverlayStyle(
  42. const SystemUiOverlayStyle(
  43. statusBarIconBrightness: Brightness.light,
  44. ),
  45. );
  46. if (kReleaseMode && Platform.isAndroid) {
  47. try {
  48. await FlutterDisplayMode.setHighRefreshRate();
  49. } catch (e) {
  50. debugPrint("Error setting high refresh rate: $e");
  51. }
  52. }
  53. runApp(
  54. EasyLocalization(
  55. supportedLocales: locales,
  56. path: translationsPath,
  57. useFallbackTranslations: true,
  58. fallbackLocale: locales.first,
  59. child: const ProviderScope(child: ImmichApp()),
  60. ),
  61. );
  62. }
  63. class ImmichApp extends ConsumerStatefulWidget {
  64. const ImmichApp({super.key});
  65. @override
  66. ImmichAppState createState() => ImmichAppState();
  67. }
  68. class ImmichAppState extends ConsumerState<ImmichApp>
  69. with WidgetsBindingObserver {
  70. @override
  71. void didChangeAppLifecycleState(AppLifecycleState state) {
  72. switch (state) {
  73. case AppLifecycleState.resumed:
  74. debugPrint("[APP STATE] resumed");
  75. ref.watch(appStateProvider.notifier).state = AppStateEnum.resumed;
  76. var isAuthenticated = ref.watch(authenticationProvider).isAuthenticated;
  77. if (isAuthenticated) {
  78. ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
  79. ref.watch(backupProvider.notifier).resumeBackup();
  80. ref.watch(assetProvider.notifier).getAllAsset();
  81. ref.watch(serverInfoProvider.notifier).getServerVersion();
  82. }
  83. ref.watch(websocketProvider.notifier).connect();
  84. ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
  85. break;
  86. case AppLifecycleState.inactive:
  87. debugPrint("[APP STATE] inactive");
  88. ref.watch(appStateProvider.notifier).state = AppStateEnum.inactive;
  89. ref.watch(websocketProvider.notifier).disconnect();
  90. ref.watch(backupProvider.notifier).cancelBackup();
  91. break;
  92. case AppLifecycleState.paused:
  93. debugPrint("[APP STATE] paused");
  94. ref.watch(appStateProvider.notifier).state = AppStateEnum.paused;
  95. break;
  96. case AppLifecycleState.detached:
  97. debugPrint("[APP STATE] detached");
  98. ref.watch(appStateProvider.notifier).state = AppStateEnum.detached;
  99. break;
  100. }
  101. }
  102. Future<void> initApp() async {
  103. WidgetsBinding.instance.addObserver(this);
  104. }
  105. @override
  106. initState() {
  107. super.initState();
  108. initApp().then((_) => debugPrint("App Init Completed"));
  109. WidgetsBinding.instance.addPostFrameCallback((_) {
  110. // needs to be delayed so that EasyLocalization is working
  111. ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
  112. });
  113. }
  114. @override
  115. void dispose() {
  116. WidgetsBinding.instance.removeObserver(this);
  117. super.dispose();
  118. }
  119. @override
  120. Widget build(BuildContext context) {
  121. var router = ref.watch(appRouterProvider);
  122. ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
  123. return MaterialApp(
  124. localizationsDelegates: context.localizationDelegates,
  125. supportedLocales: context.supportedLocales,
  126. locale: context.locale,
  127. debugShowCheckedModeBanner: false,
  128. home: Stack(
  129. children: [
  130. MaterialApp.router(
  131. title: 'Immich',
  132. debugShowCheckedModeBanner: false,
  133. themeMode: ref.watch(immichThemeProvider),
  134. darkTheme: immichDarkTheme,
  135. theme: immichLightTheme,
  136. routeInformationParser: router.defaultRouteParser(),
  137. routerDelegate: router.delegate(
  138. navigatorObservers: () => [TabNavigationObserver(ref: ref)],
  139. ),
  140. ),
  141. const ImmichLoadingOverlay(),
  142. const VersionAnnouncementOverlay(),
  143. ],
  144. ),
  145. );
  146. }
  147. }