// ignore_for_file: use_build_context_synchronously import 'dart:async'; import 'dart:io'; import 'package:cross_connectivity/cross_connectivity.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; import 'package:injectable/injectable.dart'; import 'package:install_plugin/install_plugin.dart'; import 'package:path_provider/path_provider.dart'; import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/app/app.router.dart'; import 'package:revanced_manager/models/patched_application.dart'; import 'package:revanced_manager/services/github_api.dart'; import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/services/patcher_api.dart'; import 'package:revanced_manager/services/revanced_api.dart'; import 'package:revanced_manager/services/toast.dart'; import 'package:revanced_manager/ui/views/navigation/navigation_viewmodel.dart'; import 'package:revanced_manager/ui/views/patcher/patcher_viewmodel.dart'; import 'package:revanced_manager/ui/widgets/homeView/update_confirmation_dialog.dart'; import 'package:revanced_manager/ui/widgets/shared/custom_material_button.dart'; import 'package:stacked/stacked.dart'; import 'package:stacked_services/stacked_services.dart'; @lazySingleton class HomeViewModel extends BaseViewModel { final NavigationService _navigationService = locator(); final ManagerAPI _managerAPI = locator(); final PatcherAPI _patcherAPI = locator(); final GithubAPI _githubAPI = locator(); final RevancedAPI _revancedAPI = locator(); final Toast _toast = locator(); final flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); DateTime? _lastUpdate; bool showUpdatableApps = false; List patchedInstalledApps = []; List patchedUpdatableApps = []; String? _latestManagerVersion = ''; File? downloadedApk; Future initialize(BuildContext context) async { _latestManagerVersion = await _managerAPI.getLatestManagerVersion(); if(!_managerAPI.getPatchesConsent()){ await showPatchesConsent(context); } await _patcherAPI.initialize(); await flutterLocalNotificationsPlugin.initialize( const InitializationSettings( android: AndroidInitializationSettings('ic_notification'), ), onDidReceiveNotificationResponse: (response) async { if (response.id == 0) { _toast.showBottom('homeView.installingMessage'); final File? managerApk = await _managerAPI.downloadManager(); if (managerApk != null) { await InstallPlugin.installApk(managerApk.path); } else { _toast.showBottom('homeView.errorDownloadMessage'); } } }, ); flutterLocalNotificationsPlugin .resolvePlatformSpecificImplementation< AndroidFlutterLocalNotificationsPlugin>() ?.requestPermission(); final bool isConnected = await Connectivity().checkConnection(); if (!isConnected) { _toast.showBottom('homeView.noConnection'); } final NotificationAppLaunchDetails? notificationAppLaunchDetails = await flutterLocalNotificationsPlugin.getNotificationAppLaunchDetails(); if (notificationAppLaunchDetails?.didNotificationLaunchApp ?? false) { _toast.showBottom('homeView.installingMessage'); final File? managerApk = await _managerAPI.downloadManager(); if (managerApk != null) { await InstallPlugin.installApk(managerApk.path); } else { _toast.showBottom('homeView.errorDownloadMessage'); } } _getPatchedApps(); _managerAPI.reAssessSavedApps().then((_) => _getPatchedApps()); } void navigateToAppInfo(PatchedApplication app) { _navigationService.navigateTo( Routes.appInfoView, arguments: AppInfoViewArguments(app: app), ); } void toggleUpdatableApps(bool value) { showUpdatableApps = value; notifyListeners(); } Future navigateToPatcher(PatchedApplication app) async { locator().selectedApp = app; locator().selectedPatches = await _patcherAPI.getAppliedPatches(app.appliedPatches); locator().notifyListeners(); locator().setIndex(1); } void _getPatchedApps() { patchedInstalledApps = _managerAPI.getPatchedApps().toList(); patchedUpdatableApps = _managerAPI .getPatchedApps() .where((app) => app.hasUpdates == true) .toList(); notifyListeners(); } Future hasManagerUpdates() async { String currentVersion = await _managerAPI.getCurrentManagerVersion(); // add v to current version if (!currentVersion.startsWith('v')) { currentVersion = 'v$currentVersion'; } _latestManagerVersion = await _managerAPI.getLatestManagerVersion() ?? currentVersion; if (_latestManagerVersion != currentVersion) { return true; } return false; } Future hasPatchesUpdates() async { final String? latestVersion = await _managerAPI.getLatestPatchesVersion(); final String currentVersion = await _managerAPI.getCurrentPatchesVersion(); if (latestVersion != null) { try { final int latestVersionInt = int.parse(latestVersion.replaceAll(RegExp('[^0-9]'), '')); final int currentVersionInt = int.parse(currentVersion.replaceAll(RegExp('[^0-9]'), '')); return latestVersionInt > currentVersionInt; } on Exception catch (e) { if (kDebugMode) { print(e); } return false; } } return false; } Future downloadManager() async { try { final response = await _revancedAPI.downloadManager(); final bytes = await response!.readAsBytes(); final tempDir = await getTemporaryDirectory(); final tempFile = File('${tempDir.path}/revanced-manager.apk'); await tempFile.writeAsBytes(bytes); return tempFile; } on Exception catch (e) { if (kDebugMode) { print(e); } return null; } } Future showPatchesConsent(BuildContext context) async{ final ValueNotifier autoUpdate = ValueNotifier(true); await showDialog( context: context, barrierDismissible: false, builder: (context) => AlertDialog( title: const Text('ReVanced Patches'), content: ValueListenableBuilder( valueListenable: autoUpdate, builder: (context, value, child) { return Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ I18nText( 'homeView.patchesConsentDialogText', child: Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.secondary, ), ), ), Padding( padding: const EdgeInsets.symmetric(vertical: 10), child: I18nText( 'homeView.patchesConsentDialogText2', translationParams: {'url': _managerAPI.defaultApiUrl.split('/')[2]}, child: Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.error, ), ), ), ), CheckboxListTile( value: value, contentPadding: EdgeInsets.zero, title: I18nText('homeView.patchesConsentDialogText3',), subtitle: I18nText('homeView.patchesConsentDialogText3Sub',), onChanged: (selected) { autoUpdate.value = selected!; }, ), ], ); }, ), actions: [ CustomMaterialButton( isFilled: false, onPressed: () async { await _managerAPI.setPatchesConsent(false); SystemNavigator.pop(); }, label: I18nText('quitButton'), ), CustomMaterialButton( onPressed: () async { await _managerAPI.setPatchesConsent(true); await _managerAPI.setPatchesAutoUpdate(autoUpdate.value); Navigator.of(context).pop(); }, label: I18nText('okButton'), ) ], ), ); } Future updatePatches(BuildContext context) async { _toast.showBottom('homeView.downloadingMessage'); final String patchesVersion = await _managerAPI.getLatestPatchesVersion() ?? '0.0.0'; if (patchesVersion != '0.0.0') { _toast.showBottom('homeView.downloadedMessage'); await _managerAPI.setCurrentPatchesVersion(patchesVersion); forceRefresh(context); } else { _toast.showBottom('homeView.errorDownloadMessage'); } } Future updateManager(BuildContext context) async { final ValueNotifier downloaded = ValueNotifier(false); try { _toast.showBottom('homeView.downloadingMessage'); showDialog( context: context, builder: (context) => ValueListenableBuilder( valueListenable: downloaded, builder: (context, value, child) { return SimpleDialog( contentPadding: const EdgeInsets.all(16.0), title: I18nText( !value ? 'homeView.downloadingMessage' : 'homeView.downloadedMessage', child: Text( '', style: TextStyle( fontSize: 20, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.secondary, ), ), ), children: [ Column( children: [ Row( children: [ Icon( Icons.new_releases_outlined, color: Theme.of(context).colorScheme.secondary, ), const SizedBox(width: 8.0), Text( '$_latestManagerVersion', style: TextStyle( fontSize: 18, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.secondary, ), ), ], ), const SizedBox(height: 16.0), if (!value) Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ StreamBuilder( initialData: 0.0, stream: _revancedAPI.managerUpdateProgress.stream, builder: (context, snapshot) { return LinearProgressIndicator( value: snapshot.data! * 0.01, valueColor: AlwaysStoppedAnimation( Theme.of(context).colorScheme.secondary, ), ); }, ), const SizedBox(height: 16.0), Align( alignment: Alignment.centerRight, child: CustomMaterialButton( label: I18nText('cancelButton'), onPressed: () { _revancedAPI.disposeManagerUpdateProgress(); Navigator.of(context).pop(); }, ), ), ], ), if (value) Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ I18nText( 'homeView.installUpdate', child: Text( '', style: TextStyle( fontSize: 20, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.secondary, ), ), ), const SizedBox(height: 16.0), Row( mainAxisAlignment: MainAxisAlignment.end, children: [ Align( alignment: Alignment.centerRight, child: CustomMaterialButton( isFilled: false, label: I18nText('cancelButton'), onPressed: () { Navigator.of(context).pop(); }, ), ), const SizedBox(width: 8.0), Align( alignment: Alignment.centerRight, child: CustomMaterialButton( label: I18nText('updateButton'), onPressed: () async { await InstallPlugin.installApk( downloadedApk!.path, ); }, ), ), ], ), ], ), ], ), ], ); }, ), ); final File? managerApk = await downloadManager(); if (managerApk != null) { downloaded.value = true; downloadedApk = managerApk; // await flutterLocalNotificationsPlugin.zonedSchedule( // 0, // FlutterI18n.translate( // context, // 'homeView.notificationTitle', // ), // FlutterI18n.translate( // context, // 'homeView.notificationText', // ), // tz.TZDateTime.now(tz.local).add(const Duration(seconds: 5)), // const NotificationDetails( // android: AndroidNotificationDetails( // 'revanced_manager_channel', // 'ReVanced Manager Channel', // importance: Importance.max, // priority: Priority.high, // ticker: 'ticker', // ), // ), // androidAllowWhileIdle: true, // uiLocalNotificationDateInterpretation: // UILocalNotificationDateInterpretation.absoluteTime, // ); _toast.showBottom('homeView.installingMessage'); await InstallPlugin.installApk(managerApk.path); } else { _toast.showBottom('homeView.errorDownloadMessage'); } } on Exception catch (e) { if (kDebugMode) { print(e); } _toast.showBottom('homeView.errorInstallMessage'); } } void updatesAreDisabled() { _toast.showBottom('homeView.updatesDisabled'); } Future showUpdateConfirmationDialog( BuildContext parentContext, bool isPatches, ) { return showModalBottomSheet( context: parentContext, isScrollControlled: true, shape: const RoundedRectangleBorder( borderRadius: BorderRadius.vertical(top: Radius.circular(24.0)), ), builder: (context) => UpdateConfirmationDialog( isPatches: isPatches, ), ); } Future?> getLatestManagerRelease() { return _githubAPI.getLatestManagerRelease(_managerAPI.defaultManagerRepo); } Future?> getLatestPatchesRelease() { return _githubAPI.getLatestPatchesRelease(_managerAPI.defaultPatchesRepo); } Future getLatestPatchesReleaseTime() { return _managerAPI.getLatestPatchesReleaseTime(); } Future getLatestManagerReleaseTime() { return _managerAPI.getLatestManagerReleaseTime(); } Future forceRefresh(BuildContext context) async { await Future.delayed(const Duration(seconds: 1)); if (_lastUpdate == null || _lastUpdate!.difference(DateTime.now()).inSeconds > 2) { _managerAPI.clearAllData(); } _toast.showBottom('homeView.refreshSuccess'); initialize(context); } }