2022-08-09 04:43:48 +00:00
|
|
|
import 'dart:io';
|
|
|
|
|
2022-07-07 18:40:54 +00:00
|
|
|
import 'package:easy_localization/easy_localization.dart';
|
2022-08-08 00:43:09 +00:00
|
|
|
import 'package:flutter/foundation.dart';
|
2022-02-03 16:06:44 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2023-02-03 16:26:05 +00:00
|
|
|
import 'package:flutter/services.dart';
|
2022-08-08 00:43:09 +00:00
|
|
|
import 'package:flutter_displaymode/flutter_displaymode.dart';
|
2022-02-03 16:06:44 +00:00
|
|
|
import 'package:hive_flutter/hive_flutter.dart';
|
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2022-08-18 14:41:59 +00:00
|
|
|
import 'package:immich_mobile/constants/locales.dart';
|
|
|
|
import 'package:immich_mobile/modules/backup/background_service/background.service.dart';
|
2023-03-18 14:55:11 +00:00
|
|
|
import 'package:immich_mobile/modules/backup/models/backup_album.model.dart';
|
|
|
|
import 'package:immich_mobile/modules/backup/models/duplicated_asset.model.dart';
|
2022-05-06 12:22:23 +00:00
|
|
|
import 'package:immich_mobile/modules/backup/models/hive_backup_albums.model.dart';
|
2022-10-25 14:51:03 +00:00
|
|
|
import 'package:immich_mobile/modules/backup/models/hive_duplicated_assets.model.dart';
|
2022-06-23 04:14:14 +00:00
|
|
|
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
|
2023-02-23 18:33:53 +00:00
|
|
|
import 'package:immich_mobile/modules/backup/providers/ios_background_settings.provider.dart';
|
2022-04-25 02:33:10 +00:00
|
|
|
import 'package:immich_mobile/modules/login/models/hive_saved_login_info.model.dart';
|
2022-06-11 21:12:06 +00:00
|
|
|
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
|
2023-02-28 16:22:18 +00:00
|
|
|
import 'package:immich_mobile/modules/onboarding/providers/gallery_permission.provider.dart';
|
|
|
|
import 'package:immich_mobile/modules/settings/providers/notification_permission.provider.dart';
|
2022-02-03 16:06:44 +00:00
|
|
|
import 'package:immich_mobile/routing/router.dart';
|
2022-03-16 15:19:31 +00:00
|
|
|
import 'package:immich_mobile/routing/tab_navigation_observer.dart';
|
2023-03-03 22:38:30 +00:00
|
|
|
import 'package:immich_mobile/shared/models/album.dart';
|
|
|
|
import 'package:immich_mobile/shared/models/asset.dart';
|
|
|
|
import 'package:immich_mobile/shared/models/exif_info.dart';
|
2022-11-27 20:34:19 +00:00
|
|
|
import 'package:immich_mobile/shared/models/immich_logger_message.model.dart';
|
2023-02-09 17:32:08 +00:00
|
|
|
import 'package:immich_mobile/shared/models/store.dart';
|
2023-03-03 22:38:30 +00:00
|
|
|
import 'package:immich_mobile/shared/models/user.dart';
|
2022-02-03 16:06:44 +00:00
|
|
|
import 'package:immich_mobile/shared/providers/app_state.provider.dart';
|
2022-06-23 04:14:14 +00:00
|
|
|
import 'package:immich_mobile/shared/providers/asset.provider.dart';
|
2023-02-09 17:32:08 +00:00
|
|
|
import 'package:immich_mobile/shared/providers/db.provider.dart';
|
2022-06-11 21:12:06 +00:00
|
|
|
import 'package:immich_mobile/shared/providers/release_info.provider.dart';
|
2022-03-22 06:22:04 +00:00
|
|
|
import 'package:immich_mobile/shared/providers/server_info.provider.dart';
|
2022-02-14 16:40:41 +00:00
|
|
|
import 'package:immich_mobile/shared/providers/websocket.provider.dart';
|
2022-11-27 20:34:19 +00:00
|
|
|
import 'package:immich_mobile/shared/services/immich_logger.service.dart';
|
2022-04-24 02:08:45 +00:00
|
|
|
import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
|
2022-06-11 21:12:06 +00:00
|
|
|
import 'package:immich_mobile/shared/views/version_announcement_overlay.dart';
|
2022-08-15 23:53:30 +00:00
|
|
|
import 'package:immich_mobile/utils/immich_app_theme.dart';
|
2023-02-09 17:32:08 +00:00
|
|
|
import 'package:immich_mobile/utils/migration.dart';
|
|
|
|
import 'package:isar/isar.dart';
|
2023-02-23 06:36:17 +00:00
|
|
|
import 'package:logging/logging.dart';
|
2023-02-09 17:32:08 +00:00
|
|
|
import 'package:path_provider/path_provider.dart';
|
2023-02-28 16:22:18 +00:00
|
|
|
import 'package:permission_handler/permission_handler.dart';
|
2022-02-03 16:06:44 +00:00
|
|
|
import 'constants/hive_box.dart';
|
|
|
|
|
|
|
|
void main() async {
|
2023-01-22 03:43:28 +00:00
|
|
|
await initApp();
|
2023-02-09 17:32:08 +00:00
|
|
|
final db = await loadDb();
|
|
|
|
await migrateHiveToStoreIfNecessary();
|
2023-03-03 22:38:30 +00:00
|
|
|
await migrateJsonCacheIfNecessary();
|
2023-02-09 17:32:08 +00:00
|
|
|
runApp(getMainWidget(db));
|
2023-01-22 03:43:28 +00:00
|
|
|
}
|
2022-05-06 12:22:23 +00:00
|
|
|
|
2023-01-22 03:43:28 +00:00
|
|
|
Future<void> openBoxes() async {
|
2022-10-31 14:38:24 +00:00
|
|
|
await Future.wait([
|
2022-11-27 20:34:19 +00:00
|
|
|
Hive.openBox<ImmichLoggerMessage>(immichLoggerBox),
|
2022-10-31 14:38:24 +00:00
|
|
|
Hive.openBox(userInfoBox),
|
|
|
|
Hive.openBox<HiveSavedLoginInfo>(hiveLoginInfoBox),
|
|
|
|
Hive.openBox(hiveGithubReleaseInfoBox),
|
|
|
|
Hive.openBox(userSettingInfoBox),
|
|
|
|
EasyLocalization.ensureInitialized(),
|
|
|
|
]);
|
2023-01-22 03:43:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> initApp() async {
|
|
|
|
await Hive.initFlutter();
|
|
|
|
Hive.registerAdapter(HiveSavedLoginInfoAdapter());
|
|
|
|
Hive.registerAdapter(HiveBackupAlbumsAdapter());
|
|
|
|
Hive.registerAdapter(HiveDuplicatedAssetsAdapter());
|
|
|
|
Hive.registerAdapter(ImmichLoggerMessageAdapter());
|
|
|
|
|
|
|
|
await openBoxes();
|
2022-02-03 16:06:44 +00:00
|
|
|
|
2022-08-09 04:43:48 +00:00
|
|
|
if (kReleaseMode && Platform.isAndroid) {
|
|
|
|
try {
|
|
|
|
await FlutterDisplayMode.setHighRefreshRate();
|
2023-01-18 15:59:23 +00:00
|
|
|
debugPrint("Enabled high refresh mode");
|
2022-08-09 04:43:48 +00:00
|
|
|
} catch (e) {
|
|
|
|
debugPrint("Error setting high refresh rate: $e");
|
|
|
|
}
|
2022-08-08 00:43:09 +00:00
|
|
|
}
|
|
|
|
|
2022-11-27 20:34:19 +00:00
|
|
|
// Initialize Immich Logger Service
|
|
|
|
ImmichLogger().init();
|
2023-02-23 06:36:17 +00:00
|
|
|
|
|
|
|
var log = Logger("ImmichErrorLogger");
|
|
|
|
|
|
|
|
FlutterError.onError = (details) {
|
|
|
|
FlutterError.presentError(details);
|
|
|
|
log.severe(details.toString(), details, details.stack);
|
|
|
|
};
|
|
|
|
|
|
|
|
PlatformDispatcher.instance.onError = (error, stack) {
|
|
|
|
log.severe(error.toString(), error, stack);
|
|
|
|
return true;
|
|
|
|
};
|
2023-01-22 03:43:28 +00:00
|
|
|
}
|
2022-11-27 20:34:19 +00:00
|
|
|
|
2023-02-09 17:32:08 +00:00
|
|
|
Future<Isar> loadDb() async {
|
|
|
|
final dir = await getApplicationDocumentsDirectory();
|
|
|
|
Isar db = await Isar.open(
|
2023-03-03 22:38:30 +00:00
|
|
|
[
|
|
|
|
StoreValueSchema,
|
|
|
|
ExifInfoSchema,
|
|
|
|
AssetSchema,
|
|
|
|
AlbumSchema,
|
|
|
|
UserSchema,
|
2023-03-18 14:55:11 +00:00
|
|
|
BackupAlbumSchema,
|
|
|
|
DuplicatedAssetSchema,
|
2023-03-03 22:38:30 +00:00
|
|
|
],
|
2023-02-09 17:32:08 +00:00
|
|
|
directory: dir.path,
|
|
|
|
maxSizeMiB: 256,
|
|
|
|
);
|
|
|
|
Store.init(db);
|
|
|
|
return db;
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget getMainWidget(Isar db) {
|
2023-01-22 03:43:28 +00:00
|
|
|
return EasyLocalization(
|
|
|
|
supportedLocales: locales,
|
|
|
|
path: translationsPath,
|
|
|
|
useFallbackTranslations: true,
|
|
|
|
fallbackLocale: locales.first,
|
2023-02-09 17:32:08 +00:00
|
|
|
child: ProviderScope(
|
|
|
|
overrides: [dbProvider.overrideWithValue(db)],
|
|
|
|
child: const ImmichApp(),
|
|
|
|
),
|
2022-07-13 12:23:48 +00:00
|
|
|
);
|
2022-02-03 16:06:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class ImmichApp extends ConsumerStatefulWidget {
|
2022-06-23 04:14:14 +00:00
|
|
|
const ImmichApp({super.key});
|
2022-02-03 16:06:44 +00:00
|
|
|
|
|
|
|
@override
|
2022-06-22 05:23:35 +00:00
|
|
|
ImmichAppState createState() => ImmichAppState();
|
2022-02-03 16:06:44 +00:00
|
|
|
}
|
|
|
|
|
2022-06-22 05:23:35 +00:00
|
|
|
class ImmichAppState extends ConsumerState<ImmichApp>
|
|
|
|
with WidgetsBindingObserver {
|
2022-02-03 16:06:44 +00:00
|
|
|
@override
|
|
|
|
void didChangeAppLifecycleState(AppLifecycleState state) {
|
|
|
|
switch (state) {
|
|
|
|
case AppLifecycleState.resumed:
|
|
|
|
debugPrint("[APP STATE] resumed");
|
2022-02-14 16:40:41 +00:00
|
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.resumed;
|
2022-06-11 21:12:06 +00:00
|
|
|
|
|
|
|
var isAuthenticated = ref.watch(authenticationProvider).isAuthenticated;
|
2023-02-28 16:22:18 +00:00
|
|
|
final permission = ref.watch(galleryPermissionNotifier);
|
2022-06-11 21:12:06 +00:00
|
|
|
|
2023-02-28 16:22:18 +00:00
|
|
|
// Needs to be logged in and have gallery permissions
|
|
|
|
if (isAuthenticated && (permission.isGranted || permission.isLimited)) {
|
2022-11-08 17:00:24 +00:00
|
|
|
ref.read(backupProvider.notifier).resumeBackup();
|
2022-08-18 14:41:59 +00:00
|
|
|
ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
|
2022-06-11 21:12:06 +00:00
|
|
|
ref.watch(assetProvider.notifier).getAllAsset();
|
|
|
|
ref.watch(serverInfoProvider.notifier).getServerVersion();
|
|
|
|
}
|
|
|
|
|
2022-02-14 16:40:41 +00:00
|
|
|
ref.watch(websocketProvider.notifier).connect();
|
2022-06-11 21:12:06 +00:00
|
|
|
|
|
|
|
ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
|
2022-03-22 06:22:04 +00:00
|
|
|
|
2023-03-18 14:55:11 +00:00
|
|
|
ref
|
|
|
|
.watch(notificationPermissionProvider.notifier)
|
|
|
|
.getNotificationPermission();
|
|
|
|
ref
|
|
|
|
.watch(galleryPermissionNotifier.notifier)
|
|
|
|
.getGalleryPermissionStatus();
|
2023-02-21 12:28:52 +00:00
|
|
|
|
2023-02-23 18:33:53 +00:00
|
|
|
ref.read(iOSBackgroundSettingsProvider.notifier).refresh();
|
|
|
|
|
2022-02-03 16:06:44 +00:00
|
|
|
break;
|
2022-03-22 06:22:04 +00:00
|
|
|
|
2022-02-03 16:06:44 +00:00
|
|
|
case AppLifecycleState.inactive:
|
|
|
|
debugPrint("[APP STATE] inactive");
|
2022-02-14 16:40:41 +00:00
|
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.inactive;
|
|
|
|
ref.watch(websocketProvider.notifier).disconnect();
|
2022-03-11 17:02:33 +00:00
|
|
|
ref.watch(backupProvider.notifier).cancelBackup();
|
|
|
|
|
2022-02-03 16:06:44 +00:00
|
|
|
break;
|
2022-03-22 06:22:04 +00:00
|
|
|
|
2022-02-03 16:06:44 +00:00
|
|
|
case AppLifecycleState.paused:
|
|
|
|
debugPrint("[APP STATE] paused");
|
2022-02-14 16:40:41 +00:00
|
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.paused;
|
2022-02-03 16:06:44 +00:00
|
|
|
break;
|
2022-03-22 06:22:04 +00:00
|
|
|
|
2022-02-03 16:06:44 +00:00
|
|
|
case AppLifecycleState.detached:
|
|
|
|
debugPrint("[APP STATE] detached");
|
2022-02-14 16:40:41 +00:00
|
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.detached;
|
2022-02-03 16:06:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> initApp() async {
|
2022-05-29 03:35:45 +00:00
|
|
|
WidgetsBinding.instance.addObserver(this);
|
2022-02-03 16:06:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
initState() {
|
|
|
|
super.initState();
|
|
|
|
initApp().then((_) => debugPrint("App Init Completed"));
|
2022-08-18 14:41:59 +00:00
|
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
|
|
// needs to be delayed so that EasyLocalization is working
|
|
|
|
ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
|
|
|
|
});
|
2022-02-03 16:06:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
2022-05-29 03:35:45 +00:00
|
|
|
WidgetsBinding.instance.removeObserver(this);
|
2022-02-03 16:06:44 +00:00
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2022-07-13 12:23:48 +00:00
|
|
|
var router = ref.watch(appRouterProvider);
|
2022-06-11 21:12:06 +00:00
|
|
|
ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
|
|
|
|
|
2023-02-03 16:26:05 +00:00
|
|
|
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
|
2023-03-06 04:51:18 +00:00
|
|
|
SystemChrome.setSystemUIOverlayStyle(
|
|
|
|
const SystemUiOverlayStyle(systemNavigationBarColor: Colors.transparent),
|
|
|
|
);
|
2023-02-03 16:26:05 +00:00
|
|
|
|
2022-04-24 02:08:45 +00:00
|
|
|
return MaterialApp(
|
2022-07-07 18:40:54 +00:00
|
|
|
localizationsDelegates: context.localizationDelegates,
|
|
|
|
supportedLocales: context.supportedLocales,
|
|
|
|
locale: context.locale,
|
2022-02-03 16:06:44 +00:00
|
|
|
debugShowCheckedModeBanner: false,
|
2022-04-24 02:08:45 +00:00
|
|
|
home: Stack(
|
|
|
|
children: [
|
|
|
|
MaterialApp.router(
|
|
|
|
title: 'Immich',
|
|
|
|
debugShowCheckedModeBanner: false,
|
2022-08-15 23:53:30 +00:00
|
|
|
themeMode: ref.watch(immichThemeProvider),
|
|
|
|
darkTheme: immichDarkTheme,
|
|
|
|
theme: immichLightTheme,
|
2022-07-13 12:23:48 +00:00
|
|
|
routeInformationParser: router.defaultRouteParser(),
|
|
|
|
routerDelegate: router.delegate(
|
|
|
|
navigatorObservers: () => [TabNavigationObserver(ref: ref)],
|
|
|
|
),
|
2022-04-24 02:08:45 +00:00
|
|
|
),
|
|
|
|
const ImmichLoadingOverlay(),
|
2022-06-11 21:12:06 +00:00
|
|
|
const VersionAnnouncementOverlay(),
|
2022-04-24 02:08:45 +00:00
|
|
|
],
|
2022-02-03 16:06:44 +00:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|