main.dart 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. import 'dart:io';
  2. import 'package:device_info_plus/device_info_plus.dart';
  3. import 'package:easy_localization/easy_localization.dart';
  4. import 'package:flutter/foundation.dart';
  5. import 'package:flutter/material.dart';
  6. import 'package:flutter/services.dart';
  7. import 'package:flutter_displaymode/flutter_displaymode.dart';
  8. import 'package:hooks_riverpod/hooks_riverpod.dart';
  9. import 'package:timezone/data/latest.dart';
  10. import 'package:immich_mobile/constants/locales.dart';
  11. import 'package:immich_mobile/modules/backup/background_service/background.service.dart';
  12. import 'package:immich_mobile/modules/backup/models/backup_album.model.dart';
  13. import 'package:immich_mobile/modules/backup/models/duplicated_asset.model.dart';
  14. import 'package:immich_mobile/routing/router.dart';
  15. import 'package:immich_mobile/routing/tab_navigation_observer.dart';
  16. import 'package:immich_mobile/shared/cache/widgets_binding.dart';
  17. import 'package:immich_mobile/shared/models/album.dart';
  18. import 'package:immich_mobile/shared/models/android_device_asset.dart';
  19. import 'package:immich_mobile/shared/models/asset.dart';
  20. import 'package:immich_mobile/shared/models/etag.dart';
  21. import 'package:immich_mobile/shared/models/exif_info.dart';
  22. import 'package:immich_mobile/shared/models/ios_device_asset.dart';
  23. import 'package:immich_mobile/shared/models/logger_message.model.dart';
  24. import 'package:immich_mobile/shared/models/store.dart';
  25. import 'package:immich_mobile/shared/models/user.dart';
  26. import 'package:immich_mobile/shared/providers/app_state.provider.dart';
  27. import 'package:immich_mobile/shared/providers/db.provider.dart';
  28. import 'package:immich_mobile/shared/providers/release_info.provider.dart';
  29. import 'package:immich_mobile/shared/services/immich_logger.service.dart';
  30. import 'package:immich_mobile/shared/services/local_notification.service.dart';
  31. import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
  32. import 'package:immich_mobile/shared/views/version_announcement_overlay.dart';
  33. import 'package:immich_mobile/utils/http_ssl_cert_override.dart';
  34. import 'package:immich_mobile/utils/immich_app_theme.dart';
  35. import 'package:immich_mobile/utils/migration.dart';
  36. import 'package:isar/isar.dart';
  37. import 'package:logging/logging.dart';
  38. import 'package:path_provider/path_provider.dart';
  39. void main() async {
  40. ImmichWidgetsBinding();
  41. final db = await loadDb();
  42. await initApp();
  43. await migrateDatabaseIfNeeded(db);
  44. HttpOverrides.global = HttpSSLCertOverride();
  45. runApp(getMainWidget(db));
  46. }
  47. Future<void> initApp() async {
  48. await EasyLocalization.ensureInitialized();
  49. if (kReleaseMode && Platform.isAndroid) {
  50. try {
  51. await FlutterDisplayMode.setHighRefreshRate();
  52. debugPrint("Enabled high refresh mode");
  53. } catch (e) {
  54. debugPrint("Error setting high refresh rate: $e");
  55. }
  56. }
  57. // Initialize Immich Logger Service
  58. ImmichLogger();
  59. var log = Logger("ImmichErrorLogger");
  60. FlutterError.onError = (details) {
  61. FlutterError.presentError(details);
  62. log.severe(
  63. 'Catch all error: ${details.toString()} - ${details.exception} - ${details.library} - ${details.context} - ${details.stack}',
  64. details,
  65. details.stack,
  66. );
  67. };
  68. PlatformDispatcher.instance.onError = (error, stack) {
  69. log.severe('Catch all error: ${error.toString()} - $error', error, stack);
  70. return true;
  71. };
  72. initializeTimeZones();
  73. }
  74. Future<Isar> loadDb() async {
  75. final dir = await getApplicationDocumentsDirectory();
  76. Isar db = await Isar.open(
  77. [
  78. StoreValueSchema,
  79. ExifInfoSchema,
  80. AssetSchema,
  81. AlbumSchema,
  82. UserSchema,
  83. BackupAlbumSchema,
  84. DuplicatedAssetSchema,
  85. LoggerMessageSchema,
  86. ETagSchema,
  87. if (Platform.isAndroid) AndroidDeviceAssetSchema,
  88. if (Platform.isIOS) IOSDeviceAssetSchema,
  89. ],
  90. directory: dir.path,
  91. maxSizeMiB: 256,
  92. );
  93. Store.init(db);
  94. return db;
  95. }
  96. Widget getMainWidget(Isar db) {
  97. return EasyLocalization(
  98. supportedLocales: locales,
  99. path: translationsPath,
  100. useFallbackTranslations: true,
  101. fallbackLocale: locales.first,
  102. child: ProviderScope(
  103. overrides: [dbProvider.overrideWithValue(db)],
  104. child: const ImmichApp(),
  105. ),
  106. );
  107. }
  108. class ImmichApp extends ConsumerStatefulWidget {
  109. const ImmichApp({super.key});
  110. @override
  111. ImmichAppState createState() => ImmichAppState();
  112. }
  113. class ImmichAppState extends ConsumerState<ImmichApp>
  114. with WidgetsBindingObserver {
  115. @override
  116. void didChangeAppLifecycleState(AppLifecycleState state) {
  117. switch (state) {
  118. case AppLifecycleState.resumed:
  119. debugPrint("[APP STATE] resumed");
  120. ref.read(appStateProvider.notifier).handleAppResume();
  121. break;
  122. case AppLifecycleState.inactive:
  123. debugPrint("[APP STATE] inactive");
  124. ref.read(appStateProvider.notifier).handleAppInactivity();
  125. break;
  126. case AppLifecycleState.paused:
  127. debugPrint("[APP STATE] paused");
  128. ref.read(appStateProvider.notifier).handleAppPause();
  129. break;
  130. case AppLifecycleState.detached:
  131. debugPrint("[APP STATE] detached");
  132. ref.read(appStateProvider.notifier).handleAppDetached();
  133. break;
  134. case AppLifecycleState.hidden:
  135. debugPrint("[APP STATE] hidden");
  136. ref.read(appStateProvider.notifier).handleAppHidden();
  137. break;
  138. }
  139. }
  140. Future<void> initApp() async {
  141. WidgetsBinding.instance.addObserver(this);
  142. // Draw the app from edge to edge
  143. SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
  144. // Sets the navigation bar color
  145. SystemUiOverlayStyle overlayStyle = const SystemUiOverlayStyle(
  146. systemNavigationBarColor: Colors.transparent,
  147. );
  148. if (Platform.isAndroid) {
  149. // Android 8 does not support transparent app bars
  150. final info = await DeviceInfoPlugin().androidInfo;
  151. if (info.version.sdkInt <= 26) {
  152. overlayStyle =
  153. MediaQuery.of(context).platformBrightness == Brightness.light
  154. ? SystemUiOverlayStyle.light
  155. : SystemUiOverlayStyle.dark;
  156. }
  157. }
  158. SystemChrome.setSystemUIOverlayStyle(overlayStyle);
  159. await ref.read(localNotificationService).setup();
  160. }
  161. @override
  162. initState() {
  163. super.initState();
  164. initApp().then((_) => debugPrint("App Init Completed"));
  165. WidgetsBinding.instance.addPostFrameCallback((_) {
  166. // needs to be delayed so that EasyLocalization is working
  167. ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
  168. });
  169. }
  170. @override
  171. void dispose() {
  172. WidgetsBinding.instance.removeObserver(this);
  173. super.dispose();
  174. }
  175. @override
  176. Widget build(BuildContext context) {
  177. var router = ref.watch(appRouterProvider);
  178. ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
  179. return MaterialApp(
  180. localizationsDelegates: context.localizationDelegates,
  181. supportedLocales: context.supportedLocales,
  182. locale: context.locale,
  183. debugShowCheckedModeBanner: false,
  184. home: Stack(
  185. children: [
  186. MaterialApp.router(
  187. title: 'Immich',
  188. debugShowCheckedModeBanner: false,
  189. themeMode: ref.watch(immichThemeProvider),
  190. darkTheme: immichDarkTheme,
  191. theme: immichLightTheme,
  192. routeInformationParser: router.defaultRouteParser(),
  193. routerDelegate: router.delegate(
  194. navigatorObservers: () => [TabNavigationObserver(ref: ref)],
  195. ),
  196. ),
  197. const ImmichLoadingOverlay(),
  198. const VersionAnnouncementOverlay(),
  199. ],
  200. ),
  201. );
  202. }
  203. }