// ignore_for_file: use_build_context_synchronously import 'dart:io'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:injectable/injectable.dart'; import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/app/app.router.dart'; import 'package:revanced_manager/models/patch.dart'; import 'package:revanced_manager/models/patched_application.dart'; import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/services/patcher_api.dart'; import 'package:revanced_manager/utils/about_info.dart'; import 'package:revanced_manager/utils/check_for_supported_patch.dart'; import 'package:stacked/stacked.dart'; import 'package:stacked_services/stacked_services.dart'; @lazySingleton class PatcherViewModel extends BaseViewModel { final NavigationService _navigationService = locator(); final ManagerAPI _managerAPI = locator(); final PatcherAPI _patcherAPI = locator(); Set savedPatchNames = {}; PatchedApplication? selectedApp; BuildContext? ctx; List selectedPatches = []; List removedPatches = []; void navigateToAppSelector() { _navigationService.navigateTo(Routes.appSelectorView); } void navigateToPatchesSelector() { _navigationService.navigateTo(Routes.patchesSelectorView); } void navigateToInstaller() { _navigationService.navigateTo(Routes.installerView); } bool showPatchButton() { return selectedPatches.isNotEmpty; } bool dimPatchesCard() { return selectedApp == null; } bool showRemovedPatchesDialog(BuildContext context) { if (removedPatches.isNotEmpty) { showDialog( context: context, builder: (context) => AlertDialog( title: I18nText('notice'), content: I18nText( 'patcherView.removedPatchesWarningDialogText', translationParams: {'patches': removedPatches.join('\n')}, ), actions: [ TextButton( onPressed: () { Navigator.of(context).pop(); }, child: I18nText('noButton'), ), FilledButton( onPressed: () { Navigator.of(context).pop(); showArmv7WarningDialog(context); }, child: I18nText('yesButton'), ), ], ), ); return false; } return true; } bool checkRequiredPatchOption(BuildContext context) { if (getNullRequiredOptions(selectedPatches, selectedApp!.packageName) .isNotEmpty) { showRequiredOptionDialog(context); return false; } return true; } void showRequiredOptionDialog([context]) { showDialog( context: context ?? ctx, builder: (context) => AlertDialog( title: I18nText('notice'), content: I18nText('patcherView.requiredOptionDialogText'), actions: [ TextButton( onPressed: () => { Navigator.of(context).pop(), }, child: I18nText('cancelButton'), ), FilledButton( onPressed: () => { Navigator.pop(context), navigateToPatchesSelector(), }, child: I18nText('okButton'), ), ], ), ); } Future showArmv7WarningDialog(BuildContext context) async { final bool armv7 = await AboutInfo.getInfo().then((info) { final List archs = info['supportedArch']; final supportedAbis = ['arm64-v8a', 'x86', 'x86_64']; return !archs.any((arch) => supportedAbis.contains(arch)); }); if (context.mounted && armv7) { return showDialog( context: context, builder: (context) => AlertDialog( title: I18nText('warning'), content: I18nText('patcherView.armv7WarningDialogText'), actions: [ FilledButton( onPressed: () => Navigator.of(context).pop(), child: I18nText('noButton'), ), TextButton( onPressed: () { Navigator.of(context).pop(); navigateToInstaller(); }, child: I18nText('yesButton'), ), ], ), ); } else { navigateToInstaller(); } } String getAppSelectionString() { return '${selectedApp!.name} ${selectedApp!.version}'; } Future queryVersion(String suggestedVersion) async { await openDefaultBrowser( '${selectedApp!.packageName} apk version $suggestedVersion', ); } String getSuggestedVersionString(BuildContext context) { return _patcherAPI.getSuggestedVersion(selectedApp!.packageName); } Future openDefaultBrowser(String query) async { if (Platform.isAndroid) { try { const platform = MethodChannel('app.revanced.manager.flutter/browser'); await platform.invokeMethod('openBrowser', {'query': query}); } catch (e) { if (kDebugMode) { print(e); } } } else { throw 'Platform not supported'; } } bool isPatchNew(Patch patch) { if (savedPatchNames.isEmpty) { savedPatchNames = _managerAPI .getSavedPatches(selectedApp!.packageName) .map((p) => p.name) .toSet(); } if (savedPatchNames.isEmpty) { return false; } else { return !savedPatchNames.contains(patch.name); } } Future loadLastSelectedPatches() async { this.selectedPatches.clear(); removedPatches.clear(); final List selectedPatches = await _managerAPI.getSelectedPatches(selectedApp!.packageName); final List patches = _patcherAPI.getFilteredPatches(selectedApp!.packageName); this .selectedPatches .addAll(patches.where((patch) => selectedPatches.contains(patch.name))); if (!_managerAPI.isPatchesChangeEnabled()) { this.selectedPatches.clear(); this.selectedPatches.addAll(patches.where((patch) => !patch.excluded)); } if (_managerAPI.isVersionCompatibilityCheckEnabled()) { this.selectedPatches.removeWhere((patch) => !isPatchSupported(patch)); } if (!_managerAPI.areUniversalPatchesEnabled()) { this .selectedPatches .removeWhere((patch) => patch.compatiblePackages.isEmpty); } this.selectedPatches.addAll( patches.where( (patch) => isPatchNew(patch) && !patch.excluded && !this.selectedPatches.contains(patch), ), ); final usedPatches = _managerAPI.getUsedPatches(selectedApp!.packageName); for (final patch in usedPatches) { if (!patches.any((p) => p.name == patch.name)) { removedPatches.add('• ${patch.name}'); for (final option in patch.options) { _managerAPI.clearPatchOption( selectedApp!.packageName, patch.name, option.key, ); } } } notifyListeners(); } }