revanced-manager/lib/ui/views/patcher/patcher_viewmodel.dart

274 lines
8.1 KiB
Dart
Raw Normal View History

2023-04-20 23:20:30 +05:30
// 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';
2022-08-18 15:33:33 +01:00
import 'package:injectable/injectable.dart';
2022-09-05 15:30:26 +01:00
import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/app/app.router.dart';
2022-08-18 15:33:33 +01:00
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';
2023-04-20 23:20:30 +05:30
import 'package:revanced_manager/utils/about_info.dart';
import 'package:revanced_manager/utils/check_for_supported_patch.dart';
import 'package:stacked/stacked.dart';
2022-09-05 15:30:26 +01:00
import 'package:stacked_services/stacked_services.dart';
2022-08-18 15:33:33 +01:00
@lazySingleton
class PatcherViewModel extends BaseViewModel {
2022-09-05 15:30:26 +01:00
final NavigationService _navigationService = locator<NavigationService>();
final ManagerAPI _managerAPI = locator<ManagerAPI>();
final PatcherAPI _patcherAPI = locator<PatcherAPI>();
2023-12-12 10:37:35 +05:45
Set<String> savedPatchNames = {};
2022-08-18 15:33:33 +01:00
PatchedApplication? selectedApp;
2023-10-12 00:00:39 +00:00
BuildContext? ctx;
2022-08-18 15:33:33 +01:00
List<Patch> selectedPatches = [];
List<String> removedPatches = [];
2022-08-18 15:33:33 +01:00
2022-09-05 15:30:26 +01:00
void navigateToAppSelector() {
_navigationService.navigateTo(Routes.appSelectorView);
}
void navigateToPatchesSelector() {
_navigationService.navigateTo(Routes.patchesSelectorView);
}
void navigateToInstaller() {
_navigationService.navigateTo(Routes.installerView);
}
bool showPatchButton() {
2022-08-18 15:33:33 +01:00
return selectedPatches.isNotEmpty;
}
bool dimPatchesCard() {
2022-08-18 15:33:33 +01:00
return selectedApp == null;
}
2023-10-12 00:00:39 +00:00
bool showRemovedPatchesDialog(BuildContext context) {
if (removedPatches.isNotEmpty) {
2023-10-12 00:00:39 +00:00
showDialog(
context: context,
builder: (context) => AlertDialog(
title: I18nText('notice'),
content: I18nText(
'patcherView.removedPatchesWarningDialogText',
translationParams: {'patches': removedPatches.join('\n')},
),
actions: <Widget>[
TextButton(
2023-10-12 00:00:39 +00:00
onPressed: () {
Navigator.of(context).pop();
},
child: I18nText('noButton'),
),
FilledButton(
onPressed: () {
Navigator.of(context).pop();
2023-10-12 00:00:39 +00:00
showArmv7WarningDialog(context);
},
child: I18nText('yesButton'),
),
],
),
);
2023-10-12 00:00:39 +00:00
return false;
}
2023-10-12 00:00:39 +00:00
return true;
}
bool checkRequiredPatchOption(BuildContext context) {
if (getNullRequiredOptions(selectedPatches, selectedApp!.packageName)
.isNotEmpty) {
2023-10-12 00:00:39 +00:00
showRequiredOptionDialog(context);
return false;
}
return true;
}
void showRequiredOptionDialog([context]) {
showDialog(
context: context ?? ctx,
builder: (context) => AlertDialog(
title: I18nText('notice'),
content: I18nText('patcherView.requiredOptionDialogText'),
actions: <Widget>[
TextButton(
2023-10-12 00:00:39 +00:00
onPressed: () => {
Navigator.of(context).pop(),
},
child: I18nText('cancelButton'),
2023-10-12 00:00:39 +00:00
),
FilledButton(
2023-10-12 00:00:39 +00:00
onPressed: () => {
Navigator.pop(context),
navigateToPatchesSelector(),
},
child: I18nText('okButton'),
2023-10-12 00:00:39 +00:00
),
],
),
);
}
2023-04-20 23:20:30 +05:30
Future<void> showArmv7WarningDialog(BuildContext context) async {
final bool armv7 = await AboutInfo.getInfo().then((info) {
final List<String> archs = info['supportedArch'];
final supportedAbis = ['arm64-v8a', 'x86', 'x86_64'];
return !archs.any((arch) => supportedAbis.contains(arch));
});
2023-04-20 23:20:30 +05:30
if (context.mounted && armv7) {
return showDialog(
context: context,
builder: (context) => AlertDialog(
title: I18nText('warning'),
content: I18nText('patcherView.armv7WarningDialogText'),
actions: <Widget>[
FilledButton(
2023-04-20 23:20:30 +05:30
onPressed: () => Navigator.of(context).pop(),
child: I18nText('noButton'),
2023-04-20 23:20:30 +05:30
),
TextButton(
2023-04-20 23:20:30 +05:30
onPressed: () {
Navigator.of(context).pop();
navigateToInstaller();
},
child: I18nText('yesButton'),
),
2023-04-20 23:20:30 +05:30
],
),
);
} else {
navigateToInstaller();
}
}
String getAppSelectionString() {
String text = '${selectedApp!.name} (${selectedApp!.packageName})';
if (text.length > 32) {
text = '${text.substring(0, 32)}...)';
}
return text;
}
String getCurrentVersionString(BuildContext context) {
return '${FlutterI18n.translate(
context,
'appSelectorCard.currentVersion',
)}: v${selectedApp!.version}';
}
Future<void> searchSuggestedVersionOnWeb() async {
final String suggestedVersion =
_patcherAPI.getSuggestedVersion(selectedApp!.packageName);
if (suggestedVersion.isNotEmpty) {
await openDefaultBrowser(
2023-12-12 10:37:35 +05:45
'${selectedApp!.packageName} apk version v$suggestedVersion',
);
} else {
await openDefaultBrowser('${selectedApp!.packageName} apk');
}
}
String getSuggestedVersion() {
return _patcherAPI.getSuggestedVersion(selectedApp!.packageName);
}
String getSuggestedVersionString(BuildContext context) {
String suggestedVersion =
_patcherAPI.getSuggestedVersion(selectedApp!.packageName);
if (suggestedVersion.isEmpty) {
suggestedVersion = FlutterI18n.translate(
context,
'appSelectorCard.allVersions',
);
} else {
suggestedVersion = 'v$suggestedVersion';
}
return '${FlutterI18n.translate(
context,
'appSelectorCard.suggestedVersion',
)}: $suggestedVersion';
}
Future<void> 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';
}
}
2023-12-12 10:37:35 +05:45
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<void> loadLastSelectedPatches() async {
this.selectedPatches.clear();
removedPatches.clear();
final List<String> selectedPatches =
await _managerAPI.getSelectedPatches(selectedApp!.packageName);
final List<Patch> 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);
}
2023-12-12 10:37:35 +05:45
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)) {
2023-10-12 00:00:39 +00:00
removedPatches.add('${patch.name}');
for (final option in patch.options) {
_managerAPI.clearPatchOption(
2023-12-12 10:37:35 +05:45
selectedApp!.packageName,
patch.name,
option.key,
);
2023-10-12 00:00:39 +00:00
}
}
}
notifyListeners();
}
}