mirror of
https://github.com/immich-app/immich.git
synced 2025-01-04 02:46:47 +01:00
87fea29e32
* first run of getting background sync working in iOS * got background sync calling into flutter * added background task * added necessary sync files * fixed some names and added more implementations * got as far as Hive.initFlutter * brute force got to await Hive.initFlutter * lots of print statements to figure out where execution is failing, and its failing at the root asset bundle in the localization.dart service * first time working, got plugins registered * removed broken cleanup code * refactored * linters * now can pass user settings * background service plugin uses app background processing instead of fetch * renamed backgroundFetch to backgroundProcessing to make it clearer * don't use max delay * adds fetch back in * fixes require charging default values and backup controller page * fixes background fetch * fixes ios not importing photos * guarded path provider ios * lint * adds max tries for heartbeat to work in iOS * fail after seconds * timeout instead of fail after seconds * removes release lock from system stop * restores checkLockReleasedWithHeartbeat to Future<void> * removes max tries from acquire lock * fixes lock timeout with iOS * restored for loop * adds comments, made the AppRefresh task only run while not requiring network or charge * fixed compile issue * now both are registered and added better comments. also added ability for task to cancel itself * added the podfile and pubspec * added backup diagnostics to IOS and removed iOS ignored backup options and fixed network connectivity always required * Added Alex's dev team * styled debug list item, fixed refresh task not set bug, fixed enable / disable background service on platform channel --------- Co-authored-by: Marty Fuhry <marty@fuhry.farm> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
202 lines
6.8 KiB
Dart
202 lines
6.8 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:easy_localization/easy_localization.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_displaymode/flutter_displaymode.dart';
|
|
import 'package:hive_flutter/hive_flutter.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/constants/locales.dart';
|
|
import 'package:immich_mobile/modules/backup/background_service/background.service.dart';
|
|
import 'package:immich_mobile/modules/backup/models/hive_backup_albums.model.dart';
|
|
import 'package:immich_mobile/modules/backup/models/hive_duplicated_assets.model.dart';
|
|
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
|
|
import 'package:immich_mobile/modules/login/models/hive_saved_login_info.model.dart';
|
|
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
|
|
import 'package:immich_mobile/routing/router.dart';
|
|
import 'package:immich_mobile/routing/tab_navigation_observer.dart';
|
|
import 'package:immich_mobile/shared/models/immich_logger_message.model.dart';
|
|
import 'package:immich_mobile/shared/models/store.dart';
|
|
import 'package:immich_mobile/shared/providers/app_state.provider.dart';
|
|
import 'package:immich_mobile/shared/providers/asset.provider.dart';
|
|
import 'package:immich_mobile/shared/providers/db.provider.dart';
|
|
import 'package:immich_mobile/shared/providers/release_info.provider.dart';
|
|
import 'package:immich_mobile/shared/providers/server_info.provider.dart';
|
|
import 'package:immich_mobile/shared/providers/websocket.provider.dart';
|
|
import 'package:immich_mobile/shared/services/immich_logger.service.dart';
|
|
import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
|
|
import 'package:immich_mobile/shared/views/version_announcement_overlay.dart';
|
|
import 'package:immich_mobile/utils/immich_app_theme.dart';
|
|
import 'package:immich_mobile/utils/migration.dart';
|
|
import 'package:isar/isar.dart';
|
|
import 'package:path_provider/path_provider.dart';
|
|
import 'constants/hive_box.dart';
|
|
|
|
void main() async {
|
|
await initApp();
|
|
final db = await loadDb();
|
|
await migrateHiveToStoreIfNecessary();
|
|
runApp(getMainWidget(db));
|
|
}
|
|
|
|
Future<void> openBoxes() async {
|
|
await Future.wait([
|
|
Hive.openBox<ImmichLoggerMessage>(immichLoggerBox),
|
|
Hive.openBox(userInfoBox),
|
|
Hive.openBox<HiveSavedLoginInfo>(hiveLoginInfoBox),
|
|
Hive.openBox(hiveGithubReleaseInfoBox),
|
|
Hive.openBox(userSettingInfoBox),
|
|
EasyLocalization.ensureInitialized(),
|
|
]);
|
|
}
|
|
|
|
Future<void> initApp() async {
|
|
await Hive.initFlutter();
|
|
Hive.registerAdapter(HiveSavedLoginInfoAdapter());
|
|
Hive.registerAdapter(HiveBackupAlbumsAdapter());
|
|
Hive.registerAdapter(HiveDuplicatedAssetsAdapter());
|
|
Hive.registerAdapter(ImmichLoggerMessageAdapter());
|
|
|
|
await openBoxes();
|
|
|
|
if (kReleaseMode && Platform.isAndroid) {
|
|
try {
|
|
await FlutterDisplayMode.setHighRefreshRate();
|
|
debugPrint("Enabled high refresh mode");
|
|
} catch (e) {
|
|
debugPrint("Error setting high refresh rate: $e");
|
|
}
|
|
}
|
|
|
|
// Initialize Immich Logger Service
|
|
ImmichLogger().init();
|
|
}
|
|
|
|
Future<Isar> loadDb() async {
|
|
final dir = await getApplicationDocumentsDirectory();
|
|
Isar db = await Isar.open(
|
|
[StoreValueSchema],
|
|
directory: dir.path,
|
|
maxSizeMiB: 256,
|
|
);
|
|
Store.init(db);
|
|
return db;
|
|
}
|
|
|
|
Widget getMainWidget(Isar db) {
|
|
return EasyLocalization(
|
|
supportedLocales: locales,
|
|
path: translationsPath,
|
|
useFallbackTranslations: true,
|
|
fallbackLocale: locales.first,
|
|
child: ProviderScope(
|
|
overrides: [dbProvider.overrideWithValue(db)],
|
|
child: const ImmichApp(),
|
|
),
|
|
);
|
|
}
|
|
|
|
class ImmichApp extends ConsumerStatefulWidget {
|
|
const ImmichApp({super.key});
|
|
|
|
@override
|
|
ImmichAppState createState() => ImmichAppState();
|
|
}
|
|
|
|
class ImmichAppState extends ConsumerState<ImmichApp>
|
|
with WidgetsBindingObserver {
|
|
@override
|
|
void didChangeAppLifecycleState(AppLifecycleState state) {
|
|
switch (state) {
|
|
case AppLifecycleState.resumed:
|
|
debugPrint("[APP STATE] resumed");
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.resumed;
|
|
|
|
var isAuthenticated = ref.watch(authenticationProvider).isAuthenticated;
|
|
|
|
if (isAuthenticated) {
|
|
ref.read(backupProvider.notifier).resumeBackup();
|
|
ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
|
|
ref.watch(assetProvider.notifier).getAllAsset();
|
|
ref.watch(serverInfoProvider.notifier).getServerVersion();
|
|
}
|
|
|
|
ref.watch(websocketProvider.notifier).connect();
|
|
|
|
ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
|
|
|
|
break;
|
|
|
|
case AppLifecycleState.inactive:
|
|
debugPrint("[APP STATE] inactive");
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.inactive;
|
|
ref.watch(websocketProvider.notifier).disconnect();
|
|
ref.watch(backupProvider.notifier).cancelBackup();
|
|
|
|
break;
|
|
|
|
case AppLifecycleState.paused:
|
|
debugPrint("[APP STATE] paused");
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.paused;
|
|
break;
|
|
|
|
case AppLifecycleState.detached:
|
|
debugPrint("[APP STATE] detached");
|
|
ref.watch(appStateProvider.notifier).state = AppStateEnum.detached;
|
|
break;
|
|
}
|
|
}
|
|
|
|
Future<void> initApp() async {
|
|
WidgetsBinding.instance.addObserver(this);
|
|
}
|
|
|
|
@override
|
|
initState() {
|
|
super.initState();
|
|
initApp().then((_) => debugPrint("App Init Completed"));
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
// needs to be delayed so that EasyLocalization is working
|
|
ref.read(backgroundServiceProvider).resumeServiceIfEnabled();
|
|
});
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
WidgetsBinding.instance.removeObserver(this);
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
var router = ref.watch(appRouterProvider);
|
|
ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
|
|
|
|
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
|
|
|
|
return MaterialApp(
|
|
localizationsDelegates: context.localizationDelegates,
|
|
supportedLocales: context.supportedLocales,
|
|
locale: context.locale,
|
|
debugShowCheckedModeBanner: false,
|
|
home: Stack(
|
|
children: [
|
|
MaterialApp.router(
|
|
title: 'Immich',
|
|
debugShowCheckedModeBanner: false,
|
|
themeMode: ref.watch(immichThemeProvider),
|
|
darkTheme: immichDarkTheme,
|
|
theme: immichLightTheme,
|
|
routeInformationParser: router.defaultRouteParser(),
|
|
routerDelegate: router.delegate(
|
|
navigatorObservers: () => [TabNavigationObserver(ref: ref)],
|
|
),
|
|
),
|
|
const ImmichLoadingOverlay(),
|
|
const VersionAnnouncementOverlay(),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|