revanced-manager/lib/ui/views/installer/installer_viewmodel.dart

190 lines
5.9 KiB
Dart
Raw Normal View History

2022-08-17 18:07:00 +02:00
import 'dart:convert';
import 'package:device_apps/device_apps.dart';
2022-08-18 16:33:33 +02:00
import 'package:flutter/cupertino.dart';
import 'package:flutter/services.dart';
import 'package:flutter_background/flutter_background.dart';
2022-08-18 16:33:33 +02:00
import 'package:flutter_i18n/flutter_i18n.dart';
import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/models/patch.dart';
2022-08-14 20:40:34 +02:00
import 'package:revanced_manager/models/patched_application.dart';
import 'package:revanced_manager/services/patcher_api.dart';
import 'package:revanced_manager/ui/views/patcher/patcher_viewmodel.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:stacked/stacked.dart';
class InstallerViewModel extends BaseViewModel {
2022-08-22 01:49:09 +02:00
final ScrollController scrollController = ScrollController();
2022-08-18 16:33:33 +02:00
final PatcherAPI _patcherAPI = locator<PatcherAPI>();
final PatchedApplication? _app = locator<PatcherViewModel>().selectedApp;
final List<Patch> _patches = locator<PatcherViewModel>().selectedPatches;
static const _installerChannel = MethodChannel(
'app.revanced.manager/installer',
);
double? progress = 0.0;
String logs = '';
bool isPatching = false;
bool isInstalled = false;
2022-08-18 16:33:33 +02:00
Future<void> initialize(BuildContext context) async {
try {
await FlutterBackground.initialize(
2022-08-18 16:33:33 +02:00
androidConfig: FlutterBackgroundAndroidConfig(
notificationTitle: FlutterI18n.translate(
context,
'installerView.notificationTitle',
),
notificationText: FlutterI18n.translate(
context,
'installerView.notificationText',
),
notificationImportance: AndroidNotificationImportance.Default,
2022-08-18 16:33:33 +02:00
notificationIcon: const AndroidResource(
name: 'ic_notification',
defType: 'drawable',
),
),
);
await FlutterBackground.enableBackgroundExecution();
} finally {
2022-08-18 16:33:33 +02:00
await handlePlatformChannelMethods();
await runPatcher();
}
}
2022-08-18 16:33:33 +02:00
Future<dynamic> handlePlatformChannelMethods() async {
_installerChannel.setMethodCallHandler((call) async {
switch (call.method) {
case 'updateProgress':
if (call.arguments != null) {
updateProgress(call.arguments);
}
break;
case 'updateLog':
if (call.arguments != null) {
updateLog(call.arguments);
}
break;
}
});
}
void updateProgress(double value) {
progress = value;
isInstalled = false;
isPatching = progress == 1.0 ? false : true;
if (progress == 0.0) {
logs = '';
}
notifyListeners();
}
void updateLog(String message) {
if (message.isNotEmpty && !message.startsWith('Merging L')) {
if (logs.isNotEmpty) {
logs += '\n';
}
logs += message;
2022-08-22 01:49:09 +02:00
Future.delayed(const Duration(milliseconds: 500)).then((value) {
scrollController.animateTo(
scrollController.position.maxScrollExtent,
duration: const Duration(milliseconds: 200),
curve: Curves.fastOutSlowIn,
);
});
notifyListeners();
}
}
Future<void> runPatcher() async {
updateProgress(0.0);
2022-08-18 16:33:33 +02:00
if (_app != null && _patches.isNotEmpty) {
String apkFilePath = _app!.apkFilePath;
try {
updateLog('Initializing installer');
2022-08-18 16:33:33 +02:00
if (_app!.isRooted && !_app!.isFromStorage) {
updateLog('Checking if an old patched version exists');
2022-08-18 16:33:33 +02:00
bool oldExists = await _patcherAPI.checkOldPatch(_app!);
if (oldExists) {
updateLog('Deleting old patched version');
2022-08-18 16:33:33 +02:00
await _patcherAPI.deleteOldPatch(_app!);
}
}
updateLog('Creating working directory');
bool mergeIntegrations = false;
bool resourcePatching = false;
2022-08-18 16:33:33 +02:00
if (_app!.packageName == 'com.google.android.youtube') {
mergeIntegrations = true;
resourcePatching = true;
2022-08-18 16:33:33 +02:00
} else if (_app!.packageName ==
'com.google.android.apps.youtube.music') {
resourcePatching = true;
}
2022-08-19 20:13:43 +02:00
await _patcherAPI.mergeIntegrations(mergeIntegrations);
2022-08-18 16:33:33 +02:00
await _patcherAPI.runPatcher(
apkFilePath,
2022-08-18 16:33:33 +02:00
_patches,
mergeIntegrations,
resourcePatching,
);
} on Exception {
updateLog('An error occurred! Aborting');
}
} else {
updateLog('No app or patches selected! Aborting');
}
try {
await FlutterBackground.disableBackgroundExecution();
} finally {
isPatching = false;
}
}
void installResult() async {
2022-08-18 16:33:33 +02:00
if (_app != null) {
updateLog(_app!.isRooted
? 'Installing patched file using root method'
: 'Installing patched file using nonroot method');
2022-08-18 16:33:33 +02:00
isInstalled = await _patcherAPI.installPatchedFile(_app!);
if (isInstalled) {
updateLog('Done');
2022-08-18 16:33:33 +02:00
_app!.patchDate = DateTime.now();
_app!.appliedPatches.addAll(_patches.map((p) => p.name).toList());
await saveApp();
2022-08-14 22:32:03 +02:00
} else {
updateLog('An error occurred! Aborting');
2022-08-14 22:32:03 +02:00
}
2022-08-14 20:40:34 +02:00
}
}
void shareResult() {
2022-08-18 16:33:33 +02:00
if (_app != null) {
_patcherAPI.sharePatchedFile(_app!.name, _app!.version);
}
}
Future<void> cleanPatcher() async {
2022-08-18 16:33:33 +02:00
_patcherAPI.cleanPatcher();
locator<PatcherViewModel>().selectedApp = null;
locator<PatcherViewModel>().selectedPatches.clear();
locator<PatcherViewModel>().notifyListeners();
}
void openApp() {
2022-08-18 16:33:33 +02:00
if (_app != null) {
DeviceApps.openApp(_app!.packageName);
}
}
2022-08-18 16:33:33 +02:00
Future<void> saveApp() async {
if (_app != null) {
SharedPreferences prefs = await SharedPreferences.getInstance();
List<String> patchedApps = prefs.getStringList('patchedApps') ?? [];
String appStr = json.encode(_app!.toJson());
patchedApps.removeWhere(
(a) => json.decode(a)['packageName'] == _app!.packageName);
patchedApps.add(appStr);
prefs.setStringList('patchedApps', patchedApps);
}
}
}