import 'dart:io'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:http/http.dart'; import 'package:obtainium/custom_errors.dart'; import 'package:obtainium/pages/home.dart'; import 'package:obtainium/providers/apps_provider.dart'; import 'package:obtainium/providers/logs_provider.dart'; import 'package:obtainium/providers/notifications_provider.dart'; import 'package:obtainium/providers/settings_provider.dart'; import 'package:obtainium/providers/source_provider.dart'; import 'package:permission_handler/permission_handler.dart'; import 'package:provider/provider.dart'; import 'package:dynamic_color/dynamic_color.dart'; import 'package:device_info_plus/device_info_plus.dart'; import 'package:android_alarm_manager_plus/android_alarm_manager_plus.dart'; import 'package:easy_localization/easy_localization.dart'; // ignore: implementation_imports import 'package:easy_localization/src/easy_localization_controller.dart'; // ignore: implementation_imports import 'package:easy_localization/src/localization.dart'; const String currentVersion = '0.13.26'; const String currentReleaseTag = 'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES const int bgUpdateCheckAlarmId = 666; List> supportedLocales = const [ MapEntry(Locale('en'), 'English'), MapEntry(Locale('zh'), '汉语'), MapEntry(Locale('it'), 'Italiano'), MapEntry(Locale('ja'), '日本語'), MapEntry(Locale('hu'), 'Magyar'), MapEntry(Locale('de'), 'Deutsch'), MapEntry(Locale('fa'), 'فارسی'), MapEntry(Locale('fr'), 'Français'), MapEntry(Locale('es'), 'Español'), MapEntry(Locale('pl'), 'Polski'), MapEntry(Locale('ru'), 'Русский язык'), MapEntry(Locale('bs'), 'Bosanski'), ]; const fallbackLocale = Locale('en'); const localeDir = 'assets/translations'; final globalNavigatorKey = GlobalKey(); Future loadTranslations() async { // See easy_localization/issues/210 await EasyLocalizationController.initEasyLocation(); var s = SettingsProvider(); await s.initializeSettings(); var forceLocale = s.forcedLocale; final controller = EasyLocalizationController( saveLocale: true, forceLocale: forceLocale != null ? Locale(forceLocale) : null, fallbackLocale: fallbackLocale, supportedLocales: supportedLocales.map((e) => e.key).toList(), assetLoader: const RootBundleAssetLoader(), useOnlyLangCode: true, useFallbackTranslations: true, path: localeDir, onLoadError: (FlutterError e) { throw e; }, ); await controller.loadTranslations(); Localization.load(controller.locale, translations: controller.translations, fallbackTranslations: controller.fallbackTranslations); } @pragma('vm:entry-point') Future bgUpdateCheckApps(int taskId, Map? params) async { WidgetsFlutterBinding.ensureInitialized(); await EasyLocalization.ensureInitialized(); await AndroidAlarmManager.initialize(); await loadTranslations(); LogsProvider logs = LogsProvider(); AppsProvider appsProvider = AppsProvider(); await appsProvider.loadApps(); logs.add('BG update master task started.'); var appIds = appsProvider.getAppsSortedByUpdateCheckTime(); for (var id in appIds) { AndroidAlarmManager.oneShot( const Duration(minutes: 0), id.hashCode, bgUpdateCheckApp, params: {'appId': id}); await Future.delayed(const Duration(seconds: 1)); } logs.add('BG update master task - all $appIds child tasks started.'); } @pragma('vm:entry-point') Future bgUpdateCheckApp(int taskId, Map? params) async { WidgetsFlutterBinding.ensureInitialized(); await EasyLocalization.ensureInitialized(); await AndroidAlarmManager.initialize(); await loadTranslations(); LogsProvider logs = LogsProvider(); NotificationsProvider notificationsProvider = NotificationsProvider(); AppsProvider appsProvider = AppsProvider(); String appId = params!['appId']; try { await appsProvider.loadApps(singleId: appId); AppInMemory app = appsProvider.apps[appId]!; App? newApp; if (app.app.installedVersion == app.app.latestVersion && app.app.installedVersion != null) { try { notificationsProvider.notify(checkingUpdatesNotification, cancelExisting: true); newApp = await appsProvider.checkUpdate(appId); } catch (e) { logs.add('BG update check for $appId got error \'${e.toString()}\'.'); if (e is RateLimitError || e is ClientException) { var remainingMinutes = e is RateLimitError ? e.remainingMinutes : 15; logs.add( 'BG update check for $appId will be retried in $remainingMinutes minutes.'); AndroidAlarmManager.oneShot( Duration(minutes: remainingMinutes), taskId, bgUpdateCheckApp, params: params); } else { rethrow; } } finally { notificationsProvider.cancel(checkingUpdatesNotification.id); } } if (newApp != null) { var canInstallSilently = await appsProvider.canInstallSilently(app.app); if (!canInstallSilently) { notificationsProvider .notify(UpdateNotification([newApp], id: newApp.id.hashCode * 10)); } else { logs.add('Attempting to update $appId in the background.'); await appsProvider.downloadAndInstallLatestApps([appId], null, notificationsProvider: notificationsProvider); } } } catch (e) { notificationsProvider.notify(ErrorCheckingUpdatesNotification( '$appId: ${e.toString()}', id: appId.hashCode * 20)); } } void main() async { WidgetsFlutterBinding.ensureInitialized(); try { ByteData data = await PlatformAssetBundle().load('assets/ca/lets-encrypt-r3.pem'); SecurityContext.defaultContext .setTrustedCertificatesBytes(data.buffer.asUint8List()); } catch (e) { // Already added, do nothing (see #375) } await EasyLocalization.ensureInitialized(); if ((await DeviceInfoPlugin().androidInfo).version.sdkInt >= 29) { SystemChrome.setSystemUIOverlayStyle( const SystemUiOverlayStyle(systemNavigationBarColor: Colors.transparent), ); SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge); } await AndroidAlarmManager.initialize(); runApp(MultiProvider( providers: [ ChangeNotifierProvider(create: (context) => AppsProvider()), ChangeNotifierProvider(create: (context) => SettingsProvider()), Provider(create: (context) => NotificationsProvider()), Provider(create: (context) => LogsProvider()) ], child: EasyLocalization( supportedLocales: supportedLocales.map((e) => e.key).toList(), path: localeDir, fallbackLocale: fallbackLocale, useOnlyLangCode: true, child: const Obtainium()), )); } var defaultThemeColour = Colors.deepPurple; class Obtainium extends StatefulWidget { const Obtainium({super.key}); @override State createState() => _ObtainiumState(); } class _ObtainiumState extends State { var existingUpdateInterval = -1; @override Widget build(BuildContext context) { SettingsProvider settingsProvider = context.watch(); AppsProvider appsProvider = context.read(); LogsProvider logs = context.read(); if (settingsProvider.prefs == null) { settingsProvider.initializeSettings(); } else { bool isFirstRun = settingsProvider.checkAndFlipFirstRun(); if (isFirstRun) { logs.add('This is the first ever run of Obtainium.'); // If this is the first run, ask for notification permissions and add Obtainium to the Apps list Permission.notification.request(); appsProvider.saveApps([ App( obtainiumId, 'https://github.com/ImranR98/Obtainium', 'ImranR98', 'Obtainium', currentReleaseTag, currentReleaseTag, [], 0, {'includePrereleases': true}, null, false) ], onlyIfExists: false); } if (!supportedLocales .map((e) => e.key.languageCode) .contains(context.locale.languageCode) || (settingsProvider.forcedLocale == null && context.deviceLocale.languageCode != context.locale.languageCode)) { settingsProvider.resetLocaleSafe(context); } // Register the background update task according to the user's setting if (existingUpdateInterval != settingsProvider.updateInterval) { if (existingUpdateInterval != -1) { logs.add( 'Setting update interval to ${settingsProvider.updateInterval.toString()}'); } existingUpdateInterval = settingsProvider.updateInterval; if (existingUpdateInterval == 0) { AndroidAlarmManager.cancel(bgUpdateCheckAlarmId); } else { AndroidAlarmManager.periodic( Duration(minutes: existingUpdateInterval), bgUpdateCheckAlarmId, bgUpdateCheckApps, allowWhileIdle: true, rescheduleOnReboot: true, wakeup: true); } } } return DynamicColorBuilder( builder: (ColorScheme? lightDynamic, ColorScheme? darkDynamic) { // Decide on a colour/brightness scheme based on OS and user settings ColorScheme lightColorScheme; ColorScheme darkColorScheme; if (lightDynamic != null && darkDynamic != null && settingsProvider.colour == ColourSettings.materialYou) { lightColorScheme = lightDynamic.harmonized(); darkColorScheme = darkDynamic.harmonized(); } else { lightColorScheme = ColorScheme.fromSeed(seedColor: defaultThemeColour); darkColorScheme = ColorScheme.fromSeed( seedColor: defaultThemeColour, brightness: Brightness.dark); } // set the background and surface colors to pure black in the amoled theme if (settingsProvider.useBlackTheme) { darkColorScheme = darkColorScheme .copyWith(background: Colors.black, surface: Colors.black) .harmonized(); } return MaterialApp( title: 'Obtainium', localizationsDelegates: context.localizationDelegates, supportedLocales: context.supportedLocales, locale: context.locale, navigatorKey: globalNavigatorKey, theme: ThemeData( useMaterial3: true, colorScheme: settingsProvider.theme == ThemeSettings.dark ? darkColorScheme : lightColorScheme, fontFamily: 'Metropolis'), darkTheme: ThemeData( useMaterial3: true, colorScheme: settingsProvider.theme == ThemeSettings.light ? lightColorScheme : darkColorScheme, fontFamily: 'Metropolis'), home: Shortcuts(shortcuts: { LogicalKeySet(LogicalKeyboardKey.select): const ActivateIntent(), }, child: const HomePage())); }); } }