mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-30 21:13:28 +01:00 
			
		
		
		
	Compare commits
	
		
			20 Commits
		
	
	
		
			v0.11.27-b
			...
			v0.11.30-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 537628f378 | ||
|  | c92d76df98 | ||
|  | b6959e1a8b | ||
|  | 1bf648da60 | ||
|  | 6a1275e9e4 | ||
|  | df242b91ad | ||
|  | 7ea75325bb | ||
|  | 0704dfe2ee | ||
|  | 6275cbf114 | ||
|  | 36b8ef6782 | ||
|  | d274b9a428 | ||
|  | 1c2980d1ac | ||
|  | 8f0aac057e | ||
|  | e929920a48 | ||
|  | 8ed254c7dd | ||
|  | 46a00836df | ||
|  | f144ffdded | ||
|  | d597d569e2 | ||
|  | b62475de87 | ||
|  | 334ac8d3d6 | 
| @@ -221,8 +221,8 @@ | ||||
|     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", | ||||
|     "versionDetection": "Versionserkennung", | ||||
|     "standardVersionDetection": "Standardversionserkennung", | ||||
|     "groupByCategory": "Group by Category", | ||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||
|     "groupByCategory": "Nach Kategorie gruppieren", | ||||
|     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App entfernen?", | ||||
|         "other": "App entfernen?" | ||||
|   | ||||
| @@ -207,7 +207,7 @@ | ||||
|     "addCategory": "اضافه کردن دسته", | ||||
|     "label": "برچسب", | ||||
|     "language": "زبان", | ||||
|     "copiedToClipboard": "Copied to Clipboard", | ||||
|     "copiedToClipboard": "در کلیپ بورد کپی شد", | ||||
|     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", | ||||
|     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", | ||||
|     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", | ||||
| @@ -221,8 +221,8 @@ | ||||
|     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", | ||||
|     "versionDetection": "تشخیص نسخه", | ||||
|     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||
|     "groupByCategory": "Group by Category", | ||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||
|     "groupByCategory": "گروه بر اساس دسته", | ||||
|     "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "برنامه حذف شود؟", | ||||
|         "other": "برنامه ها حذف شوند؟" | ||||
|   | ||||
| @@ -221,7 +221,7 @@ | ||||
|     "versionDetection": "Verzió érzékelés", | ||||
|     "standardVersionDetection": "Alapért. verzió érzékelés", | ||||
|     "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": { | ||||
|         "one": "Eltávolítja az alkalmazást?", | ||||
|         "other": "Eltávolítja az alkalmazást?" | ||||
|   | ||||
| @@ -207,7 +207,7 @@ | ||||
|     "addCategory": "Aggiungi categoria", | ||||
|     "label": "Etichetta", | ||||
|     "language": "Lingua", | ||||
|     "copiedToClipboard": "Copied to Clipboard", | ||||
|     "copiedToClipboard": "Copiato negli appunti", | ||||
|     "storagePermissionDenied": "Accesso ai file non autorizzato", | ||||
|     "selectedCategorizeWarning": "Ciò sostituirà le impostazioni di categoria esistenti per le App selezionate.", | ||||
|     "filterAPKsByRegEx": "Filtra file APK con espressioni regolari", | ||||
| @@ -221,8 +221,8 @@ | ||||
|     "importFromURLsInFile": "Importa da URL in file (come OPML)", | ||||
|     "versionDetection": "Rilevamento di versione", | ||||
|     "standardVersionDetection": "Rilevamento di versione standard", | ||||
|     "groupByCategory": "Group by Category", | ||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||
|     "groupByCategory": "Raggruppa per categoria", | ||||
|     "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Rimuovere l'App?", | ||||
|         "other": "Rimuovere le App?" | ||||
|   | ||||
| @@ -221,8 +221,8 @@ | ||||
|     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", | ||||
|     "versionDetection": "バージョン検出", | ||||
|     "standardVersionDetection": "標準のバージョン検出", | ||||
|     "groupByCategory": "Group by Category", | ||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||
|     "groupByCategory": "カテゴリ別にグループ化する", | ||||
|     "autoApkFilterByArch": "可能であれば,CPUアーキテクチャによるAPKのフィルタリングを試みる", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "アプリを削除しますか?", | ||||
|         "other": "アプリを削除しますか?" | ||||
|   | ||||
| @@ -81,10 +81,11 @@ class Codeberg extends AppSource { | ||||
|           []; | ||||
|  | ||||
|       dynamic targetRelease; | ||||
|  | ||||
|       var prerrelsSkipped = 0; | ||||
|       for (int i = 0; i < releases.length; i++) { | ||||
|         if (!fallbackToOlderReleases && i > 0) break; | ||||
|         if (!fallbackToOlderReleases && i > prerrelsSkipped) break; | ||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { | ||||
|           prerrelsSkipped++; | ||||
|           continue; | ||||
|         } | ||||
|         if (releases[i]['draft'] == true) { | ||||
|   | ||||
| @@ -127,10 +127,11 @@ class GitHub extends AppSource { | ||||
|           []; | ||||
|  | ||||
|       dynamic targetRelease; | ||||
|  | ||||
|       var prerrelsSkipped = 0; | ||||
|       for (int i = 0; i < releases.length; i++) { | ||||
|         if (!fallbackToOlderReleases && i > 0) break; | ||||
|         if (!fallbackToOlderReleases && i > prerrelsSkipped) break; | ||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { | ||||
|           prerrelsSkipped++; | ||||
|           continue; | ||||
|         } | ||||
|         var nameToFilter = releases[i]['name'] as String?; | ||||
|   | ||||
| @@ -34,14 +34,20 @@ class HTML extends AppSource { | ||||
|       var rel = links.last; | ||||
|       var apkName = rel.split('/').last; | ||||
|       var version = apkName.substring(0, apkName.length - 4); | ||||
|       List<String> apkUrls = [rel] | ||||
|           .map((e) => e.toLowerCase().startsWith('http://') || | ||||
|                   e.toLowerCase().startsWith('https://') | ||||
|               ? e | ||||
|               : e.startsWith('/') | ||||
|                   ? '${uri.origin}/$e' | ||||
|                   : '${uri.origin}/${uri.path}/$e') | ||||
|           .toList(); | ||||
|       List<String> apkUrls = [rel].map((e) { | ||||
|         try { | ||||
|           Uri.parse(e).origin; | ||||
|           return e; | ||||
|         } catch (err) { | ||||
|           // is relative | ||||
|         } | ||||
|         var currPathSegments = uri.path.split('/'); | ||||
|         if (e.startsWith('/') || currPathSegments.isEmpty) { | ||||
|           return '${uri.origin}/$e'; | ||||
|         } else { | ||||
|           return '${uri.origin}/${currPathSegments.sublist(0, currPathSegments.length - 1).join('/')}/$e'; | ||||
|         } | ||||
|       }).toList(); | ||||
|       return APKDetails( | ||||
|           version, getApkUrlsFromUrls(apkUrls), AppNames(uri.host, tr('app'))); | ||||
|     } else { | ||||
|   | ||||
| @@ -21,7 +21,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart'; | ||||
| // ignore: implementation_imports | ||||
| import 'package:easy_localization/src/localization.dart'; | ||||
|  | ||||
| const String currentVersion = '0.11.27'; | ||||
| const String currentVersion = '0.11.30'; | ||||
| const String currentReleaseTag = | ||||
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES | ||||
|  | ||||
|   | ||||
| @@ -38,7 +38,7 @@ class _AppPageState extends State<AppPage> { | ||||
|     bool areDownloadsRunning = appsProvider.areDownloadsRunning(); | ||||
|  | ||||
|     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; | ||||
|     if (!areDownloadsRunning && prevApp == null && app != null) { | ||||
|       prevApp = app; | ||||
| @@ -153,7 +153,7 @@ class _AppPageState extends State<AppPage> { | ||||
|               height: 25, | ||||
|             ), | ||||
|             Text( | ||||
|               app?.app.name ?? tr('app'), | ||||
|               app?.name ?? tr('app'), | ||||
|               textAlign: TextAlign.center, | ||||
|               style: Theme.of(context).textTheme.displayLarge, | ||||
|             ), | ||||
| @@ -386,7 +386,7 @@ class _AppPageState extends State<AppPage> { | ||||
|                                       scrollable: true, | ||||
|                                       content: getInfoColumn(), | ||||
|                                       title: Text( | ||||
|                                           '${app.app.name} ${tr('byX', args: [ | ||||
|                                           '${app.name} ${tr('byX', args: [ | ||||
|                                             app.app.author | ||||
|                                           ])}'), | ||||
|                                       actions: [ | ||||
|   | ||||
| @@ -29,13 +29,13 @@ class AppsPageState extends State<AppsPage> { | ||||
|   final AppsFilter neutralFilter = AppsFilter(); | ||||
|   var updatesOnlyFilter = | ||||
|       AppsFilter(includeUptodate: false, includeNonInstalled: false); | ||||
|   Set<App> selectedApps = {}; | ||||
|   Set<String> selectedAppIds = {}; | ||||
|   DateTime? refreshingSince; | ||||
|  | ||||
|   clearSelected() { | ||||
|     if (selectedApps.isNotEmpty) { | ||||
|     if (selectedAppIds.isNotEmpty) { | ||||
|       setState(() { | ||||
|         selectedApps.clear(); | ||||
|         selectedAppIds.clear(); | ||||
|       }); | ||||
|       return true; | ||||
|     } | ||||
| @@ -43,10 +43,10 @@ class AppsPageState extends State<AppsPage> { | ||||
|   } | ||||
|  | ||||
|   selectThese(List<App> apps) { | ||||
|     if (selectedApps.isEmpty) { | ||||
|     if (selectedAppIds.isEmpty) { | ||||
|       setState(() { | ||||
|         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 settingsProvider = context.watch<SettingsProvider>(); | ||||
|     var sourceProvider = SourceProvider(); | ||||
|     var listedApps = appsProvider.apps.values.toList(); | ||||
|     var listedApps = appsProvider.getAppValues().toList(); | ||||
|     var currentFilterIsUpdatesOnly = | ||||
|         filter.isIdenticalTo(updatesOnlyFilter, settingsProvider); | ||||
|  | ||||
|     selectedApps = selectedApps | ||||
|         .where((element) => listedApps.map((e) => e.app).contains(element)) | ||||
|     selectedAppIds = selectedAppIds | ||||
|         .where((element) => listedApps.map((e) => e.app.id).contains(element)) | ||||
|         .toSet(); | ||||
|  | ||||
|     toggleAppSelected(App app) { | ||||
|       setState(() { | ||||
|         if (selectedApps.contains(app)) { | ||||
|           selectedApps.remove(app); | ||||
|         if (selectedAppIds.map((e) => e).contains(app.id)) { | ||||
|           selectedAppIds.removeWhere((a) => a == app.id); | ||||
|         } else { | ||||
|           selectedApps.add(app); | ||||
|           selectedAppIds.add(app.id); | ||||
|         } | ||||
|       }); | ||||
|     } | ||||
| @@ -94,8 +94,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|             .toList(); | ||||
|  | ||||
|         for (var t in nameTokens) { | ||||
|           var name = app.installedInfo?.name ?? app.app.name; | ||||
|           if (!name.toLowerCase().contains(t.toLowerCase())) { | ||||
|           if (!app.name.toLowerCase().contains(t.toLowerCase())) { | ||||
|             return false; | ||||
|           } | ||||
|         } | ||||
| @@ -120,13 +119,13 @@ class AppsPageState extends State<AppsPage> { | ||||
|     }).toList(); | ||||
|  | ||||
|     listedApps.sort((a, b) { | ||||
|       var nameA = a.installedInfo?.name ?? a.app.name; | ||||
|       var nameB = b.installedInfo?.name ?? b.app.name; | ||||
|       int result = 0; | ||||
|       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) { | ||||
|         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 == | ||||
|           SortColumnSettings.releaseDate) { | ||||
|         result = (a.app.releaseDate)?.compareTo( | ||||
| @@ -143,15 +142,15 @@ class AppsPageState extends State<AppsPage> { | ||||
|     var existingUpdates = appsProvider.findExistingUpdates(installedOnly: true); | ||||
|  | ||||
|     var existingUpdateIdsAllOrSelected = existingUpdates | ||||
|         .where((element) => selectedApps.isEmpty | ||||
|         .where((element) => selectedAppIds.isEmpty | ||||
|             ? listedApps.where((a) => a.app.id == element).isNotEmpty | ||||
|             : selectedApps.map((e) => e.id).contains(element)) | ||||
|             : selectedAppIds.map((e) => e).contains(element)) | ||||
|         .toList(); | ||||
|     var newInstallIdsAllOrSelected = appsProvider | ||||
|         .findExistingUpdates(nonInstalledOnly: true) | ||||
|         .where((element) => selectedApps.isEmpty | ||||
|         .where((element) => selectedAppIds.isEmpty | ||||
|             ? listedApps.where((a) => a.app.id == element).isNotEmpty | ||||
|             : selectedApps.map((e) => e.id).contains(element)) | ||||
|             : selectedAppIds.map((e) => e).contains(element)) | ||||
|         .toList(); | ||||
|  | ||||
|     List<String> trackOnlyUpdateIdsAllOrSelected = []; | ||||
| @@ -206,12 +205,17 @@ class AppsPageState extends State<AppsPage> { | ||||
|     var listedCategories = getListedCategories(); | ||||
|     listedCategories.sort((a, b) { | ||||
|       return a != null && b != null | ||||
|           ? a.compareTo(b) | ||||
|           ? a.toLowerCase().compareTo(b.toLowerCase()) | ||||
|           : a == null | ||||
|               ? 1 | ||||
|               : -1; | ||||
|     }); | ||||
|  | ||||
|     Set<App> selectedApps = listedApps | ||||
|         .map((e) => e.app) | ||||
|         .where((a) => selectedAppIds.contains(a.id)) | ||||
|         .toSet(); | ||||
|  | ||||
|     showChangeLogDialog( | ||||
|         String? changesUrl, AppSource appSource, String changeLog, int index) { | ||||
|       showDialog( | ||||
| @@ -288,7 +292,8 @@ class AppsPageState extends State<AppsPage> { | ||||
|         if (refreshingSince != null) | ||||
|           SliverToBoxAdapter( | ||||
|             child: LinearProgressIndicator( | ||||
|               value: appsProvider.apps.values | ||||
|               value: appsProvider | ||||
|                       .getAppValues() | ||||
|                       .where((element) => !(element.app.lastUpdateCheck | ||||
|                               ?.isBefore(refreshingSince!) ?? | ||||
|                           true)) | ||||
| @@ -467,15 +472,15 @@ class AppsPageState extends State<AppsPage> { | ||||
|                 .colorScheme | ||||
|                 .primary | ||||
|                 .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: () { | ||||
|               toggleAppSelected(listedApps[index].app); | ||||
|             }, | ||||
|             leading: getAppIcon(index), | ||||
|             title: Text( | ||||
|               maxLines: 1, | ||||
|               listedApps[index].installedInfo?.name ?? | ||||
|                   listedApps[index].app.name, | ||||
|               listedApps[index].name, | ||||
|               style: TextStyle( | ||||
|                 overflow: TextOverflow.ellipsis, | ||||
|                 fontWeight: listedApps[index].app.pinned | ||||
| @@ -497,7 +502,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|                   ])) | ||||
|                 : trailingRow, | ||||
|             onTap: () { | ||||
|               if (selectedApps.isNotEmpty) { | ||||
|               if (selectedAppIds.isNotEmpty) { | ||||
|                 toggleAppSelected(listedApps[index].app); | ||||
|               } else { | ||||
|                 Navigator.push( | ||||
| @@ -534,7 +539,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|     } | ||||
|  | ||||
|     getSelectAllButton() { | ||||
|       return selectedApps.isEmpty | ||||
|       return selectedAppIds.isEmpty | ||||
|           ? TextButton.icon( | ||||
|               style: const ButtonStyle(visualDensity: VisualDensity.compact), | ||||
|               onPressed: () { | ||||
| @@ -548,17 +553,17 @@ class AppsPageState extends State<AppsPage> { | ||||
|           : TextButton.icon( | ||||
|               style: const ButtonStyle(visualDensity: VisualDensity.compact), | ||||
|               onPressed: () { | ||||
|                 selectedApps.isEmpty | ||||
|                 selectedAppIds.isEmpty | ||||
|                     ? selectThese(listedApps.map((e) => e.app).toList()) | ||||
|                     : clearSelected(); | ||||
|               }, | ||||
|               icon: Icon( | ||||
|                 selectedApps.isEmpty | ||||
|                 selectedAppIds.isEmpty | ||||
|                     ? Icons.select_all_outlined | ||||
|                     : Icons.deselect_outlined, | ||||
|                 color: Theme.of(context).colorScheme.primary, | ||||
|               ), | ||||
|               label: Text(selectedApps.length.toString())); | ||||
|               label: Text(selectedAppIds.length.toString())); | ||||
|     } | ||||
|  | ||||
|     getMassObtainFunction() { | ||||
| @@ -706,7 +711,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|           builder: (BuildContext ctx) { | ||||
|             return AlertDialog( | ||||
|               title: Text(tr('markXSelectedAppsAsUpdated', | ||||
|                   args: [selectedApps.length.toString()])), | ||||
|                   args: [selectedAppIds.length.toString()])), | ||||
|               content: Text( | ||||
|                 tr('onlyWorksWithNonVersionDetectApps'), | ||||
|                 style: const TextStyle( | ||||
| @@ -760,7 +765,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|                   items: const [], | ||||
|                   initValid: true, | ||||
|                   message: tr('installStatusOfXWillBeResetExplanation', | ||||
|                       args: [plural('app', selectedApps.length)]), | ||||
|                       args: [plural('app', selectedAppIds.length)]), | ||||
|                 ); | ||||
|               }); | ||||
|           if (values != null) { | ||||
| @@ -836,7 +841,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|         children: [ | ||||
|           IconButton( | ||||
|             visualDensity: VisualDensity.compact, | ||||
|             onPressed: selectedApps.isEmpty | ||||
|             onPressed: selectedAppIds.isEmpty | ||||
|                 ? null | ||||
|                 : () { | ||||
|                     appsProvider.removeAppsWithModal( | ||||
| @@ -848,7 +853,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|           IconButton( | ||||
|               visualDensity: VisualDensity.compact, | ||||
|               onPressed: getMassObtainFunction(), | ||||
|               tooltip: selectedApps.isEmpty | ||||
|               tooltip: selectedAppIds.isEmpty | ||||
|                   ? tr('installUpdateApps') | ||||
|                   : tr('installUpdateSelectedApps'), | ||||
|               icon: const Icon( | ||||
| @@ -856,13 +861,13 @@ class AppsPageState extends State<AppsPage> { | ||||
|               )), | ||||
|           IconButton( | ||||
|             visualDensity: VisualDensity.compact, | ||||
|             onPressed: selectedApps.isEmpty ? null : launchCategorizeDialog(), | ||||
|             onPressed: selectedAppIds.isEmpty ? null : launchCategorizeDialog(), | ||||
|             tooltip: tr('categorize'), | ||||
|             icon: const Icon(Icons.category_outlined), | ||||
|           ), | ||||
|           IconButton( | ||||
|             visualDensity: VisualDensity.compact, | ||||
|             onPressed: selectedApps.isEmpty ? null : showMoreOptionsDialog, | ||||
|             onPressed: selectedAppIds.isEmpty ? null : showMoreOptionsDialog, | ||||
|             tooltip: tr('more'), | ||||
|             icon: const Icon(Icons.more_horiz), | ||||
|           ), | ||||
|   | ||||
| @@ -34,6 +34,10 @@ class AppInMemory { | ||||
|   AppInfo? 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 { | ||||
| @@ -97,6 +101,8 @@ class AppsProvider with ChangeNotifier { | ||||
|   late Stream<FGBGType>? foregroundStream; | ||||
|   late StreamSubscription<FGBGType>? foregroundSubscription; | ||||
|  | ||||
|   Iterable<AppInMemory> getAppValues() => apps.values.map((a) => a.deepCopy()); | ||||
|  | ||||
|   AppsProvider() { | ||||
|     // Subscribe to changes in the app foreground status | ||||
|     foregroundStream = FGBGEvents.stream.asBroadcastStream(); | ||||
| @@ -159,7 +165,7 @@ class AppsProvider with ChangeNotifier { | ||||
|   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { | ||||
|     NotificationsProvider? notificationsProvider = | ||||
|         context?.read<NotificationsProvider>(); | ||||
|     var notifId = DownloadNotification(app.name, 0).id; | ||||
|     var notifId = DownloadNotification(app.finalName, 0).id; | ||||
|     if (apps[app.id] != null) { | ||||
|       apps[app.id]!.downloadProgress = 0; | ||||
|       notifyListeners(); | ||||
| @@ -169,7 +175,7 @@ class AppsProvider with ChangeNotifier { | ||||
|           .getSource(app.url) | ||||
|           .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value); | ||||
|       var fileName = '${app.id}-${downloadUrl.hashCode}.apk'; | ||||
|       var notif = DownloadNotification(app.name, 100); | ||||
|       var notif = DownloadNotification(app.finalName, 100); | ||||
|       notificationsProvider?.cancel(notif.id); | ||||
|       int? prevProg; | ||||
|       File downloadedFile = | ||||
| @@ -179,7 +185,7 @@ class AppsProvider with ChangeNotifier { | ||||
|           apps[app.id]!.downloadProgress = progress; | ||||
|           notifyListeners(); | ||||
|         } | ||||
|         notif = DownloadNotification(app.name, prog ?? 100); | ||||
|         notif = DownloadNotification(app.finalName, prog ?? 100); | ||||
|         if (prog != null && prevProg != prog) { | ||||
|           notificationsProvider?.notify(notif); | ||||
|         } | ||||
| @@ -637,7 +643,7 @@ class AppsProvider with ChangeNotifier { | ||||
|         sp.getSource(newApps[i].url); | ||||
|         apps[newApps[i].id] = AppInMemory(newApps[i], null, info); | ||||
|       } 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) { | ||||
| @@ -667,12 +673,10 @@ class AppsProvider with ChangeNotifier { | ||||
|       bool onlyIfExists = true}) async { | ||||
|     attemptToCorrectInstallStatus = | ||||
|         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); | ||||
|     for (var app in apps) { | ||||
|     for (var a in apps) { | ||||
|       var app = a.deepCopy(); | ||||
|       AppInfo? info = await getInstalledInfo(app.id); | ||||
|       app.name = info?.name ?? app.name; | ||||
|       if (app.additionalSettings['appName']?.toString().isNotEmpty == true) { | ||||
|         app.name = app.additionalSettings['appName'].toString().trim(); | ||||
|       } | ||||
|       if (attemptToCorrectInstallStatus) { | ||||
|         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; | ||||
|       } | ||||
| @@ -940,7 +944,7 @@ class _APKPickerState extends State<APKPicker> { | ||||
|       scrollable: true, | ||||
|       title: Text(tr('pickAnAPK')), | ||||
|       content: Column(children: [ | ||||
|         Text(tr('appHasMoreThanOnePackage', args: [widget.app.name])), | ||||
|         Text(tr('appHasMoreThanOnePackage', args: [widget.app.finalName])), | ||||
|         const SizedBox(height: 16), | ||||
|         ...widget.app.apkUrls.map( | ||||
|           (u) => RadioListTile<String>( | ||||
|   | ||||
| @@ -34,9 +34,9 @@ class UpdateNotification extends ObtainiumNotification { | ||||
|     message = updates.isEmpty | ||||
|         ? tr('noNewUpdates') | ||||
|         : updates.length == 1 | ||||
|             ? tr('xHasAnUpdate', args: [updates[0].name]) | ||||
|             ? tr('xHasAnUpdate', args: [updates[0].finalName]) | ||||
|             : 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) { | ||||
|     message = updates.length == 1 | ||||
|         ? tr('xWasUpdatedToY', | ||||
|             args: [updates[0].name, updates[0].latestVersion]) | ||||
|             args: [updates[0].finalName, updates[0].latestVersion]) | ||||
|         : plural('xAndNMoreUpdatesInstalled', updates.length - 1, | ||||
|             args: [updates[0].name, (updates.length - 1).toString()]); | ||||
|             args: [updates[0].finalName, (updates.length - 1).toString()]); | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -164,7 +164,8 @@ class SettingsProvider with ChangeNotifier { | ||||
|  | ||||
|   void setCategories(Map<String, int> cats, {AppsProvider? appsProvider}) { | ||||
|     if (appsProvider != null) { | ||||
|       List<App> changedApps = appsProvider.apps.values | ||||
|       List<App> changedApps = appsProvider | ||||
|           .getAppValues() | ||||
|           .map((a) { | ||||
|             var n1 = a.app.categories.length; | ||||
|             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'; | ||||
|   } | ||||
|  | ||||
|   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) { | ||||
|     var source = SourceProvider().getSource(json['url']); | ||||
|     var formItems = source.combinedAppSpecificSettingFormItems | ||||
|   | ||||
							
								
								
									
										64
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										64
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -5,18 +5,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: android_alarm_manager_plus | ||||
|       sha256: "8647cc5f9339f3955a2bd9ec40e0f10c3a80049f31f80b3ffdd87e07bb73fce2" | ||||
|       sha256: f6d0347734fa2ea716349a5a3e16ffdc1800ca64e5640112896d128c6815c178 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.1" | ||||
|     version: "2.1.2" | ||||
|   android_intent_plus: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: android_intent_plus | ||||
|       sha256: "54810cb33945c2c10742cd746ea994822c115e9dbe189919bc63cb436e45a6af" | ||||
|       sha256: "6bcdcd20461ac7a0c785f6298cdda96ad275d5bcbc1ecf28829cbe03ec6690be" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.6" | ||||
|     version: "3.1.7" | ||||
|   animations: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -117,10 +117,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: device_info_plus | ||||
|       sha256: "1d6e5a61674ba3a68fb048a7c7b4ff4bebfed8d7379dbe8f2b718231be9a7c95" | ||||
|       sha256: "435383ca05f212760b0a70426b5a90354fe6bd65992b3a5e27ab6ede74c02f5c" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "8.1.0" | ||||
|     version: "8.2.0" | ||||
|   device_info_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -133,10 +133,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: dynamic_color | ||||
|       sha256: c4a508284b14ec4dda5adba2c28b2cdd34fbae1afead7e8c52cad87d51c5405b | ||||
|       sha256: bbebb1b7ebed819e0ec83d4abdc2a8482d934f6a85289ffc1c6acf7589fa2aad | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.6.2" | ||||
|     version: "1.6.3" | ||||
|   easy_localization: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -181,10 +181,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: file_picker | ||||
|       sha256: d8e9ca7e5d1983365c277f12c21b4362df6cf659c99af146ad4d04eb33033013 | ||||
|       sha256: dd328189f2f4ccea042bb5b382d5e981691cc74b5a3429b9317bff2b19704489 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.2.6" | ||||
|     version: "5.2.8" | ||||
|   flutter: | ||||
|     dependency: "direct main" | ||||
|     description: flutter | ||||
| @@ -425,10 +425,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_foundation | ||||
|       sha256: "818b2dc38b0f178e0ea3f7cf3b28146faab11375985d815942a68eee11c2d0f7" | ||||
|       sha256: ad4c4d011830462633f03eb34445a45345673dfd4faf1ab0b4735fbd93b19183 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.1" | ||||
|     version: "2.2.2" | ||||
|   path_provider_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -537,18 +537,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: share_plus | ||||
|       sha256: "8c6892037b1824e2d7e8f59d54b3105932899008642e6372e5079c6939b4b625" | ||||
|       sha256: "692261968a494e47323dcc8bc66d8d52e81bc27cb4b808e4e8d7e8079d4cc01a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.3.1" | ||||
|     version: "6.3.2" | ||||
|   share_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: share_plus_platform_interface | ||||
|       sha256: "82ddd4ab9260c295e6e39612d4ff00390b9a7a21f1bb1da771e2f232d80ab8a1" | ||||
|       sha256: "0c6e61471bd71b04a138b8b588fa388e66d8b005e6f2deda63371c5c505a0981" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.2.0" | ||||
|     version: "3.2.1" | ||||
|   shared_preferences: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -569,10 +569,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_foundation | ||||
|       sha256: cf2a42fb20148502022861f71698db12d937c7459345a1bdaa88fc91a91b3603 | ||||
|       sha256: "0c1c16c56c9708aa9c361541a6f0e5cc6fc12a3232d866a687a7b7db30032b07" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.0" | ||||
|     version: "2.2.1" | ||||
|   shared_preferences_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -710,18 +710,18 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_android | ||||
|       sha256: dd729390aa936bf1bdf5cd1bc7468ff340263f80a2c4f569416507667de8e3c8 | ||||
|       sha256: a52628068d282d01a07cd86e6ba99e497aa45ce8c91159015b2416907d78e411 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.0.26" | ||||
|     version: "6.0.27" | ||||
|   url_launcher_ios: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_ios | ||||
|       sha256: "3dedc66ca3c0bef9e6a93c0999aee102556a450afcc1b7bcfeace7a424927d92" | ||||
|       sha256: "9af7ea73259886b92199f9e42c116072f05ff9bea2dcb339ab935dfc957392c2" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.1.3" | ||||
|     version: "6.1.4" | ||||
|   url_launcher_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -734,10 +734,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_macos | ||||
|       sha256: "0ef2b4f97942a16523e51256b799e9aa1843da6c60c55eefbfa9dbc2dcb8331a" | ||||
|       sha256: "91ee3e75ea9dadf38036200c5d3743518f4a5eb77a8d13fda1ee5764373f185e" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.0.4" | ||||
|     version: "3.0.5" | ||||
|   url_launcher_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -790,34 +790,34 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_android | ||||
|       sha256: "9e223788e1954087dac30d813dc151f8e12f09f1139f116ce20b5658893f3627" | ||||
|       sha256: "5906c9aa8c88ed372b2ad3c88c942790b4fb16f73fdd1c0647b4d747d9cf5b3f" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.4.4" | ||||
|     version: "3.4.5" | ||||
|   webview_flutter_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_platform_interface | ||||
|       sha256: "1939c39e2150fb4d30fd3cc59a891a49fed9935db53007df633ed83581b6117b" | ||||
|       sha256: "6341f92977609be71391f4d4dcd64bfaa8ac657af1dfb2e231b5c1724e8c6c36" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.0" | ||||
|     version: "2.2.0" | ||||
|   webview_flutter_wkwebview: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_wkwebview | ||||
|       sha256: d601aba11ad8d4481e17a34a76fa1d30dee92dcbbe2c58b0df3120e9453099c7 | ||||
|       sha256: "9a78d963cce191dd6a9df547301fc5c008bf3dae95a323ec281fff1284e0a037" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.2.3" | ||||
|     version: "3.2.4" | ||||
|   win32: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: win32 | ||||
|       sha256: c9ebe7ee4ab0c2194e65d3a07d8c54c5d00bb001b76081c4a04cdb8448b59e46 | ||||
|       sha256: a6f0236dbda0f63aa9a25ad1ff9a9d8a4eaaa5012da0dc59d21afdb1dc361ca4 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.3" | ||||
|     version: "3.1.4" | ||||
|   xdg_directories: | ||||
|     dependency: transitive | ||||
|     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 | ||||
| # 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. | ||||
| version: 0.11.27+149 # When changing this, update the tag in main() accordingly | ||||
| version: 0.11.30+152 # When changing this, update the tag in main() accordingly | ||||
|  | ||||
| environment: | ||||
|   sdk: '>=2.18.2 <3.0.0' | ||||
|   | ||||
		Reference in New Issue
	
	Block a user