import 'dart:io'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_file_dialog/flutter_file_dialog.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:logcat/logcat.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/services/manager_api.dart'; import 'package:revanced_manager/services/toast.dart'; import 'package:revanced_manager/ui/views/patcher/patcher_viewmodel.dart'; import 'package:revanced_manager/ui/views/patches_selector/patches_selector_viewmodel.dart'; import 'package:revanced_manager/ui/views/settings/settingsFragment/settings_update_language.dart'; import 'package:share_plus/share_plus.dart'; import 'package:stacked/stacked.dart'; import 'package:stacked_services/stacked_services.dart'; class SettingsViewModel extends BaseViewModel { final NavigationService _navigationService = locator(); final ManagerAPI _managerAPI = locator(); final PatchesSelectorViewModel _patchesSelectorViewModel = PatchesSelectorViewModel(); final PatcherViewModel _patcherViewModel = locator(); final Toast _toast = locator(); final SUpdateLanguage sUpdateLanguage = SUpdateLanguage(); void navigateToContributors() { _navigationService.navigateTo(Routes.contributorsView); } bool isPatchesAutoUpdate() { return _managerAPI.isPatchesAutoUpdate(); } void setPatchesAutoUpdate(bool value) { _managerAPI.setPatchesAutoUpdate(value); notifyListeners(); } bool showUpdateDialog() { return _managerAPI.showUpdateDialog(); } void setShowUpdateDialog(bool value) { _managerAPI.setShowUpdateDialog(value); notifyListeners(); } bool isPatchesChangeEnabled() { return _managerAPI.isPatchesChangeEnabled(); } Future showPatchesChangeEnableDialog( bool value, BuildContext context, ) async { if (value) { return showDialog( context: context, builder: (context) => AlertDialog( title: I18nText('warning'), content: I18nText( 'settingsView.enablePatchesSelectionWarningText', child: const Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, ), ), ), actions: [ TextButton( onPressed: () { _managerAPI.setChangingToggleModified(true); _managerAPI.setPatchesChangeEnabled(true); Navigator.of(context).pop(); }, child: I18nText('yesButton'), ), FilledButton( onPressed: () { Navigator.of(context).pop(); }, child: I18nText('noButton'), ), ], ), ); } else { return showDialog( context: context, builder: (context) => AlertDialog( title: I18nText('warning'), content: I18nText( 'settingsView.disablePatchesSelectionWarningText', child: const Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, ), ), ), actions: [ TextButton( onPressed: () { Navigator.of(context).pop(); }, child: I18nText('noButton'), ), FilledButton( onPressed: () { _managerAPI.setChangingToggleModified(true); _patchesSelectorViewModel.selectDefaultPatches(); _managerAPI.setPatchesChangeEnabled(false); Navigator.of(context).pop(); }, child: I18nText('yesButton'), ), ], ), ); } } bool areUniversalPatchesEnabled() { return _managerAPI.areUniversalPatchesEnabled(); } void showUniversalPatches(bool value) { _managerAPI.enableUniversalPatchesStatus(value); notifyListeners(); } bool isVersionCompatibilityCheckEnabled() { return _managerAPI.isVersionCompatibilityCheckEnabled(); } void useVersionCompatibilityCheck(bool value) { _managerAPI.enableVersionCompatibilityCheckStatus(value); notifyListeners(); } bool isRequireSuggestedAppVersionEnabled() { return _managerAPI.isRequireSuggestedAppVersionEnabled(); } Future? showRequireSuggestedAppVersionDialog( BuildContext context, bool value, ) { if (!value) { return showDialog( context: context, builder: (context) => AlertDialog( title: I18nText('warning'), content: I18nText( 'settingsView.requireSuggestedAppVersionDialogText', child: const Text( '', style: TextStyle( fontSize: 16, fontWeight: FontWeight.w500, ), ), ), actions: [ TextButton( onPressed: () { _managerAPI.enableRequireSuggestedAppVersionStatus(false); Navigator.of(context).pop(); }, child: I18nText('yesButton'), ), FilledButton( onPressed: () { Navigator.of(context).pop(); }, child: I18nText('noButton'), ), ], ), ); } else { _managerAPI.enableRequireSuggestedAppVersionStatus(true); if (!_managerAPI.suggestedAppVersionSelected) { _patcherViewModel.selectedApp = null; } return null; } } void deleteKeystore() { _managerAPI.deleteKeystore(); _toast.showBottom('settingsView.regeneratedKeystore'); notifyListeners(); } void deleteTempDir() { _managerAPI.deleteTempFolder(); _toast.showBottom('settingsView.deletedTempDir'); notifyListeners(); } Future exportPatches() async { try { final File outFile = File(_managerAPI.storedPatchesFile); if (outFile.existsSync()) { final String dateTime = DateTime.now().toString().replaceAll(' ', '_').split('.').first; await FlutterFileDialog.saveFile( params: SaveFileDialogParams( sourceFilePath: outFile.path, fileName: 'selected_patches_$dateTime.json', ), ); _toast.showBottom('settingsView.exportedPatches'); } else { _toast.showBottom('settingsView.noExportFileFound'); } } on Exception catch (e) { if (kDebugMode) { print(e); } } } Future importPatches(BuildContext context) async { if (isPatchesChangeEnabled()) { try { final String? result = await FlutterFileDialog.pickFile( params: const OpenFileDialogParams( fileExtensionsFilter: ['json'], ), ); if (result != null) { final File inFile = File(result); inFile.copySync(_managerAPI.storedPatchesFile); inFile.delete(); if (_patcherViewModel.selectedApp != null) { _patcherViewModel.loadLastSelectedPatches(); } _toast.showBottom('settingsView.importedPatches'); } } on Exception catch (e) { if (kDebugMode) { print(e); } _toast.showBottom('settingsView.jsonSelectorErrorMessage'); } } else { _managerAPI.showPatchesChangeWarningDialog(context); } } Future exportKeystore() async { try { final File outFile = File(_managerAPI.keystoreFile); if (outFile.existsSync()) { final String dateTime = DateTime.now().toString().replaceAll(' ', '_').split('.').first; await FlutterFileDialog.saveFile( params: SaveFileDialogParams( sourceFilePath: outFile.path, fileName: 'keystore_$dateTime.keystore', ), ); _toast.showBottom('settingsView.exportedKeystore'); } else { _toast.showBottom('settingsView.noKeystoreExportFileFound'); } } on Exception catch (e) { if (kDebugMode) { print(e); } } } Future importKeystore() async { try { final String? result = await FlutterFileDialog.pickFile(); if (result != null) { final File inFile = File(result); inFile.copySync(_managerAPI.keystoreFile); _toast.showBottom('settingsView.importedKeystore'); } } on Exception catch (e) { if (kDebugMode) { print(e); } _toast.showBottom('settingsView.keystoreSelectorErrorMessage'); } } void resetAllOptions() { _managerAPI.resetAllOptions(); _toast.showBottom('settingsView.resetStoredOptions'); } void resetSelectedPatches() { _managerAPI.resetLastSelectedPatches(); _toast.showBottom('settingsView.resetStoredPatches'); } Future deleteLogs() async { final Directory appCacheDir = await getTemporaryDirectory(); final Directory logsDir = Directory('${appCacheDir.path}/logs'); if (logsDir.existsSync()) { logsDir.deleteSync(recursive: true); } _toast.showBottom('settingsView.deletedLogs'); } Future exportLogcatLogs() async { final Directory appCache = await getTemporaryDirectory(); final Directory logDir = Directory('${appCache.path}/logs'); logDir.createSync(); final String dateTime = DateTime.now() .toIso8601String() .replaceAll('-', '') .replaceAll(':', '') .replaceAll('T', '') .replaceAll('.', ''); final File logcat = File('${logDir.path}/revanced-manager_logcat_$dateTime.log'); final String logs = await Logcat.execute(); logcat.writeAsStringSync(logs); await Share.shareXFiles([XFile(logcat.path)]); } }