Merge branch 'flutter' of https://github.com/revanced/revanced-manager into flutter

This commit is contained in:
Aunali321 2022-09-12 18:23:39 +05:30
commit a3c14e0a51
22 changed files with 580 additions and 371 deletions

View File

@ -25,7 +25,8 @@
"downloadingMessage": "Downloading update!", "downloadingMessage": "Downloading update!",
"installingMessage": "Installing update!", "installingMessage": "Installing update!",
"errorDownloadMessage": "Unable to download update!", "errorDownloadMessage": "Unable to download update!",
"errorInstallMessage": "Unable to download update!" "errorInstallMessage": "Unable to download update!",
"noConnection": "No internet connection"
}, },
"applicationItem": { "applicationItem": {
"patchButton": "Patch", "patchButton": "Patch",

View File

@ -1,5 +1,7 @@
import 'package:revanced_manager/services/github_api.dart';
import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/services/manager_api.dart';
import 'package:revanced_manager/services/patcher_api.dart'; import 'package:revanced_manager/services/patcher_api.dart';
import 'package:revanced_manager/services/revanced_api.dart';
import 'package:revanced_manager/ui/views/app_selector/app_selector_view.dart'; import 'package:revanced_manager/ui/views/app_selector/app_selector_view.dart';
import 'package:revanced_manager/ui/views/contributors/contributors_view.dart'; import 'package:revanced_manager/ui/views/contributors/contributors_view.dart';
import 'package:revanced_manager/ui/views/home/home_viewmodel.dart'; import 'package:revanced_manager/ui/views/home/home_viewmodel.dart';
@ -32,6 +34,8 @@ import 'package:stacked_services/stacked_services.dart';
LazySingleton(classType: NavigationService), LazySingleton(classType: NavigationService),
LazySingleton(classType: ManagerAPI), LazySingleton(classType: ManagerAPI),
LazySingleton(classType: PatcherAPI), LazySingleton(classType: PatcherAPI),
LazySingleton(classType: RevancedAPI),
LazySingleton(classType: GithubAPI),
], ],
) )
class AppSetup {} class AppSetup {}

View File

@ -1,10 +1,11 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:flutter_i18n/flutter_i18n.dart';
// ignore: depend_on_referenced_packages
import 'package:flutter_localizations/flutter_localizations.dart'; import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/services/github_api.dart';
import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/services/manager_api.dart';
import 'package:revanced_manager/services/patcher_api.dart'; import 'package:revanced_manager/services/patcher_api.dart';
import 'package:revanced_manager/services/revanced_api.dart';
import 'package:revanced_manager/ui/theme/dynamic_theme_builder.dart'; import 'package:revanced_manager/ui/theme/dynamic_theme_builder.dart';
import 'package:revanced_manager/ui/views/navigation/navigation_view.dart'; import 'package:revanced_manager/ui/views/navigation/navigation_view.dart';
import 'package:stacked_themes/stacked_themes.dart'; import 'package:stacked_themes/stacked_themes.dart';
@ -15,6 +16,8 @@ Future main() async {
WidgetsFlutterBinding.ensureInitialized(); WidgetsFlutterBinding.ensureInitialized();
await locator<ManagerAPI>().initialize(); await locator<ManagerAPI>().initialize();
await locator<PatcherAPI>().initialize(); await locator<PatcherAPI>().initialize();
locator<RevancedAPI>().initialize();
locator<GithubAPI>().initialize();
runApp(const MyApp()); runApp(const MyApp());
} }

View File

@ -1,11 +1,21 @@
import 'dart:convert';
import 'dart:io'; import 'dart:io';
import 'package:collection/collection.dart';
import 'package:dio/dio.dart';
import 'package:dio_http_cache_lts/dio_http_cache_lts.dart';
import 'package:flutter_cache_manager/flutter_cache_manager.dart'; import 'package:flutter_cache_manager/flutter_cache_manager.dart';
import 'package:github/github.dart'; import 'package:injectable/injectable.dart';
import 'package:timeago/timeago.dart'; import 'package:revanced_manager/models/patch.dart';
@lazySingleton
class GithubAPI { class GithubAPI {
final GitHub _github = GitHub(); final String apiUrl = 'https://api.github.com';
final Dio _dio = Dio();
final DioCacheManager _dioCacheManager = DioCacheManager(CacheConfig());
final Options _cacheOptions = buildCacheOptions(
const Duration(hours: 1),
maxStale: const Duration(days: 7),
);
final Map<String, String> repoAppPath = { final Map<String, String> repoAppPath = {
'com.google.android.youtube': 'youtube', 'com.google.android.youtube': 'youtube',
'com.google.android.apps.youtube.music': 'music', 'com.google.android.apps.youtube.music': 'music',
@ -16,31 +26,66 @@ class GithubAPI {
'com.garzotto.pflotsh.ecmwf_a': 'ecmwf', 'com.garzotto.pflotsh.ecmwf_a': 'ecmwf',
}; };
Future<String?> latestReleaseVersion(String repoName) async { void initialize() {
_dio.interceptors.add(_dioCacheManager.interceptor);
}
Future<void> clearAllCache() async {
await _dioCacheManager.clearAll();
}
Future<Map<String, dynamic>?> _getLatestRelease(String repoName) async {
try { try {
var latestRelease = await _github.repositories.getLatestRelease( var response = await _dio.get(
RepositorySlug.full(repoName), '$apiUrl/repos/$repoName/releases/latest',
options: _cacheOptions,
); );
return latestRelease.tagName; return response.data;
} on Exception { } on Exception {
return null; return null;
} }
} }
Future<File?> latestReleaseFile(String extension, String repoName) async { Future<List<String>> getCommits(
String packageName,
String repoName,
DateTime since,
) async {
String path =
'src/main/kotlin/app/revanced/patches/${repoAppPath[packageName]}';
try { try {
var latestRelease = await _github.repositories.getLatestRelease( var response = await _dio.get(
RepositorySlug.full(repoName), '$apiUrl/repos/$repoName/commits',
queryParameters: {
'path': path,
'per_page': 3,
'since': since.toIso8601String(),
},
options: _cacheOptions,
); );
String? url = latestRelease.assets List<dynamic> commits = response.data;
?.firstWhere((asset) => return commits
asset.name != null && .map((commit) =>
asset.name!.endsWith(extension) && (commit['commit']['message'] as String).split('\n')[0])
!asset.name!.contains('-sources') && .toList();
!asset.name!.contains('-javadoc')) } on Exception {
.browserDownloadUrl; return List.empty();
if (url != null) { }
return await DefaultCacheManager().getSingleFile(url); }
Future<File?> getLatestReleaseFile(String extension, String repoName) async {
try {
Map<String, dynamic>? release = await _getLatestRelease(repoName);
if (release != null) {
Map<String, dynamic>? asset =
(release['assets'] as List<dynamic>).firstWhereOrNull(
(asset) => (asset['name'] as String).endsWith(extension),
);
if (asset != null) {
return await DefaultCacheManager().getSingleFile(
asset['browser_download_url'],
);
}
} }
} on Exception { } on Exception {
return null; return null;
@ -48,37 +93,17 @@ class GithubAPI {
return null; return null;
} }
Future<String> latestCommitTime(String repoName) async { Future<List<Patch>> getPatches(String repoName) async {
List<Patch> patches = [];
try { try {
var repo = await _github.repositories.getRepository( File? f = await getLatestReleaseFile('.json', repoName);
RepositorySlug.full(repoName), if (f != null) {
); List<dynamic> list = jsonDecode(f.readAsStringSync());
return repo.pushedAt != null patches = list.map((patch) => Patch.fromJson(patch)).toList();
? format(repo.pushedAt!, locale: 'en_short') }
: '';
} on Exception { } on Exception {
return ''; return List.empty();
} }
} return patches;
Future<List<Contributor>> getContributors(String repoName) async {
return await (_github.repositories.listContributors(
RepositorySlug.full(repoName),
)).toList();
}
Future<List<RepositoryCommit>> getCommits(
String packageName,
String repoName,
) async {
String path =
'src/main/kotlin/app/revanced/patches/${repoAppPath[packageName]}';
return await (PaginationHelper(_github)
.objects<Map<String, dynamic>, RepositoryCommit>(
'GET',
'/repos/$repoName/commits',
(i) => RepositoryCommit.fromJson(i),
params: <String, dynamic>{'path': path},
)).toList();
} }
} }

View File

@ -1,17 +1,20 @@
import 'dart:convert'; import 'dart:convert';
import 'dart:io'; import 'dart:io';
import 'package:device_apps/device_apps.dart'; import 'package:device_apps/device_apps.dart';
import 'package:github/github.dart';
import 'package:injectable/injectable.dart'; import 'package:injectable/injectable.dart';
import 'package:package_info_plus/package_info_plus.dart'; import 'package:package_info_plus/package_info_plus.dart';
import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/models/patch.dart';
import 'package:revanced_manager/models/patched_application.dart'; import 'package:revanced_manager/models/patched_application.dart';
import 'package:revanced_manager/services/github_api.dart'; import 'package:revanced_manager/services/github_api.dart';
import 'package:revanced_manager/services/revanced_api.dart';
import 'package:revanced_manager/services/root_api.dart'; import 'package:revanced_manager/services/root_api.dart';
import 'package:shared_preferences/shared_preferences.dart'; import 'package:shared_preferences/shared_preferences.dart';
@lazySingleton @lazySingleton
class ManagerAPI { class ManagerAPI {
final GithubAPI _githubAPI = GithubAPI(); final RevancedAPI _revancedAPI = locator<RevancedAPI>();
final GithubAPI _githubAPI = locator<GithubAPI>();
final RootAPI _rootAPI = RootAPI(); final RootAPI _rootAPI = RootAPI();
final String patcherRepo = 'revanced-patcher'; final String patcherRepo = 'revanced-patcher';
final String cliRepo = 'revanced-cli'; final String cliRepo = 'revanced-cli';
@ -26,10 +29,6 @@ class ManagerAPI {
_prefs = await SharedPreferences.getInstance(); _prefs = await SharedPreferences.getInstance();
} }
String getPatcherRepo() {
return defaultPatcherRepo;
}
String getPatchesRepo() { String getPatchesRepo() {
return _prefs.getString('patchesRepo') ?? defaultPatchesRepo; return _prefs.getString('patchesRepo') ?? defaultPatchesRepo;
} }
@ -52,46 +51,6 @@ class ManagerAPI {
await _prefs.setString('integrationsRepo', value); await _prefs.setString('integrationsRepo', value);
} }
String getCliRepo() {
return defaultCliRepo;
}
String getManagerRepo() {
return _prefs.getString('managerRepo') ?? defaultManagerRepo;
}
Future<void> setManagerRepo(String value) async {
if (value.isEmpty || value.startsWith('/') || value.endsWith('/')) {
value = defaultManagerRepo;
}
await _prefs.setString('managerRepo', value);
}
Future<File?> downloadPatches(String extension) async {
return await _githubAPI.latestReleaseFile(extension, getPatchesRepo());
}
Future<File?> downloadIntegrations(String extension) async {
return await _githubAPI.latestReleaseFile(extension, getIntegrationsRepo());
}
Future<File?> downloadManager(String extension) async {
return await _githubAPI.latestReleaseFile(extension, getManagerRepo());
}
Future<String?> getLatestPatchesVersion() async {
return await _githubAPI.latestReleaseVersion(getPatchesRepo());
}
Future<String?> getLatestManagerVersion() async {
return await _githubAPI.latestReleaseVersion(getManagerRepo());
}
Future<String> getCurrentManagerVersion() async {
PackageInfo packageInfo = await PackageInfo.fromPlatform();
return packageInfo.version;
}
bool getUseDynamicTheme() { bool getUseDynamicTheme() {
return _prefs.getBool('useDynamicTheme') ?? false; return _prefs.getBool('useDynamicTheme') ?? false;
} }
@ -110,9 +69,7 @@ class ManagerAPI {
List<PatchedApplication> getPatchedApps() { List<PatchedApplication> getPatchedApps() {
List<String> apps = _prefs.getStringList('patchedApps') ?? []; List<String> apps = _prefs.getStringList('patchedApps') ?? [];
return apps return apps.map((a) => PatchedApplication.fromJson(jsonDecode(a))).toList();
.map((a) => PatchedApplication.fromJson(json.decode(a)))
.toList();
} }
Future<void> setPatchedApps(List<PatchedApplication> patchedApps) async { Future<void> setPatchedApps(List<PatchedApplication> patchedApps) async {
@ -143,6 +100,71 @@ class ManagerAPI {
await setPatchedApps(patchedApps); await setPatchedApps(patchedApps);
} }
void clearAllData() {
_revancedAPI.clearAllCache();
_githubAPI.clearAllCache();
}
Future<Map<String, List<dynamic>>> getContributors() async {
return await _revancedAPI.getContributors();
}
Future<List<Patch>> getPatches() async {
if (getPatchesRepo() == defaultPatchesRepo) {
return await _revancedAPI.getPatches();
} else {
return await _githubAPI.getPatches(getPatchesRepo());
}
}
Future<File?> downloadPatches() async {
String repoName = getPatchesRepo();
if (repoName == defaultPatchesRepo) {
return await _revancedAPI.getLatestReleaseFile(
'.jar',
defaultPatchesRepo,
);
} else {
return await _githubAPI.getLatestReleaseFile('.jar', repoName);
}
}
Future<File?> downloadIntegrations() async {
String repoName = getIntegrationsRepo();
if (repoName == defaultIntegrationsRepo) {
return await _revancedAPI.getLatestReleaseFile(
'.apk',
defaultIntegrationsRepo,
);
} else {
return await _githubAPI.getLatestReleaseFile('.apk', repoName);
}
}
Future<File?> downloadManager() async {
return await _revancedAPI.getLatestReleaseFile('.apk', defaultManagerRepo);
}
Future<String?> getLatestPatcherReleaseTime() async {
return await _revancedAPI.getLatestReleaseTime('.gz', defaultPatcherRepo);
}
Future<String?> getLatestManagerReleaseTime() async {
return await _revancedAPI.getLatestReleaseTime('.apk', defaultManagerRepo);
}
Future<String?> getLatestManagerVersion() async {
return await _revancedAPI.getLatestReleaseVersion(
'.apk',
defaultManagerRepo,
);
}
Future<String> getCurrentManagerVersion() async {
PackageInfo packageInfo = await PackageInfo.fromPlatform();
return packageInfo.version;
}
Future<void> reAssessSavedApps() async { Future<void> reAssessSavedApps() async {
List<PatchedApplication> patchedApps = getPatchedApps(); List<PatchedApplication> patchedApps = getPatchedApps();
List<PatchedApplication> toRemove = []; List<PatchedApplication> toRemove = [];
@ -183,40 +205,27 @@ class ManagerAPI {
} }
Future<bool> hasAppUpdates(String packageName, DateTime patchDate) async { Future<bool> hasAppUpdates(String packageName, DateTime patchDate) async {
List<RepositoryCommit> commits = await _githubAPI.getCommits( List<String> commits = await _githubAPI.getCommits(
packageName, packageName,
getPatchesRepo(), getPatchesRepo(),
patchDate,
); );
return commits.any((c) => return commits.isNotEmpty;
c.commit != null &&
c.commit!.author != null &&
c.commit!.author!.date != null &&
c.commit!.author!.date!.isAfter(patchDate));
} }
Future<List<String>> getAppChangelog( Future<List<String>> getAppChangelog(
String packageName, String packageName, DateTime patchDate) async {
DateTime patchDate, List<String> newCommits = await _githubAPI.getCommits(
) async {
List<RepositoryCommit> commits = await _githubAPI.getCommits(
packageName, packageName,
getPatchesRepo(), getPatchesRepo(),
patchDate,
); );
List<String> newCommits = commits
.where((c) =>
c.commit != null &&
c.commit!.author != null &&
c.commit!.author!.date != null &&
c.commit!.author!.date!.isAfter(patchDate) &&
c.commit!.message != null)
.map((c) => c.commit!.message!)
.toList();
if (newCommits.isEmpty) { if (newCommits.isEmpty) {
newCommits = commits newCommits = await _githubAPI.getCommits(
.where((c) => c.commit != null && c.commit!.message != null) packageName,
.take(3) getPatchesRepo(),
.map((c) => c.commit!.message!) DateTime(2022, 3, 20, 21, 06, 01),
.toList(); );
} }
return newCommits; return newCommits;
} }

View File

@ -1,6 +1,6 @@
import 'dart:convert';
import 'dart:io'; import 'dart:io';
import 'package:app_installer/app_installer.dart'; import 'package:app_installer/app_installer.dart';
import 'package:collection/collection.dart';
import 'package:device_apps/device_apps.dart'; import 'package:device_apps/device_apps.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:injectable/injectable.dart'; import 'package:injectable/injectable.dart';
@ -40,11 +40,7 @@ class PatcherAPI {
Future<void> _loadPatches() async { Future<void> _loadPatches() async {
try { try {
if (_patches.isEmpty) { if (_patches.isEmpty) {
File? patchJsonFile = await _managerAPI.downloadPatches('.json'); _patches = await _managerAPI.getPatches();
if (patchJsonFile != null) {
List<dynamic> list = json.decode(patchJsonFile.readAsStringSync());
_patches = list.map((patch) => Patch.fromJson(patch)).toList();
}
} }
} on Exception { } on Exception {
_patches = List.empty(); _patches = List.empty();
@ -53,7 +49,6 @@ class PatcherAPI {
Future<List<ApplicationWithIcon>> getFilteredInstalledApps() async { Future<List<ApplicationWithIcon>> getFilteredInstalledApps() async {
List<ApplicationWithIcon> filteredApps = []; List<ApplicationWithIcon> filteredApps = [];
await _loadPatches();
for (Patch patch in _patches) { for (Patch patch in _patches) {
for (Package package in patch.compatiblePackages) { for (Package package in patch.compatiblePackages) {
try { try {
@ -74,7 +69,6 @@ class PatcherAPI {
} }
Future<List<Patch>> getFilteredPatches(String packageName) async { Future<List<Patch>> getFilteredPatches(String packageName) async {
await _loadPatches();
return _patches return _patches
.where((patch) => .where((patch) =>
!patch.name.contains('settings') && !patch.name.contains('settings') &&
@ -83,7 +77,6 @@ class PatcherAPI {
} }
Future<List<Patch>> getAppliedPatches(List<String> appliedPatches) async { Future<List<Patch>> getAppliedPatches(List<String> appliedPatches) async {
await _loadPatches();
return _patches return _patches
.where((patch) => appliedPatches.contains(patch.name)) .where((patch) => appliedPatches.contains(patch.name))
.toList(); .toList();
@ -105,20 +98,22 @@ class PatcherAPI {
); );
if (includeSettings) { if (includeSettings) {
try { try {
Patch settingsPatch = _patches.firstWhere( Patch? settingsPatch = _patches.firstWhereOrNull(
(patch) => (patch) =>
patch.name.contains('settings') && patch.name.contains('settings') &&
patch.compatiblePackages.any((pack) => pack.name == packageName), patch.compatiblePackages.any((pack) => pack.name == packageName),
); );
selectedPatches.add(settingsPatch); if (settingsPatch != null) {
selectedPatches.add(settingsPatch);
}
} catch (e) { } catch (e) {
// ignore // ignore
} }
} }
File? patchBundleFile = await _managerAPI.downloadPatches('.jar'); File? patchBundleFile = await _managerAPI.downloadPatches();
File? integrationsFile; File? integrationsFile;
if (mergeIntegrations) { if (mergeIntegrations) {
integrationsFile = await _managerAPI.downloadIntegrations('.apk'); integrationsFile = await _managerAPI.downloadIntegrations();
} }
if (patchBundleFile != null) { if (patchBundleFile != null) {
_tmpDir.createSync(); _tmpDir.createSync();

View File

@ -0,0 +1,121 @@
import 'dart:io';
import 'package:collection/collection.dart';
import 'package:dio/dio.dart';
import 'package:dio_http_cache_lts/dio_http_cache_lts.dart';
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
import 'package:injectable/injectable.dart';
import 'package:revanced_manager/models/patch.dart';
import 'package:timeago/timeago.dart';
@lazySingleton
class RevancedAPI {
final String apiUrl = 'https://revanced-releases-api.afterst0rm.xyz';
final Dio _dio = Dio();
final DioCacheManager _dioCacheManager = DioCacheManager(CacheConfig());
final Options _cacheOptions = buildCacheOptions(
const Duration(hours: 1),
maxStale: const Duration(days: 7),
);
void initialize() {
_dio.interceptors.add(_dioCacheManager.interceptor);
}
Future<void> clearAllCache() async {
await _dioCacheManager.clearAll();
}
Future<Map<String, List<dynamic>>> getContributors() async {
Map<String, List<dynamic>> contributors = {};
try {
var response = await _dio.get(
'$apiUrl/contributors',
options: _cacheOptions,
);
List<dynamic> repositories = response.data['repositories'];
for (Map<String, dynamic> repo in repositories) {
String name = repo['name'];
contributors[name] = repo['contributors'];
}
} on Exception {
return {};
}
return contributors;
}
Future<List<Patch>> getPatches() async {
try {
var response = await _dio.get('$apiUrl/patches', options: _cacheOptions);
List<dynamic> patches = response.data;
return patches.map((patch) => Patch.fromJson(patch)).toList();
} on Exception {
return List.empty();
}
}
Future<Map<String, dynamic>?> _getLatestRelease(
String extension,
String repoName,
) async {
try {
var response = await _dio.get('$apiUrl/tools', options: _cacheOptions);
List<dynamic> tools = response.data['tools'];
return tools.firstWhereOrNull(
(t) =>
t['repository'] == repoName &&
(t['name'] as String).endsWith(extension),
);
} on Exception {
return null;
}
}
Future<String?> getLatestReleaseVersion(
String extension, String repoName) async {
try {
Map<String, dynamic>? release =
await _getLatestRelease(extension, repoName);
if (release != null) {
return release['version'];
}
} on Exception {
return null;
}
return null;
}
Future<File?> getLatestReleaseFile(String extension, String repoName) async {
try {
Map<String, dynamic>? release = await _getLatestRelease(
extension,
repoName,
);
if (release != null) {
String url = release['browser_download_url'];
return await DefaultCacheManager().getSingleFile(url);
}
} on Exception {
return null;
}
return null;
}
Future<String?> getLatestReleaseTime(
String extension,
String repoName,
) async {
try {
Map<String, dynamic>? release = await _getLatestRelease(
extension,
repoName,
);
if (release != null) {
DateTime timestamp = DateTime.parse(release['timestamp'] as String);
return format(timestamp, locale: 'en_short');
}
} on Exception {
return null;
}
return null;
}
}

View File

@ -35,29 +35,29 @@ class _AppSelectorViewState extends State<AppSelectorView> {
child: Padding( child: Padding(
padding: padding:
const EdgeInsets.symmetric(vertical: 4.0, horizontal: 12.0), const EdgeInsets.symmetric(vertical: 4.0, horizontal: 12.0),
child: model.noApps child: Column(
? Center( children: <Widget>[
child: I18nText('appSelectorCard.noAppsLabel'), SearchBar(
) showSelectIcon: false,
: model.apps.isEmpty hintText: FlutterI18n.translate(
? const AppSkeletonLoader() context,
: Column( 'appSelectorView.searchBarHint',
children: <Widget>[ ),
SearchBar( onQueryChanged: (searchQuery) {
showSelectIcon: false, setState(() {
hintText: FlutterI18n.translate( _query = searchQuery;
context, });
'appSelectorView.searchBarHint', },
), ),
onQueryChanged: (searchQuery) { const SizedBox(height: 12),
setState(() { Expanded(
_query = searchQuery; child: model.noApps
}); ? Center(
}, child: I18nText('appSelectorCard.noAppsLabel'),
), )
const SizedBox(height: 12), : model.apps.isEmpty
Expanded( ? const AppSkeletonLoader()
child: ListView( : ListView(
padding: const EdgeInsets.only(bottom: 80), padding: const EdgeInsets.only(bottom: 80),
children: model children: model
.getFilteredApps(_query) .getFilteredApps(_query)
@ -74,9 +74,9 @@ class _AppSelectorViewState extends State<AppSelectorView> {
)) ))
.toList(), .toList(),
), ),
), ),
], ],
), ),
), ),
), ),
), ),

View File

@ -1,34 +1,24 @@
import 'package:github/github.dart';
import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/services/github_api.dart';
import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/services/manager_api.dart';
import 'package:stacked/stacked.dart'; import 'package:stacked/stacked.dart';
class ContributorsViewModel extends BaseViewModel { class ContributorsViewModel extends BaseViewModel {
final ManagerAPI _managerAPI = locator<ManagerAPI>(); final ManagerAPI _managerAPI = locator<ManagerAPI>();
final GithubAPI _githubAPI = GithubAPI(); List<dynamic> patcherContributors = [];
List<Contributor> patcherContributors = []; List<dynamic> patchesContributors = [];
List<Contributor> patchesContributors = []; List<dynamic> integrationsContributors = [];
List<Contributor> integrationsContributors = []; List<dynamic> cliContributors = [];
List<Contributor> cliContributors = []; List<dynamic> managerContributors = [];
List<Contributor> managerContributors = [];
Future<void> getContributors() async { Future<void> getContributors() async {
patcherContributors = await _githubAPI.getContributors( Map<String, List<dynamic>> contributors =
_managerAPI.getPatcherRepo(), await _managerAPI.getContributors();
); patcherContributors = contributors[_managerAPI.defaultPatcherRepo] ?? [];
patchesContributors = await _githubAPI.getContributors( patchesContributors = contributors[_managerAPI.getPatchesRepo()] ?? [];
_managerAPI.getPatchesRepo(), integrationsContributors =
); contributors[_managerAPI.getIntegrationsRepo()] ?? [];
integrationsContributors = await _githubAPI.getContributors( cliContributors = contributors[_managerAPI.defaultCliRepo] ?? [];
_managerAPI.getIntegrationsRepo(), managerContributors = contributors[_managerAPI.defaultManagerRepo] ?? [];
);
cliContributors = await _githubAPI.getContributors(
_managerAPI.getCliRepo(),
);
managerContributors = await _githubAPI.getContributors(
_managerAPI.getManagerRepo(),
);
notifyListeners(); notifyListeners();
} }
} }

View File

@ -17,76 +17,81 @@ class HomeView extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return ViewModelBuilder<HomeViewModel>.reactive( return ViewModelBuilder<HomeViewModel>.reactive(
disposeViewModel: false, disposeViewModel: false,
onModelReady: (model) => model.initialize(), onModelReady: (model) => model.initialize(context),
viewModelBuilder: () => locator<HomeViewModel>(), viewModelBuilder: () => locator<HomeViewModel>(),
builder: (context, model, child) => Scaffold( builder: (context, model, child) => Scaffold(
body: CustomScrollView( body: RefreshIndicator(
slivers: <Widget>[ color: Theme.of(context).colorScheme.secondary,
CustomSliverAppBar( backgroundColor: Theme.of(context).colorScheme.secondaryContainer,
title: I18nText( onRefresh: () => model.forceRefresh(context),
'homeView.widgetTitle', child: CustomScrollView(
child: Text( slivers: <Widget>[
'', CustomSliverAppBar(
style: GoogleFonts.inter( title: I18nText(
color: Theme.of(context).textTheme.headline6!.color, 'homeView.widgetTitle',
child: Text(
'',
style: GoogleFonts.inter(
color: Theme.of(context).textTheme.headline6!.color,
),
), ),
), ),
), ),
), SliverPadding(
SliverPadding( padding: const EdgeInsets.symmetric(horizontal: 20.0),
padding: const EdgeInsets.symmetric(horizontal: 20.0), sliver: SliverList(
sliver: SliverList( delegate: SliverChildListDelegate.fixed(
delegate: SliverChildListDelegate.fixed( <Widget>[
<Widget>[ I18nText(
I18nText( 'homeView.updatesSubtitle',
'homeView.updatesSubtitle', child: Text(
child: Text( '',
'', style: Theme.of(context).textTheme.headline6!,
style: Theme.of(context).textTheme.headline6!,
),
),
const SizedBox(height: 10),
LatestCommitCard(
onPressed: () =>
model.showUpdateConfirmationDialog(context),
),
const SizedBox(height: 23),
I18nText(
'homeView.patchedSubtitle',
child: Text(
'',
style: Theme.of(context).textTheme.headline6!,
),
),
const SizedBox(height: 8),
Row(
children: <Widget>[
DashboardChip(
label: I18nText('homeView.updatesAvailable'),
isSelected: model.showUpdatableApps,
onSelected: (value) {
model.toggleUpdatableApps(true);
},
), ),
const SizedBox(width: 10), ),
DashboardChip( const SizedBox(height: 10),
label: I18nText('homeView.installed'), LatestCommitCard(
isSelected: !model.showUpdatableApps, onPressed: () =>
onSelected: (value) { model.showUpdateConfirmationDialog(context),
model.toggleUpdatableApps(false); ),
}, const SizedBox(height: 23),
) I18nText(
], 'homeView.patchedSubtitle',
), child: Text(
const SizedBox(height: 14), '',
model.showUpdatableApps style: Theme.of(context).textTheme.headline6!,
? AvailableUpdatesCard() ),
: InstalledAppsCard(), ),
], const SizedBox(height: 8),
Row(
children: <Widget>[
DashboardChip(
label: I18nText('homeView.updatesAvailable'),
isSelected: model.showUpdatableApps,
onSelected: (value) {
model.toggleUpdatableApps(true);
},
),
const SizedBox(width: 10),
DashboardChip(
label: I18nText('homeView.installed'),
isSelected: !model.showUpdatableApps,
onSelected: (value) {
model.toggleUpdatableApps(false);
},
)
],
),
const SizedBox(height: 14),
model.showUpdatableApps
? AvailableUpdatesCard()
: InstalledAppsCard(),
],
),
), ),
), ),
), ],
], ),
), ),
), ),
); );

View File

@ -1,6 +1,7 @@
// ignore_for_file: use_build_context_synchronously // ignore_for_file: use_build_context_synchronously
import 'dart:io'; import 'dart:io';
import 'package:app_installer/app_installer.dart'; import 'package:app_installer/app_installer.dart';
import 'package:cross_connectivity/cross_connectivity.dart';
import 'package:device_apps/device_apps.dart'; import 'package:device_apps/device_apps.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:flutter_i18n/flutter_i18n.dart';
@ -23,13 +24,13 @@ class HomeViewModel extends BaseViewModel {
final NavigationService _navigationService = locator<NavigationService>(); final NavigationService _navigationService = locator<NavigationService>();
final ManagerAPI _managerAPI = locator<ManagerAPI>(); final ManagerAPI _managerAPI = locator<ManagerAPI>();
final PatcherAPI _patcherAPI = locator<PatcherAPI>(); final PatcherAPI _patcherAPI = locator<PatcherAPI>();
final FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = final flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin();
FlutterLocalNotificationsPlugin(); DateTime? _lastUpdate;
bool showUpdatableApps = true; bool showUpdatableApps = true;
List<PatchedApplication> patchedInstalledApps = []; List<PatchedApplication> patchedInstalledApps = [];
List<PatchedApplication> patchedUpdatableApps = []; List<PatchedApplication> patchedUpdatableApps = [];
Future<void> initialize() async { Future<void> initialize(BuildContext context) async {
await flutterLocalNotificationsPlugin.initialize( await flutterLocalNotificationsPlugin.initialize(
const InitializationSettings( const InitializationSettings(
android: AndroidInitializationSettings('ic_notification'), android: AndroidInitializationSettings('ic_notification'),
@ -37,6 +38,17 @@ class HomeViewModel extends BaseViewModel {
onSelectNotification: (p) => onSelectNotification: (p) =>
DeviceApps.openApp('app.revanced.manager.flutter'), DeviceApps.openApp('app.revanced.manager.flutter'),
); );
bool isConnected = await Connectivity().checkConnection();
if (!isConnected) {
Fluttertoast.showToast(
msg: FlutterI18n.translate(
context,
'homeView.noConnection',
),
toastLength: Toast.LENGTH_LONG,
gravity: ToastGravity.CENTER,
);
}
_getPatchedApps(); _getPatchedApps();
_managerAPI.reAssessSavedApps().then((_) => _getPatchedApps()); _managerAPI.reAssessSavedApps().then((_) => _getPatchedApps());
} }
@ -62,10 +74,7 @@ class HomeViewModel extends BaseViewModel {
} }
void _getPatchedApps() { void _getPatchedApps() {
patchedInstalledApps = _managerAPI patchedInstalledApps = _managerAPI.getPatchedApps().toList();
.getPatchedApps()
.where((app) => app.hasUpdates == false)
.toList();
patchedUpdatableApps = _managerAPI patchedUpdatableApps = _managerAPI
.getPatchedApps() .getPatchedApps()
.where((app) => app.hasUpdates == true) .where((app) => app.hasUpdates == true)
@ -99,7 +108,7 @@ class HomeViewModel extends BaseViewModel {
toastLength: Toast.LENGTH_LONG, toastLength: Toast.LENGTH_LONG,
gravity: ToastGravity.CENTER, gravity: ToastGravity.CENTER,
); );
File? managerApk = await _managerAPI.downloadManager('.apk'); File? managerApk = await _managerAPI.downloadManager();
if (managerApk != null) { if (managerApk != null) {
flutterLocalNotificationsPlugin.show( flutterLocalNotificationsPlugin.show(
0, 0,
@ -171,4 +180,21 @@ class HomeViewModel extends BaseViewModel {
), ),
); );
} }
Future<String?> getLatestPatcherReleaseTime() async {
return _managerAPI.getLatestPatcherReleaseTime();
}
Future<String?> getLatestManagerReleaseTime() async {
return _managerAPI.getLatestManagerReleaseTime();
}
Future<void> forceRefresh(BuildContext context) async {
await Future.delayed(const Duration(seconds: 1));
if (_lastUpdate == null ||
_lastUpdate!.difference(DateTime.now()).inSeconds > 60) {
_managerAPI.clearAllData();
}
initialize(context);
}
} }

View File

@ -60,13 +60,14 @@ class InstallerView extends StatelessWidget {
preferredSize: const Size(double.infinity, 1.0), preferredSize: const Size(double.infinity, 1.0),
child: LinearProgressIndicator( child: LinearProgressIndicator(
color: Theme.of(context).colorScheme.primary, color: Theme.of(context).colorScheme.primary,
backgroundColor: Theme.of(context).colorScheme.secondary, backgroundColor:
Theme.of(context).colorScheme.primaryContainer,
value: model.progress, value: model.progress,
), ),
), ),
), ),
SliverPadding( SliverPadding(
padding: const EdgeInsets.all(20.0), padding: const EdgeInsets.all(20.0).copyWith(bottom: 20.0),
sliver: SliverList( sliver: SliverList(
delegate: SliverChildListDelegate.fixed( delegate: SliverChildListDelegate.fixed(
<Widget>[ <Widget>[
@ -79,61 +80,64 @@ class InstallerView extends StatelessWidget {
), ),
), ),
), ),
Padding(
padding: const EdgeInsets.symmetric(
vertical: 16,
horizontal: 0,
),
child: Visibility(
visible: !model.isPatching,
child: Row(
mainAxisAlignment: MainAxisAlignment.end,
children: <Widget>[
Visibility(
visible: model.isInstalled,
child: CustomMaterialButton(
label: I18nText('installerView.openButton'),
isExpanded: true,
onPressed: () {
model.openApp();
model.cleanPatcher();
Navigator.of(context).pop();
},
),
),
Visibility(
visible: !model.isInstalled,
child: CustomMaterialButton(
isFilled: false,
label: I18nText(
'installerView.installRootButton'),
isExpanded: true,
onPressed: () => model.installResult(true),
),
),
Visibility(
visible: !model.isInstalled,
child: const SizedBox(
width: 16,
),
),
Visibility(
visible: !model.isInstalled,
child: CustomMaterialButton(
label:
I18nText('installerView.installButton'),
isExpanded: true,
onPressed: () => model.installResult(false),
),
),
],
),
),
),
], ],
), ),
), ),
), ),
SliverFillRemaining(
hasScrollBody: false,
child: Align(
alignment: Alignment.bottomCenter,
child: Visibility(
visible: !model.isPatching,
child: Padding(
padding: const EdgeInsets.all(20.0).copyWith(top: 0.0),
child: Row(
mainAxisAlignment: MainAxisAlignment.end,
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Visibility(
visible: model.isInstalled,
child: CustomMaterialButton(
label: I18nText('installerView.openButton'),
isExpanded: true,
onPressed: () {
model.openApp();
model.cleanPatcher();
Navigator.of(context).pop();
},
),
),
Visibility(
visible: !model.isInstalled,
child: CustomMaterialButton(
isFilled: false,
label:
I18nText('installerView.installRootButton'),
isExpanded: true,
onPressed: () => model.installResult(true),
),
),
Visibility(
visible: !model.isInstalled,
child: const SizedBox(
width: 16,
),
),
Visibility(
visible: !model.isInstalled,
child: CustomMaterialButton(
label: I18nText('installerView.installButton'),
isExpanded: true,
onPressed: () => model.installResult(false),
),
),
],
),
),
),
),
),
], ],
), ),
), ),

View File

@ -10,6 +10,7 @@ import 'package:revanced_manager/services/manager_api.dart';
import 'package:revanced_manager/services/patcher_api.dart'; import 'package:revanced_manager/services/patcher_api.dart';
import 'package:revanced_manager/ui/views/patcher/patcher_viewmodel.dart'; import 'package:revanced_manager/ui/views/patcher/patcher_viewmodel.dart';
import 'package:stacked/stacked.dart'; import 'package:stacked/stacked.dart';
import 'package:wakelock/wakelock.dart';
class InstallerViewModel extends BaseViewModel { class InstallerViewModel extends BaseViewModel {
final ManagerAPI _managerAPI = locator<ManagerAPI>(); final ManagerAPI _managerAPI = locator<ManagerAPI>();
@ -46,10 +47,12 @@ class InstallerViewModel extends BaseViewModel {
), ),
); );
await FlutterBackground.enableBackgroundExecution(); await FlutterBackground.enableBackgroundExecution();
} finally { await Wakelock.enable();
await handlePlatformChannelMethods(); } on Exception {
await runPatcher(); // ignore
} }
await handlePlatformChannelMethods();
await runPatcher();
} }
Future<dynamic> handlePlatformChannelMethods() async { Future<dynamic> handlePlatformChannelMethods() async {
@ -119,9 +122,11 @@ class InstallerViewModel extends BaseViewModel {
} }
try { try {
await FlutterBackground.disableBackgroundExecution(); await FlutterBackground.disableBackgroundExecution();
} finally { await Wakelock.disable();
isPatching = false; } on Exception {
// ignore
} }
isPatching = false;
} }
void installResult(bool installAsRoot) async { void installResult(bool installAsRoot) async {

View File

@ -37,30 +37,30 @@ class _PatchesSelectorViewState extends State<PatchesSelectorView> {
child: Padding( child: Padding(
padding: padding:
const EdgeInsets.symmetric(vertical: 4.0, horizontal: 12.0), const EdgeInsets.symmetric(vertical: 4.0, horizontal: 12.0),
child: model.patches.isEmpty child: Column(
? Center( children: <Widget>[
child: CircularProgressIndicator( SearchBar(
color: Theme.of(context).colorScheme.primary, showSelectIcon: true,
), hintText: FlutterI18n.translate(
) context,
: Column( 'patchesSelectorView.searchBarHint',
children: <Widget>[ ),
SearchBar( onQueryChanged: (searchQuery) {
showSelectIcon: true, setState(() {
hintText: FlutterI18n.translate( _query = searchQuery;
context, });
'patchesSelectorView.searchBarHint', },
), onSelectAll: (value) => model.selectAllPatches(value),
onQueryChanged: (searchQuery) { ),
setState(() { const SizedBox(height: 12),
_query = searchQuery; Expanded(
}); child: model.patches.isEmpty
}, ? Center(
onSelectAll: (value) => model.selectAllPatches(value), child: CircularProgressIndicator(
), color: Theme.of(context).colorScheme.primary,
const SizedBox(height: 12), ),
Expanded( )
child: ListView( : ListView(
padding: const EdgeInsets.only(bottom: 80), padding: const EdgeInsets.only(bottom: 80),
children: model children: model
.getQueriedPatches(_query) .getQueriedPatches(_query)
@ -160,9 +160,9 @@ class _PatchesSelectorViewState extends State<PatchesSelectorView> {
) )
.toList(), .toList(),
), ),
), ),
], ],
), ),
), ),
), ),
), ),

View File

@ -1,3 +1,4 @@
import 'package:collection/collection.dart';
import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/models/patch.dart'; import 'package:revanced_manager/models/patch.dart';
import 'package:revanced_manager/models/patched_application.dart'; import 'package:revanced_manager/models/patched_application.dart';
@ -71,9 +72,14 @@ class PatchesSelectorViewModel extends BaseViewModel {
List<String> getSupportedVersions(Patch patch) { List<String> getSupportedVersions(Patch patch) {
PatchedApplication app = locator<PatcherViewModel>().selectedApp!; PatchedApplication app = locator<PatcherViewModel>().selectedApp!;
return patch.compatiblePackages Package? package = patch.compatiblePackages.firstWhereOrNull(
.firstWhere((pack) => pack.name == app.packageName) (pack) => pack.name == app.packageName,
.versions; );
if (package != null) {
return package.versions;
} else {
return List.empty();
}
} }
bool isPatchSupported(Patch patch) { bool isPatchSupported(Patch patch) {

View File

@ -1,5 +1,4 @@
// ignore_for_file: use_build_context_synchronously // ignore_for_file: use_build_context_synchronously
import 'package:device_info_plus/device_info_plus.dart'; import 'package:device_info_plus/device_info_plus.dart';
import 'package:dynamic_themes/dynamic_themes.dart'; import 'package:dynamic_themes/dynamic_themes.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';

View File

@ -7,6 +7,7 @@ import 'package:revanced_manager/models/patched_application.dart';
import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/services/manager_api.dart';
import 'package:revanced_manager/services/patcher_api.dart'; import 'package:revanced_manager/services/patcher_api.dart';
import 'package:revanced_manager/services/root_api.dart'; import 'package:revanced_manager/services/root_api.dart';
import 'package:revanced_manager/ui/views/home/home_viewmodel.dart';
import 'package:revanced_manager/ui/views/navigation/navigation_viewmodel.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/views/patcher/patcher_viewmodel.dart';
import 'package:revanced_manager/ui/widgets/installerView/custom_material_button.dart'; import 'package:revanced_manager/ui/widgets/installerView/custom_material_button.dart';
@ -73,7 +74,7 @@ class AppInfoViewModel extends BaseViewModel {
label: I18nText('okButton'), label: I18nText('okButton'),
onPressed: () { onPressed: () {
uninstallApp(app); uninstallApp(app);
locator<NavigationViewModel>().notifyListeners(); locator<HomeViewModel>().initialize(context);
Navigator.of(context).pop(); Navigator.of(context).pop();
Navigator.of(context).pop(); Navigator.of(context).pop();
}, },

View File

@ -30,7 +30,7 @@ class AppSkeletonLoader extends StatelessWidget {
children: [ children: [
Container( Container(
color: Colors.white, color: Colors.white,
height: 25, height: 34,
width: screenWidth * 0.4, width: screenWidth * 0.4,
child: SkeletonParagraph( child: SkeletonParagraph(
style: const SkeletonParagraphStyle( style: const SkeletonParagraphStyle(
@ -42,7 +42,7 @@ class AppSkeletonLoader extends StatelessWidget {
Container( Container(
margin: const EdgeInsets.only(bottom: 4), margin: const EdgeInsets.only(bottom: 4),
color: Colors.white, color: Colors.white,
height: 25, height: 34,
width: screenWidth * 0.6, width: screenWidth * 0.6,
child: SkeletonParagraph( child: SkeletonParagraph(
style: const SkeletonParagraphStyle( style: const SkeletonParagraphStyle(

View File

@ -1,11 +1,12 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:github/github.dart'; import 'package:flutter_cache_manager/file.dart';
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
import 'package:revanced_manager/ui/widgets/shared/custom_card.dart'; import 'package:revanced_manager/ui/widgets/shared/custom_card.dart';
import 'package:url_launcher/url_launcher.dart'; import 'package:url_launcher/url_launcher.dart';
class ContributorsCard extends StatefulWidget { class ContributorsCard extends StatefulWidget {
final String title; final String title;
final List<Contributor> contributors; final List<dynamic> contributors;
final double height; final double height;
const ContributorsCard({ const ContributorsCard({
@ -52,11 +53,25 @@ class _ContributorsCardState extends State<ContributorsCard> {
borderRadius: BorderRadius.circular(100), borderRadius: BorderRadius.circular(100),
child: GestureDetector( child: GestureDetector(
onTap: () => launchUrl( onTap: () => launchUrl(
Uri.parse(widget.contributors[index].htmlUrl!)), Uri.parse(
child: Image.network( widget.contributors[index]['html_url'],
widget.contributors[index].avatarUrl!, ),
height: 40, ),
width: 40, child: FutureBuilder<File?>(
future: DefaultCacheManager().getSingleFile(
widget.contributors[index]['avatar_url'],
),
builder: (context, snapshot) => snapshot.hasData
? Image.file(
snapshot.data!,
height: 40,
width: 40,
)
: Image.network(
widget.contributors[index]['avatar_url'],
height: 40,
width: 40,
),
), ),
), ),
); );

View File

@ -1,8 +1,6 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:flutter_i18n/flutter_i18n.dart';
import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/app/app.locator.dart';
import 'package:revanced_manager/services/github_api.dart';
import 'package:revanced_manager/services/manager_api.dart';
import 'package:revanced_manager/ui/views/home/home_viewmodel.dart'; import 'package:revanced_manager/ui/views/home/home_viewmodel.dart';
import 'package:revanced_manager/ui/widgets/installerView/custom_material_button.dart'; import 'package:revanced_manager/ui/widgets/installerView/custom_material_button.dart';
import 'package:revanced_manager/ui/widgets/shared/custom_card.dart'; import 'package:revanced_manager/ui/widgets/shared/custom_card.dart';
@ -20,8 +18,7 @@ class LatestCommitCard extends StatefulWidget {
} }
class _LatestCommitCardState extends State<LatestCommitCard> { class _LatestCommitCardState extends State<LatestCommitCard> {
final ManagerAPI _managerAPI = locator<ManagerAPI>(); final HomeViewModel model = locator<HomeViewModel>();
final GithubAPI _githubAPI = GithubAPI();
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -35,10 +32,8 @@ class _LatestCommitCardState extends State<LatestCommitCard> {
Row( Row(
children: <Widget>[ children: <Widget>[
I18nText('latestCommitCard.patcherLabel'), I18nText('latestCommitCard.patcherLabel'),
FutureBuilder<String>( FutureBuilder<String?>(
future: _githubAPI.latestCommitTime( future: model.getLatestPatcherReleaseTime(),
_managerAPI.getPatcherRepo(),
),
builder: (context, snapshot) => Text( builder: (context, snapshot) => Text(
snapshot.hasData && snapshot.data!.isNotEmpty snapshot.hasData && snapshot.data!.isNotEmpty
? FlutterI18n.translate( ? FlutterI18n.translate(
@ -58,10 +53,8 @@ class _LatestCommitCardState extends State<LatestCommitCard> {
Row( Row(
children: <Widget>[ children: <Widget>[
I18nText('latestCommitCard.managerLabel'), I18nText('latestCommitCard.managerLabel'),
FutureBuilder<String>( FutureBuilder<String?>(
future: _githubAPI.latestCommitTime( future: model.getLatestManagerReleaseTime(),
_managerAPI.getManagerRepo(),
),
builder: (context, snapshot) => builder: (context, snapshot) =>
snapshot.hasData && snapshot.data!.isNotEmpty snapshot.hasData && snapshot.data!.isNotEmpty
? I18nText( ? I18nText(

View File

@ -49,7 +49,7 @@ class ApplicationItem extends StatelessWidget {
fontWeight: FontWeight.w500, fontWeight: FontWeight.w500,
), ),
), ),
Text(format(patchDate, locale: 'en_short')), Text(format(patchDate)),
], ],
), ),
const Spacer(), const Spacer(),

View File

@ -1,6 +1,6 @@
name: revanced_manager name: revanced_manager
description: An unofficial ReVanced Manager based on Flutter. description: The official ReVanced Manager.
homepage: https://github.com/Aunali321/revanced-manager homepage: https://github.com/revanced/revanced-manager
publish_to: 'none' publish_to: 'none'
@ -12,11 +12,15 @@ environment:
dependencies: dependencies:
animations: ^2.0.4 animations: ^2.0.4
app_installer: ^1.1.0 app_installer: ^1.1.0
collection: ^1.16.0
cross_connectivity: ^3.0.5
device_apps: device_apps:
git: git:
url: https://github.com/ponces/flutter_plugin_device_apps url: https://github.com/ponces/flutter_plugin_device_apps
ref: appinfo-from-storage ref: appinfo-from-storage
device_info_plus: ^4.1.2 device_info_plus: ^4.1.2
dio: ^4.0.6
dio_http_cache_lts: ^0.4.1
dynamic_color: ^1.5.4 dynamic_color: ^1.5.4
dynamic_themes: ^1.1.0 dynamic_themes: ^1.1.0
expandable: ^5.0.1 expandable: ^5.0.1
@ -28,11 +32,12 @@ dependencies:
flutter_cache_manager: ^3.3.0 flutter_cache_manager: ^3.3.0
flutter_i18n: ^0.32.4 flutter_i18n: ^0.32.4
flutter_local_notifications: ^9.8.0+1 flutter_local_notifications: ^9.8.0+1
flutter_localizations:
sdk: flutter
flutter_svg: ^1.1.1+1 flutter_svg: ^1.1.1+1
fluttertoast: ^8.0.9 fluttertoast: ^8.0.9
font_awesome_flutter: ^10.1.0 font_awesome_flutter: ^10.1.0
get_it: ^7.2.0 get_it: ^7.2.0
github: ^9.4.0
google_fonts: ^3.0.1 google_fonts: ^3.0.1
http: ^0.13.4 http: ^0.13.4
injectable: ^1.5.3 injectable: ^1.5.3
@ -40,6 +45,7 @@ dependencies:
json_annotation: ^4.6.0 json_annotation: ^4.6.0
package_info_plus: ^1.4.3+1 package_info_plus: ^1.4.3+1
path_provider: ^2.0.11 path_provider: ^2.0.11
pull_to_refresh: ^2.0.0
root: ^2.0.2 root: ^2.0.2
share_extend: ^2.0.0 share_extend: ^2.0.0
shared_preferences: ^2.0.15 shared_preferences: ^2.0.15
@ -50,6 +56,7 @@ dependencies:
stacked_themes: ^0.3.9 stacked_themes: ^0.3.9
timeago: ^3.2.2 timeago: ^3.2.2
url_launcher: ^6.1.5 url_launcher: ^6.1.5
wakelock: ^0.6.2
dev_dependencies: dev_dependencies:
build_runner: any build_runner: any