mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-31 13:33:28 +01:00 
			
		
		
		
	Compare commits
	
		
			46 Commits
		
	
	
		
			v0.11.26-b
			...
			v0.11.34-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 02da24aa75 | ||
|  | 3c6e66ce12 | ||
|  | 0213b542e3 | ||
|  | b0e8a4a297 | ||
|  | e72b33ebf2 | ||
|  | 283722319b | ||
|  | b406bb5c6a | ||
|  | de2b7fa7a1 | ||
|  | be61220af4 | ||
|  | 3e732a4317 | ||
|  | 9f2db4e4e7 | ||
|  | 78141998f4 | ||
|  | 934f237e34 | ||
|  | 1b2a9a39e3 | ||
|  | dc52fb6181 | ||
|  | 9e4ac397d8 | ||
|  | 0ec944eae9 | ||
|  | ad250c30e4 | ||
|  | 1090f15508 | ||
|  | 666941350e | ||
|  | eeadbce8b0 | ||
|  | ce8aeff342 | ||
|  | 0d8362a2ed | ||
|  | 3b28143a4e | ||
|  | 537628f378 | ||
|  | c92d76df98 | ||
|  | b6959e1a8b | ||
|  | 1bf648da60 | ||
|  | 6a1275e9e4 | ||
|  | df242b91ad | ||
|  | 7ea75325bb | ||
|  | 0704dfe2ee | ||
|  | 6275cbf114 | ||
|  | 36b8ef6782 | ||
|  | d274b9a428 | ||
|  | 1c2980d1ac | ||
|  | 8f0aac057e | ||
|  | e929920a48 | ||
|  | 8ed254c7dd | ||
|  | 46a00836df | ||
|  | f144ffdded | ||
|  | d597d569e2 | ||
|  | b62475de87 | ||
|  | 334ac8d3d6 | ||
|  | 9193788356 | ||
|  | 8f75ddd43f | 
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "System folgen", |     "followSystem": "System folgen", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App sortieren nach", |     "appSortBy": "App sortieren nach", | ||||||
|     "authorName": "Autor/Name", |     "authorName": "Autor/Name", | ||||||
|     "nameAuthor": "Name/Autor", |     "nameAuthor": "Name/Autor", | ||||||
| @@ -221,11 +222,11 @@ | |||||||
|     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", |     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", | ||||||
|     "versionDetection": "Versionserkennung", |     "versionDetection": "Versionserkennung", | ||||||
|     "standardVersionDetection": "Standardversionserkennung", |     "standardVersionDetection": "Standardversionserkennung", | ||||||
|     "groupByCategory": "Group by Category", |     "groupByCategory": "Nach Kategorie gruppieren", | ||||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", |     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "App entfernen?", |         "one": "App entfernen?", | ||||||
|         "other": "App entfernen?" |         "other": "Apps entfernen?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", |         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Follow System", |     "followSystem": "Follow System", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App Sort By", |     "appSortBy": "App Sort By", | ||||||
|     "authorName": "Author/Name", |     "authorName": "Author/Name", | ||||||
|     "nameAuthor": "Name/Author", |     "nameAuthor": "Name/Author", | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "هماهنگ با سیستم", |     "followSystem": "هماهنگ با سیستم", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "مرتب سازی برنامه بر اساس", |     "appSortBy": "مرتب سازی برنامه بر اساس", | ||||||
|     "authorName": "سازنده/اسم", |     "authorName": "سازنده/اسم", | ||||||
|     "nameAuthor": "اسم/سازنده", |     "nameAuthor": "اسم/سازنده", | ||||||
| @@ -207,7 +208,7 @@ | |||||||
|     "addCategory": "اضافه کردن دسته", |     "addCategory": "اضافه کردن دسته", | ||||||
|     "label": "برچسب", |     "label": "برچسب", | ||||||
|     "language": "زبان", |     "language": "زبان", | ||||||
|     "copiedToClipboard": "Copied to Clipboard", |     "copiedToClipboard": "در کلیپ بورد کپی شد", | ||||||
|     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", |     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", | ||||||
|     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", |     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", | ||||||
|     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", |     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", | ||||||
| @@ -221,8 +222,8 @@ | |||||||
|     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", |     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", | ||||||
|     "versionDetection": "تشخیص نسخه", |     "versionDetection": "تشخیص نسخه", | ||||||
|     "standardVersionDetection": "تشخیص نسخه استاندارد", |     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||||
|     "groupByCategory": "Group by Category", |     "groupByCategory": "گروه بر اساس دسته", | ||||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", |     "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "برنامه حذف شود؟", |         "one": "برنامه حذف شود؟", | ||||||
|         "other": "برنامه ها حذف شوند؟" |         "other": "برنامه ها حذف شوند؟" | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Suivre le système", |     "followSystem": "Suivre le système", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "Applications triées par", |     "appSortBy": "Applications triées par", | ||||||
|     "authorName": "Auteur/Nom", |     "authorName": "Auteur/Nom", | ||||||
|     "nameAuthor": "Nom/Auteur", |     "nameAuthor": "Nom/Auteur", | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Rendszer szerint", |     "followSystem": "Rendszer szerint", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App rendezés...", |     "appSortBy": "App rendezés...", | ||||||
|     "authorName": "Szerző/Név", |     "authorName": "Szerző/Név", | ||||||
|     "nameAuthor": "Név/Szerző", |     "nameAuthor": "Név/Szerző", | ||||||
| @@ -221,7 +222,7 @@ | |||||||
|     "versionDetection": "Verzió érzékelés", |     "versionDetection": "Verzió érzékelés", | ||||||
|     "standardVersionDetection": "Alapért. verzió érzékelés", |     "standardVersionDetection": "Alapért. verzió érzékelés", | ||||||
|     "groupByCategory": "Csoportosítás Kategória alapján", |     "groupByCategory": "Csoportosítás Kategória alapján", | ||||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", |     "autoApkFilterByArch": "Ha lehetséges, próbálja CPU architektúra szerint szűrni az APK-kat", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Eltávolítja az alkalmazást?", |         "one": "Eltávolítja az alkalmazást?", | ||||||
|         "other": "Eltávolítja az alkalmazást?" |         "other": "Eltávolítja az alkalmazást?" | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Segui sistema", |     "followSystem": "Segui sistema", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App ordinate per", |     "appSortBy": "App ordinate per", | ||||||
|     "authorName": "Autore/Nome", |     "authorName": "Autore/Nome", | ||||||
|     "nameAuthor": "Nome/Autore", |     "nameAuthor": "Nome/Autore", | ||||||
| @@ -207,7 +208,7 @@ | |||||||
|     "addCategory": "Aggiungi categoria", |     "addCategory": "Aggiungi categoria", | ||||||
|     "label": "Etichetta", |     "label": "Etichetta", | ||||||
|     "language": "Lingua", |     "language": "Lingua", | ||||||
|     "copiedToClipboard": "Copied to Clipboard", |     "copiedToClipboard": "Copiato negli appunti", | ||||||
|     "storagePermissionDenied": "Accesso ai file non autorizzato", |     "storagePermissionDenied": "Accesso ai file non autorizzato", | ||||||
|     "selectedCategorizeWarning": "Ciò sostituirà le impostazioni di categoria esistenti per le App selezionate.", |     "selectedCategorizeWarning": "Ciò sostituirà le impostazioni di categoria esistenti per le App selezionate.", | ||||||
|     "filterAPKsByRegEx": "Filtra file APK con espressioni regolari", |     "filterAPKsByRegEx": "Filtra file APK con espressioni regolari", | ||||||
| @@ -221,8 +222,8 @@ | |||||||
|     "importFromURLsInFile": "Importa da URL in file (come OPML)", |     "importFromURLsInFile": "Importa da URL in file (come OPML)", | ||||||
|     "versionDetection": "Rilevamento di versione", |     "versionDetection": "Rilevamento di versione", | ||||||
|     "standardVersionDetection": "Rilevamento di versione standard", |     "standardVersionDetection": "Rilevamento di versione standard", | ||||||
|     "groupByCategory": "Group by Category", |     "groupByCategory": "Raggruppa per categoria", | ||||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", |     "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Rimuovere l'App?", |         "one": "Rimuovere l'App?", | ||||||
|         "other": "Rimuovere le App?" |         "other": "Rimuovere le App?" | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "システムに従う", |     "followSystem": "システムに従う", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "アプリの並び方", |     "appSortBy": "アプリの並び方", | ||||||
|     "authorName": "作者名/アプリ名", |     "authorName": "作者名/アプリ名", | ||||||
|     "nameAuthor": "アプリ名/作者名", |     "nameAuthor": "アプリ名/作者名", | ||||||
| @@ -221,8 +222,8 @@ | |||||||
|     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", |     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", | ||||||
|     "versionDetection": "バージョン検出", |     "versionDetection": "バージョン検出", | ||||||
|     "standardVersionDetection": "標準のバージョン検出", |     "standardVersionDetection": "標準のバージョン検出", | ||||||
|     "groupByCategory": "Group by Category", |     "groupByCategory": "カテゴリ別にグループ化する", | ||||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", |     "autoApkFilterByArch": "可能であれば,CPUアーキテクチャによるAPKのフィルタリングを試みる", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "アプリを削除しますか?", |         "one": "アプリを削除しますか?", | ||||||
|         "other": "アプリを削除しますか?" |         "other": "アプリを削除しますか?" | ||||||
|   | |||||||
| @@ -123,6 +123,7 @@ | |||||||
|     "followSystem": "跟随系统", |     "followSystem": "跟随系统", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "排列方式", |     "appSortBy": "排列方式", | ||||||
|     "authorName": "作者 / 名字", |     "authorName": "作者 / 名字", | ||||||
|     "nameAuthor": "名字 / 作者", |     "nameAuthor": "名字 / 作者", | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/app_sources/github.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/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| @@ -35,6 +36,8 @@ class Codeberg extends AppSource { | |||||||
|     canSearch = true; |     canSearch = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   var gh = GitHub(); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String standardizeURL(String url) { |   String standardizeURL(String url) { | ||||||
|     RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+'); |     RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+'); | ||||||
| @@ -54,79 +57,10 @@ class Codeberg extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases'] == true; |     return gh.getLatestAPKDetailsCommon( | ||||||
|     bool fallbackToOlderReleases = |         'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100', | ||||||
|         additionalSettings['fallbackToOlderReleases'] == true; |         standardUrl, | ||||||
|     String? regexFilter = |         additionalSettings); | ||||||
|         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) |  | ||||||
|                     ?.isNotEmpty == |  | ||||||
|                 true |  | ||||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] |  | ||||||
|             : null; |  | ||||||
|     Response res = await get(Uri.parse( |  | ||||||
|         'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100')); |  | ||||||
|     if (res.statusCode == 200) { |  | ||||||
|       var releases = jsonDecode(res.body) as List<dynamic>; |  | ||||||
|  |  | ||||||
|       List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) => |  | ||||||
|           (release['assets'] as List<dynamic>?) |  | ||||||
|               ?.map((e) { |  | ||||||
|                 return e['name'] != null && e['browser_download_url'] != null |  | ||||||
|                     ? MapEntry(e['name'] as String, |  | ||||||
|                         e['browser_download_url'] as String) |  | ||||||
|                     : const MapEntry('', ''); |  | ||||||
|               }) |  | ||||||
|               .where((element) => element.key.toLowerCase().endsWith('.apk')) |  | ||||||
|               .toList() ?? |  | ||||||
|           []; |  | ||||||
|  |  | ||||||
|       dynamic targetRelease; |  | ||||||
|  |  | ||||||
|       for (int i = 0; i < releases.length; i++) { |  | ||||||
|         if (!fallbackToOlderReleases && i > 0) break; |  | ||||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|         if (releases[i]['draft'] == true) { |  | ||||||
|           // Draft releases not supported |  | ||||||
|         } |  | ||||||
|         var nameToFilter = releases[i]['name'] as String?; |  | ||||||
|         if (nameToFilter == null || nameToFilter.trim().isEmpty) { |  | ||||||
|           // Some leave titles empty so tag is used |  | ||||||
|           nameToFilter = releases[i]['tag_name'] as String; |  | ||||||
|         } |  | ||||||
|         if (regexFilter != null && |  | ||||||
|             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|         var apkUrls = getReleaseAPKUrls(releases[i]); |  | ||||||
|         if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) { |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|         targetRelease = releases[i]; |  | ||||||
|         targetRelease['apkUrls'] = apkUrls; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       if (targetRelease == null) { |  | ||||||
|         throw NoReleasesError(); |  | ||||||
|       } |  | ||||||
|       String? version = targetRelease['tag_name']; |  | ||||||
|       DateTime? releaseDate = targetRelease['published_at'] != null |  | ||||||
|           ? DateTime.parse(targetRelease['published_at']) |  | ||||||
|           : null; |  | ||||||
|       if (version == null) { |  | ||||||
|         throw NoVersionError(); |  | ||||||
|       } |  | ||||||
|       var changeLog = targetRelease['body'].toString(); |  | ||||||
|       return APKDetails( |  | ||||||
|           version, |  | ||||||
|           targetRelease['apkUrls'] as List<MapEntry<String, String>>, |  | ||||||
|           getAppNames(standardUrl), |  | ||||||
|           releaseDate: releaseDate, |  | ||||||
|           changeLog: changeLog.isEmpty ? null : changeLog); |  | ||||||
|     } else { |  | ||||||
|       throw getObtainiumHttpError(res); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   AppNames getAppNames(String standardUrl) { |   AppNames getAppNames(String standardUrl) { | ||||||
| @@ -137,20 +71,9 @@ class Codeberg extends AppSource { | |||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<Map<String, String>> search(String query) async { |   Future<Map<String, String>> search(String query) async { | ||||||
|     Response res = await get(Uri.parse( |     return gh.searchCommon( | ||||||
|         'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100')); |         query, | ||||||
|     if (res.statusCode == 200) { |         'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100', | ||||||
|       Map<String, String> urlsWithDescriptions = {}; |         'data'); | ||||||
|       for (var e in (jsonDecode(res.body)['data'] as List<dynamic>)) { |  | ||||||
|         urlsWithDescriptions.addAll({ |  | ||||||
|           e['html_url'] as String: e['description'] != null |  | ||||||
|               ? e['description'] as String |  | ||||||
|               : tr('noDescription') |  | ||||||
|         }); |  | ||||||
|       } |  | ||||||
|       return urlsWithDescriptions; |  | ||||||
|     } else { |  | ||||||
|       throw getObtainiumHttpError(res); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -96,11 +96,9 @@ class GitHub extends AppSource { | |||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => |   String? changeLogPageFromStandardUrl(String standardUrl) => | ||||||
|       '$standardUrl/releases'; |       '$standardUrl/releases'; | ||||||
|  |  | ||||||
|   @override |   Future<APKDetails> getLatestAPKDetailsCommon(String requestUrl, | ||||||
|   Future<APKDetails> getLatestAPKDetails( |       String standardUrl, Map<String, dynamic> additionalSettings, | ||||||
|     String standardUrl, |       {Function(Response)? onHttpErrorCode}) async { | ||||||
|     Map<String, dynamic> additionalSettings, |  | ||||||
|   ) async { |  | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases'] == true; |     bool includePrereleases = additionalSettings['includePrereleases'] == true; | ||||||
|     bool fallbackToOlderReleases = |     bool fallbackToOlderReleases = | ||||||
|         additionalSettings['fallbackToOlderReleases'] == true; |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
| @@ -110,27 +108,50 @@ class GitHub extends AppSource { | |||||||
|                 true |                 true | ||||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|             : null; |             : null; | ||||||
|     Response res = await get(Uri.parse( |     Response res = await get(Uri.parse(requestUrl)); | ||||||
|         'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100')); |  | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var releases = jsonDecode(res.body) as List<dynamic>; |       var releases = jsonDecode(res.body) as List<dynamic>; | ||||||
|  |  | ||||||
|       List<String> getReleaseAPKUrls(dynamic release) => |       List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) => | ||||||
|           (release['assets'] as List<dynamic>?) |           (release['assets'] as List<dynamic>?) | ||||||
|               ?.map((e) { |               ?.map((e) { | ||||||
|                 return e['browser_download_url'] != null |                 return e['name'] != null && e['browser_download_url'] != null | ||||||
|                     ? e['browser_download_url'] as String |                     ? MapEntry(e['name'] as String, | ||||||
|                     : ''; |                         e['browser_download_url'] as String) | ||||||
|  |                     : const MapEntry('', ''); | ||||||
|               }) |               }) | ||||||
|               .where((element) => element.toLowerCase().endsWith('.apk')) |               .where((element) => element.key.toLowerCase().endsWith('.apk')) | ||||||
|               .toList() ?? |               .toList() ?? | ||||||
|           []; |           []; | ||||||
|  |  | ||||||
|  |       DateTime? getReleaseDateFromRelease(dynamic rel) => | ||||||
|  |           rel?['published_at'] != null | ||||||
|  |               ? DateTime.parse(rel['published_at']) | ||||||
|  |               : null; | ||||||
|  |       releases.sort((a, b) { | ||||||
|  |         // See #478 | ||||||
|  |         if (a == b) { | ||||||
|  |           return 0; | ||||||
|  |         } else if (a == null) { | ||||||
|  |           return -1; | ||||||
|  |         } else if (b == null) { | ||||||
|  |           return 1; | ||||||
|  |         } else { | ||||||
|  |           return getReleaseDateFromRelease(a)! | ||||||
|  |               .compareTo(getReleaseDateFromRelease(b)!); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |       releases = releases.reversed.toList(); | ||||||
|       dynamic targetRelease; |       dynamic targetRelease; | ||||||
|  |       var prerrelsSkipped = 0; | ||||||
|       for (int i = 0; i < releases.length; i++) { |       for (int i = 0; i < releases.length; i++) { | ||||||
|         if (!fallbackToOlderReleases && i > 0) break; |         if (!fallbackToOlderReleases && i > prerrelsSkipped) break; | ||||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { |         if (!includePrereleases && releases[i]['prerelease'] == true) { | ||||||
|  |           prerrelsSkipped++; | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         if (releases[i]['draft'] == true) { | ||||||
|  |           // Draft releases not supported | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|         var nameToFilter = releases[i]['name'] as String?; |         var nameToFilter = releases[i]['name'] as String?; | ||||||
| @@ -154,38 +175,51 @@ class GitHub extends AppSource { | |||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String? version = targetRelease['tag_name']; |       String? version = targetRelease['tag_name']; | ||||||
|       DateTime? releaseDate = targetRelease['published_at'] != null |       DateTime? releaseDate = getReleaseDateFromRelease(targetRelease); | ||||||
|           ? DateTime.parse(targetRelease['published_at']) |  | ||||||
|           : null; |  | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       var changeLog = targetRelease['body'].toString(); |       var changeLog = targetRelease['body'].toString(); | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, |           version, | ||||||
|           getApkUrlsFromUrls(targetRelease['apkUrls'] as List<String>), |           targetRelease['apkUrls'] as List<MapEntry<String, String>>, | ||||||
|           getAppNames(standardUrl), |           getAppNames(standardUrl), | ||||||
|           releaseDate: releaseDate, |           releaseDate: releaseDate, | ||||||
|           changeLog: changeLog.isEmpty ? null : changeLog); |           changeLog: changeLog.isEmpty ? null : changeLog); | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       if (onHttpErrorCode != null) { | ||||||
|  |         onHttpErrorCode(res); | ||||||
|  |       } | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     return getLatestAPKDetailsCommon( | ||||||
|  |         'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100', | ||||||
|  |         standardUrl, | ||||||
|  |         additionalSettings, onHttpErrorCode: (Response res) { | ||||||
|  |       rateLimitErrorCheck(res); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   AppNames getAppNames(String standardUrl) { |   AppNames getAppNames(String standardUrl) { | ||||||
|     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); |     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); | ||||||
|     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); |     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); | ||||||
|     return AppNames(names[0], names[1]); |     return AppNames(names[0], names[1]); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   Future<Map<String, String>> searchCommon( | ||||||
|   Future<Map<String, String>> search(String query) async { |       String query, String requestUrl, String rootProp, | ||||||
|     Response res = await get(Uri.parse( |       {Function(Response)? onHttpErrorCode}) async { | ||||||
|         'https://${await getCredentialPrefixIfAny()}api.$host/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100')); |     Response res = await get(Uri.parse(requestUrl)); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       Map<String, String> urlsWithDescriptions = {}; |       Map<String, String> urlsWithDescriptions = {}; | ||||||
|       for (var e in (jsonDecode(res.body)['items'] as List<dynamic>)) { |       for (var e in (jsonDecode(res.body)[rootProp] as List<dynamic>)) { | ||||||
|         urlsWithDescriptions.addAll({ |         urlsWithDescriptions.addAll({ | ||||||
|           e['html_url'] as String: |           e['html_url'] as String: | ||||||
|               ((e['archived'] == true ? '[ARCHIVED] ' : '') + |               ((e['archived'] == true ? '[ARCHIVED] ' : '') + | ||||||
| @@ -196,11 +230,23 @@ class GitHub extends AppSource { | |||||||
|       } |       } | ||||||
|       return urlsWithDescriptions; |       return urlsWithDescriptions; | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       if (onHttpErrorCode != null) { | ||||||
|  |         onHttpErrorCode(res); | ||||||
|  |       } | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<Map<String, String>> search(String query) async { | ||||||
|  |     return searchCommon( | ||||||
|  |         query, | ||||||
|  |         'https://${await getCredentialPrefixIfAny()}api.$host/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100', | ||||||
|  |         'items', onHttpErrorCode: (Response res) { | ||||||
|  |       rateLimitErrorCheck(res); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   rateLimitErrorCheck(Response res) { |   rateLimitErrorCheck(Response res) { | ||||||
|     if (res.headers['x-ratelimit-remaining'] == '0') { |     if (res.headers['x-ratelimit-remaining'] == '0') { | ||||||
|       throw RateLimitError( |       throw RateLimitError( | ||||||
|   | |||||||
| @@ -3,10 +3,19 @@ import 'package:http/http.dart'; | |||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  |  | ||||||
| class GitLab extends AppSource { | class GitLab extends AppSource { | ||||||
|   GitLab() { |   GitLab() { | ||||||
|     host = 'gitlab.com'; |     host = 'gitlab.com'; | ||||||
|  |  | ||||||
|  |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('fallbackToOlderReleases', | ||||||
|  |             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||||
|  |       ] | ||||||
|  |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -28,41 +37,58 @@ class GitLab extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|  |     bool fallbackToOlderReleases = | ||||||
|  |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|     Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom')); |     Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var standardUri = Uri.parse(standardUrl); |       var standardUri = Uri.parse(standardUrl); | ||||||
|       var parsedHtml = parse(res.body); |       var parsedHtml = parse(res.body); | ||||||
|       var entry = parsedHtml.querySelector('entry'); |       var apkDetailsList = parsedHtml.querySelectorAll('entry').map((entry) { | ||||||
|       var entryContent = |         var entryContent = parse( | ||||||
|           parse(parseFragment(entry?.querySelector('content')!.innerHtml).text); |             parseFragment(entry.querySelector('content')!.innerHtml).text); | ||||||
|       var apkUrls = [ |         var apkUrls = [ | ||||||
|         ...getLinksFromParsedHTML( |           ...getLinksFromParsedHTML( | ||||||
|             entryContent, |               entryContent, | ||||||
|             RegExp( |               RegExp( | ||||||
|                 '^${standardUri.path.replaceAllMapped(RegExp(r'[.*+?^${}()|[\]\\]'), (x) { |                   '^${standardUri.path.replaceAllMapped(RegExp(r'[.*+?^${}()|[\]\\]'), (x) { | ||||||
|                   return '\\${x[0]}'; |                     return '\\${x[0]}'; | ||||||
|                 })}/uploads/[^/]+/[^/]+\\.apk\$', |                   })}/uploads/[^/]+/[^/]+\\.apk\$', | ||||||
|                 caseSensitive: false), |                   caseSensitive: false), | ||||||
|             standardUri.origin), |               standardUri.origin), | ||||||
|         // GitLab releases may contain links to externally hosted APKs |           // GitLab releases may contain links to externally hosted APKs | ||||||
|         ...getLinksFromParsedHTML(entryContent, |           ...getLinksFromParsedHTML(entryContent, | ||||||
|                 RegExp('/[^/]+\\.apk\$', caseSensitive: false), '') |                   RegExp('/[^/]+\\.apk\$', caseSensitive: false), '') | ||||||
|             .where((element) => Uri.parse(element).host != '') |               .where((element) => Uri.parse(element).host != '') | ||||||
|             .toList() |               .toList() | ||||||
|       ]; |         ]; | ||||||
|  |  | ||||||
|       var entryId = entry?.querySelector('id')?.innerHtml; |         var entryId = entry.querySelector('id')?.innerHtml; | ||||||
|       var version = |         var version = | ||||||
|           entryId == null ? null : Uri.parse(entryId).pathSegments.last; |             entryId == null ? null : Uri.parse(entryId).pathSegments.last; | ||||||
|       var releaseDateString = entry?.querySelector('updated')?.innerHtml; |         var releaseDateString = entry.querySelector('updated')?.innerHtml; | ||||||
|       DateTime? releaseDate = |         DateTime? releaseDate = releaseDateString != null | ||||||
|           releaseDateString != null ? DateTime.parse(releaseDateString) : null; |             ? DateTime.parse(releaseDateString) | ||||||
|       if (version == null) { |             : null; | ||||||
|         throw NoVersionError(); |         if (version == null) { | ||||||
|  |           throw NoVersionError(); | ||||||
|  |         } | ||||||
|  |         return APKDetails(version, getApkUrlsFromUrls(apkUrls), | ||||||
|  |             GitHub().getAppNames(standardUrl), | ||||||
|  |             releaseDate: releaseDate); | ||||||
|  |       }); | ||||||
|  |       if (apkDetailsList.isEmpty) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, getApkUrlsFromUrls(apkUrls), |       if (fallbackToOlderReleases) { | ||||||
|           GitHub().getAppNames(standardUrl), |         if (additionalSettings['trackOnly'] != true) { | ||||||
|           releaseDate: releaseDate); |           apkDetailsList = | ||||||
|  |               apkDetailsList.where((e) => e.apkUrls.isNotEmpty).toList(); | ||||||
|  |         } | ||||||
|  |         if (apkDetailsList.isEmpty) { | ||||||
|  |           throw NoReleasesError(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       return apkDetailsList.first; | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -34,14 +34,20 @@ class HTML extends AppSource { | |||||||
|       var rel = links.last; |       var rel = links.last; | ||||||
|       var apkName = rel.split('/').last; |       var apkName = rel.split('/').last; | ||||||
|       var version = apkName.substring(0, apkName.length - 4); |       var version = apkName.substring(0, apkName.length - 4); | ||||||
|       List<String> apkUrls = [rel] |       List<String> apkUrls = [rel].map((e) { | ||||||
|           .map((e) => e.toLowerCase().startsWith('http://') || |         try { | ||||||
|                   e.toLowerCase().startsWith('https://') |           Uri.parse(e).origin; | ||||||
|               ? e |           return e; | ||||||
|               : e.startsWith('/') |         } catch (err) { | ||||||
|                   ? '${uri.origin}/$e' |           // is relative | ||||||
|                   : '${uri.origin}/${uri.path}/$e') |         } | ||||||
|           .toList(); |         var currPathSegments = uri.path.split('/'); | ||||||
|  |         if (e.startsWith('/') || currPathSegments.isEmpty) { | ||||||
|  |           return '${uri.origin}/$e'; | ||||||
|  |         } else { | ||||||
|  |           return '${uri.origin}/${currPathSegments.sublist(0, currPathSegments.length - 1).join('/')}/$e'; | ||||||
|  |         } | ||||||
|  |       }).toList(); | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, getApkUrlsFromUrls(apkUrls), AppNames(uri.host, tr('app'))); |           version, getApkUrlsFromUrls(apkUrls), AppNames(uri.host, tr('app'))); | ||||||
|     } else { |     } else { | ||||||
|   | |||||||
| @@ -31,7 +31,8 @@ class SourceForge extends AppSource { | |||||||
|       getVersion(String url) { |       getVersion(String url) { | ||||||
|         try { |         try { | ||||||
|           var tokens = url.split('/'); |           var tokens = url.split('/'); | ||||||
|           return tokens[tokens.length - 3]; |           var fi = tokens.indexOf('files'); | ||||||
|  |           return tokens[tokens[fi + 2] == 'download' ? fi - 1 : fi + 1]; | ||||||
|         } catch (e) { |         } catch (e) { | ||||||
|           return null; |           return null; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -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.26'; | const String currentVersion = '0.11.34'; | ||||||
| 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 | ||||||
|  |  | ||||||
| @@ -263,6 +263,14 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|         darkColorScheme = ColorScheme.fromSeed( |         darkColorScheme = ColorScheme.fromSeed( | ||||||
|             seedColor: defaultThemeColour, brightness: Brightness.dark); |             seedColor: defaultThemeColour, brightness: Brightness.dark); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       // set the background and surface colors to pure black in the amoled theme | ||||||
|  |       if (settingsProvider.useBlackTheme) { | ||||||
|  |         darkColorScheme = darkColorScheme | ||||||
|  |             .copyWith(background: Colors.black, surface: Colors.black) | ||||||
|  |             .harmonized(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|       return MaterialApp( |       return MaterialApp( | ||||||
|           title: 'Obtainium', |           title: 'Obtainium', | ||||||
|           localizationsDelegates: context.localizationDelegates, |           localizationsDelegates: context.localizationDelegates, | ||||||
|   | |||||||
| @@ -127,7 +127,8 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|             if (apkUrl == null) { |             if (apkUrl == null) { | ||||||
|               throw ObtainiumError(tr('cancelled')); |               throw ObtainiumError(tr('cancelled')); | ||||||
|             } |             } | ||||||
|             app.preferredApkIndex = app.apkUrls.indexOf(apkUrl); |             app.preferredApkIndex = | ||||||
|  |                 app.apkUrls.map((e) => e.value).toList().indexOf(apkUrl.value); | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
|             var downloadedApk = await appsProvider.downloadApp( |             var downloadedApk = await appsProvider.downloadApp( | ||||||
|                 app, globalNavigatorKey.currentContext); |                 app, globalNavigatorKey.currentContext); | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|     bool areDownloadsRunning = appsProvider.areDownloadsRunning(); |     bool areDownloadsRunning = appsProvider.areDownloadsRunning(); | ||||||
|  |  | ||||||
|     var sourceProvider = SourceProvider(); |     var sourceProvider = SourceProvider(); | ||||||
|     AppInMemory? app = appsProvider.apps[widget.appId]; |     AppInMemory? app = appsProvider.apps[widget.appId]?.deepCopy(); | ||||||
|     var source = app != null ? sourceProvider.getSource(app.app.url) : null; |     var source = app != null ? sourceProvider.getSource(app.app.url) : null; | ||||||
|     if (!areDownloadsRunning && prevApp == null && app != null) { |     if (!areDownloadsRunning && prevApp == null && app != null) { | ||||||
|       prevApp = app; |       prevApp = app; | ||||||
| @@ -153,7 +153,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|               height: 25, |               height: 25, | ||||||
|             ), |             ), | ||||||
|             Text( |             Text( | ||||||
|               app?.app.name ?? tr('app'), |               app?.name ?? tr('app'), | ||||||
|               textAlign: TextAlign.center, |               textAlign: TextAlign.center, | ||||||
|               style: Theme.of(context).textTheme.displayLarge, |               style: Theme.of(context).textTheme.displayLarge, | ||||||
|             ), |             ), | ||||||
| @@ -268,9 +268,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|             }).toList(); |             }).toList(); | ||||||
|  |  | ||||||
|             return GeneratedFormModal( |             return GeneratedFormModal( | ||||||
|               title: tr('additionalOptions'), |                 title: tr('additionalOptions'), items: items); | ||||||
|               items: items, |  | ||||||
|             ); |  | ||||||
|           }); |           }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -307,6 +305,15 @@ class _AppPageState extends State<AppPage> { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     getResetInstallStatusButton() => TextButton( | ||||||
|  |         onPressed: app?.app == null | ||||||
|  |             ? null | ||||||
|  |             : () { | ||||||
|  |                 app!.app.installedVersion = null; | ||||||
|  |                 appsProvider.saveApps([app.app]); | ||||||
|  |               }, | ||||||
|  |         child: Text(tr('resetInstallStatus'))); | ||||||
|  |  | ||||||
|     getInstallOrUpdateButton() => TextButton( |     getInstallOrUpdateButton() => TextButton( | ||||||
|         onPressed: (app?.app.installedVersion == null || |         onPressed: (app?.app.installedVersion == null || | ||||||
|                     app?.app.installedVersion != app?.app.latestVersion) && |                     app?.app.installedVersion != app?.app.latestVersion) && | ||||||
| @@ -386,7 +393,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|                                       scrollable: true, |                                       scrollable: true, | ||||||
|                                       content: getInfoColumn(), |                                       content: getInfoColumn(), | ||||||
|                                       title: Text( |                                       title: Text( | ||||||
|                                           '${app.app.name} ${tr('byX', args: [ |                                           '${app.name} ${tr('byX', args: [ | ||||||
|                                             app.app.author |                                             app.app.author | ||||||
|                                           ])}'), |                                           ])}'), | ||||||
|                                       actions: [ |                                       actions: [ | ||||||
| @@ -402,7 +409,13 @@ class _AppPageState extends State<AppPage> { | |||||||
|                             icon: const Icon(Icons.more_horiz), |                             icon: const Icon(Icons.more_horiz), | ||||||
|                             tooltip: tr('more')), |                             tooltip: tr('more')), | ||||||
|                       const SizedBox(width: 16.0), |                       const SizedBox(width: 16.0), | ||||||
|                       Expanded(child: getInstallOrUpdateButton()), |                       Expanded( | ||||||
|  |                           child: !isVersionDetectionStandard && | ||||||
|  |                                   app?.app.installedVersion != null && | ||||||
|  |                                   app?.app.installedVersion == | ||||||
|  |                                       app?.app.latestVersion | ||||||
|  |                               ? getResetInstallStatusButton() | ||||||
|  |                               : getInstallOrUpdateButton()), | ||||||
|                       const SizedBox(width: 16.0), |                       const SizedBox(width: 16.0), | ||||||
|                       Expanded( |                       Expanded( | ||||||
|                           child: TextButton( |                           child: TextButton( | ||||||
|   | |||||||
| @@ -29,13 +29,13 @@ class AppsPageState extends State<AppsPage> { | |||||||
|   final AppsFilter neutralFilter = AppsFilter(); |   final AppsFilter neutralFilter = AppsFilter(); | ||||||
|   var updatesOnlyFilter = |   var updatesOnlyFilter = | ||||||
|       AppsFilter(includeUptodate: false, includeNonInstalled: false); |       AppsFilter(includeUptodate: false, includeNonInstalled: false); | ||||||
|   Set<App> selectedApps = {}; |   Set<String> selectedAppIds = {}; | ||||||
|   DateTime? refreshingSince; |   DateTime? refreshingSince; | ||||||
|  |  | ||||||
|   clearSelected() { |   clearSelected() { | ||||||
|     if (selectedApps.isNotEmpty) { |     if (selectedAppIds.isNotEmpty) { | ||||||
|       setState(() { |       setState(() { | ||||||
|         selectedApps.clear(); |         selectedAppIds.clear(); | ||||||
|       }); |       }); | ||||||
|       return true; |       return true; | ||||||
|     } |     } | ||||||
| @@ -43,10 +43,10 @@ class AppsPageState extends State<AppsPage> { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   selectThese(List<App> apps) { |   selectThese(List<App> apps) { | ||||||
|     if (selectedApps.isEmpty) { |     if (selectedAppIds.isEmpty) { | ||||||
|       setState(() { |       setState(() { | ||||||
|         for (var a in apps) { |         for (var a in apps) { | ||||||
|           selectedApps.add(a); |           selectedAppIds.add(a.id); | ||||||
|         } |         } | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
| @@ -57,20 +57,20 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     var appsProvider = context.watch<AppsProvider>(); |     var appsProvider = context.watch<AppsProvider>(); | ||||||
|     var settingsProvider = context.watch<SettingsProvider>(); |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|     var sourceProvider = SourceProvider(); |     var sourceProvider = SourceProvider(); | ||||||
|     var listedApps = appsProvider.apps.values.toList(); |     var listedApps = appsProvider.getAppValues().toList(); | ||||||
|     var currentFilterIsUpdatesOnly = |     var currentFilterIsUpdatesOnly = | ||||||
|         filter.isIdenticalTo(updatesOnlyFilter, settingsProvider); |         filter.isIdenticalTo(updatesOnlyFilter, settingsProvider); | ||||||
|  |  | ||||||
|     selectedApps = selectedApps |     selectedAppIds = selectedAppIds | ||||||
|         .where((element) => listedApps.map((e) => e.app).contains(element)) |         .where((element) => listedApps.map((e) => e.app.id).contains(element)) | ||||||
|         .toSet(); |         .toSet(); | ||||||
|  |  | ||||||
|     toggleAppSelected(App app) { |     toggleAppSelected(App app) { | ||||||
|       setState(() { |       setState(() { | ||||||
|         if (selectedApps.contains(app)) { |         if (selectedAppIds.map((e) => e).contains(app.id)) { | ||||||
|           selectedApps.remove(app); |           selectedAppIds.removeWhere((a) => a == app.id); | ||||||
|         } else { |         } else { | ||||||
|           selectedApps.add(app); |           selectedAppIds.add(app.id); | ||||||
|         } |         } | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
| @@ -94,8 +94,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|             .toList(); |             .toList(); | ||||||
|  |  | ||||||
|         for (var t in nameTokens) { |         for (var t in nameTokens) { | ||||||
|           var name = app.installedInfo?.name ?? app.app.name; |           if (!app.name.toLowerCase().contains(t.toLowerCase())) { | ||||||
|           if (!name.toLowerCase().contains(t.toLowerCase())) { |  | ||||||
|             return false; |             return false; | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
| @@ -120,13 +119,13 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     }).toList(); |     }).toList(); | ||||||
|  |  | ||||||
|     listedApps.sort((a, b) { |     listedApps.sort((a, b) { | ||||||
|       var nameA = a.installedInfo?.name ?? a.app.name; |  | ||||||
|       var nameB = b.installedInfo?.name ?? b.app.name; |  | ||||||
|       int result = 0; |       int result = 0; | ||||||
|       if (settingsProvider.sortColumn == SortColumnSettings.authorName) { |       if (settingsProvider.sortColumn == SortColumnSettings.authorName) { | ||||||
|         result = (a.app.author + nameA).compareTo(b.app.author + nameB); |         result = ((a.app.author + a.name).toLowerCase()) | ||||||
|  |             .compareTo((b.app.author + b.name).toLowerCase()); | ||||||
|       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { |       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { | ||||||
|         result = (nameA + a.app.author).compareTo(nameB + b.app.author); |         result = ((a.name + a.app.author).toLowerCase()) | ||||||
|  |             .compareTo((b.name + b.app.author).toLowerCase()); | ||||||
|       } else if (settingsProvider.sortColumn == |       } else if (settingsProvider.sortColumn == | ||||||
|           SortColumnSettings.releaseDate) { |           SortColumnSettings.releaseDate) { | ||||||
|         result = (a.app.releaseDate)?.compareTo( |         result = (a.app.releaseDate)?.compareTo( | ||||||
| @@ -143,15 +142,15 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     var existingUpdates = appsProvider.findExistingUpdates(installedOnly: true); |     var existingUpdates = appsProvider.findExistingUpdates(installedOnly: true); | ||||||
|  |  | ||||||
|     var existingUpdateIdsAllOrSelected = existingUpdates |     var existingUpdateIdsAllOrSelected = existingUpdates | ||||||
|         .where((element) => selectedApps.isEmpty |         .where((element) => selectedAppIds.isEmpty | ||||||
|             ? listedApps.where((a) => a.app.id == element).isNotEmpty |             ? listedApps.where((a) => a.app.id == element).isNotEmpty | ||||||
|             : selectedApps.map((e) => e.id).contains(element)) |             : selectedAppIds.map((e) => e).contains(element)) | ||||||
|         .toList(); |         .toList(); | ||||||
|     var newInstallIdsAllOrSelected = appsProvider |     var newInstallIdsAllOrSelected = appsProvider | ||||||
|         .findExistingUpdates(nonInstalledOnly: true) |         .findExistingUpdates(nonInstalledOnly: true) | ||||||
|         .where((element) => selectedApps.isEmpty |         .where((element) => selectedAppIds.isEmpty | ||||||
|             ? listedApps.where((a) => a.app.id == element).isNotEmpty |             ? listedApps.where((a) => a.app.id == element).isNotEmpty | ||||||
|             : selectedApps.map((e) => e.id).contains(element)) |             : selectedAppIds.map((e) => e).contains(element)) | ||||||
|         .toList(); |         .toList(); | ||||||
|  |  | ||||||
|     List<String> trackOnlyUpdateIdsAllOrSelected = []; |     List<String> trackOnlyUpdateIdsAllOrSelected = []; | ||||||
| @@ -206,12 +205,17 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     var listedCategories = getListedCategories(); |     var listedCategories = getListedCategories(); | ||||||
|     listedCategories.sort((a, b) { |     listedCategories.sort((a, b) { | ||||||
|       return a != null && b != null |       return a != null && b != null | ||||||
|           ? a.compareTo(b) |           ? a.toLowerCase().compareTo(b.toLowerCase()) | ||||||
|           : a == null |           : a == null | ||||||
|               ? 1 |               ? 1 | ||||||
|               : -1; |               : -1; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     Set<App> selectedApps = listedApps | ||||||
|  |         .map((e) => e.app) | ||||||
|  |         .where((a) => selectedAppIds.contains(a.id)) | ||||||
|  |         .toSet(); | ||||||
|  |  | ||||||
|     showChangeLogDialog( |     showChangeLogDialog( | ||||||
|         String? changesUrl, AppSource appSource, String changeLog, int index) { |         String? changesUrl, AppSource appSource, String changeLog, int index) { | ||||||
|       showDialog( |       showDialog( | ||||||
| @@ -220,6 +224,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|             return GeneratedFormModal( |             return GeneratedFormModal( | ||||||
|               title: tr('changes'), |               title: tr('changes'), | ||||||
|               items: const [], |               items: const [], | ||||||
|  |               message: listedApps[index].app.latestVersion, | ||||||
|               additionalWidgets: [ |               additionalWidgets: [ | ||||||
|                 changesUrl != null |                 changesUrl != null | ||||||
|                     ? GestureDetector( |                     ? GestureDetector( | ||||||
| @@ -288,7 +293,8 @@ class AppsPageState extends State<AppsPage> { | |||||||
|         if (refreshingSince != null) |         if (refreshingSince != null) | ||||||
|           SliverToBoxAdapter( |           SliverToBoxAdapter( | ||||||
|             child: LinearProgressIndicator( |             child: LinearProgressIndicator( | ||||||
|               value: appsProvider.apps.values |               value: appsProvider | ||||||
|  |                       .getAppValues() | ||||||
|                       .where((element) => !(element.app.lastUpdateCheck |                       .where((element) => !(element.app.lastUpdateCheck | ||||||
|                               ?.isBefore(refreshingSince!) ?? |                               ?.isBefore(refreshingSince!) ?? | ||||||
|                           true)) |                           true)) | ||||||
| @@ -400,7 +406,8 @@ class AppsPageState extends State<AppsPage> { | |||||||
|             children: [ |             children: [ | ||||||
|               Row(mainAxisSize: MainAxisSize.min, children: [ |               Row(mainAxisSize: MainAxisSize.min, children: [ | ||||||
|                 Container( |                 Container( | ||||||
|                     constraints: const BoxConstraints(maxWidth: 150), |                     constraints: BoxConstraints( | ||||||
|  |                         maxWidth: MediaQuery.of(context).size.width / 4), | ||||||
|                     child: Text( |                     child: Text( | ||||||
|                       getVersionText(index), |                       getVersionText(index), | ||||||
|                       overflow: TextOverflow.ellipsis, |                       overflow: TextOverflow.ellipsis, | ||||||
| @@ -467,15 +474,15 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                 .colorScheme |                 .colorScheme | ||||||
|                 .primary |                 .primary | ||||||
|                 .withOpacity(listedApps[index].app.pinned ? 0.2 : 0.1), |                 .withOpacity(listedApps[index].app.pinned ? 0.2 : 0.1), | ||||||
|             selected: selectedApps.contains(listedApps[index].app), |             selected: | ||||||
|  |                 selectedAppIds.map((e) => e).contains(listedApps[index].app.id), | ||||||
|             onLongPress: () { |             onLongPress: () { | ||||||
|               toggleAppSelected(listedApps[index].app); |               toggleAppSelected(listedApps[index].app); | ||||||
|             }, |             }, | ||||||
|             leading: getAppIcon(index), |             leading: getAppIcon(index), | ||||||
|             title: Text( |             title: Text( | ||||||
|               maxLines: 1, |               maxLines: 1, | ||||||
|               listedApps[index].installedInfo?.name ?? |               listedApps[index].name, | ||||||
|                   listedApps[index].app.name, |  | ||||||
|               style: TextStyle( |               style: TextStyle( | ||||||
|                 overflow: TextOverflow.ellipsis, |                 overflow: TextOverflow.ellipsis, | ||||||
|                 fontWeight: listedApps[index].app.pinned |                 fontWeight: listedApps[index].app.pinned | ||||||
| @@ -497,7 +504,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                   ])) |                   ])) | ||||||
|                 : trailingRow, |                 : trailingRow, | ||||||
|             onTap: () { |             onTap: () { | ||||||
|               if (selectedApps.isNotEmpty) { |               if (selectedAppIds.isNotEmpty) { | ||||||
|                 toggleAppSelected(listedApps[index].app); |                 toggleAppSelected(listedApps[index].app); | ||||||
|               } else { |               } else { | ||||||
|                 Navigator.push( |                 Navigator.push( | ||||||
| @@ -534,7 +541,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     getSelectAllButton() { |     getSelectAllButton() { | ||||||
|       return selectedApps.isEmpty |       return selectedAppIds.isEmpty | ||||||
|           ? TextButton.icon( |           ? TextButton.icon( | ||||||
|               style: const ButtonStyle(visualDensity: VisualDensity.compact), |               style: const ButtonStyle(visualDensity: VisualDensity.compact), | ||||||
|               onPressed: () { |               onPressed: () { | ||||||
| @@ -548,17 +555,17 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           : TextButton.icon( |           : TextButton.icon( | ||||||
|               style: const ButtonStyle(visualDensity: VisualDensity.compact), |               style: const ButtonStyle(visualDensity: VisualDensity.compact), | ||||||
|               onPressed: () { |               onPressed: () { | ||||||
|                 selectedApps.isEmpty |                 selectedAppIds.isEmpty | ||||||
|                     ? selectThese(listedApps.map((e) => e.app).toList()) |                     ? selectThese(listedApps.map((e) => e.app).toList()) | ||||||
|                     : clearSelected(); |                     : clearSelected(); | ||||||
|               }, |               }, | ||||||
|               icon: Icon( |               icon: Icon( | ||||||
|                 selectedApps.isEmpty |                 selectedAppIds.isEmpty | ||||||
|                     ? Icons.select_all_outlined |                     ? Icons.select_all_outlined | ||||||
|                     : Icons.deselect_outlined, |                     : Icons.deselect_outlined, | ||||||
|                 color: Theme.of(context).colorScheme.primary, |                 color: Theme.of(context).colorScheme.primary, | ||||||
|               ), |               ), | ||||||
|               label: Text(selectedApps.length.toString())); |               label: Text(selectedAppIds.length.toString())); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     getMassObtainFunction() { |     getMassObtainFunction() { | ||||||
| @@ -706,7 +713,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           builder: (BuildContext ctx) { |           builder: (BuildContext ctx) { | ||||||
|             return AlertDialog( |             return AlertDialog( | ||||||
|               title: Text(tr('markXSelectedAppsAsUpdated', |               title: Text(tr('markXSelectedAppsAsUpdated', | ||||||
|                   args: [selectedApps.length.toString()])), |                   args: [selectedAppIds.length.toString()])), | ||||||
|               content: Text( |               content: Text( | ||||||
|                 tr('onlyWorksWithNonVersionDetectApps'), |                 tr('onlyWorksWithNonVersionDetectApps'), | ||||||
|                 style: const TextStyle( |                 style: const TextStyle( | ||||||
| @@ -749,30 +756,28 @@ class AppsPageState extends State<AppsPage> { | |||||||
|       Navigator.of(context).pop(); |       Navigator.of(context).pop(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     resetSelectedAppsInstallStatuses() { |     resetSelectedAppsInstallStatuses() async { | ||||||
|       () async { |       try { | ||||||
|         try { |         var values = await showDialog( | ||||||
|           var values = await showDialog( |             context: context, | ||||||
|               context: context, |             builder: (BuildContext ctx) { | ||||||
|               builder: (BuildContext ctx) { |               return GeneratedFormModal( | ||||||
|                 return GeneratedFormModal( |                 title: tr('resetInstallStatusForSelectedAppsQuestion'), | ||||||
|                   title: tr('resetInstallStatusForSelectedAppsQuestion'), |                 items: const [], | ||||||
|                   items: const [], |                 initValid: true, | ||||||
|                   initValid: true, |                 message: tr('installStatusOfXWillBeResetExplanation', | ||||||
|                   message: tr('installStatusOfXWillBeResetExplanation', |                     args: [plural('app', selectedAppIds.length)]), | ||||||
|                       args: [plural('app', selectedApps.length)]), |               ); | ||||||
|                 ); |             }); | ||||||
|               }); |         if (values != null) { | ||||||
|           if (values != null) { |           appsProvider.saveApps(selectedApps.map((e) { | ||||||
|             appsProvider.saveApps(selectedApps.map((e) { |             e.installedVersion = null; | ||||||
|               e.installedVersion = null; |             return e; | ||||||
|               return e; |           }).toList()); | ||||||
|             }).toList()); |  | ||||||
|           } |  | ||||||
|         } finally { |  | ||||||
|           Navigator.of(context).pop(); |  | ||||||
|         } |         } | ||||||
|       }; |       } finally { | ||||||
|  |         Navigator.of(context).pop(); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     showMoreOptionsDialog() { |     showMoreOptionsDialog() { | ||||||
| @@ -820,7 +825,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                         icon: const Icon(Icons.share), |                         icon: const Icon(Icons.share), | ||||||
|                       ), |                       ), | ||||||
|                       IconButton( |                       IconButton( | ||||||
|                         onPressed: resetSelectedAppsInstallStatuses(), |                         onPressed: resetSelectedAppsInstallStatuses, | ||||||
|                         tooltip: tr('resetInstallStatus'), |                         tooltip: tr('resetInstallStatus'), | ||||||
|                         icon: const Icon(Icons.restore_page_outlined), |                         icon: const Icon(Icons.restore_page_outlined), | ||||||
|                       ), |                       ), | ||||||
| @@ -836,7 +841,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|         children: [ |         children: [ | ||||||
|           IconButton( |           IconButton( | ||||||
|             visualDensity: VisualDensity.compact, |             visualDensity: VisualDensity.compact, | ||||||
|             onPressed: selectedApps.isEmpty |             onPressed: selectedAppIds.isEmpty | ||||||
|                 ? null |                 ? null | ||||||
|                 : () { |                 : () { | ||||||
|                     appsProvider.removeAppsWithModal( |                     appsProvider.removeAppsWithModal( | ||||||
| @@ -848,7 +853,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           IconButton( |           IconButton( | ||||||
|               visualDensity: VisualDensity.compact, |               visualDensity: VisualDensity.compact, | ||||||
|               onPressed: getMassObtainFunction(), |               onPressed: getMassObtainFunction(), | ||||||
|               tooltip: selectedApps.isEmpty |               tooltip: selectedAppIds.isEmpty | ||||||
|                   ? tr('installUpdateApps') |                   ? tr('installUpdateApps') | ||||||
|                   : tr('installUpdateSelectedApps'), |                   : tr('installUpdateSelectedApps'), | ||||||
|               icon: const Icon( |               icon: const Icon( | ||||||
| @@ -856,13 +861,13 @@ class AppsPageState extends State<AppsPage> { | |||||||
|               )), |               )), | ||||||
|           IconButton( |           IconButton( | ||||||
|             visualDensity: VisualDensity.compact, |             visualDensity: VisualDensity.compact, | ||||||
|             onPressed: selectedApps.isEmpty ? null : launchCategorizeDialog(), |             onPressed: selectedAppIds.isEmpty ? null : launchCategorizeDialog(), | ||||||
|             tooltip: tr('categorize'), |             tooltip: tr('categorize'), | ||||||
|             icon: const Icon(Icons.category_outlined), |             icon: const Icon(Icons.category_outlined), | ||||||
|           ), |           ), | ||||||
|           IconButton( |           IconButton( | ||||||
|             visualDensity: VisualDensity.compact, |             visualDensity: VisualDensity.compact, | ||||||
|             onPressed: selectedApps.isEmpty ? null : showMoreOptionsDialog, |             onPressed: selectedAppIds.isEmpty ? null : showMoreOptionsDialog, | ||||||
|             tooltip: tr('more'), |             tooltip: tr('more'), | ||||||
|             icon: const Icon(Icons.more_horiz), |             icon: const Icon(Icons.more_horiz), | ||||||
|           ), |           ), | ||||||
|   | |||||||
| @@ -506,7 +506,7 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|           widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')), |           widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')), | ||||||
|       content: Column(children: [ |       content: Column(children: [ | ||||||
|         ...urlWithDescriptionSelections.keys.map((urlWithD) { |         ...urlWithDescriptionSelections.keys.map((urlWithD) { | ||||||
|           select(bool? value) { |           selectThis(bool? value) { | ||||||
|             setState(() { |             setState(() { | ||||||
|               value ??= false; |               value ??= false; | ||||||
|               if (value! && widget.onlyOneSelectionAllowed) { |               if (value! && widget.onlyOneSelectionAllowed) { | ||||||
| @@ -517,11 +517,56 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|             }); |             }); | ||||||
|           } |           } | ||||||
|  |  | ||||||
|           return Row(children: [ |           var urlLink = GestureDetector( | ||||||
|  |               onTap: () { | ||||||
|  |                 launchUrlString(urlWithD.key, | ||||||
|  |                     mode: LaunchMode.externalApplication); | ||||||
|  |               }, | ||||||
|  |               child: Text( | ||||||
|  |                 Uri.parse(urlWithD.key).path.substring(1), | ||||||
|  |                 style: const TextStyle(decoration: TextDecoration.underline), | ||||||
|  |                 textAlign: TextAlign.start, | ||||||
|  |               )); | ||||||
|  |  | ||||||
|  |           var descriptionText = Text( | ||||||
|  |             urlWithD.value.length > 128 | ||||||
|  |                 ? '${urlWithD.value.substring(0, 128)}...' | ||||||
|  |                 : urlWithD.value, | ||||||
|  |             style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12), | ||||||
|  |           ); | ||||||
|  |  | ||||||
|  |           var selectedUrlsWithDs = urlWithDescriptionSelections.entries | ||||||
|  |               .where((e) => e.value) | ||||||
|  |               .toList(); | ||||||
|  |  | ||||||
|  |           var singleSelectTile = ListTile( | ||||||
|  |             title: urlLink, | ||||||
|  |             subtitle: GestureDetector( | ||||||
|  |               onTap: () { | ||||||
|  |                 setState(() { | ||||||
|  |                   selectOnlyOne(urlWithD.key); | ||||||
|  |                 }); | ||||||
|  |               }, | ||||||
|  |               child: descriptionText, | ||||||
|  |             ), | ||||||
|  |             leading: Radio<String>( | ||||||
|  |               value: urlWithD.key, | ||||||
|  |               groupValue: selectedUrlsWithDs.isEmpty | ||||||
|  |                   ? null | ||||||
|  |                   : selectedUrlsWithDs.first.key.key, | ||||||
|  |               onChanged: (value) { | ||||||
|  |                 setState(() { | ||||||
|  |                   selectOnlyOne(urlWithD.key); | ||||||
|  |                 }); | ||||||
|  |               }, | ||||||
|  |             ), | ||||||
|  |           ); | ||||||
|  |  | ||||||
|  |           var multiSelectTile = Row(children: [ | ||||||
|             Checkbox( |             Checkbox( | ||||||
|                 value: urlWithDescriptionSelections[urlWithD], |                 value: urlWithDescriptionSelections[urlWithD], | ||||||
|                 onChanged: (value) { |                 onChanged: (value) { | ||||||
|                   select(value); |                   selectThis(value); | ||||||
|                 }), |                 }), | ||||||
|             const SizedBox( |             const SizedBox( | ||||||
|               width: 8, |               width: 8, | ||||||
| @@ -534,28 +579,13 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
|                   height: 8, |                   height: 8, | ||||||
|                 ), |                 ), | ||||||
|                 GestureDetector( |                 urlLink, | ||||||
|                     onTap: () { |  | ||||||
|                       launchUrlString(urlWithD.key, |  | ||||||
|                           mode: LaunchMode.externalApplication); |  | ||||||
|                     }, |  | ||||||
|                     child: Text( |  | ||||||
|                       Uri.parse(urlWithD.key).path.substring(1), |  | ||||||
|                       style: |  | ||||||
|                           const TextStyle(decoration: TextDecoration.underline), |  | ||||||
|                       textAlign: TextAlign.start, |  | ||||||
|                     )), |  | ||||||
|                 GestureDetector( |                 GestureDetector( | ||||||
|                   onTap: () { |                   onTap: () { | ||||||
|                     select(!(urlWithDescriptionSelections[urlWithD] ?? false)); |                     selectThis( | ||||||
|  |                         !(urlWithDescriptionSelections[urlWithD] ?? false)); | ||||||
|                   }, |                   }, | ||||||
|                   child: Text( |                   child: descriptionText, | ||||||
|                     urlWithD.value.length > 128 |  | ||||||
|                         ? '${urlWithD.value.substring(0, 128)}...' |  | ||||||
|                         : urlWithD.value, |  | ||||||
|                     style: const TextStyle( |  | ||||||
|                         fontStyle: FontStyle.italic, fontSize: 12), |  | ||||||
|                   ), |  | ||||||
|                 ), |                 ), | ||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
|                   height: 8, |                   height: 8, | ||||||
| @@ -563,6 +593,10 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|               ], |               ], | ||||||
|             )) |             )) | ||||||
|           ]); |           ]); | ||||||
|  |  | ||||||
|  |           return widget.onlyOneSelectionAllowed | ||||||
|  |               ? singleSelectTile | ||||||
|  |               : multiSelectTile; | ||||||
|         }) |         }) | ||||||
|       ]), |       ]), | ||||||
|       actions: [ |       actions: [ | ||||||
|   | |||||||
| @@ -224,6 +224,17 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                             ), |                             ), | ||||||
|                             themeDropdown, |                             themeDropdown, | ||||||
|                             height16, |                             height16, | ||||||
|  |                             Row( | ||||||
|  |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|  |                               children: [ | ||||||
|  |                                 Text(tr('useBlackTheme')), | ||||||
|  |                                 Switch( | ||||||
|  |                                     value: settingsProvider.useBlackTheme, | ||||||
|  |                                     onChanged: (value) { | ||||||
|  |                                       settingsProvider.useBlackTheme = value; | ||||||
|  |                                     }) | ||||||
|  |                               ], | ||||||
|  |                             ), | ||||||
|                             colourDropdown, |                             colourDropdown, | ||||||
|                             height16, |                             height16, | ||||||
|                             Row( |                             Row( | ||||||
|   | |||||||
| @@ -34,6 +34,10 @@ class AppInMemory { | |||||||
|   AppInfo? installedInfo; |   AppInfo? installedInfo; | ||||||
|  |  | ||||||
|   AppInMemory(this.app, this.downloadProgress, this.installedInfo); |   AppInMemory(this.app, this.downloadProgress, this.installedInfo); | ||||||
|  |   AppInMemory deepCopy() => | ||||||
|  |       AppInMemory(app.deepCopy(), downloadProgress, installedInfo); | ||||||
|  |  | ||||||
|  |   String get name => app.overrideName ?? installedInfo?.name ?? app.finalName; | ||||||
| } | } | ||||||
|  |  | ||||||
| class DownloadedApk { | class DownloadedApk { | ||||||
| @@ -97,6 +101,8 @@ class AppsProvider with ChangeNotifier { | |||||||
|   late Stream<FGBGType>? foregroundStream; |   late Stream<FGBGType>? foregroundStream; | ||||||
|   late StreamSubscription<FGBGType>? foregroundSubscription; |   late StreamSubscription<FGBGType>? foregroundSubscription; | ||||||
|  |  | ||||||
|  |   Iterable<AppInMemory> getAppValues() => apps.values.map((a) => a.deepCopy()); | ||||||
|  |  | ||||||
|   AppsProvider() { |   AppsProvider() { | ||||||
|     // Subscribe to changes in the app foreground status |     // Subscribe to changes in the app foreground status | ||||||
|     foregroundStream = FGBGEvents.stream.asBroadcastStream(); |     foregroundStream = FGBGEvents.stream.asBroadcastStream(); | ||||||
| @@ -159,18 +165,17 @@ class AppsProvider with ChangeNotifier { | |||||||
|   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { |   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { | ||||||
|     NotificationsProvider? notificationsProvider = |     NotificationsProvider? notificationsProvider = | ||||||
|         context?.read<NotificationsProvider>(); |         context?.read<NotificationsProvider>(); | ||||||
|     var notifId = DownloadNotification(app.name, 0).id; |     var notifId = DownloadNotification(app.finalName, 0).id; | ||||||
|     if (apps[app.id] != null) { |     if (apps[app.id] != null) { | ||||||
|       apps[app.id]!.downloadProgress = 0; |       apps[app.id]!.downloadProgress = 0; | ||||||
|       notifyListeners(); |       notifyListeners(); | ||||||
|     } |     } | ||||||
|     try { |     try { | ||||||
|       var fileName = |  | ||||||
|           '${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'; |  | ||||||
|       String downloadUrl = await SourceProvider() |       String downloadUrl = await SourceProvider() | ||||||
|           .getSource(app.url) |           .getSource(app.url) | ||||||
|           .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value); |           .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value); | ||||||
|       var notif = DownloadNotification(app.name, 100); |       var fileName = '${app.id}-${downloadUrl.hashCode}.apk'; | ||||||
|  |       var notif = DownloadNotification(app.finalName, 100); | ||||||
|       notificationsProvider?.cancel(notif.id); |       notificationsProvider?.cancel(notif.id); | ||||||
|       int? prevProg; |       int? prevProg; | ||||||
|       File downloadedFile = |       File downloadedFile = | ||||||
| @@ -180,7 +185,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|           apps[app.id]!.downloadProgress = progress; |           apps[app.id]!.downloadProgress = progress; | ||||||
|           notifyListeners(); |           notifyListeners(); | ||||||
|         } |         } | ||||||
|         notif = DownloadNotification(app.name, prog ?? 100); |         notif = DownloadNotification(app.finalName, prog ?? 100); | ||||||
|         if (prog != null && prevProg != prog) { |         if (prog != null && prevProg != prog) { | ||||||
|           notificationsProvider?.notify(notif); |           notificationsProvider?.notify(notif); | ||||||
|         } |         } | ||||||
| @@ -199,16 +204,17 @@ class AppsProvider with ChangeNotifier { | |||||||
|       // The former case should be handled (give the App its real ID), the latter is a security issue |       // The former case should be handled (give the App its real ID), the latter is a security issue | ||||||
|       var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path); |       var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path); | ||||||
|       if (app.id != newInfo.packageName) { |       if (app.id != newInfo.packageName) { | ||||||
|         if (apps[app.id] != null && !SourceProvider().isTempId(app)) { |         var isTempId = SourceProvider().isTempId(app); | ||||||
|  |         if (apps[app.id] != null && !isTempId) { | ||||||
|           throw IDChangedError(); |           throw IDChangedError(); | ||||||
|         } |         } | ||||||
|         var originalAppId = app.id; |         var originalAppId = app.id; | ||||||
|         app.id = newInfo.packageName; |         app.id = newInfo.packageName; | ||||||
|         downloadedFile = downloadedFile.renameSync( |         downloadedFile = downloadedFile.renameSync( | ||||||
|             '${downloadedFile.parent.path}/${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'); |             '${downloadedFile.parent.path}/${app.id}-${downloadUrl.hashCode}.apk'); | ||||||
|         if (apps[originalAppId] != null) { |         if (apps[originalAppId] != null) { | ||||||
|           await removeApps([originalAppId]); |           await removeApps([originalAppId]); | ||||||
|           await saveApps([app]); |           await saveApps([app], onlyIfExists: !isTempId); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       return DownloadedApk(app.id, downloadedFile); |       return DownloadedApk(app.id, downloadedFile); | ||||||
| @@ -299,7 +305,8 @@ class AppsProvider with ChangeNotifier { | |||||||
|   Future<MapEntry<String, String>?> confirmApkUrl( |   Future<MapEntry<String, String>?> confirmApkUrl( | ||||||
|       App app, BuildContext? context) async { |       App app, BuildContext? context) async { | ||||||
|     // If the App has more than one APK, the user should pick one (if context provided) |     // If the App has more than one APK, the user should pick one (if context provided) | ||||||
|     MapEntry<String, String>? apkUrl = app.apkUrls[app.preferredApkIndex]; |     MapEntry<String, String>? apkUrl = | ||||||
|  |         app.apkUrls[app.preferredApkIndex >= 0 ? app.preferredApkIndex : 0]; | ||||||
|     // get device supported architecture |     // get device supported architecture | ||||||
|     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; |     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||||
|  |  | ||||||
| @@ -360,8 +367,13 @@ class AppsProvider with ChangeNotifier { | |||||||
|         apkUrl = await confirmApkUrl(apps[id]!.app, context); |         apkUrl = await confirmApkUrl(apps[id]!.app, context); | ||||||
|       } |       } | ||||||
|       if (apkUrl != null) { |       if (apkUrl != null) { | ||||||
|         int urlInd = apps[id]!.app.apkUrls.indexOf(apkUrl); |         int urlInd = apps[id]! | ||||||
|         if (urlInd != apps[id]!.app.preferredApkIndex) { |             .app | ||||||
|  |             .apkUrls | ||||||
|  |             .map((e) => e.value) | ||||||
|  |             .toList() | ||||||
|  |             .indexOf(apkUrl.value); | ||||||
|  |         if (urlInd >= 0 && urlInd != apps[id]!.app.preferredApkIndex) { | ||||||
|           apps[id]!.app.preferredApkIndex = urlInd; |           apps[id]!.app.preferredApkIndex = urlInd; | ||||||
|           await saveApps([apps[id]!.app]); |           await saveApps([apps[id]!.app]); | ||||||
|         } |         } | ||||||
| @@ -638,7 +650,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|         sp.getSource(newApps[i].url); |         sp.getSource(newApps[i].url); | ||||||
|         apps[newApps[i].id] = AppInMemory(newApps[i], null, info); |         apps[newApps[i].id] = AppInMemory(newApps[i], null, info); | ||||||
|       } catch (e) { |       } catch (e) { | ||||||
|         errors.add([newApps[i].id, newApps[i].name, e.toString()]); |         errors.add([newApps[i].id, newApps[i].finalName, e.toString()]); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (errors.isNotEmpty) { |     if (errors.isNotEmpty) { | ||||||
| @@ -668,12 +680,10 @@ class AppsProvider with ChangeNotifier { | |||||||
|       bool onlyIfExists = true}) async { |       bool onlyIfExists = true}) async { | ||||||
|     attemptToCorrectInstallStatus = |     attemptToCorrectInstallStatus = | ||||||
|         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); |         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); | ||||||
|     for (var app in apps) { |     for (var a in apps) { | ||||||
|  |       var app = a.deepCopy(); | ||||||
|       AppInfo? info = await getInstalledInfo(app.id); |       AppInfo? info = await getInstalledInfo(app.id); | ||||||
|       app.name = info?.name ?? app.name; |       app.name = info?.name ?? app.name; | ||||||
|       if (app.additionalSettings['appName']?.toString().isNotEmpty == true) { |  | ||||||
|         app.name = app.additionalSettings['appName'].toString().trim(); |  | ||||||
|       } |  | ||||||
|       if (attemptToCorrectInstallStatus) { |       if (attemptToCorrectInstallStatus) { | ||||||
|         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; |         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; | ||||||
|       } |       } | ||||||
| @@ -904,7 +914,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|   Future<List<List<String>>> addAppsByURL(List<String> urls) async { |   Future<List<List<String>>> addAppsByURL(List<String> urls) async { | ||||||
|     List<dynamic> results = await SourceProvider().getAppsByURLNaive(urls, |     List<dynamic> results = await SourceProvider().getAppsByURLNaive(urls, | ||||||
|         ignoreUrls: apps.values.map((e) => e.app.url).toList()); |         alreadyAddedUrls: apps.values.map((e) => e.app.url).toList()); | ||||||
|     List<App> pps = results[0]; |     List<App> pps = results[0]; | ||||||
|     Map<String, dynamic> errorsMap = results[1]; |     Map<String, dynamic> errorsMap = results[1]; | ||||||
|     for (var app in pps) { |     for (var app in pps) { | ||||||
| @@ -941,7 +951,7 @@ class _APKPickerState extends State<APKPicker> { | |||||||
|       scrollable: true, |       scrollable: true, | ||||||
|       title: Text(tr('pickAnAPK')), |       title: Text(tr('pickAnAPK')), | ||||||
|       content: Column(children: [ |       content: Column(children: [ | ||||||
|         Text(tr('appHasMoreThanOnePackage', args: [widget.app.name])), |         Text(tr('appHasMoreThanOnePackage', args: [widget.app.finalName])), | ||||||
|         const SizedBox(height: 16), |         const SizedBox(height: 16), | ||||||
|         ...widget.app.apkUrls.map( |         ...widget.app.apkUrls.map( | ||||||
|           (u) => RadioListTile<String>( |           (u) => RadioListTile<String>( | ||||||
|   | |||||||
| @@ -34,9 +34,9 @@ class UpdateNotification extends ObtainiumNotification { | |||||||
|     message = updates.isEmpty |     message = updates.isEmpty | ||||||
|         ? tr('noNewUpdates') |         ? tr('noNewUpdates') | ||||||
|         : updates.length == 1 |         : updates.length == 1 | ||||||
|             ? tr('xHasAnUpdate', args: [updates[0].name]) |             ? tr('xHasAnUpdate', args: [updates[0].finalName]) | ||||||
|             : plural('xAndNMoreUpdatesAvailable', updates.length - 1, |             : plural('xAndNMoreUpdatesAvailable', updates.length - 1, | ||||||
|                 args: [updates[0].name, (updates.length - 1).toString()]); |                 args: [updates[0].finalName, (updates.length - 1).toString()]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -46,9 +46,9 @@ class SilentUpdateNotification extends ObtainiumNotification { | |||||||
|             tr('appsUpdatedNotifDescription'), Importance.defaultImportance) { |             tr('appsUpdatedNotifDescription'), Importance.defaultImportance) { | ||||||
|     message = updates.length == 1 |     message = updates.length == 1 | ||||||
|         ? tr('xWasUpdatedToY', |         ? tr('xWasUpdatedToY', | ||||||
|             args: [updates[0].name, updates[0].latestVersion]) |             args: [updates[0].finalName, updates[0].latestVersion]) | ||||||
|         : plural('xAndNMoreUpdatesInstalled', updates.length - 1, |         : plural('xAndNMoreUpdatesInstalled', updates.length - 1, | ||||||
|             args: [updates[0].name, (updates.length - 1).toString()]); |             args: [updates[0].finalName, (updates.length - 1).toString()]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -64,6 +64,15 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool get useBlackTheme { | ||||||
|  |     return prefs?.getBool('useBlackTheme') ?? false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set useBlackTheme(bool useBlackTheme) { | ||||||
|  |     prefs?.setBool('useBlackTheme', useBlackTheme); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   int get updateInterval { |   int get updateInterval { | ||||||
|     var min = prefs?.getInt('updateInterval') ?? 360; |     var min = prefs?.getInt('updateInterval') ?? 360; | ||||||
|     if (!updateIntervals.contains(min)) { |     if (!updateIntervals.contains(min)) { | ||||||
| @@ -164,7 +173,8 @@ class SettingsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|   void setCategories(Map<String, int> cats, {AppsProvider? appsProvider}) { |   void setCategories(Map<String, int> cats, {AppsProvider? appsProvider}) { | ||||||
|     if (appsProvider != null) { |     if (appsProvider != null) { | ||||||
|       List<App> changedApps = appsProvider.apps.values |       List<App> changedApps = appsProvider | ||||||
|  |           .getAppValues() | ||||||
|           .map((a) { |           .map((a) { | ||||||
|             var n1 = a.app.categories.length; |             var n1 = a.app.categories.length; | ||||||
|             a.app.categories.removeWhere((c) => !cats.keys.contains(c)); |             a.app.categories.removeWhere((c) => !cats.keys.contains(c)); | ||||||
|   | |||||||
| @@ -80,6 +80,31 @@ class App { | |||||||
|     return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALSETTINGS: ${additionalSettings.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned'; |     return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALSETTINGS: ${additionalSettings.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   String? get overrideName => | ||||||
|  |       additionalSettings['appName']?.toString().trim().isNotEmpty == true | ||||||
|  |           ? additionalSettings['appName'] | ||||||
|  |           : null; | ||||||
|  |  | ||||||
|  |   String get finalName { | ||||||
|  |     return overrideName ?? name; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   App deepCopy() => App( | ||||||
|  |       id, | ||||||
|  |       url, | ||||||
|  |       author, | ||||||
|  |       name, | ||||||
|  |       installedVersion, | ||||||
|  |       latestVersion, | ||||||
|  |       apkUrls, | ||||||
|  |       preferredApkIndex, | ||||||
|  |       Map.from(additionalSettings), | ||||||
|  |       lastUpdateCheck, | ||||||
|  |       pinned, | ||||||
|  |       categories: categories, | ||||||
|  |       changeLog: changeLog, | ||||||
|  |       releaseDate: releaseDate); | ||||||
|  |  | ||||||
|   factory App.fromJson(Map<String, dynamic> json) { |   factory App.fromJson(Map<String, dynamic> json) { | ||||||
|     var source = SourceProvider().getSource(json['url']); |     var source = SourceProvider().getSource(json['url']); | ||||||
|     var formItems = source.combinedAppSpecificSettingFormItems |     var formItems = source.combinedAppSpecificSettingFormItems | ||||||
| @@ -241,10 +266,12 @@ Map<String, dynamic> getDefaultValuesFromFormItems( | |||||||
|       .reduce((value, element) => [...value, ...element])); |       .reduce((value, element) => [...value, ...element])); | ||||||
| } | } | ||||||
|  |  | ||||||
| getApkUrlsFromUrls(List<String> urls) => urls | List<MapEntry<String, String>> getApkUrlsFromUrls(List<String> urls) => | ||||||
|     .map((e) => |     urls.map((e) { | ||||||
|         MapEntry(e.split('/').where((el) => el.trim().isNotEmpty).last, e)) |       var segments = e.split('/').where((el) => el.trim().isNotEmpty); | ||||||
|     .toList(); |       var apkSegs = segments.where((s) => s.toLowerCase().endsWith('.apk')); | ||||||
|  |       return MapEntry(apkSegs.isNotEmpty ? apkSegs.last : segments.last, e); | ||||||
|  |     }).toList(); | ||||||
|  |  | ||||||
| class AppSource { | class AppSource { | ||||||
|   String? host; |   String? host; | ||||||
| @@ -492,11 +519,14 @@ class SourceProvider { | |||||||
|  |  | ||||||
|   // Returns errors in [results, errors] instead of throwing them |   // Returns errors in [results, errors] instead of throwing them | ||||||
|   Future<List<dynamic>> getAppsByURLNaive(List<String> urls, |   Future<List<dynamic>> getAppsByURLNaive(List<String> urls, | ||||||
|       {List<String> ignoreUrls = const []}) async { |       {List<String> alreadyAddedUrls = const []}) async { | ||||||
|     List<App> apps = []; |     List<App> apps = []; | ||||||
|     Map<String, dynamic> errors = {}; |     Map<String, dynamic> errors = {}; | ||||||
|     for (var url in urls.where((element) => !ignoreUrls.contains(element))) { |     for (var url in urls) { | ||||||
|       try { |       try { | ||||||
|  |         if (alreadyAddedUrls.contains(url)) { | ||||||
|  |           throw ObtainiumError(tr('appAlreadyAdded')); | ||||||
|  |         } | ||||||
|         var source = getSource(url); |         var source = getSource(url); | ||||||
|         apps.add(await getApp( |         apps.add(await getApp( | ||||||
|             source, |             source, | ||||||
|   | |||||||
							
								
								
									
										92
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										92
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -5,18 +5,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: android_alarm_manager_plus |       name: android_alarm_manager_plus | ||||||
|       sha256: "8647cc5f9339f3955a2bd9ec40e0f10c3a80049f31f80b3ffdd87e07bb73fce2" |       sha256: f6d0347734fa2ea716349a5a3e16ffdc1800ca64e5640112896d128c6815c178 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.1" |     version: "2.1.2" | ||||||
|   android_intent_plus: |   android_intent_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: android_intent_plus |       name: android_intent_plus | ||||||
|       sha256: "54810cb33945c2c10742cd746ea994822c115e9dbe189919bc63cb436e45a6af" |       sha256: "6bcdcd20461ac7a0c785f6298cdda96ad275d5bcbc1ecf28829cbe03ec6690be" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.1.6" |     version: "3.1.7" | ||||||
|   animations: |   animations: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -117,10 +117,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: device_info_plus |       name: device_info_plus | ||||||
|       sha256: "1d6e5a61674ba3a68fb048a7c7b4ff4bebfed8d7379dbe8f2b718231be9a7c95" |       sha256: "435383ca05f212760b0a70426b5a90354fe6bd65992b3a5e27ab6ede74c02f5c" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "8.1.0" |     version: "8.2.0" | ||||||
|   device_info_plus_platform_interface: |   device_info_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -133,10 +133,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: dynamic_color |       name: dynamic_color | ||||||
|       sha256: c4a508284b14ec4dda5adba2c28b2cdd34fbae1afead7e8c52cad87d51c5405b |       sha256: bbebb1b7ebed819e0ec83d4abdc2a8482d934f6a85289ffc1c6acf7589fa2aad | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.6.2" |     version: "1.6.3" | ||||||
|   easy_localization: |   easy_localization: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -181,10 +181,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: file_picker |       name: file_picker | ||||||
|       sha256: d8e9ca7e5d1983365c277f12c21b4362df6cf659c99af146ad4d04eb33033013 |       sha256: b85eb92b175767fdaa0c543bf3b0d1f610fe966412ea72845fe5ba7801e763ff | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.2.6" |     version: "5.2.10" | ||||||
|   flutter: |   flutter: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -417,18 +417,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_android |       name: path_provider_android | ||||||
|       sha256: "019f18c9c10ae370b08dce1f3e3b73bc9f58e7f087bb5e921f06529438ac0ae7" |       sha256: da97262be945a72270513700a92b39dd2f4a54dad55d061687e2e37a6390366a | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.24" |     version: "2.0.25" | ||||||
|   path_provider_foundation: |   path_provider_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_foundation |       name: path_provider_foundation | ||||||
|       sha256: "818b2dc38b0f178e0ea3f7cf3b28146faab11375985d815942a68eee11c2d0f7" |       sha256: ad4c4d011830462633f03eb34445a45345673dfd4faf1ab0b4735fbd93b19183 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.2.1" |     version: "2.2.2" | ||||||
|   path_provider_linux: |   path_provider_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -537,18 +537,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: share_plus |       name: share_plus | ||||||
|       sha256: "8c6892037b1824e2d7e8f59d54b3105932899008642e6372e5079c6939b4b625" |       sha256: "692261968a494e47323dcc8bc66d8d52e81bc27cb4b808e4e8d7e8079d4cc01a" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.1" |     version: "6.3.2" | ||||||
|   share_plus_platform_interface: |   share_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: share_plus_platform_interface |       name: share_plus_platform_interface | ||||||
|       sha256: "82ddd4ab9260c295e6e39612d4ff00390b9a7a21f1bb1da771e2f232d80ab8a1" |       sha256: "0c6e61471bd71b04a138b8b588fa388e66d8b005e6f2deda63371c5c505a0981" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.2.0" |     version: "3.2.1" | ||||||
|   shared_preferences: |   shared_preferences: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -561,18 +561,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_android |       name: shared_preferences_android | ||||||
|       sha256: "8304d8a1f7d21a429f91dee552792249362b68a331ac5c3c1caf370f658873f6" |       sha256: "7fa90471a6875d26ad78c7e4a675874b2043874586891128dc5899662c97db46" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.0" |     version: "2.1.2" | ||||||
|   shared_preferences_foundation: |   shared_preferences_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_foundation |       name: shared_preferences_foundation | ||||||
|       sha256: cf2a42fb20148502022861f71698db12d937c7459345a1bdaa88fc91a91b3603 |       sha256: "0c1c16c56c9708aa9c361541a6f0e5cc6fc12a3232d866a687a7b7db30032b07" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.2.0" |     version: "2.2.1" | ||||||
|   shared_preferences_linux: |   shared_preferences_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -622,18 +622,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: sqflite |       name: sqflite | ||||||
|       sha256: "500d6fec583d2c021f2d25a056d96654f910662c64f836cd2063167b8f1fa758" |       sha256: e7dfb6482d5d02b661d0b2399efa72b98909e5aa7b8336e1fb37e226264ade00 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.2.6" |     version: "2.2.7" | ||||||
|   sqflite_common: |   sqflite_common: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: sqflite_common |       name: sqflite_common | ||||||
|       sha256: "963dad8c4aa2f814ce7d2d5b1da2f36f31bd1a439d8f27e3dc189bb9d26bc684" |       sha256: "220831bf0bd5333ff2445eee35ec131553b804e6b5d47a4a37ca6f5eb66e282c" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.3" |     version: "2.4.4" | ||||||
|   stack_trace: |   stack_trace: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -662,10 +662,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: synchronized |       name: synchronized | ||||||
|       sha256: "33b31b6beb98100bf9add464a36a8dd03eb10c7a8cf15aeec535e9b054aaf04b" |       sha256: "5fcbd27688af6082f5abd611af56ee575342c30e87541d0245f7ff99faa02c60" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.1" |     version: "3.1.0" | ||||||
|   term_glyph: |   term_glyph: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -686,10 +686,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: timezone |       name: timezone | ||||||
|       sha256: "24c8fcdd49a805d95777a39064862133ff816ebfffe0ceff110fb5960e557964" |       sha256: "1cfd8ddc2d1cfd836bc93e67b9be88c3adaeca6f40a00ca999104c30693cdca0" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.9.1" |     version: "0.9.2" | ||||||
|   typed_data: |   typed_data: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -710,18 +710,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_android |       name: url_launcher_android | ||||||
|       sha256: dd729390aa936bf1bdf5cd1bc7468ff340263f80a2c4f569416507667de8e3c8 |       sha256: a52628068d282d01a07cd86e6ba99e497aa45ce8c91159015b2416907d78e411 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.0.26" |     version: "6.0.27" | ||||||
|   url_launcher_ios: |   url_launcher_ios: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_ios |       name: url_launcher_ios | ||||||
|       sha256: "3dedc66ca3c0bef9e6a93c0999aee102556a450afcc1b7bcfeace7a424927d92" |       sha256: "9af7ea73259886b92199f9e42c116072f05ff9bea2dcb339ab935dfc957392c2" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.1.3" |     version: "6.1.4" | ||||||
|   url_launcher_linux: |   url_launcher_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -734,10 +734,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_macos |       name: url_launcher_macos | ||||||
|       sha256: "0ef2b4f97942a16523e51256b799e9aa1843da6c60c55eefbfa9dbc2dcb8331a" |       sha256: "91ee3e75ea9dadf38036200c5d3743518f4a5eb77a8d13fda1ee5764373f185e" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.4" |     version: "3.0.5" | ||||||
|   url_launcher_platform_interface: |   url_launcher_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -782,42 +782,42 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter |       name: webview_flutter | ||||||
|       sha256: "47663d51a9061451aa3880a214ee9a65dcbb933b77bc44388e194279ab3ccaf6" |       sha256: "1a37bdbaaf5fbe09ad8579ab09ecfd473284ce482f900b5aea27cf834386a567" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.0.7" |     version: "4.2.0" | ||||||
|   webview_flutter_android: |   webview_flutter_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_android |       name: webview_flutter_android | ||||||
|       sha256: "9e223788e1954087dac30d813dc151f8e12f09f1139f116ce20b5658893f3627" |       sha256: "134ed5d36127b6f5865e86a82174886eae0b983dacd8df14b0448371debde755" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.4.4" |     version: "3.6.0" | ||||||
|   webview_flutter_platform_interface: |   webview_flutter_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_platform_interface |       name: webview_flutter_platform_interface | ||||||
|       sha256: "1939c39e2150fb4d30fd3cc59a891a49fed9935db53007df633ed83581b6117b" |       sha256: "78715dc442b7849dbde74e92bb67de1cecf5addf95531c5fb474e72f5fe9a507" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.0" |     version: "2.3.0" | ||||||
|   webview_flutter_wkwebview: |   webview_flutter_wkwebview: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_wkwebview |       name: webview_flutter_wkwebview | ||||||
|       sha256: d601aba11ad8d4481e17a34a76fa1d30dee92dcbbe2c58b0df3120e9453099c7 |       sha256: c94d242d8cbe1012c06ba7ac790c46d6e6b68723b7d34f8c74ed19f68d166f49 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.2.3" |     version: "3.4.0" | ||||||
|   win32: |   win32: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: win32 |       name: win32 | ||||||
|       sha256: c9ebe7ee4ab0c2194e65d3a07d8c54c5d00bb001b76081c4a04cdb8448b59e46 |       sha256: a6f0236dbda0f63aa9a25ad1ff9a9d8a4eaaa5012da0dc59d21afdb1dc361ca4 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.1.3" |     version: "3.1.4" | ||||||
|   xdg_directories: |   xdg_directories: | ||||||
|     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.26+148 # When changing this, update the tag in main() accordingly | version: 0.11.34+156 # 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' | ||||||
| @@ -49,7 +49,7 @@ dependencies: | |||||||
|   permission_handler: ^10.0.0 |   permission_handler: ^10.0.0 | ||||||
|   fluttertoast: ^8.0.9 |   fluttertoast: ^8.0.9 | ||||||
|   device_info_plus: ^8.0.0 |   device_info_plus: ^8.0.0 | ||||||
|   file_picker: ^5.1.0 |   file_picker: ^5.2.10 | ||||||
|   animations: ^2.0.4 |   animations: ^2.0.4 | ||||||
|   install_plugin_v2: ^1.0.0 |   install_plugin_v2: ^1.0.0 | ||||||
|   share_plus: ^6.0.1 |   share_plus: ^6.0.1 | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user