mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-11-04 07:13:28 +01:00 
			
		
		
		
	Compare commits
	
		
			52 Commits
		
	
	
		
			v0.11.27-b
			...
			v0.11.36-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					e1d914118f | ||
| 
						 | 
					4a07cf9951 | ||
| 
						 | 
					ce44e200a5 | ||
| 
						 | 
					e8ebf53626 | ||
| 
						 | 
					cdd6a4124c | ||
| 
						 | 
					09c71e4e9f | ||
| 
						 | 
					28a996441c | ||
| 
						 | 
					396bf012c9 | ||
| 
						 | 
					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 | 
@@ -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": "Használjon tiszta fekete sötét témát",
 | 
				
			||||||
    "appSortBy": "App rendezés...",
 | 
					    "appSortBy": "App rendezés...",
 | 
				
			||||||
    "authorName": "Szerző/Név",
 | 
					    "authorName": "Szerző/Név",
 | 
				
			||||||
    "nameAuthor": "Név/Szerző",
 | 
					    "nameAuthor": "Név/Szerző",
 | 
				
			||||||
@@ -206,7 +207,7 @@
 | 
				
			|||||||
    "addCategory": "Új kategória",
 | 
					    "addCategory": "Új kategória",
 | 
				
			||||||
    "label": "Címke",
 | 
					    "label": "Címke",
 | 
				
			||||||
    "language": "Nyelv",
 | 
					    "language": "Nyelv",
 | 
				
			||||||
    "copiedToClipboard": "Copied to Clipboard",
 | 
					    "copiedToClipboard": "Másolva a vágólapra",
 | 
				
			||||||
    "storagePermissionDenied": "Tárhely engedély megtagadva",
 | 
					    "storagePermissionDenied": "Tárhely engedély megtagadva",
 | 
				
			||||||
    "selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.",
 | 
					    "selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.",
 | 
				
			||||||
    "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel",
 | 
					    "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel",
 | 
				
			||||||
@@ -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,25 @@ 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('/')
 | 
				
			||||||
 | 
					            .where((element) => element.trim().isNotEmpty)
 | 
				
			||||||
 | 
					            .toList();
 | 
				
			||||||
 | 
					        if (e.startsWith('/') || currPathSegments.isEmpty) {
 | 
				
			||||||
 | 
					          return '${uri.origin}/$e';
 | 
				
			||||||
 | 
					        } else if (e.split('/').length == 1) {
 | 
				
			||||||
 | 
					          return '${uri.origin}/${currPathSegments.join('/')}/$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.27';
 | 
					const String currentVersion = '0.11.36';
 | 
				
			||||||
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,18 @@ class _AppPageState extends State<AppPage> {
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    getResetInstallStatusButton() => TextButton(
 | 
				
			||||||
 | 
					        onPressed: app?.app == null
 | 
				
			||||||
 | 
					            ? null
 | 
				
			||||||
 | 
					            : () {
 | 
				
			||||||
 | 
					                app!.app.installedVersion = null;
 | 
				
			||||||
 | 
					                appsProvider.saveApps([app.app]);
 | 
				
			||||||
 | 
					              },
 | 
				
			||||||
 | 
					        child: Text(
 | 
				
			||||||
 | 
					          tr('resetInstallStatus'),
 | 
				
			||||||
 | 
					          textAlign: TextAlign.center,
 | 
				
			||||||
 | 
					        ));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    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 +396,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 +412,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 || trackOnly) &&
 | 
				
			||||||
 | 
					                                  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))
 | 
				
			||||||
@@ -358,7 +364,7 @@ class AppsPageState extends State<AppsPage> {
 | 
				
			|||||||
                        child: Image(
 | 
					                        child: Image(
 | 
				
			||||||
                          image: const AssetImage(
 | 
					                          image: const AssetImage(
 | 
				
			||||||
                              'assets/graphics/icon_small.png'),
 | 
					                              'assets/graphics/icon_small.png'),
 | 
				
			||||||
                          color: Colors.white.withOpacity(0.1),
 | 
					                          color: Colors.white.withOpacity(0.3),
 | 
				
			||||||
                          colorBlendMode: BlendMode.modulate,
 | 
					                          colorBlendMode: BlendMode.modulate,
 | 
				
			||||||
                          gaplessPlayback: true,
 | 
					                          gaplessPlayback: 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,7 +165,7 @@ 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();
 | 
				
			||||||
@@ -169,7 +175,7 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
          .getSource(app.url)
 | 
					          .getSource(app.url)
 | 
				
			||||||
          .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value);
 | 
					          .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value);
 | 
				
			||||||
      var fileName = '${app.id}-${downloadUrl.hashCode}.apk';
 | 
					      var fileName = '${app.id}-${downloadUrl.hashCode}.apk';
 | 
				
			||||||
      var notif = DownloadNotification(app.name, 100);
 | 
					      var notif = DownloadNotification(app.finalName, 100);
 | 
				
			||||||
      notificationsProvider?.cancel(notif.id);
 | 
					      notificationsProvider?.cancel(notif.id);
 | 
				
			||||||
      int? prevProg;
 | 
					      int? prevProg;
 | 
				
			||||||
      File downloadedFile =
 | 
					      File downloadedFile =
 | 
				
			||||||
@@ -179,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);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -198,7 +204,8 @@ 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;
 | 
				
			||||||
@@ -207,7 +214,7 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
            '${downloadedFile.parent.path}/${app.id}-${downloadUrl.hashCode}.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);
 | 
				
			||||||
@@ -298,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;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -359,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]);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -637,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) {
 | 
				
			||||||
@@ -667,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;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
@@ -903,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) {
 | 
				
			||||||
@@ -940,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,
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										124
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										124
									
								
								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: "88a8001851fdc9bd54fa4e30d0277bb900a50f3d86ff244da7f027400bf23ac0"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.1.1"
 | 
					    version: "2.1.4"
 | 
				
			||||||
  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: "04cbc7c332a6f0bba88fed354de78813e9d24049c1800aaf10f449c7adc22603"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "3.1.6"
 | 
					    version: "3.1.9"
 | 
				
			||||||
  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: f52ab3b76b36ede4d135aab80194df8925b553686f0fa12226b4e2d658e45903
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "8.1.0"
 | 
					    version: "8.2.2"
 | 
				
			||||||
  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
 | 
				
			||||||
@@ -210,26 +210,26 @@ packages:
 | 
				
			|||||||
    dependency: "direct main"
 | 
					    dependency: "direct main"
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: flutter_local_notifications
 | 
					      name: flutter_local_notifications
 | 
				
			||||||
      sha256: "293995f94e120c8afce768981bd1fa9c5d6de67c547568e3b42ae2defdcbb4a0"
 | 
					      sha256: "2876372952b65ca7f684e698eba22bda1cf581fa071dd30ba2f01900f507d0d1"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "13.0.0"
 | 
					    version: "14.0.0+1"
 | 
				
			||||||
  flutter_local_notifications_linux:
 | 
					  flutter_local_notifications_linux:
 | 
				
			||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: flutter_local_notifications_linux
 | 
					      name: flutter_local_notifications_linux
 | 
				
			||||||
      sha256: ccb08b93703aeedb58856e5637450bf3ffec899adb66dc325630b68994734b89
 | 
					      sha256: "909bb95de05a2e793503a2437146285a2f600cd0b3f826e26b870a334d8586d7"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "3.0.0+1"
 | 
					    version: "4.0.0"
 | 
				
			||||||
  flutter_local_notifications_platform_interface:
 | 
					  flutter_local_notifications_platform_interface:
 | 
				
			||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: flutter_local_notifications_platform_interface
 | 
					      name: flutter_local_notifications_platform_interface
 | 
				
			||||||
      sha256: "5ec1feac5f7f7d9266759488bc5f76416152baba9aa1b26fe572246caa00d1ab"
 | 
					      sha256: "63235c42de5b6c99846969a27ad0209c401e6b77b0498939813725b5791c107c"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "6.0.0"
 | 
					    version: "7.0.0"
 | 
				
			||||||
  flutter_localizations:
 | 
					  flutter_localizations:
 | 
				
			||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description: flutter
 | 
					    description: flutter
 | 
				
			||||||
@@ -247,10 +247,10 @@ packages:
 | 
				
			|||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: flutter_plugin_android_lifecycle
 | 
					      name: flutter_plugin_android_lifecycle
 | 
				
			||||||
      sha256: c224ac897bed083dabf11f238dd11a239809b446740be0c2044608c50029ffdf
 | 
					      sha256: "8ffe990dac54a4a5492747added38571a5ab474c8e5d196809ea08849c69b1bb"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.0.9"
 | 
					    version: "2.0.13"
 | 
				
			||||||
  flutter_test:
 | 
					  flutter_test:
 | 
				
			||||||
    dependency: "direct dev"
 | 
					    dependency: "direct dev"
 | 
				
			||||||
    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: "2cec049d282c7f13c594b4a73976b0b4f2d7a1838a6dd5aaf7bd9719196bee86"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.0.24"
 | 
					    version: "2.0.27"
 | 
				
			||||||
  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:
 | 
				
			||||||
@@ -449,10 +449,10 @@ packages:
 | 
				
			|||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: path_provider_windows
 | 
					      name: path_provider_windows
 | 
				
			||||||
      sha256: f53720498d5a543f9607db4b0e997c4b5438884de25b0f73098cc2671a51b130
 | 
					      sha256: d3f80b32e83ec208ac95253e0cd4d298e104fbc63cb29c5c69edaed43b0c69d6
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.1.5"
 | 
					    version: "2.1.6"
 | 
				
			||||||
  permission_handler:
 | 
					  permission_handler:
 | 
				
			||||||
    dependency: "direct main"
 | 
					    dependency: "direct main"
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
@@ -537,18 +537,18 @@ packages:
 | 
				
			|||||||
    dependency: "direct main"
 | 
					    dependency: "direct main"
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: share_plus
 | 
					      name: share_plus
 | 
				
			||||||
      sha256: "8c6892037b1824e2d7e8f59d54b3105932899008642e6372e5079c6939b4b625"
 | 
					      sha256: b1f15232d41e9701ab2f04181f21610c36c83a12ae426b79b4bd011c567934b1
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "6.3.1"
 | 
					    version: "6.3.4"
 | 
				
			||||||
  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: "6478c6bbbecfe9aced34c483171e90d7c078f5883558b30ec3163cf18402c749"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.1.0"
 | 
					    version: "2.1.4"
 | 
				
			||||||
  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: "8453780d1f703ead201a39673deb93decf85d543f359f750e2afc4908b55533f"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.2.6"
 | 
					    version: "2.2.8"
 | 
				
			||||||
  sqflite_common:
 | 
					  sqflite_common:
 | 
				
			||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: sqflite_common
 | 
					      name: sqflite_common
 | 
				
			||||||
      sha256: "963dad8c4aa2f814ce7d2d5b1da2f36f31bd1a439d8f27e3dc189bb9d26bc684"
 | 
					      sha256: e77abf6ff961d69dfef41daccbb66b51e9983cdd5cb35bf30733598057401555
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "2.4.3"
 | 
					    version: "2.4.5"
 | 
				
			||||||
  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,34 +710,34 @@ packages:
 | 
				
			|||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: url_launcher_android
 | 
					      name: url_launcher_android
 | 
				
			||||||
      sha256: dd729390aa936bf1bdf5cd1bc7468ff340263f80a2c4f569416507667de8e3c8
 | 
					      sha256: "22f8db4a72be26e9e3a4aa3f194b1f7afbc76d20ec141f84be1d787db2155cbd"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "6.0.26"
 | 
					    version: "6.0.31"
 | 
				
			||||||
  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:
 | 
				
			||||||
      name: url_launcher_linux
 | 
					      name: url_launcher_linux
 | 
				
			||||||
      sha256: "206fb8334a700ef7754d6a9ed119e7349bc830448098f21a69bf1b4ed038cabc"
 | 
					      sha256: "207f4ddda99b95b4d4868320a352d374b0b7e05eefad95a4a26f57da413443f5"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "3.0.4"
 | 
					    version: "3.0.5"
 | 
				
			||||||
  url_launcher_macos:
 | 
					  url_launcher_macos:
 | 
				
			||||||
    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:
 | 
				
			||||||
@@ -758,10 +758,10 @@ packages:
 | 
				
			|||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
      name: url_launcher_windows
 | 
					      name: url_launcher_windows
 | 
				
			||||||
      sha256: a83ba3607a507758669cfafb03f9de09bf6e6280c14d9b9cb18f013e406dcacd
 | 
					      sha256: "254708f17f7c20a9c8c471f67d86d76d4a3f9c1591aad1e15292008aceb82771"
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "3.0.5"
 | 
					    version: "3.0.6"
 | 
				
			||||||
  uuid:
 | 
					  uuid:
 | 
				
			||||||
    dependency: transitive
 | 
					    dependency: transitive
 | 
				
			||||||
    description:
 | 
					    description:
 | 
				
			||||||
@@ -782,50 +782,50 @@ 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: d6cf18cd6c809c5a9294cd99707a21986aac4e08c87e1916ce2590315fb55d3a
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "3.4.4"
 | 
					    version: "3.6.2"
 | 
				
			||||||
  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:
 | 
				
			||||||
      name: xdg_directories
 | 
					      name: xdg_directories
 | 
				
			||||||
      sha256: bd512f03919aac5f1313eb8249f223bacf4927031bf60b02601f81f687689e86
 | 
					      sha256: ee1505df1426458f7f60aac270645098d318a8b4766d85fde75f76f2e21807d1
 | 
				
			||||||
      url: "https://pub.dev"
 | 
					      url: "https://pub.dev"
 | 
				
			||||||
    source: hosted
 | 
					    source: hosted
 | 
				
			||||||
    version: "0.2.0+3"
 | 
					    version: "1.0.0"
 | 
				
			||||||
  xml:
 | 
					  xml:
 | 
				
			||||||
    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.27+149 # When changing this, update the tag in main() accordingly
 | 
					version: 0.11.36+158 # 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'
 | 
				
			||||||
@@ -38,7 +38,7 @@ dependencies:
 | 
				
			|||||||
  cupertino_icons: ^1.0.5
 | 
					  cupertino_icons: ^1.0.5
 | 
				
			||||||
  path_provider: ^2.0.11
 | 
					  path_provider: ^2.0.11
 | 
				
			||||||
  flutter_fgbg: ^0.2.0 # Try removing reliance on this
 | 
					  flutter_fgbg: ^0.2.0 # Try removing reliance on this
 | 
				
			||||||
  flutter_local_notifications: ^13.0.0
 | 
					  flutter_local_notifications: ^14.0.0+1
 | 
				
			||||||
  provider: ^6.0.3
 | 
					  provider: ^6.0.3
 | 
				
			||||||
  http: ^0.13.5
 | 
					  http: ^0.13.5
 | 
				
			||||||
  webview_flutter: ^4.0.0
 | 
					  webview_flutter: ^4.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