mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-11-04 15:23:28 +01:00 
			
		
		
		
	Compare commits
	
		
			18 Commits
		
	
	
		
			v0.11.20-b
			...
			v0.11.26-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					a2edc86bfa | ||
| 
						 | 
					0804e680b2 | ||
| 
						 | 
					49affd1bd4 | ||
| 
						 | 
					202ce4f0d5 | ||
| 
						 | 
					361a3e1bc2 | ||
| 
						 | 
					f33a26d4f4 | ||
| 
						 | 
					7aaf56ec8c | ||
| 
						 | 
					ed120016d9 | ||
| 
						 | 
					e8cbac8657 | ||
| 
						 | 
					b66c13d319 | ||
| 
						 | 
					782d055bc3 | ||
| 
						 | 
					d557746965 | ||
| 
						 | 
					e6b05d50b9 | ||
| 
						 | 
					dea635fa6a | ||
| 
						 | 
					682026ed0a | ||
| 
						 | 
					9fe8a200ef | ||
| 
						 | 
					210100da2b | ||
| 
						 | 
					d52660235b | 
@@ -207,6 +207,7 @@
 | 
				
			|||||||
    "addCategory": "Kategorie hinzufügen",
 | 
					    "addCategory": "Kategorie hinzufügen",
 | 
				
			||||||
    "label": "Bezeichnung",
 | 
					    "label": "Bezeichnung",
 | 
				
			||||||
    "language": "Sprache",
 | 
					    "language": "Sprache",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "storagePermissionDenied": "Speicherberechtigung verweigert",
 | 
					    "storagePermissionDenied": "Speicherberechtigung verweigert",
 | 
				
			||||||
    "selectedCategorizeWarning": "Dadurch werden alle bestehenden Kategorieeinstellungen für die ausgewählten Apps ersetzt.",
 | 
					    "selectedCategorizeWarning": "Dadurch werden alle bestehenden Kategorieeinstellungen für die ausgewählten Apps ersetzt.",
 | 
				
			||||||
    "filterAPKsByRegEx": "APKs nach regulärem Ausdruck filtern",
 | 
					    "filterAPKsByRegEx": "APKs nach regulärem Ausdruck filtern",
 | 
				
			||||||
@@ -221,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "Versionserkennung",
 | 
					    "versionDetection": "Versionserkennung",
 | 
				
			||||||
    "standardVersionDetection": "Standardversionserkennung",
 | 
					    "standardVersionDetection": "Standardversionserkennung",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "App entfernen?",
 | 
					        "one": "App entfernen?",
 | 
				
			||||||
        "other": "App entfernen?"
 | 
					        "other": "App entfernen?"
 | 
				
			||||||
@@ -269,4 +271,4 @@
 | 
				
			|||||||
        "one": "{} und 1 weitere Anwendung wurden aktualisiert.",
 | 
					        "one": "{} und 1 weitere Anwendung wurden aktualisiert.",
 | 
				
			||||||
        "other": "{} und {} weitere Anwendungen wurden aktualisiert."
 | 
					        "other": "{} und {} weitere Anwendungen wurden aktualisiert."
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -207,6 +207,7 @@
 | 
				
			|||||||
    "addCategory": "Add Category",
 | 
					    "addCategory": "Add Category",
 | 
				
			||||||
    "label": "Label",
 | 
					    "label": "Label",
 | 
				
			||||||
    "language": "Language",
 | 
					    "language": "Language",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "storagePermissionDenied": "Storage permission denied",
 | 
					    "storagePermissionDenied": "Storage permission denied",
 | 
				
			||||||
    "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.",
 | 
					    "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.",
 | 
				
			||||||
    "filterAPKsByRegEx": "Filter APKs by Regular Expression",
 | 
					    "filterAPKsByRegEx": "Filter APKs by Regular Expression",
 | 
				
			||||||
@@ -221,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "Version Detection",
 | 
					    "versionDetection": "Version Detection",
 | 
				
			||||||
    "standardVersionDetection": "Standard version detection",
 | 
					    "standardVersionDetection": "Standard version detection",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "Remove App?",
 | 
					        "one": "Remove App?",
 | 
				
			||||||
        "other": "Remove Apps?"
 | 
					        "other": "Remove Apps?"
 | 
				
			||||||
@@ -269,4 +271,4 @@
 | 
				
			|||||||
        "one": "{} and 1 more app were updated.",
 | 
					        "one": "{} and 1 more app were updated.",
 | 
				
			||||||
        "other": "{} and {} more apps were updated."
 | 
					        "other": "{} and {} more apps were updated."
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -207,6 +207,7 @@
 | 
				
			|||||||
    "addCategory": "اضافه کردن دسته",
 | 
					    "addCategory": "اضافه کردن دسته",
 | 
				
			||||||
    "label": "برچسب",
 | 
					    "label": "برچسب",
 | 
				
			||||||
    "language": "زبان",
 | 
					    "language": "زبان",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "storagePermissionDenied": "مجوز ذخیره سازی رد شد",
 | 
					    "storagePermissionDenied": "مجوز ذخیره سازی رد شد",
 | 
				
			||||||
    "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.",
 | 
					    "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.",
 | 
				
			||||||
    "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید",
 | 
					    "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید",
 | 
				
			||||||
@@ -221,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "تشخیص نسخه",
 | 
					    "versionDetection": "تشخیص نسخه",
 | 
				
			||||||
    "standardVersionDetection": "تشخیص نسخه استاندارد",
 | 
					    "standardVersionDetection": "تشخیص نسخه استاندارد",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "برنامه حذف شود؟",
 | 
					        "one": "برنامه حذف شود؟",
 | 
				
			||||||
        "other": "برنامه ها حذف شوند؟"
 | 
					        "other": "برنامه ها حذف شوند؟"
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -207,6 +207,7 @@
 | 
				
			|||||||
    "addCategory": "Ajouter une catégorie",
 | 
					    "addCategory": "Ajouter une catégorie",
 | 
				
			||||||
    "label": "Étiquette",
 | 
					    "label": "Étiquette",
 | 
				
			||||||
    "language": "Langue",
 | 
					    "language": "Langue",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "storagePermissionDenied": "Autorisation de stockage refusée",
 | 
					    "storagePermissionDenied": "Autorisation de stockage refusée",
 | 
				
			||||||
    "selectedCategorizeWarning": "Cela remplacera tous les paramètres de catégorie existants pour les applications sélectionnées.",
 | 
					    "selectedCategorizeWarning": "Cela remplacera tous les paramètres de catégorie existants pour les applications sélectionnées.",
 | 
				
			||||||
    "filterAPKsByRegEx": "Filtrer les APK par expression régulière",
 | 
					    "filterAPKsByRegEx": "Filtrer les APK par expression régulière",
 | 
				
			||||||
@@ -221,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "Détection des versions",
 | 
					    "versionDetection": "Détection des versions",
 | 
				
			||||||
    "standardVersionDetection": "Détection de version standard",
 | 
					    "standardVersionDetection": "Détection de version standard",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "Supprimer l'application ?",
 | 
					        "one": "Supprimer l'application ?",
 | 
				
			||||||
        "other": "Supprimer les applications ?"
 | 
					        "other": "Supprimer les applications ?"
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -206,6 +206,7 @@
 | 
				
			|||||||
    "addCategory": "Új kategória",
 | 
					    "addCategory": "Új kategória",
 | 
				
			||||||
    "label": "Címke",
 | 
					    "label": "Címke",
 | 
				
			||||||
    "language": "Nyelv",
 | 
					    "language": "Nyelv",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "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",
 | 
				
			||||||
@@ -219,7 +220,8 @@
 | 
				
			|||||||
    "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)",
 | 
					    "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)",
 | 
				
			||||||
    "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": "Group by Category",
 | 
					    "groupByCategory": "Csoportosítás Kategória alapján",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "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?"
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -207,6 +207,7 @@
 | 
				
			|||||||
    "addCategory": "Aggiungi categoria",
 | 
					    "addCategory": "Aggiungi categoria",
 | 
				
			||||||
    "label": "Etichetta",
 | 
					    "label": "Etichetta",
 | 
				
			||||||
    "language": "Lingua",
 | 
					    "language": "Lingua",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "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,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "Rilevamento di versione",
 | 
					    "versionDetection": "Rilevamento di versione",
 | 
				
			||||||
    "standardVersionDetection": "Rilevamento di versione standard",
 | 
					    "standardVersionDetection": "Rilevamento di versione standard",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "Rimuovere l'App?",
 | 
					        "one": "Rimuovere l'App?",
 | 
				
			||||||
        "other": "Rimuovere le App?"
 | 
					        "other": "Rimuovere le App?"
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -207,6 +207,7 @@
 | 
				
			|||||||
    "addCategory": "カテゴリを追加",
 | 
					    "addCategory": "カテゴリを追加",
 | 
				
			||||||
    "label": "ラベル",
 | 
					    "label": "ラベル",
 | 
				
			||||||
    "language": "言語",
 | 
					    "language": "言語",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "クリップボードにコピーしました",
 | 
				
			||||||
    "storagePermissionDenied": "ストレージ権限が拒否されました",
 | 
					    "storagePermissionDenied": "ストレージ権限が拒否されました",
 | 
				
			||||||
    "selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。",
 | 
					    "selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。",
 | 
				
			||||||
    "filterAPKsByRegEx": "正規表現でAPKを絞り込む",
 | 
					    "filterAPKsByRegEx": "正規表現でAPKを絞り込む",
 | 
				
			||||||
@@ -221,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "バージョン検出",
 | 
					    "versionDetection": "バージョン検出",
 | 
				
			||||||
    "standardVersionDetection": "標準のバージョン検出",
 | 
					    "standardVersionDetection": "標準のバージョン検出",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "アプリを削除しますか?",
 | 
					        "one": "アプリを削除しますか?",
 | 
				
			||||||
        "other": "アプリを削除しますか?"
 | 
					        "other": "アプリを削除しますか?"
 | 
				
			||||||
@@ -269,4 +271,4 @@
 | 
				
			|||||||
        "one": "{} とさらに {} 個のアプリがアップデートされました",
 | 
					        "one": "{} とさらに {} 個のアプリがアップデートされました",
 | 
				
			||||||
        "other": "{} とさらに {} 個のアプリがアップデートされました"
 | 
					        "other": "{} とさらに {} 個のアプリがアップデートされました"
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -208,6 +208,7 @@
 | 
				
			|||||||
    "addCategory": "添加类别",
 | 
					    "addCategory": "添加类别",
 | 
				
			||||||
    "label": "标签",
 | 
					    "label": "标签",
 | 
				
			||||||
    "language": "语言",
 | 
					    "language": "语言",
 | 
				
			||||||
 | 
					    "copiedToClipboard": "Copied to Clipboard",
 | 
				
			||||||
    "storagePermissionDenied": "存储权限已被拒绝",
 | 
					    "storagePermissionDenied": "存储权限已被拒绝",
 | 
				
			||||||
    "selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别",
 | 
					    "selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别",
 | 
				
			||||||
    "filterAPKsByRegEx": "Filter APKs by Regular Expression",
 | 
					    "filterAPKsByRegEx": "Filter APKs by Regular Expression",
 | 
				
			||||||
@@ -221,6 +222,7 @@
 | 
				
			|||||||
    "versionDetection": "Version Detection",
 | 
					    "versionDetection": "Version Detection",
 | 
				
			||||||
    "standardVersionDetection": "Standard version detection",
 | 
					    "standardVersionDetection": "Standard version detection",
 | 
				
			||||||
    "groupByCategory": "Group by Category",
 | 
					    "groupByCategory": "Group by Category",
 | 
				
			||||||
 | 
					    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
				
			||||||
    "removeAppQuestion": {
 | 
					    "removeAppQuestion": {
 | 
				
			||||||
        "one": "删除应用?",
 | 
					        "one": "删除应用?",
 | 
				
			||||||
        "other": "删除应用?"
 | 
					        "other": "删除应用?"
 | 
				
			||||||
@@ -269,4 +271,4 @@
 | 
				
			|||||||
        "one": "{} 和 {} 更多应用已被安装",
 | 
					        "one": "{} 和 {} 更多应用已被安装",
 | 
				
			||||||
        "other": "{} 和 {} 更多应用已被安装"
 | 
					        "other": "{} 和 {} 更多应用已被安装"
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -68,7 +68,7 @@ class Codeberg extends AppSource {
 | 
				
			|||||||
    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['name'] != null && e['browser_download_url'] != null
 | 
					                return e['name'] != null && e['browser_download_url'] != null
 | 
				
			||||||
@@ -77,7 +77,6 @@ class Codeberg extends AppSource {
 | 
				
			|||||||
                    : const MapEntry('', '');
 | 
					                    : const MapEntry('', '');
 | 
				
			||||||
              })
 | 
					              })
 | 
				
			||||||
              .where((element) => element.key.toLowerCase().endsWith('.apk'))
 | 
					              .where((element) => element.key.toLowerCase().endsWith('.apk'))
 | 
				
			||||||
              .map((e) => e.value)
 | 
					 | 
				
			||||||
              .toList() ??
 | 
					              .toList() ??
 | 
				
			||||||
          [];
 | 
					          [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -119,7 +118,9 @@ class Codeberg extends AppSource {
 | 
				
			|||||||
        throw NoVersionError();
 | 
					        throw NoVersionError();
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      var changeLog = targetRelease['body'].toString();
 | 
					      var changeLog = targetRelease['body'].toString();
 | 
				
			||||||
      return APKDetails(version, targetRelease['apkUrls'] as List<String>,
 | 
					      return APKDetails(
 | 
				
			||||||
 | 
					          version,
 | 
				
			||||||
 | 
					          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);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -14,12 +14,14 @@ class FDroid extends AppSource {
 | 
				
			|||||||
  @override
 | 
					  @override
 | 
				
			||||||
  String standardizeURL(String url) {
 | 
					  String standardizeURL(String url) {
 | 
				
			||||||
    RegExp standardUrlRegExB =
 | 
					    RegExp standardUrlRegExB =
 | 
				
			||||||
        RegExp('^https?://$host/+[^/]+/+packages/+[^/]+');
 | 
					        RegExp('^https?://(cloudflare\\.)?$host/+[^/]+/+packages/+[^/]+');
 | 
				
			||||||
    RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase());
 | 
					    RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase());
 | 
				
			||||||
    if (match != null) {
 | 
					    if (match != null) {
 | 
				
			||||||
      url = 'https://$host/packages/${Uri.parse(url).pathSegments.last}';
 | 
					      url =
 | 
				
			||||||
 | 
					          'https://${Uri.parse(url.substring(0, match.end)).host}/packages/${Uri.parse(url).pathSegments.last}';
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    RegExp standardUrlRegExA = RegExp('^https?://$host/+packages/+[^/]+');
 | 
					    RegExp standardUrlRegExA =
 | 
				
			||||||
 | 
					        RegExp('^https?://(cloudflare\\.)?$host/+packages/+[^/]+');
 | 
				
			||||||
    match = standardUrlRegExA.firstMatch(url.toLowerCase());
 | 
					    match = standardUrlRegExA.firstMatch(url.toLowerCase());
 | 
				
			||||||
    if (match == null) {
 | 
					    if (match == null) {
 | 
				
			||||||
      throw InvalidURLError(name);
 | 
					      throw InvalidURLError(name);
 | 
				
			||||||
@@ -48,7 +50,7 @@ class FDroid extends AppSource {
 | 
				
			|||||||
          .where((element) => element['versionName'] == latestVersion)
 | 
					          .where((element) => element['versionName'] == latestVersion)
 | 
				
			||||||
          .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk')
 | 
					          .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk')
 | 
				
			||||||
          .toList();
 | 
					          .toList();
 | 
				
			||||||
      return APKDetails(latestVersion, apkUrls,
 | 
					      return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls),
 | 
				
			||||||
          AppNames(name, Uri.parse(standardUrl).pathSegments.last));
 | 
					          AppNames(name, Uri.parse(standardUrl).pathSegments.last));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
@@ -61,9 +63,10 @@ class FDroid extends AppSource {
 | 
				
			|||||||
    Map<String, dynamic> additionalSettings,
 | 
					    Map<String, dynamic> additionalSettings,
 | 
				
			||||||
  ) async {
 | 
					  ) async {
 | 
				
			||||||
    String? appId = tryInferringAppId(standardUrl);
 | 
					    String? appId = tryInferringAppId(standardUrl);
 | 
				
			||||||
 | 
					    String host = Uri.parse(standardUrl).host;
 | 
				
			||||||
    return getAPKUrlsFromFDroidPackagesAPIResponse(
 | 
					    return getAPKUrlsFromFDroidPackagesAPIResponse(
 | 
				
			||||||
        await get(Uri.parse('https://f-droid.org/api/v1/packages/$appId')),
 | 
					        await get(Uri.parse('https://$host/api/v1/packages/$appId')),
 | 
				
			||||||
        'https://f-droid.org/repo/$appId',
 | 
					        'https://$host/repo/$appId',
 | 
				
			||||||
        standardUrl);
 | 
					        standardUrl);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -80,7 +80,8 @@ class FDroidRepo extends AppSource {
 | 
				
			|||||||
              element.querySelector('apkname') != null)
 | 
					              element.querySelector('apkname') != null)
 | 
				
			||||||
          .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}')
 | 
					          .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}')
 | 
				
			||||||
          .toList();
 | 
					          .toList();
 | 
				
			||||||
      return APKDetails(latestVersion, apkUrls, AppNames(authorName, appName),
 | 
					      return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls),
 | 
				
			||||||
 | 
					          AppNames(authorName, appName),
 | 
				
			||||||
          releaseDate: releaseDate);
 | 
					          releaseDate: releaseDate);
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -161,7 +161,9 @@ class GitHub extends AppSource {
 | 
				
			|||||||
        throw NoVersionError();
 | 
					        throw NoVersionError();
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      var changeLog = targetRelease['body'].toString();
 | 
					      var changeLog = targetRelease['body'].toString();
 | 
				
			||||||
      return APKDetails(version, targetRelease['apkUrls'] as List<String>,
 | 
					      return APKDetails(
 | 
				
			||||||
 | 
					          version,
 | 
				
			||||||
 | 
					          getApkUrlsFromUrls(targetRelease['apkUrls'] as List<String>),
 | 
				
			||||||
          getAppNames(standardUrl),
 | 
					          getAppNames(standardUrl),
 | 
				
			||||||
          releaseDate: releaseDate,
 | 
					          releaseDate: releaseDate,
 | 
				
			||||||
          changeLog: changeLog.isEmpty ? null : changeLog);
 | 
					          changeLog: changeLog.isEmpty ? null : changeLog);
 | 
				
			||||||
@@ -185,9 +187,11 @@ class GitHub extends AppSource {
 | 
				
			|||||||
      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)['items'] as List<dynamic>)) {
 | 
				
			||||||
        urlsWithDescriptions.addAll({
 | 
					        urlsWithDescriptions.addAll({
 | 
				
			||||||
          e['html_url'] as String: e['description'] != null
 | 
					          e['html_url'] as String:
 | 
				
			||||||
              ? e['description'] as String
 | 
					              ((e['archived'] == true ? '[ARCHIVED] ' : '') +
 | 
				
			||||||
              : tr('noDescription')
 | 
					                  (e['description'] != null
 | 
				
			||||||
 | 
					                      ? e['description'] as String
 | 
				
			||||||
 | 
					                      : tr('noDescription')))
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      return urlsWithDescriptions;
 | 
					      return urlsWithDescriptions;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -60,7 +60,8 @@ class GitLab extends AppSource {
 | 
				
			|||||||
      if (version == null) {
 | 
					      if (version == null) {
 | 
				
			||||||
        throw NoVersionError();
 | 
					        throw NoVersionError();
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl),
 | 
					      return APKDetails(version, getApkUrlsFromUrls(apkUrls),
 | 
				
			||||||
 | 
					          GitHub().getAppNames(standardUrl),
 | 
				
			||||||
          releaseDate: releaseDate);
 | 
					          releaseDate: releaseDate);
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -42,7 +42,8 @@ class HTML extends AppSource {
 | 
				
			|||||||
                  ? '${uri.origin}/$e'
 | 
					                  ? '${uri.origin}/$e'
 | 
				
			||||||
                  : '${uri.origin}/${uri.path}/$e')
 | 
					                  : '${uri.origin}/${uri.path}/$e')
 | 
				
			||||||
          .toList();
 | 
					          .toList();
 | 
				
			||||||
      return APKDetails(version, apkUrls, AppNames(uri.host, tr('app')));
 | 
					      return APKDetails(
 | 
				
			||||||
 | 
					          version, getApkUrlsFromUrls(apkUrls), AppNames(uri.host, tr('app')));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -58,7 +58,7 @@ class Mullvad extends AppSource {
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
      return APKDetails(
 | 
					      return APKDetails(
 | 
				
			||||||
          versions[0],
 | 
					          versions[0],
 | 
				
			||||||
          ['https://mullvad.net/download/app/apk/latest'],
 | 
					          getApkUrlsFromUrls(['https://mullvad.net/download/app/apk/latest']),
 | 
				
			||||||
          AppNames(name, 'Mullvad-VPN'),
 | 
					          AppNames(name, 'Mullvad-VPN'),
 | 
				
			||||||
          changeLog: changeLog);
 | 
					          changeLog: changeLog);
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -98,7 +98,7 @@ class NeutronCode extends AppSource {
 | 
				
			|||||||
          ? (customDateParse(dateStringOriginal))
 | 
					          ? (customDateParse(dateStringOriginal))
 | 
				
			||||||
          : null;
 | 
					          : null;
 | 
				
			||||||
      var changeLogElements = http.querySelectorAll('.pd-fdesc p');
 | 
					      var changeLogElements = http.querySelectorAll('.pd-fdesc p');
 | 
				
			||||||
      return APKDetails(version, [apkUrl],
 | 
					      return APKDetails(version, getApkUrlsFromUrls([apkUrl]),
 | 
				
			||||||
          AppNames(runtimeType.toString(), name ?? standardUrl.split('/').last),
 | 
					          AppNames(runtimeType.toString(), name ?? standardUrl.split('/').last),
 | 
				
			||||||
          releaseDate: dateString != null ? DateTime.parse(dateString) : null,
 | 
					          releaseDate: dateString != null ? DateTime.parse(dateString) : null,
 | 
				
			||||||
          changeLog: changeLogElements.isNotEmpty
 | 
					          changeLog: changeLogElements.isNotEmpty
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -28,7 +28,8 @@ class Signal extends AppSource {
 | 
				
			|||||||
      if (version == null) {
 | 
					      if (version == null) {
 | 
				
			||||||
        throw NoVersionError();
 | 
					        throw NoVersionError();
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      return APKDetails(version, apkUrls, AppNames(name, 'Signal'));
 | 
					      return APKDetails(
 | 
				
			||||||
 | 
					          version, getApkUrlsFromUrls(apkUrls), AppNames(name, 'Signal'));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -50,7 +50,7 @@ class SourceForge extends AppSource {
 | 
				
			|||||||
              .toList();
 | 
					              .toList();
 | 
				
			||||||
      return APKDetails(
 | 
					      return APKDetails(
 | 
				
			||||||
          version,
 | 
					          version,
 | 
				
			||||||
          apkUrlList,
 | 
					          getApkUrlsFromUrls(apkUrlList),
 | 
				
			||||||
          AppNames(
 | 
					          AppNames(
 | 
				
			||||||
              name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1)));
 | 
					              name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1)));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -53,7 +53,8 @@ class SteamMobile extends AppSource {
 | 
				
			|||||||
      var version = links[0].substring(
 | 
					      var version = links[0].substring(
 | 
				
			||||||
          versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4);
 | 
					          versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4);
 | 
				
			||||||
      var apkUrls = [links[0]];
 | 
					      var apkUrls = [links[0]];
 | 
				
			||||||
      return APKDetails(version, apkUrls, AppNames(name, apks[apkNamePrefix]!));
 | 
					      return APKDetails(version, getApkUrlsFromUrls(apkUrls),
 | 
				
			||||||
 | 
					          AppNames(name, apks[apkNamePrefix]!));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -32,7 +32,8 @@ class TelegramApp extends AppSource {
 | 
				
			|||||||
        throw NoVersionError();
 | 
					        throw NoVersionError();
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      String? apkUrl = 'https://telegram.org/dl/android/apk';
 | 
					      String? apkUrl = 'https://telegram.org/dl/android/apk';
 | 
				
			||||||
      return APKDetails(version, [apkUrl], AppNames('Telegram', 'Telegram'));
 | 
					      return APKDetails(version, getApkUrlsFromUrls([apkUrl]),
 | 
				
			||||||
 | 
					          AppNames('Telegram', 'Telegram'));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -54,7 +54,8 @@ class VLC extends AppSource {
 | 
				
			|||||||
        throw getObtainiumHttpError(res2);
 | 
					        throw getObtainiumHttpError(res2);
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      return APKDetails(version, apkUrls, AppNames('VideoLAN', 'VLC'));
 | 
					      return APKDetails(
 | 
				
			||||||
 | 
					          version, getApkUrlsFromUrls(apkUrls), AppNames('VideoLAN', 'VLC'));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -64,9 +64,9 @@ class WhatsApp extends AppSource {
 | 
				
			|||||||
          vLines[0].substring(versionMatch.start, versionMatch.end);
 | 
					          vLines[0].substring(versionMatch.start, versionMatch.end);
 | 
				
			||||||
      return APKDetails(
 | 
					      return APKDetails(
 | 
				
			||||||
          version,
 | 
					          version,
 | 
				
			||||||
          [
 | 
					          getApkUrlsFromUrls([
 | 
				
			||||||
            'https://www.whatsapp.com/android?v=$version&=thisIsaPlaceholder&a=realURLPrefetchedAtDownloadTime'
 | 
					            'https://www.whatsapp.com/android?v=$version&=thisIsaPlaceholder&a=realURLPrefetchedAtDownloadTime'
 | 
				
			||||||
          ],
 | 
					          ]),
 | 
				
			||||||
          AppNames('Meta', 'WhatsApp'));
 | 
					          AppNames('Meta', 'WhatsApp'));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      throw getObtainiumHttpError(res);
 | 
					      throw getObtainiumHttpError(res);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -267,7 +267,10 @@ class _GeneratedFormState extends State<GeneratedForm> {
 | 
				
			|||||||
          formInputs[r][e] = Row(
 | 
					          formInputs[r][e] = Row(
 | 
				
			||||||
            mainAxisAlignment: MainAxisAlignment.spaceBetween,
 | 
					            mainAxisAlignment: MainAxisAlignment.spaceBetween,
 | 
				
			||||||
            children: [
 | 
					            children: [
 | 
				
			||||||
              Text(widget.items[r][e].label),
 | 
					              Flexible(child: Text(widget.items[r][e].label)),
 | 
				
			||||||
 | 
					              const SizedBox(
 | 
				
			||||||
 | 
					                width: 8,
 | 
				
			||||||
 | 
					              ),
 | 
				
			||||||
              Switch(
 | 
					              Switch(
 | 
				
			||||||
                  value: values[widget.items[r][e].key],
 | 
					                  value: values[widget.items[r][e].key],
 | 
				
			||||||
                  onChanged: (value) {
 | 
					                  onChanged: (value) {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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.20';
 | 
					const String currentVersion = '0.11.26';
 | 
				
			||||||
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
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -334,11 +334,10 @@ class _AddAppPageState extends State<AddAppPage> {
 | 
				
			|||||||
          ],
 | 
					          ],
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    Widget getSourcesListWidget() => Expanded(
 | 
					    Widget getSourcesListWidget() => Column(
 | 
				
			||||||
            child: Column(
 | 
					            crossAxisAlignment: CrossAxisAlignment.center,
 | 
				
			||||||
                crossAxisAlignment: CrossAxisAlignment.center,
 | 
					            mainAxisAlignment: MainAxisAlignment.center,
 | 
				
			||||||
                mainAxisAlignment: MainAxisAlignment.center,
 | 
					            children: [
 | 
				
			||||||
                children: [
 | 
					 | 
				
			||||||
              const SizedBox(
 | 
					              const SizedBox(
 | 
				
			||||||
                height: 48,
 | 
					                height: 48,
 | 
				
			||||||
              ),
 | 
					              ),
 | 
				
			||||||
@@ -365,16 +364,17 @@ class _AddAppPageState extends State<AddAppPage> {
 | 
				
			|||||||
                            fontStyle: FontStyle.italic),
 | 
					                            fontStyle: FontStyle.italic),
 | 
				
			||||||
                      )))
 | 
					                      )))
 | 
				
			||||||
                  .toList()
 | 
					                  .toList()
 | 
				
			||||||
            ]));
 | 
					            ]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return Scaffold(
 | 
					    return Scaffold(
 | 
				
			||||||
        backgroundColor: Theme.of(context).colorScheme.surface,
 | 
					        backgroundColor: Theme.of(context).colorScheme.surface,
 | 
				
			||||||
        body: CustomScrollView(slivers: <Widget>[
 | 
					        body: CustomScrollView(shrinkWrap: true, slivers: <Widget>[
 | 
				
			||||||
          CustomAppBar(title: tr('addApp')),
 | 
					          CustomAppBar(title: tr('addApp')),
 | 
				
			||||||
          SliverFillRemaining(
 | 
					          SliverToBoxAdapter(
 | 
				
			||||||
            child: Padding(
 | 
					            child: Padding(
 | 
				
			||||||
                padding: const EdgeInsets.all(16),
 | 
					                padding: const EdgeInsets.all(16),
 | 
				
			||||||
                child: Column(
 | 
					                child: Column(
 | 
				
			||||||
 | 
					                    mainAxisSize: MainAxisSize.min,
 | 
				
			||||||
                    crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
					                    crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
				
			||||||
                    children: [
 | 
					                    children: [
 | 
				
			||||||
                      getUrlInputRow(),
 | 
					                      getUrlInputRow(),
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -61,6 +61,12 @@ class _AppPageState extends State<AppPage> {
 | 
				
			|||||||
                        mode: LaunchMode.externalApplication);
 | 
					                        mode: LaunchMode.externalApplication);
 | 
				
			||||||
                  }
 | 
					                  }
 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
 | 
					                onLongPress: () {
 | 
				
			||||||
 | 
					                  Clipboard.setData(ClipboardData(text: app?.app.url ?? ''));
 | 
				
			||||||
 | 
					                  ScaffoldMessenger.of(context).showSnackBar(SnackBar(
 | 
				
			||||||
 | 
					                    content: Text(tr('copiedToClipboard')),
 | 
				
			||||||
 | 
					                  ));
 | 
				
			||||||
 | 
					                },
 | 
				
			||||||
                child: Text(
 | 
					                child: Text(
 | 
				
			||||||
                  app?.app.url ?? '',
 | 
					                  app?.app.url ?? '',
 | 
				
			||||||
                  textAlign: TextAlign.center,
 | 
					                  textAlign: TextAlign.center,
 | 
				
			||||||
@@ -147,7 +153,7 @@ class _AppPageState extends State<AppPage> {
 | 
				
			|||||||
              height: 25,
 | 
					              height: 25,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            Text(
 | 
					            Text(
 | 
				
			||||||
              app?.installedInfo?.name ?? app?.app.name ?? tr('app'),
 | 
					              app?.app.name ?? tr('app'),
 | 
				
			||||||
              textAlign: TextAlign.center,
 | 
					              textAlign: TextAlign.center,
 | 
				
			||||||
              style: Theme.of(context).textTheme.displayLarge,
 | 
					              style: Theme.of(context).textTheme.displayLarge,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -56,6 +56,7 @@ class AppsPageState extends State<AppsPage> {
 | 
				
			|||||||
  Widget build(BuildContext context) {
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
    var appsProvider = context.watch<AppsProvider>();
 | 
					    var appsProvider = context.watch<AppsProvider>();
 | 
				
			||||||
    var settingsProvider = context.watch<SettingsProvider>();
 | 
					    var settingsProvider = context.watch<SettingsProvider>();
 | 
				
			||||||
 | 
					    var sourceProvider = SourceProvider();
 | 
				
			||||||
    var listedApps = appsProvider.apps.values.toList();
 | 
					    var listedApps = appsProvider.apps.values.toList();
 | 
				
			||||||
    var currentFilterIsUpdatesOnly =
 | 
					    var currentFilterIsUpdatesOnly =
 | 
				
			||||||
        filter.isIdenticalTo(updatesOnlyFilter, settingsProvider);
 | 
					        filter.isIdenticalTo(updatesOnlyFilter, settingsProvider);
 | 
				
			||||||
@@ -110,6 +111,11 @@ class AppsPageState extends State<AppsPage> {
 | 
				
			|||||||
              .isEmpty) {
 | 
					              .isEmpty) {
 | 
				
			||||||
        return false;
 | 
					        return false;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					      if (filter.sourceFilter.isNotEmpty &&
 | 
				
			||||||
 | 
					          sourceProvider.getSource(app.app.url).runtimeType.toString() !=
 | 
				
			||||||
 | 
					              filter.sourceFilter) {
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
      return true;
 | 
					      return true;
 | 
				
			||||||
    }).toList();
 | 
					    }).toList();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -892,6 +898,19 @@ class AppsPageState extends State<AppsPage> {
 | 
				
			|||||||
                  GeneratedFormSwitch('nonInstalledApps',
 | 
					                  GeneratedFormSwitch('nonInstalledApps',
 | 
				
			||||||
                      label: tr('nonInstalledApps'),
 | 
					                      label: tr('nonInstalledApps'),
 | 
				
			||||||
                      defaultValue: vals['nonInstalledApps'])
 | 
					                      defaultValue: vals['nonInstalledApps'])
 | 
				
			||||||
 | 
					                ],
 | 
				
			||||||
 | 
					                [
 | 
				
			||||||
 | 
					                  GeneratedFormDropdown(
 | 
				
			||||||
 | 
					                      'sourceFilter',
 | 
				
			||||||
 | 
					                      label: tr('appSource'),
 | 
				
			||||||
 | 
					                      defaultValue: filter.sourceFilter,
 | 
				
			||||||
 | 
					                      [
 | 
				
			||||||
 | 
					                        MapEntry('', tr('none')),
 | 
				
			||||||
 | 
					                        ...sourceProvider.sources
 | 
				
			||||||
 | 
					                            .map((e) =>
 | 
				
			||||||
 | 
					                                MapEntry(e.runtimeType.toString(), e.name))
 | 
				
			||||||
 | 
					                            .toList()
 | 
				
			||||||
 | 
					                      ])
 | 
				
			||||||
                ]
 | 
					                ]
 | 
				
			||||||
              ],
 | 
					              ],
 | 
				
			||||||
              additionalWidgets: [
 | 
					              additionalWidgets: [
 | 
				
			||||||
@@ -1015,20 +1034,23 @@ class AppsFilter {
 | 
				
			|||||||
  late bool includeUptodate;
 | 
					  late bool includeUptodate;
 | 
				
			||||||
  late bool includeNonInstalled;
 | 
					  late bool includeNonInstalled;
 | 
				
			||||||
  late Set<String> categoryFilter;
 | 
					  late Set<String> categoryFilter;
 | 
				
			||||||
 | 
					  late String sourceFilter;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  AppsFilter(
 | 
					  AppsFilter(
 | 
				
			||||||
      {this.nameFilter = '',
 | 
					      {this.nameFilter = '',
 | 
				
			||||||
      this.authorFilter = '',
 | 
					      this.authorFilter = '',
 | 
				
			||||||
      this.includeUptodate = true,
 | 
					      this.includeUptodate = true,
 | 
				
			||||||
      this.includeNonInstalled = true,
 | 
					      this.includeNonInstalled = true,
 | 
				
			||||||
      this.categoryFilter = const {}});
 | 
					      this.categoryFilter = const {},
 | 
				
			||||||
 | 
					      this.sourceFilter = ''});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Map<String, dynamic> toFormValuesMap() {
 | 
					  Map<String, dynamic> toFormValuesMap() {
 | 
				
			||||||
    return {
 | 
					    return {
 | 
				
			||||||
      'appName': nameFilter,
 | 
					      'appName': nameFilter,
 | 
				
			||||||
      'author': authorFilter,
 | 
					      'author': authorFilter,
 | 
				
			||||||
      'upToDateApps': includeUptodate,
 | 
					      'upToDateApps': includeUptodate,
 | 
				
			||||||
      'nonInstalledApps': includeNonInstalled
 | 
					      'nonInstalledApps': includeNonInstalled,
 | 
				
			||||||
 | 
					      'sourceFilter': sourceFilter
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1037,6 +1059,7 @@ class AppsFilter {
 | 
				
			|||||||
    authorFilter = values['author']!;
 | 
					    authorFilter = values['author']!;
 | 
				
			||||||
    includeUptodate = values['upToDateApps'];
 | 
					    includeUptodate = values['upToDateApps'];
 | 
				
			||||||
    includeNonInstalled = values['nonInstalledApps'];
 | 
					    includeNonInstalled = values['nonInstalledApps'];
 | 
				
			||||||
 | 
					    sourceFilter = values['sourceFilter'];
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  bool isIdenticalTo(AppsFilter other, SettingsProvider settingsProvider) =>
 | 
					  bool isIdenticalTo(AppsFilter other, SettingsProvider settingsProvider) =>
 | 
				
			||||||
@@ -1044,5 +1067,6 @@ class AppsFilter {
 | 
				
			|||||||
      nameFilter.trim() == other.nameFilter.trim() &&
 | 
					      nameFilter.trim() == other.nameFilter.trim() &&
 | 
				
			||||||
      includeUptodate == other.includeUptodate &&
 | 
					      includeUptodate == other.includeUptodate &&
 | 
				
			||||||
      includeNonInstalled == other.includeNonInstalled &&
 | 
					      includeNonInstalled == other.includeNonInstalled &&
 | 
				
			||||||
      settingsProvider.setEqual(categoryFilter, other.categoryFilter);
 | 
					      settingsProvider.setEqual(categoryFilter, other.categoryFilter) &&
 | 
				
			||||||
 | 
					      sourceFilter.trim() == other.sourceFilter.trim();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -169,7 +169,7 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
          '${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk';
 | 
					          '${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk';
 | 
				
			||||||
      String downloadUrl = await SourceProvider()
 | 
					      String downloadUrl = await SourceProvider()
 | 
				
			||||||
          .getSource(app.url)
 | 
					          .getSource(app.url)
 | 
				
			||||||
          .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex]);
 | 
					          .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value);
 | 
				
			||||||
      var notif = DownloadNotification(app.name, 100);
 | 
					      var notif = DownloadNotification(app.name, 100);
 | 
				
			||||||
      notificationsProvider?.cancel(notif.id);
 | 
					      notificationsProvider?.cancel(notif.id);
 | 
				
			||||||
      int? prevProg;
 | 
					      int? prevProg;
 | 
				
			||||||
@@ -296,9 +296,10 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
    await intent.launch();
 | 
					    await intent.launch();
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Future<String?> confirmApkUrl(App app, BuildContext? context) async {
 | 
					  Future<MapEntry<String, String>?> confirmApkUrl(
 | 
				
			||||||
 | 
					      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)
 | 
				
			||||||
    String? apkUrl = app.apkUrls[app.preferredApkIndex];
 | 
					    MapEntry<String, String>? apkUrl = app.apkUrls[app.preferredApkIndex];
 | 
				
			||||||
    // get device supported architecture
 | 
					    // get device supported architecture
 | 
				
			||||||
    List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis;
 | 
					    List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -321,14 +322,14 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    // If the picked APK comes from an origin different from the source, get user confirmation (if context provided)
 | 
					    // If the picked APK comes from an origin different from the source, get user confirmation (if context provided)
 | 
				
			||||||
    if (apkUrl != null &&
 | 
					    if (apkUrl != null &&
 | 
				
			||||||
        getHost(apkUrl) != getHost(app.url) &&
 | 
					        getHost(apkUrl.value) != getHost(app.url) &&
 | 
				
			||||||
        context != null) {
 | 
					        context != null) {
 | 
				
			||||||
      // ignore: use_build_context_synchronously
 | 
					      // ignore: use_build_context_synchronously
 | 
				
			||||||
      if (await showDialog(
 | 
					      if (await showDialog(
 | 
				
			||||||
              context: context,
 | 
					              context: context,
 | 
				
			||||||
              builder: (BuildContext ctx) {
 | 
					              builder: (BuildContext ctx) {
 | 
				
			||||||
                return APKOriginWarningDialog(
 | 
					                return APKOriginWarningDialog(
 | 
				
			||||||
                    sourceUrl: app.url, apkUrl: apkUrl!);
 | 
					                    sourceUrl: app.url, apkUrl: apkUrl!.value);
 | 
				
			||||||
              }) !=
 | 
					              }) !=
 | 
				
			||||||
          true) {
 | 
					          true) {
 | 
				
			||||||
        apkUrl = null;
 | 
					        apkUrl = null;
 | 
				
			||||||
@@ -353,7 +354,7 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
      if (apps[id] == null) {
 | 
					      if (apps[id] == null) {
 | 
				
			||||||
        throw ObtainiumError(tr('appNotFound'));
 | 
					        throw ObtainiumError(tr('appNotFound'));
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      String? apkUrl;
 | 
					      MapEntry<String, String>? apkUrl;
 | 
				
			||||||
      var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true;
 | 
					      var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true;
 | 
				
			||||||
      if (!trackOnly) {
 | 
					      if (!trackOnly) {
 | 
				
			||||||
        apkUrl = await confirmApkUrl(apps[id]!.app, context);
 | 
					        apkUrl = await confirmApkUrl(apps[id]!.app, context);
 | 
				
			||||||
@@ -670,6 +671,9 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
    for (var app in apps) {
 | 
					    for (var app in apps) {
 | 
				
			||||||
      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;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
@@ -848,12 +852,6 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Future<String> exportApps() async {
 | 
					  Future<String> exportApps() async {
 | 
				
			||||||
    Directory? exportDir = Directory('/storage/emulated/0/Download');
 | 
					 | 
				
			||||||
    String path = 'Downloads'; // TODO: See if hardcoding this can be avoided
 | 
					 | 
				
			||||||
    if (!exportDir.existsSync()) {
 | 
					 | 
				
			||||||
      exportDir = await getExternalStorageDirectory();
 | 
					 | 
				
			||||||
      path = exportDir!.path;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if ((await DeviceInfoPlugin().androidInfo).version.sdkInt <= 29) {
 | 
					    if ((await DeviceInfoPlugin().androidInfo).version.sdkInt <= 29) {
 | 
				
			||||||
      if (await Permission.storage.isDenied) {
 | 
					      if (await Permission.storage.isDenied) {
 | 
				
			||||||
        await Permission.storage.request();
 | 
					        await Permission.storage.request();
 | 
				
			||||||
@@ -862,6 +860,18 @@ class AppsProvider with ChangeNotifier {
 | 
				
			|||||||
        throw ObtainiumError(tr('storagePermissionDenied'));
 | 
					        throw ObtainiumError(tr('storagePermissionDenied'));
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    Directory? exportDir = Directory('/storage/emulated/0/Download');
 | 
				
			||||||
 | 
					    String path = 'Downloads'; // TODO: See if hardcoding this can be avoided
 | 
				
			||||||
 | 
					    var downloadsAccessible = false;
 | 
				
			||||||
 | 
					    try {
 | 
				
			||||||
 | 
					      downloadsAccessible = exportDir.existsSync();
 | 
				
			||||||
 | 
					    } catch (e) {
 | 
				
			||||||
 | 
					      logs.add('Error accessing Downloads (will use fallback): $e');
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (!downloadsAccessible) {
 | 
				
			||||||
 | 
					      exportDir = await getExternalStorageDirectory();
 | 
				
			||||||
 | 
					      path = exportDir!.path;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    File export = File(
 | 
					    File export = File(
 | 
				
			||||||
        '${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json');
 | 
					        '${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json');
 | 
				
			||||||
    export.writeAsStringSync(
 | 
					    export.writeAsStringSync(
 | 
				
			||||||
@@ -914,7 +924,7 @@ class APKPicker extends StatefulWidget {
 | 
				
			|||||||
  const APKPicker({super.key, required this.app, this.initVal, this.archs});
 | 
					  const APKPicker({super.key, required this.app, this.initVal, this.archs});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  final App app;
 | 
					  final App app;
 | 
				
			||||||
  final String? initVal;
 | 
					  final MapEntry<String, String>? initVal;
 | 
				
			||||||
  final List<String>? archs;
 | 
					  final List<String>? archs;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
@@ -922,7 +932,7 @@ class APKPicker extends StatefulWidget {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class _APKPickerState extends State<APKPicker> {
 | 
					class _APKPickerState extends State<APKPicker> {
 | 
				
			||||||
  String? apkUrl;
 | 
					  MapEntry<String, String>? apkUrl;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  Widget build(BuildContext context) {
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
@@ -935,15 +945,13 @@ class _APKPickerState extends State<APKPicker> {
 | 
				
			|||||||
        const SizedBox(height: 16),
 | 
					        const SizedBox(height: 16),
 | 
				
			||||||
        ...widget.app.apkUrls.map(
 | 
					        ...widget.app.apkUrls.map(
 | 
				
			||||||
          (u) => RadioListTile<String>(
 | 
					          (u) => RadioListTile<String>(
 | 
				
			||||||
              title: Text(Uri.parse(u)
 | 
					              title: Text(u.key),
 | 
				
			||||||
                  .pathSegments
 | 
					              value: u.value,
 | 
				
			||||||
                  .where((element) => element.isNotEmpty)
 | 
					              groupValue: apkUrl!.value,
 | 
				
			||||||
                  .last),
 | 
					 | 
				
			||||||
              value: u,
 | 
					 | 
				
			||||||
              groupValue: apkUrl,
 | 
					 | 
				
			||||||
              onChanged: (String? val) {
 | 
					              onChanged: (String? val) {
 | 
				
			||||||
                setState(() {
 | 
					                setState(() {
 | 
				
			||||||
                  apkUrl = val;
 | 
					                  apkUrl =
 | 
				
			||||||
 | 
					                      widget.app.apkUrls.where((e) => e.value == val).first;
 | 
				
			||||||
                });
 | 
					                });
 | 
				
			||||||
              }),
 | 
					              }),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,6 +3,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import 'dart:convert';
 | 
					import 'dart:convert';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import 'package:device_info_plus/device_info_plus.dart';
 | 
				
			||||||
import 'package:easy_localization/easy_localization.dart';
 | 
					import 'package:easy_localization/easy_localization.dart';
 | 
				
			||||||
import 'package:html/dom.dart';
 | 
					import 'package:html/dom.dart';
 | 
				
			||||||
import 'package:http/http.dart';
 | 
					import 'package:http/http.dart';
 | 
				
			||||||
@@ -34,7 +35,7 @@ class AppNames {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
class APKDetails {
 | 
					class APKDetails {
 | 
				
			||||||
  late String version;
 | 
					  late String version;
 | 
				
			||||||
  late List<String> apkUrls;
 | 
					  late List<MapEntry<String, String>> apkUrls;
 | 
				
			||||||
  late AppNames names;
 | 
					  late AppNames names;
 | 
				
			||||||
  late DateTime? releaseDate;
 | 
					  late DateTime? releaseDate;
 | 
				
			||||||
  late String? changeLog;
 | 
					  late String? changeLog;
 | 
				
			||||||
@@ -50,7 +51,7 @@ class App {
 | 
				
			|||||||
  late String name;
 | 
					  late String name;
 | 
				
			||||||
  String? installedVersion;
 | 
					  String? installedVersion;
 | 
				
			||||||
  late String latestVersion;
 | 
					  late String latestVersion;
 | 
				
			||||||
  List<String> apkUrls = [];
 | 
					  List<MapEntry<String, String>> apkUrls = [];
 | 
				
			||||||
  late int preferredApkIndex;
 | 
					  late int preferredApkIndex;
 | 
				
			||||||
  late Map<String, dynamic> additionalSettings;
 | 
					  late Map<String, dynamic> additionalSettings;
 | 
				
			||||||
  late DateTime? lastUpdateCheck;
 | 
					  late DateTime? lastUpdateCheck;
 | 
				
			||||||
@@ -134,6 +135,23 @@ class App {
 | 
				
			|||||||
    if (preferredApkIndex < 0) {
 | 
					    if (preferredApkIndex < 0) {
 | 
				
			||||||
      preferredApkIndex = 0;
 | 
					      preferredApkIndex = 0;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    // apkUrls can either be old list or new named list apkUrls
 | 
				
			||||||
 | 
					    List<MapEntry<String, String>> apkUrls = [];
 | 
				
			||||||
 | 
					    if (json['apkUrls'] != null) {
 | 
				
			||||||
 | 
					      var apkUrlJson = jsonDecode(json['apkUrls']);
 | 
				
			||||||
 | 
					      try {
 | 
				
			||||||
 | 
					        apkUrls = getApkUrlsFromUrls(List<String>.from(apkUrlJson));
 | 
				
			||||||
 | 
					      } catch (e) {
 | 
				
			||||||
 | 
					        apkUrls = List<dynamic>.from(apkUrlJson)
 | 
				
			||||||
 | 
					            .map((e) => MapEntry(e[0] as String, e[1] as String))
 | 
				
			||||||
 | 
					            .toList();
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    // Arch based APK filter option should be disabled if it previously did not exist
 | 
				
			||||||
 | 
					    if (json['additionalSettings'] != null &&
 | 
				
			||||||
 | 
					        jsonDecode(json['additionalSettings'])['autoApkFilterByArch'] == null) {
 | 
				
			||||||
 | 
					      additionalSettings['autoApkFilterByArch'] = false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    return App(
 | 
					    return App(
 | 
				
			||||||
        json['id'] as String,
 | 
					        json['id'] as String,
 | 
				
			||||||
        json['url'] as String,
 | 
					        json['url'] as String,
 | 
				
			||||||
@@ -143,9 +161,7 @@ class App {
 | 
				
			|||||||
            ? null
 | 
					            ? null
 | 
				
			||||||
            : json['installedVersion'] as String,
 | 
					            : json['installedVersion'] as String,
 | 
				
			||||||
        json['latestVersion'] as String,
 | 
					        json['latestVersion'] as String,
 | 
				
			||||||
        json['apkUrls'] == null
 | 
					        apkUrls,
 | 
				
			||||||
            ? []
 | 
					 | 
				
			||||||
            : List<String>.from(jsonDecode(json['apkUrls'])),
 | 
					 | 
				
			||||||
        preferredApkIndex,
 | 
					        preferredApkIndex,
 | 
				
			||||||
        additionalSettings,
 | 
					        additionalSettings,
 | 
				
			||||||
        json['lastUpdateCheck'] == null
 | 
					        json['lastUpdateCheck'] == null
 | 
				
			||||||
@@ -173,7 +189,7 @@ class App {
 | 
				
			|||||||
        'name': name,
 | 
					        'name': name,
 | 
				
			||||||
        'installedVersion': installedVersion,
 | 
					        'installedVersion': installedVersion,
 | 
				
			||||||
        'latestVersion': latestVersion,
 | 
					        'latestVersion': latestVersion,
 | 
				
			||||||
        'apkUrls': jsonEncode(apkUrls),
 | 
					        'apkUrls': jsonEncode(apkUrls.map((e) => [e.key, e.value]).toList()),
 | 
				
			||||||
        'preferredApkIndex': preferredApkIndex,
 | 
					        'preferredApkIndex': preferredApkIndex,
 | 
				
			||||||
        'additionalSettings': jsonEncode(additionalSettings),
 | 
					        'additionalSettings': jsonEncode(additionalSettings),
 | 
				
			||||||
        'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch,
 | 
					        'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch,
 | 
				
			||||||
@@ -225,6 +241,11 @@ Map<String, dynamic> getDefaultValuesFromFormItems(
 | 
				
			|||||||
      .reduce((value, element) => [...value, ...element]));
 | 
					      .reduce((value, element) => [...value, ...element]));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					getApkUrlsFromUrls(List<String> urls) => urls
 | 
				
			||||||
 | 
					    .map((e) =>
 | 
				
			||||||
 | 
					        MapEntry(e.split('/').where((el) => el.trim().isNotEmpty).last, e))
 | 
				
			||||||
 | 
					    .toList();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class AppSource {
 | 
					class AppSource {
 | 
				
			||||||
  String? host;
 | 
					  String? host;
 | 
				
			||||||
  late String name;
 | 
					  late String name;
 | 
				
			||||||
@@ -278,7 +299,12 @@ class AppSource {
 | 
				
			|||||||
              return regExValidator(value);
 | 
					              return regExValidator(value);
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
          ])
 | 
					          ])
 | 
				
			||||||
    ]
 | 
					    ],
 | 
				
			||||||
 | 
					    [
 | 
				
			||||||
 | 
					      GeneratedFormSwitch('autoApkFilterByArch',
 | 
				
			||||||
 | 
					          label: tr('autoApkFilterByArch'), defaultValue: true)
 | 
				
			||||||
 | 
					    ],
 | 
				
			||||||
 | 
					    [GeneratedFormTextField('appName', label: tr('appName'), required: false)]
 | 
				
			||||||
  ];
 | 
					  ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // Previous 2 variables combined into one at runtime for convenient usage
 | 
					  // Previous 2 variables combined into one at runtime for convenient usage
 | 
				
			||||||
@@ -362,7 +388,7 @@ class SourceProvider {
 | 
				
			|||||||
    url = preStandardizeUrl(url);
 | 
					    url = preStandardizeUrl(url);
 | 
				
			||||||
    AppSource? source;
 | 
					    AppSource? source;
 | 
				
			||||||
    for (var s in sources.where((element) => element.host != null)) {
 | 
					    for (var s in sources.where((element) => element.host != null)) {
 | 
				
			||||||
      if (url.contains('://${s.host}')) {
 | 
					      if (RegExp('://(.+\\.)?${s.host}').hasMatch(url)) {
 | 
				
			||||||
        source = s;
 | 
					        source = s;
 | 
				
			||||||
        break;
 | 
					        break;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
@@ -421,14 +447,29 @@ class SourceProvider {
 | 
				
			|||||||
    if (additionalSettings['apkFilterRegEx'] != null) {
 | 
					    if (additionalSettings['apkFilterRegEx'] != null) {
 | 
				
			||||||
      var reg = RegExp(additionalSettings['apkFilterRegEx']);
 | 
					      var reg = RegExp(additionalSettings['apkFilterRegEx']);
 | 
				
			||||||
      apk.apkUrls =
 | 
					      apk.apkUrls =
 | 
				
			||||||
          apk.apkUrls.where((element) => reg.hasMatch(element)).toList();
 | 
					          apk.apkUrls.where((element) => reg.hasMatch(element.key)).toList();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (apk.apkUrls.isEmpty && !trackOnly) {
 | 
					    if (apk.apkUrls.isEmpty && !trackOnly) {
 | 
				
			||||||
      throw NoAPKError();
 | 
					      throw NoAPKError();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    if (apk.apkUrls.length > 1 &&
 | 
				
			||||||
 | 
					        additionalSettings['autoApkFilterByArch'] == true) {
 | 
				
			||||||
 | 
					      var abis = (await DeviceInfoPlugin().androidInfo).supportedAbis;
 | 
				
			||||||
 | 
					      for (var abi in abis) {
 | 
				
			||||||
 | 
					        var urls2 = apk.apkUrls
 | 
				
			||||||
 | 
					            .where((element) => RegExp('.*$abi.*').hasMatch(element.key))
 | 
				
			||||||
 | 
					            .toList();
 | 
				
			||||||
 | 
					        if (urls2.isNotEmpty && urls2.length < apk.apkUrls.length) {
 | 
				
			||||||
 | 
					          apk.apkUrls = urls2;
 | 
				
			||||||
 | 
					          break;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    String apkVersion = apk.version.replaceAll('/', '-');
 | 
					    String apkVersion = apk.version.replaceAll('/', '-');
 | 
				
			||||||
    var name = currentApp?.name.trim() ??
 | 
					    var name = currentApp != null ? currentApp.name.trim() : '';
 | 
				
			||||||
        apk.names.name[0].toUpperCase() + apk.names.name.substring(1);
 | 
					    name = name.isNotEmpty
 | 
				
			||||||
 | 
					        ? name
 | 
				
			||||||
 | 
					        : apk.names.name[0].toUpperCase() + apk.names.name.substring(1);
 | 
				
			||||||
    return App(
 | 
					    return App(
 | 
				
			||||||
        currentApp?.id ??
 | 
					        currentApp?.id ??
 | 
				
			||||||
            source.tryInferringAppId(standardUrl,
 | 
					            source.tryInferringAppId(standardUrl,
 | 
				
			||||||
@@ -436,9 +477,7 @@ class SourceProvider {
 | 
				
			|||||||
            generateTempID(standardUrl, additionalSettings),
 | 
					            generateTempID(standardUrl, additionalSettings),
 | 
				
			||||||
        standardUrl,
 | 
					        standardUrl,
 | 
				
			||||||
        apk.names.author[0].toUpperCase() + apk.names.author.substring(1),
 | 
					        apk.names.author[0].toUpperCase() + apk.names.author.substring(1),
 | 
				
			||||||
        name.trim().isNotEmpty
 | 
					        name,
 | 
				
			||||||
            ? name
 | 
					 | 
				
			||||||
            : apk.names.name[0].toUpperCase() + apk.names.name.substring(1),
 | 
					 | 
				
			||||||
        currentApp?.installedVersion,
 | 
					        currentApp?.installedVersion,
 | 
				
			||||||
        apkVersion,
 | 
					        apkVersion,
 | 
				
			||||||
        apk.apkUrls,
 | 
					        apk.apkUrls,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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.20+142 # When changing this, update the tag in main() accordingly
 | 
					version: 0.11.26+148 # 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'
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user