mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-23 02:43:46 +02:00 
			
		
		
		
	- All Sources now have a "Track-Only" option that will prevent Obtainium from looking for APKs (though the App must still have a release of some kind so that a version string can be grabbed).
    - These Apps cannot be installed through Obtainium, but update notifications will still be sent.
    - The user needs to manually mark them as updated when appropriate.
    - This addresses issue #119.
    - It also partially addresses #44 by allowing some sources to be configured as "Track-Only"-only. The first such source (APKMirror) will be added later.
- Includes various UI changes to accommodate the above change.
- Also makes App loading a bit more responsive (sending Obtainium to the background then returning will now cause App re-load to pick up changes in App versioning that may have been made in the meantime, for instance through update checking).
		
	
		
			
				
	
	
		
			214 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			214 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:io';
 | |
| import 'dart:math';
 | |
| 
 | |
| import 'package:flutter/material.dart';
 | |
| import 'package:flutter/services.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';
 | |
| 
 | |
| const String currentVersion = '0.7.6';
 | |
| const String currentReleaseTag =
 | |
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES
 | |
| 
 | |
| const int bgUpdateCheckAlarmId = 666;
 | |
| 
 | |
| @pragma('vm:entry-point')
 | |
| Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async {
 | |
|   LogsProvider logs = LogsProvider();
 | |
|   logs.add('Started BG update check task');
 | |
|   int? ignoreAfterMicroseconds = params?['ignoreAfterMicroseconds'];
 | |
|   WidgetsFlutterBinding.ensureInitialized();
 | |
|   await AndroidAlarmManager.initialize();
 | |
|   DateTime? ignoreAfter = ignoreAfterMicroseconds != null
 | |
|       ? DateTime.fromMicrosecondsSinceEpoch(ignoreAfterMicroseconds)
 | |
|       : null;
 | |
|   logs.add('Bg update ignoreAfter is $ignoreAfter');
 | |
|   var notificationsProvider = NotificationsProvider();
 | |
|   await notificationsProvider.notify(checkingUpdatesNotification);
 | |
|   try {
 | |
|     var appsProvider = AppsProvider(forBGTask: true);
 | |
|     await notificationsProvider.cancel(ErrorCheckingUpdatesNotification('').id);
 | |
|     await appsProvider.loadApps();
 | |
|     List<String> existingUpdateIds =
 | |
|         appsProvider.findExistingUpdates(installedOnly: true);
 | |
|     DateTime nextIgnoreAfter = DateTime.now();
 | |
|     String? err;
 | |
|     try {
 | |
|       logs.add('Started actual BG update checking');
 | |
|       await appsProvider.checkUpdates(
 | |
|           ignoreAppsCheckedAfter: ignoreAfter, throwErrorsForRetry: true);
 | |
|     } catch (e) {
 | |
|       if (e is RateLimitError || e is SocketException) {
 | |
|         var remainingMinutes = e is RateLimitError ? e.remainingMinutes : 15;
 | |
|         logs.add(
 | |
|             'BG update checking encountered a ${e.runtimeType}, will schedule a retry check in $remainingMinutes minutes');
 | |
|         AndroidAlarmManager.oneShot(Duration(minutes: remainingMinutes),
 | |
|             Random().nextInt(pow(2, 31) as int), bgUpdateCheck, params: {
 | |
|           'ignoreAfterMicroseconds': nextIgnoreAfter.microsecondsSinceEpoch
 | |
|         });
 | |
|       } else {
 | |
|         err = e.toString();
 | |
|       }
 | |
|     }
 | |
|     List<App> newUpdates = appsProvider
 | |
|         .findExistingUpdates(installedOnly: true)
 | |
|         .where((id) => !existingUpdateIds.contains(id))
 | |
|         .map((e) => appsProvider.apps[e]!.app)
 | |
|         .toList();
 | |
| 
 | |
|     // TODO: This silent update code doesn't work yet
 | |
|     // List<String> silentlyUpdated = await appsProvider
 | |
|     //     .downloadAndInstallLatestApp(
 | |
|     //         [...newUpdates.map((e) => e.id), ...existingUpdateIds], null);
 | |
|     // if (silentlyUpdated.isNotEmpty) {
 | |
|     //   newUpdates = newUpdates
 | |
|     //       .where((element) => !silentlyUpdated.contains(element.id))
 | |
|     //       .toList();
 | |
|     //   notificationsProvider.notify(
 | |
|     //       SilentUpdateNotification(
 | |
|     //           silentlyUpdated.map((e) => appsProvider.apps[e]!.app).toList()),
 | |
|     //       cancelExisting: true);
 | |
|     // }
 | |
|     logs.add(
 | |
|         'BG update checking found ${newUpdates.length} updates - will notify user if needed');
 | |
|     if (newUpdates.isNotEmpty) {
 | |
|       notificationsProvider.notify(UpdateNotification(newUpdates));
 | |
|     }
 | |
|     if (err != null) {
 | |
|       throw err;
 | |
|     }
 | |
|   } catch (e) {
 | |
|     notificationsProvider
 | |
|         .notify(ErrorCheckingUpdatesNotification(e.toString()));
 | |
|   } finally {
 | |
|     logs.add('Finished BG update check task');
 | |
|     await notificationsProvider.cancel(checkingUpdatesNotification.id);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void main() async {
 | |
|   WidgetsFlutterBinding.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: const Obtainium(),
 | |
|   ));
 | |
| }
 | |
| 
 | |
| var defaultThemeColour = Colors.deepPurple;
 | |
| 
 | |
| class Obtainium extends StatefulWidget {
 | |
|   const Obtainium({super.key});
 | |
| 
 | |
|   @override
 | |
|   State<Obtainium> createState() => _ObtainiumState();
 | |
| }
 | |
| 
 | |
| class _ObtainiumState extends State<Obtainium> {
 | |
|   var existingUpdateInterval = -1;
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context) {
 | |
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>();
 | |
|     AppsProvider appsProvider = context.read<AppsProvider>();
 | |
|     LogsProvider logs = context.read<LogsProvider>();
 | |
| 
 | |
|     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,
 | |
|               ['true'],
 | |
|               null,
 | |
|               false,
 | |
|               false)
 | |
|         ]);
 | |
|       }
 | |
|       // 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}');
 | |
|         }
 | |
|         existingUpdateInterval = settingsProvider.updateInterval;
 | |
|         if (existingUpdateInterval == 0) {
 | |
|           AndroidAlarmManager.cancel(bgUpdateCheckAlarmId);
 | |
|         } else {
 | |
|           AndroidAlarmManager.periodic(
 | |
|               Duration(minutes: existingUpdateInterval),
 | |
|               bgUpdateCheckAlarmId,
 | |
|               bgUpdateCheck,
 | |
|               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);
 | |
|       }
 | |
|       return MaterialApp(
 | |
|           title: 'Obtainium',
 | |
|           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: const HomePage());
 | |
|     });
 | |
|   }
 | |
| }
 |