mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-25 11:53:45 +02:00 
			
		
		
		
	Compare commits
	
		
			15 Commits
		
	
	
		
			v0.11.9-be
			...
			v0.11.12-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | cd86d6112b | ||
|  | 1112c79c14 | ||
|  | 08555bac75 | ||
|  | 6db31e2b24 | ||
|  | 48d2532323 | ||
|  | f1fc43a3e7 | ||
|  | 280827d8ec | ||
|  | 05ee0f9c48 | ||
|  | ef06ae289e | ||
|  | bd0e322465 | ||
|  | a93a2411fa | ||
|  | 26e6eef72e | ||
|  | e49a6e311b | ||
|  | 53d3397651 | ||
|  | fe540f5e61 | 
| @@ -19,6 +19,9 @@ Currently supported App sources: | |||||||
| - Third Party F-Droid Repos | - Third Party F-Droid Repos | ||||||
|   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` |   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` | ||||||
| - [Steam](https://store.steampowered.com/mobile) | - [Steam](https://store.steampowered.com/mobile) | ||||||
|  | - [Telegram App](https://telegram.org) | ||||||
|  | - [VLC](https://www.videolan.org/vlc/download-android.html) | ||||||
|  | - [Neutron Code](https://neutroncode.com) | ||||||
| - "HTML" (Fallback) | - "HTML" (Fallback) | ||||||
|   - Any other URL that returns an HTML page with links to APK files (if multiple, the last file alphabetically is picked) |   - Any other URL that returns an HTML page with links to APK files (if multiple, the last file alphabetically is picked) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -118,9 +118,11 @@ class Codeberg extends AppSource { | |||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|  |       var changeLog = targetRelease['body'].toString(); | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |       return APKDetails(version, targetRelease['apkUrls'] as List<String>, | ||||||
|           getAppNames(standardUrl), |           getAppNames(standardUrl), | ||||||
|           releaseDate: releaseDate); |           releaseDate: releaseDate, | ||||||
|  |           changeLog: changeLog.isEmpty ? null : changeLog); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -27,9 +27,6 @@ class FDroid extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String? tryInferringAppId(String standardUrl, |   String? tryInferringAppId(String standardUrl, | ||||||
|       {Map<String, dynamic> additionalSettings = const {}}) { |       {Map<String, dynamic> additionalSettings = const {}}) { | ||||||
|   | |||||||
| @@ -160,9 +160,11 @@ class GitHub extends AppSource { | |||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|  |       var changeLog = targetRelease['body'].toString(); | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |       return APKDetails(version, targetRelease['apkUrls'] as List<String>, | ||||||
|           getAppNames(standardUrl), |           getAppNames(standardUrl), | ||||||
|           releaseDate: releaseDate); |           releaseDate: releaseDate, | ||||||
|  |           changeLog: changeLog.isEmpty ? null : changeLog); | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       rateLimitErrorCheck(res); | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|   | |||||||
| @@ -10,9 +10,6 @@ class HTML extends AppSource { | |||||||
|     return url; |     return url; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
|   | |||||||
| @@ -18,9 +18,6 @@ class IzzyOnDroid extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String? tryInferringAppId(String standardUrl, |   String? tryInferringAppId(String standardUrl, | ||||||
|       {Map<String, dynamic> additionalSettings = const {}}) { |       {Map<String, dynamic> additionalSettings = const {}}) { | ||||||
|   | |||||||
							
								
								
									
										111
									
								
								lib/app_sources/neutroncode.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								lib/app_sources/neutroncode.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,111 @@ | |||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class NeutronCode extends AppSource { | ||||||
|  |   NeutronCode() { | ||||||
|  |     host = 'neutroncode.com'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     RegExp standardUrlRegEx = RegExp('^https?://$host/downloads/file/[^/]+'); | ||||||
|  |     RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase()); | ||||||
|  |     if (match == null) { | ||||||
|  |       throw InvalidURLError(name); | ||||||
|  |     } | ||||||
|  |     return url.substring(0, match.end); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String? changeLogPageFromStandardUrl(String standardUrl) => standardUrl; | ||||||
|  |  | ||||||
|  |   String monthNameToNumberString(String s) { | ||||||
|  |     switch (s.toLowerCase()) { | ||||||
|  |       case 'january': | ||||||
|  |         return '01'; | ||||||
|  |       case 'february': | ||||||
|  |         return '02'; | ||||||
|  |       case 'march': | ||||||
|  |         return '03'; | ||||||
|  |       case 'april': | ||||||
|  |         return '04'; | ||||||
|  |       case 'may': | ||||||
|  |         return '05'; | ||||||
|  |       case 'june': | ||||||
|  |         return '06'; | ||||||
|  |       case 'july': | ||||||
|  |         return '07'; | ||||||
|  |       case 'august': | ||||||
|  |         return '08'; | ||||||
|  |       case 'september': | ||||||
|  |         return '09'; | ||||||
|  |       case 'october': | ||||||
|  |         return '10'; | ||||||
|  |       case 'november': | ||||||
|  |         return '11'; | ||||||
|  |       case 'december': | ||||||
|  |         return '12'; | ||||||
|  |       default: | ||||||
|  |         throw ArgumentError('Invalid month name: $s'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   customDateParse(String dateString) { | ||||||
|  |     List<String> parts = dateString.split(' '); | ||||||
|  |     if (parts.length != 3) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |     String result = ''; | ||||||
|  |     for (var s in parts.reversed) { | ||||||
|  |       try { | ||||||
|  |         try { | ||||||
|  |           int.parse(s); | ||||||
|  |           result += '$s-'; | ||||||
|  |         } catch (e) { | ||||||
|  |           result += '${monthNameToNumberString(s)}-'; | ||||||
|  |         } | ||||||
|  |       } catch (e) { | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return result.substring(0, result.length - 1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     Response res = await get(Uri.parse(standardUrl)); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var http = parse(res.body); | ||||||
|  |       var name = http.querySelector('.pd-title')?.innerHtml; | ||||||
|  |       var filename = http.querySelector('.pd-filename .pd-float')?.innerHtml; | ||||||
|  |       if (filename == null) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|  |       } | ||||||
|  |       var version = | ||||||
|  |           http.querySelector('.pd-version-txt')?.nextElementSibling?.innerHtml; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String? apkUrl = 'https://$host/download/$filename'; | ||||||
|  |       var dateStringOriginal = | ||||||
|  |           http.querySelector('.pd-date-txt')?.nextElementSibling?.innerHtml; | ||||||
|  |       var dateString = dateStringOriginal != null | ||||||
|  |           ? (customDateParse(dateStringOriginal)) | ||||||
|  |           : null; | ||||||
|  |       var changeLogElements = http.querySelectorAll('.pd-fdesc p'); | ||||||
|  |       return APKDetails(version, [apkUrl], | ||||||
|  |           AppNames(runtimeType.toString(), name ?? standardUrl.split('/').last), | ||||||
|  |           releaseDate: dateString != null ? DateTime.parse(dateString) : null, | ||||||
|  |           changeLog: changeLogElements.isNotEmpty | ||||||
|  |               ? changeLogElements.last.innerHtml | ||||||
|  |               : null); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -13,9 +13,6 @@ class Signal extends AppSource { | |||||||
|     return 'https://$host'; |     return 'https://$host'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
|   | |||||||
| @@ -18,9 +18,6 @@ class SourceForge extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
|   | |||||||
| @@ -24,9 +24,6 @@ class SteamMobile extends AppSource { | |||||||
|     return 'https://$host'; |     return 'https://$host'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								lib/app_sources/telegramapp.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								lib/app_sources/telegramapp.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class TelegramApp extends AppSource { | ||||||
|  |   TelegramApp() { | ||||||
|  |     host = 'telegram.org'; | ||||||
|  |     name = 'Telegram ${tr('app')}'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     return 'https://$host'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     Response res = await get(Uri.parse('https://t.me/s/TAndroidAPK')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var http = parse(res.body); | ||||||
|  |       var messages = | ||||||
|  |           http.querySelectorAll('.tgme_widget_message_text.js-message_text'); | ||||||
|  |       var version = messages.isNotEmpty | ||||||
|  |           ? messages.last.innerHtml.split('\n').first.trim().split(' ').first | ||||||
|  |           : null; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String? apkUrl = 'https://telegram.org/dl/android/apk'; | ||||||
|  |       return APKDetails(version, [apkUrl], AppNames('Telegram', 'Telegram')); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										62
									
								
								lib/app_sources/vlc.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								lib/app_sources/vlc.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/app_sources/html.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class VLC extends AppSource { | ||||||
|  |   VLC() { | ||||||
|  |     host = 'videolan.org'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     return 'https://$host'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     Response res = await get( | ||||||
|  |         Uri.parse('https://www.videolan.org/vlc/download-android.html')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var dwUrlBase = 'get.videolan.org/vlc-android'; | ||||||
|  |       var dwLinks = parse(res.body) | ||||||
|  |           .querySelectorAll('a') | ||||||
|  |           .where((element) => | ||||||
|  |               element.attributes['href']?.contains(dwUrlBase) ?? false) | ||||||
|  |           .toList(); | ||||||
|  |       String? version = dwLinks.isNotEmpty | ||||||
|  |           ? dwLinks.first.attributes['href'] | ||||||
|  |               ?.split('/') | ||||||
|  |               .where((s) => s.isNotEmpty) | ||||||
|  |               .last | ||||||
|  |           : null; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String? targetUrl = 'https://$dwUrlBase/$version/'; | ||||||
|  |       Response res2 = await get(Uri.parse(targetUrl)); | ||||||
|  |       String mirrorDwBase = | ||||||
|  |           'https://plug-mirror.rcac.purdue.edu/vlc/vlc-android/$version/'; | ||||||
|  |       List<String> apkUrls = []; | ||||||
|  |       if (res2.statusCode == 200) { | ||||||
|  |         apkUrls = parse(res2.body) | ||||||
|  |             .querySelectorAll('a') | ||||||
|  |             .map((e) => e.attributes['href']) | ||||||
|  |             .where((h) => | ||||||
|  |                 h != null && h.isNotEmpty && h.toLowerCase().endsWith('.apk')) | ||||||
|  |             .map((e) => mirrorDwBase + e!) | ||||||
|  |             .toList(); | ||||||
|  |       } else { | ||||||
|  |         throw getObtainiumHttpError(res2); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       return APKDetails(version, apkUrls, AppNames('VideoLAN', 'VLC')); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -476,6 +476,7 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|         rowItems.add(Expanded( |         rowItems.add(Expanded( | ||||||
|             child: Column( |             child: Column( | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.stretch, |                 crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |                 mainAxisSize: MainAxisSize.min, | ||||||
|                 children: [ |                 children: [ | ||||||
|               rowInput.value, |               rowInput.value, | ||||||
|               ...widget.items[rowInputs.key][rowInput.key].belowWidgets |               ...widget.items[rowInputs.key][rowInput.key].belowWidgets | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart'; | |||||||
| // ignore: implementation_imports | // ignore: implementation_imports | ||||||
| import 'package:easy_localization/src/localization.dart'; | import 'package:easy_localization/src/localization.dart'; | ||||||
|  |  | ||||||
| const String currentVersion = '0.11.9'; | const String currentVersion = '0.11.12'; | ||||||
| const String currentReleaseTag = | const String currentReleaseTag = | ||||||
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES |     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
|  | import 'package:flutter_markdown/flutter_markdown.dart'; | ||||||
| import 'package:obtainium/components/custom_app_bar.dart'; | import 'package:obtainium/components/custom_app_bar.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| @@ -14,6 +15,7 @@ import 'package:obtainium/providers/source_provider.dart'; | |||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
| import 'package:share_plus/share_plus.dart'; | import 'package:share_plus/share_plus.dart'; | ||||||
| import 'package:url_launcher/url_launcher_string.dart'; | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
|  | import 'package:markdown/markdown.dart' as md; | ||||||
|  |  | ||||||
| class AppsPage extends StatefulWidget { | class AppsPage extends StatefulWidget { | ||||||
|   const AppsPage({super.key}); |   const AppsPage({super.key}); | ||||||
| @@ -229,9 +231,88 @@ class AppsPageState extends State<AppsPage> { | |||||||
|             SliverList( |             SliverList( | ||||||
|                 delegate: SliverChildBuilderDelegate( |                 delegate: SliverChildBuilderDelegate( | ||||||
|                     (BuildContext context, int index) { |                     (BuildContext context, int index) { | ||||||
|               String? changesUrl = SourceProvider() |               AppSource appSource = | ||||||
|                   .getSource(listedApps[index].app.url) |                   SourceProvider().getSource(listedApps[index].app.url); | ||||||
|  |               String? changesUrl = appSource | ||||||
|                   .changeLogPageFromStandardUrl(listedApps[index].app.url); |                   .changeLogPageFromStandardUrl(listedApps[index].app.url); | ||||||
|  |               String? changeLog = listedApps[index].app.changeLog; | ||||||
|  |               var showChanges = (changeLog == null && changesUrl == null) | ||||||
|  |                   ? null | ||||||
|  |                   : () { | ||||||
|  |                       if (changeLog != null) { | ||||||
|  |                         showDialog( | ||||||
|  |                             context: context, | ||||||
|  |                             builder: (BuildContext context) { | ||||||
|  |                               return GeneratedFormModal( | ||||||
|  |                                 title: tr('changes'), | ||||||
|  |                                 items: const [], | ||||||
|  |                                 additionalWidgets: [ | ||||||
|  |                                   changesUrl != null | ||||||
|  |                                       ? GestureDetector( | ||||||
|  |                                           child: Text( | ||||||
|  |                                             changesUrl, | ||||||
|  |                                             style: const TextStyle( | ||||||
|  |                                                 decoration: | ||||||
|  |                                                     TextDecoration.underline, | ||||||
|  |                                                 fontStyle: FontStyle.italic), | ||||||
|  |                                           ), | ||||||
|  |                                           onTap: () { | ||||||
|  |                                             launchUrlString(changesUrl, | ||||||
|  |                                                 mode: LaunchMode | ||||||
|  |                                                     .externalApplication); | ||||||
|  |                                           }, | ||||||
|  |                                         ) | ||||||
|  |                                       : const SizedBox.shrink(), | ||||||
|  |                                   changesUrl != null | ||||||
|  |                                       ? const SizedBox( | ||||||
|  |                                           height: 16, | ||||||
|  |                                         ) | ||||||
|  |                                       : const SizedBox.shrink(), | ||||||
|  |                                   appSource.changeLogIfAnyIsMarkDown | ||||||
|  |                                       ? SizedBox( | ||||||
|  |                                           width: | ||||||
|  |                                               MediaQuery.of(context).size.width, | ||||||
|  |                                           height: MediaQuery.of(context) | ||||||
|  |                                                   .size | ||||||
|  |                                                   .height - | ||||||
|  |                                               350, | ||||||
|  |                                           child: Markdown( | ||||||
|  |                                             data: changeLog, | ||||||
|  |                                             onTapLink: (text, href, title) { | ||||||
|  |                                               if (href != null) { | ||||||
|  |                                                 launchUrlString( | ||||||
|  |                                                     href.startsWith( | ||||||
|  |                                                                 'http://') || | ||||||
|  |                                                             href.startsWith( | ||||||
|  |                                                                 'https://') | ||||||
|  |                                                         ? href | ||||||
|  |                                                         : '${Uri.parse(listedApps[index].app.url).origin}/$href', | ||||||
|  |                                                     mode: LaunchMode | ||||||
|  |                                                         .externalApplication); | ||||||
|  |                                               } | ||||||
|  |                                             }, | ||||||
|  |                                             extensionSet: md.ExtensionSet( | ||||||
|  |                                               md.ExtensionSet.gitHubFlavored | ||||||
|  |                                                   .blockSyntaxes, | ||||||
|  |                                               [ | ||||||
|  |                                                 md.EmojiSyntax(), | ||||||
|  |                                                 ...md | ||||||
|  |                                                     .ExtensionSet | ||||||
|  |                                                     .gitHubFlavored | ||||||
|  |                                                     .inlineSyntaxes | ||||||
|  |                                               ], | ||||||
|  |                                             ), | ||||||
|  |                                           )) | ||||||
|  |                                       : Text(changeLog), | ||||||
|  |                                 ], | ||||||
|  |                                 singleNullReturnButton: tr('ok'), | ||||||
|  |                               ); | ||||||
|  |                             }); | ||||||
|  |                       } else { | ||||||
|  |                         launchUrlString(changesUrl!, | ||||||
|  |                             mode: LaunchMode.externalApplication); | ||||||
|  |                       } | ||||||
|  |                     }; | ||||||
|               var transparent = const Color.fromARGB(0, 0, 0, 0).value; |               var transparent = const Color.fromARGB(0, 0, 0, 0).value; | ||||||
|               var hasUpdate = listedApps[index].app.installedVersion != null && |               var hasUpdate = listedApps[index].app.installedVersion != null && | ||||||
|                   listedApps[index].app.installedVersion != |                   listedApps[index].app.installedVersion != | ||||||
| @@ -366,25 +447,22 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                                     mainAxisSize: MainAxisSize.min, |                                     mainAxisSize: MainAxisSize.min, | ||||||
|                                     children: [ |                                     children: [ | ||||||
|                                       GestureDetector( |                                       GestureDetector( | ||||||
|                                           onTap: changesUrl == null |                                           onTap: showChanges, | ||||||
|                                               ? null |  | ||||||
|                                               : () { |  | ||||||
|                                                   launchUrlString(changesUrl, |  | ||||||
|                                                       mode: LaunchMode |  | ||||||
|                                                           .externalApplication); |  | ||||||
|                                                 }, |  | ||||||
|                                           child: Text( |                                           child: Text( | ||||||
|                                             listedApps[index].app.releaseDate == |                                             listedApps[index].app.releaseDate == | ||||||
|                                                     null |                                                     null | ||||||
|                                                 ? tr('changes') |                                                 ? showChanges != null | ||||||
|  |                                                     ? tr('changes') | ||||||
|  |                                                     : '' | ||||||
|                                                 : DateFormat('yyyy-MM-dd') |                                                 : DateFormat('yyyy-MM-dd') | ||||||
|                                                     .format(listedApps[index] |                                                     .format(listedApps[index] | ||||||
|                                                         .app |                                                         .app | ||||||
|                                                         .releaseDate!), |                                                         .releaseDate!), | ||||||
|                                             style: const TextStyle( |                                             style: TextStyle( | ||||||
|                                                 fontStyle: FontStyle.italic, |                                                 fontStyle: FontStyle.italic, | ||||||
|                                                 decoration: |                                                 decoration: showChanges != null | ||||||
|                                                     TextDecoration.underline), |                                                     ? TextDecoration.underline | ||||||
|  |                                                     : TextDecoration.none), | ||||||
|                                           )) |                                           )) | ||||||
|                                     ], |                                     ], | ||||||
|                                   ), |                                   ), | ||||||
|   | |||||||
| @@ -145,6 +145,9 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { |   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { | ||||||
|  |     NotificationsProvider? notificationsProvider = | ||||||
|  |         context?.read<NotificationsProvider>(); | ||||||
|  |     var notifId = DownloadNotification(app.name, 0).id; | ||||||
|     if (apps[app.id] != null) { |     if (apps[app.id] != null) { | ||||||
|       apps[app.id]!.downloadProgress = 0; |       apps[app.id]!.downloadProgress = 0; | ||||||
|       notifyListeners(); |       notifyListeners(); | ||||||
| @@ -155,8 +158,6 @@ class AppsProvider with ChangeNotifier { | |||||||
|       String downloadUrl = await SourceProvider() |       String downloadUrl = await SourceProvider() | ||||||
|           .getSource(app.url) |           .getSource(app.url) | ||||||
|           .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex]); |           .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex]); | ||||||
|       NotificationsProvider? notificationsProvider = |  | ||||||
|           context?.read<NotificationsProvider>(); |  | ||||||
|       var notif = DownloadNotification(app.name, 100); |       var notif = DownloadNotification(app.name, 100); | ||||||
|       notificationsProvider?.cancel(notif.id); |       notificationsProvider?.cancel(notif.id); | ||||||
|       int? prevProg; |       int? prevProg; | ||||||
| @@ -173,7 +174,6 @@ class AppsProvider with ChangeNotifier { | |||||||
|         } |         } | ||||||
|         prevProg = prog; |         prevProg = prog; | ||||||
|       }); |       }); | ||||||
|       notificationsProvider?.cancel(notif.id); |  | ||||||
|       // Delete older versions of the APK if any |       // Delete older versions of the APK if any | ||||||
|       for (var file in downloadedFile.parent.listSync()) { |       for (var file in downloadedFile.parent.listSync()) { | ||||||
|         var fn = file.path.split('/').last; |         var fn = file.path.split('/').last; | ||||||
| @@ -201,6 +201,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|       } |       } | ||||||
|       return DownloadedApk(app.id, downloadedFile); |       return DownloadedApk(app.id, downloadedFile); | ||||||
|     } finally { |     } finally { | ||||||
|  |       notificationsProvider?.cancel(notifId); | ||||||
|       if (apps[app.id] != null) { |       if (apps[app.id] != null) { | ||||||
|         apps[app.id]!.downloadProgress = null; |         apps[app.id]!.downloadProgress = null; | ||||||
|         notifyListeners(); |         notifyListeners(); | ||||||
|   | |||||||
| @@ -15,9 +15,12 @@ import 'package:obtainium/app_sources/gitlab.dart'; | |||||||
| import 'package:obtainium/app_sources/izzyondroid.dart'; | import 'package:obtainium/app_sources/izzyondroid.dart'; | ||||||
| import 'package:obtainium/app_sources/html.dart'; | import 'package:obtainium/app_sources/html.dart'; | ||||||
| import 'package:obtainium/app_sources/mullvad.dart'; | import 'package:obtainium/app_sources/mullvad.dart'; | ||||||
|  | import 'package:obtainium/app_sources/neutroncode.dart'; | ||||||
| import 'package:obtainium/app_sources/signal.dart'; | import 'package:obtainium/app_sources/signal.dart'; | ||||||
| import 'package:obtainium/app_sources/sourceforge.dart'; | import 'package:obtainium/app_sources/sourceforge.dart'; | ||||||
| import 'package:obtainium/app_sources/steammobile.dart'; | import 'package:obtainium/app_sources/steammobile.dart'; | ||||||
|  | import 'package:obtainium/app_sources/telegramapp.dart'; | ||||||
|  | import 'package:obtainium/app_sources/vlc.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/mass_app_sources/githubstars.dart'; | import 'package:obtainium/mass_app_sources/githubstars.dart'; | ||||||
| @@ -34,8 +37,10 @@ class APKDetails { | |||||||
|   late List<String> apkUrls; |   late List<String> apkUrls; | ||||||
|   late AppNames names; |   late AppNames names; | ||||||
|   late DateTime? releaseDate; |   late DateTime? releaseDate; | ||||||
|  |   late String? changeLog; | ||||||
|  |  | ||||||
|   APKDetails(this.version, this.apkUrls, this.names, {this.releaseDate}); |   APKDetails(this.version, this.apkUrls, this.names, | ||||||
|  |       {this.releaseDate, this.changeLog}); | ||||||
| } | } | ||||||
|  |  | ||||||
| class App { | class App { | ||||||
| @@ -52,6 +57,7 @@ class App { | |||||||
|   bool pinned = false; |   bool pinned = false; | ||||||
|   List<String> categories; |   List<String> categories; | ||||||
|   late DateTime? releaseDate; |   late DateTime? releaseDate; | ||||||
|  |   late String? changeLog; | ||||||
|   App( |   App( | ||||||
|       this.id, |       this.id, | ||||||
|       this.url, |       this.url, | ||||||
| @@ -65,7 +71,8 @@ class App { | |||||||
|       this.lastUpdateCheck, |       this.lastUpdateCheck, | ||||||
|       this.pinned, |       this.pinned, | ||||||
|       {this.categories = const [], |       {this.categories = const [], | ||||||
|       this.releaseDate}); |       this.releaseDate, | ||||||
|  |       this.changeLog}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
| @@ -128,34 +135,35 @@ class App { | |||||||
|       preferredApkIndex = 0; |       preferredApkIndex = 0; | ||||||
|     } |     } | ||||||
|     return App( |     return App( | ||||||
|       json['id'] as String, |         json['id'] as String, | ||||||
|       json['url'] as String, |         json['url'] as String, | ||||||
|       json['author'] as String, |         json['author'] as String, | ||||||
|       json['name'] as String, |         json['name'] as String, | ||||||
|       json['installedVersion'] == null |         json['installedVersion'] == null | ||||||
|           ? null |             ? null | ||||||
|           : json['installedVersion'] as String, |             : json['installedVersion'] as String, | ||||||
|       json['latestVersion'] as String, |         json['latestVersion'] as String, | ||||||
|       json['apkUrls'] == null |         json['apkUrls'] == null | ||||||
|           ? [] |             ? [] | ||||||
|           : List<String>.from(jsonDecode(json['apkUrls'])), |             : List<String>.from(jsonDecode(json['apkUrls'])), | ||||||
|       preferredApkIndex, |         preferredApkIndex, | ||||||
|       additionalSettings, |         additionalSettings, | ||||||
|       json['lastUpdateCheck'] == null |         json['lastUpdateCheck'] == null | ||||||
|           ? null |             ? null | ||||||
|           : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), |             : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), | ||||||
|       json['pinned'] ?? false, |         json['pinned'] ?? false, | ||||||
|       categories: json['categories'] != null |         categories: json['categories'] != null | ||||||
|           ? (json['categories'] as List<dynamic>) |             ? (json['categories'] as List<dynamic>) | ||||||
|               .map((e) => e.toString()) |                 .map((e) => e.toString()) | ||||||
|               .toList() |                 .toList() | ||||||
|           : json['category'] != null |             : json['category'] != null | ||||||
|               ? [json['category'] as String] |                 ? [json['category'] as String] | ||||||
|               : [], |                 : [], | ||||||
|       releaseDate: json['releaseDate'] == null |         releaseDate: json['releaseDate'] == null | ||||||
|           ? null |             ? null | ||||||
|           : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']), |             : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']), | ||||||
|     ); |         changeLog: | ||||||
|  |             json['changeLog'] == null ? null : json['changeLog'] as String); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> toJson() => { |   Map<String, dynamic> toJson() => { | ||||||
| @@ -171,7 +179,8 @@ class App { | |||||||
|         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, |         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, | ||||||
|         'pinned': pinned, |         'pinned': pinned, | ||||||
|         'categories': categories, |         'categories': categories, | ||||||
|         'releaseDate': releaseDate?.microsecondsSinceEpoch |         'releaseDate': releaseDate?.microsecondsSinceEpoch, | ||||||
|  |         'changeLog': changeLog | ||||||
|       }; |       }; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -220,6 +229,7 @@ class AppSource { | |||||||
|   String? host; |   String? host; | ||||||
|   late String name; |   late String name; | ||||||
|   bool enforceTrackOnly = false; |   bool enforceTrackOnly = false; | ||||||
|  |   bool changeLogIfAnyIsMarkDown = true; | ||||||
|  |  | ||||||
|   AppSource() { |   AppSource() { | ||||||
|     name = runtimeType.toString(); |     name = runtimeType.toString(); | ||||||
| @@ -338,6 +348,9 @@ class SourceProvider { | |||||||
|     APKMirror(), |     APKMirror(), | ||||||
|     FDroidRepo(), |     FDroidRepo(), | ||||||
|     SteamMobile(), |     SteamMobile(), | ||||||
|  |     TelegramApp(), | ||||||
|  |     VLC(), | ||||||
|  |     NeutronCode(), | ||||||
|     HTML() // This should ALWAYS be the last option as they are tried in order |     HTML() // This should ALWAYS be the last option as they are tried in order | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
| @@ -433,7 +446,8 @@ class SourceProvider { | |||||||
|         DateTime.now(), |         DateTime.now(), | ||||||
|         currentApp?.pinned ?? false, |         currentApp?.pinned ?? false, | ||||||
|         categories: currentApp?.categories ?? const [], |         categories: currentApp?.categories ?? const [], | ||||||
|         releaseDate: apk.releaseDate); |         releaseDate: apk.releaseDate, | ||||||
|  |         changeLog: apk.changeLog); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Returns errors in [results, errors] instead of throwing them |   // Returns errors in [results, errors] instead of throwing them | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -235,6 +235,14 @@ packages: | |||||||
|     description: flutter |     description: flutter | ||||||
|     source: sdk |     source: sdk | ||||||
|     version: "0.0.0" |     version: "0.0.0" | ||||||
|  |   flutter_markdown: | ||||||
|  |     dependency: "direct main" | ||||||
|  |     description: | ||||||
|  |       name: flutter_markdown | ||||||
|  |       sha256: "7b25c10de1fea883f3c4f9b8389506b54053cd00807beab69fd65c8653a2711f" | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "0.6.14" | ||||||
|   flutter_plugin_android_lifecycle: |   flutter_plugin_android_lifecycle: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -325,6 +333,14 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.1" |     version: "2.0.1" | ||||||
|  |   markdown: | ||||||
|  |     dependency: transitive | ||||||
|  |     description: | ||||||
|  |       name: markdown | ||||||
|  |       sha256: b3c60dee8c2af50ad0e6e90cceba98e47718a6ee0a7a6772c77846a0cc21f78b | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "7.0.1" | ||||||
|   matcher: |   matcher: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -702,10 +718,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_ios |       name: url_launcher_ios | ||||||
|       sha256: "7ab1e5b646623d6a2537aa59d5d039f90eebef75a7c25e105f6f75de1f7750c3" |       sha256: "3dedc66ca3c0bef9e6a93c0999aee102556a450afcc1b7bcfeace7a424927d92" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.1.2" |     version: "6.1.3" | ||||||
|   url_launcher_linux: |   url_launcher_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -766,10 +782,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter |       name: webview_flutter | ||||||
|       sha256: b6cd42db3ced5411f3d01599906156885b18e4188f7065a8a351eb84bee347e0 |       sha256: "47663d51a9061451aa3880a214ee9a65dcbb933b77bc44388e194279ab3ccaf6" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.0.6" |     version: "4.0.7" | ||||||
|   webview_flutter_android: |   webview_flutter_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev | |||||||
| # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | ||||||
| # In Windows, build-name is used as the major, minor, and patch parts | # In Windows, build-name is used as the major, minor, and patch parts | ||||||
| # of the product and file versions while build-number is used as the build suffix. | # of the product and file versions while build-number is used as the build suffix. | ||||||
| version: 0.11.9+130 # When changing this, update the tag in main() accordingly | version: 0.11.12+133 # When changing this, update the tag in main() accordingly | ||||||
|  |  | ||||||
| environment: | environment: | ||||||
|   sdk: '>=2.18.2 <3.0.0' |   sdk: '>=2.18.2 <3.0.0' | ||||||
| @@ -59,6 +59,7 @@ dependencies: | |||||||
|   sqflite: ^2.2.0+3 |   sqflite: ^2.2.0+3 | ||||||
|   easy_localization: ^3.0.1 |   easy_localization: ^3.0.1 | ||||||
|   android_intent_plus: ^3.1.5 |   android_intent_plus: ^3.1.5 | ||||||
|  |   flutter_markdown: ^0.6.14 | ||||||
|  |  | ||||||
|  |  | ||||||
| dev_dependencies: | dev_dependencies: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user