mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-04 03:39:37 -05:00 
			
		
		
		
	* 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>
		
			
				
	
	
		
			203 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			203 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(),
 | 
						|
        ],
 | 
						|
      ),
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |