// ignore_for_file: use_build_context_synchronously import 'dart:async'; import 'dart:io'; import 'package:connectivity_plus/connectivity_plus.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: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_sheet.dart'; import 'package:revanced_manager/ui/widgets/shared/haptics/haptic_checkbox_list_tile.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(); bool showUpdatableApps = false; List patchedInstalledApps = []; String _currentManagerVersion = ''; String _currentPatchesVersion = ''; String? _latestManagerVersion = ''; File? downloadedApk; Future initialize(BuildContext context) async { _managerAPI.rePatchedSavedApps().then((_) => _getPatchedApps()); _currentManagerVersion = await _managerAPI.getCurrentManagerVersion(); if (!_managerAPI.getDownloadConsent()) { await showDownloadConsent(context); await forceRefresh(context); return; } _latestManagerVersion = await _managerAPI.getLatestManagerVersion(); _currentPatchesVersion = await _managerAPI.getCurrentPatchesVersion(); if (_managerAPI.showUpdateDialog() && await hasManagerUpdates()) { showUpdateDialog(context, false); } if (!_managerAPI.isPatchesAutoUpdate() && _managerAPI.showUpdateDialog() && await hasPatchesUpdates()) { showUpdateDialog(context, true); } 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 _patcherAPI.installApk(context, managerApk.path); } else { _toast.showBottom('homeView.errorDownloadMessage'); } } }, ); flutterLocalNotificationsPlugin .resolvePlatformSpecificImplementation< AndroidFlutterLocalNotificationsPlugin>() ?.requestNotificationsPermission(); final bool isConnected = await Connectivity().checkConnectivity() != ConnectivityResult.none; 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 _patcherAPI.installApk(context, managerApk.path); } else { _toast.showBottom('homeView.errorDownloadMessage'); } } } 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(); notifyListeners(); } Future hasManagerUpdates() async { _latestManagerVersion = await _managerAPI.getLatestManagerVersion() ?? _currentManagerVersion; if (_latestManagerVersion != _currentManagerVersion) { return true; } return false; } Future hasPatchesUpdates() async { final String? latestVersion = await _managerAPI.getLatestPatchesVersion(); if (latestVersion != null) { try { final int latestVersionInt = int.parse(latestVersion.replaceAll(RegExp('[^0-9]'), '')); final int currentVersionInt = int.parse(_currentPatchesVersion.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 showDownloadConsent(BuildContext context) async { final ValueNotifier autoUpdate = ValueNotifier(true); await showDialog( context: context, barrierDismissible: false, builder: (context) => PopScope( canPop: false, child: AlertDialog( title: I18nText('homeView.downloadConsentDialogTitle'), content: ValueListenableBuilder( valueListenable: autoUpdate, builder: (context, value, child) { return Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ I18nText( 'homeView.downloadConsentDialogText', 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.downloadConsentDialogText2', translationParams: { 'url': _managerAPI.defaultApiUrl.split('/')[2], }, child: Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.error, ), ), ), ), ], ); }, ), actions: [ TextButton( onPressed: () async { _managerAPI.setDownloadConsent(false); SystemNavigator.pop(); }, child: I18nText('quitButton'), ), FilledButton( onPressed: () async { _managerAPI.setDownloadConsent(true); _managerAPI.setPatchesAutoUpdate(autoUpdate.value); Navigator.of(context).pop(); }, child: I18nText('okButton'), ), ], ), ), ); } void showUpdateDialog(BuildContext context, bool isPatches) { final ValueNotifier noShow = ValueNotifier(!_managerAPI.showUpdateDialog()); showDialog( context: context, builder: (innerContext) => AlertDialog( title: I18nText('homeView.updateDialogTitle'), content: ValueListenableBuilder( valueListenable: noShow, builder: (context, value, child) { return Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ I18nText( 'homeView.updateDialogText', translationParams: { 'file': isPatches ? 'ReVanced Patches' : 'ReVanced Manager', 'version': isPatches ? _currentPatchesVersion : _currentManagerVersion, }, child: Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.secondary, ), ), ), const SizedBox(height: 10), HapticCheckboxListTile( value: value, contentPadding: EdgeInsets.zero, title: I18nText( 'noShowAgain', ), subtitle: I18nText( 'homeView.changeLaterSubtitle', ), onChanged: (selected) { noShow.value = selected!; }, ), ], ); }, ), actions: [ TextButton( onPressed: () async { _managerAPI.setShowUpdateDialog(!noShow.value); Navigator.pop(innerContext); }, child: I18nText('dismissButton'), // Decide later ), FilledButton( onPressed: () async { _managerAPI.setShowUpdateDialog(!noShow.value); Navigator.pop(innerContext); await showUpdateConfirmationDialog(context, isPatches); }, child: I18nText('showUpdateButton'), ), ], ), ); } Future updatePatches(BuildContext context) async { _toast.showBottom('homeView.downloadingMessage'); final String patchesVersion = await _managerAPI.getLatestPatchesVersion() ?? '0.0.0'; final String integrationsVersion = await _managerAPI.getLatestIntegrationsVersion() ?? '0.0.0'; if (patchesVersion != '0.0.0' && integrationsVersion != '0.0.0') { await _managerAPI.setCurrentPatchesVersion(patchesVersion); await _managerAPI.setCurrentIntegrationsVersion(integrationsVersion); _toast.showBottom('homeView.downloadedMessage'); 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 AlertDialog( title: I18nText( !value ? 'homeView.downloadingMessage' : 'homeView.downloadedMessage', ), content: Column( mainAxisSize: MainAxisSize.min, children: [ 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: FilledButton( onPressed: () { _revancedAPI.disposeManagerUpdateProgress(); Navigator.of(context).pop(); }, child: I18nText('cancelButton'), ), ), ], ), if (value) Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ I18nText( 'homeView.installUpdate', child: Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, color: Theme.of(context).colorScheme.secondary, ), ), ), const SizedBox(height: 16.0), Row( mainAxisAlignment: MainAxisAlignment.end, children: [ Align( alignment: Alignment.centerRight, child: TextButton( onPressed: () { Navigator.of(context).pop(); }, child: I18nText('cancelButton'), ), ), const SizedBox(width: 8.0), Align( alignment: Alignment.centerRight, child: FilledButton( onPressed: () async { await _patcherAPI.installApk( context, downloadedApk!.path, ); }, child: I18nText('updateButton'), ), ), ], ), ], ), ], ), ); }, ), ); 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 _patcherAPI.installApk(context, 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, [ bool changelog = false, ]) { return showModalBottomSheet( context: parentContext, isScrollControlled: true, shape: const RoundedRectangleBorder( borderRadius: BorderRadius.vertical(top: Radius.circular(24.0)), ), builder: (context) => UpdateConfirmationSheet( isPatches: isPatches, changelog: changelog, ), ); } 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 { _managerAPI.clearAllData(); _toast.showBottom('homeView.refreshSuccess'); initialize(context); } }