mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-26 19:23:45 +01:00 
			
		
		
		
	Merge branch 'main' into main
This commit is contained in:
		| @@ -15,7 +15,8 @@ Currently supported App sources: | |||||||
| - [Signal](https://signal.org/) | - [Signal](https://signal.org/) | ||||||
| - [SourceForge](https://sourceforge.net/) | - [SourceForge](https://sourceforge.net/) | ||||||
| - [APKMirror](https://apkmirror.com/) (Track-Only) | - [APKMirror](https://apkmirror.com/) (Track-Only) | ||||||
| - Third Party F-Droid Repos (URLs ending with `/fdroid/repo`) | - Third Party F-Droid Repos | ||||||
|  |   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` | ||||||
| - [Steam](https://store.steampowered.com/mobile) | - [Steam](https://store.steampowered.com/mobile) | ||||||
|  |  | ||||||
| ## Limitations | ## Limitations | ||||||
|   | |||||||
| @@ -188,25 +188,27 @@ | |||||||
|     "steam": "Steam", |     "steam": "Steam", | ||||||
|     "steamMobile": "Steam Mobile", |     "steamMobile": "Steam Mobile", | ||||||
|     "steamChat": "Steam Chat", |     "steamChat": "Steam Chat", | ||||||
|     "install": "Install", |     "install": "Installieren", | ||||||
|     "markInstalled": "Mark Installed", |     "markInstalled": "Als Installiert markieren", | ||||||
|     "update": "Update", |     "update": "Aktualisieren", | ||||||
|     "markUpdated": "Mark Updated", |     "markUpdated": "Als Aktuell markieren", | ||||||
|     "additionalOptions": "Additional Options", |     "additionalOptions": "Zusätzliche Optionen", | ||||||
|     "disableVersionDetection": "Disable Version Detection", |     "disableVersionDetection": "Versionsermittlung deaktivieren", | ||||||
|     "noVersionDetectionExplanation": "This option should only be used for Apps where version detection does not work correctly.", |     "noVersionDetectionExplanation": "Diese Option sollte nur für Apps verwendet werden, bei denen die Versionserkennung nicht korrekt funktioniert.", | ||||||
|     "downloadingX": "Downloading {}", |     "downloadingX": "Lade {} herunter", | ||||||
|     "downloadNotifDescription": "Notifies the user of the progress in downloading an App", |     "downloadNotifDescription": "Benachrichtigt den Nutzer über den Fortschritt beim Herunterladen einer App", | ||||||
|     "noAPKFound": "No APK found", |     "noAPKFound": "Keine APK gefunden", | ||||||
|     "noVersionDetection": "No version detection", |     "noVersionDetection": "Keine Versionserkennung", | ||||||
|     "categorize": "Categorize", |     "categorize": "Kategorisieren", | ||||||
|     "categories": "Categories", |     "categories": "Kategorien", | ||||||
|     "category": "Category", |     "category": "Kategorie", | ||||||
|     "noCategory": "No Category", |     "noCategory": "Keine Kategorie", | ||||||
|     "deleteCategoryQuestion": "Delete Category?", |     "noCategories": "Keine Kategorien", | ||||||
|     "categoryDeleteWarning": "All Apps in {} will be set to uncategorized.", |     "deleteCategoriesQuestion": "Kategorien löschen?", | ||||||
|     "addCategory": "Add Category", |     "categoryDeleteWarning": "Alle Apps in gelöschten Kategorien werden auf nicht kategorisiert gesetzt.", | ||||||
|     "label": "Label", |     "addCategory": "Kategorie hinzufügen", | ||||||
|  |     "label": "Bezeichnung", | ||||||
|  |     "language": "Sprache", | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", |         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", | ||||||
|         "other": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minuten erneut" |         "other": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minuten erneut" | ||||||
|   | |||||||
| @@ -203,10 +203,12 @@ | |||||||
|     "categories": "Categories", |     "categories": "Categories", | ||||||
|     "category": "Category", |     "category": "Category", | ||||||
|     "noCategory": "No Category", |     "noCategory": "No Category", | ||||||
|     "deleteCategoryQuestion": "Delete Category?", |     "noCategories": "No Categories", | ||||||
|     "categoryDeleteWarning": "All Apps in {} will be set to uncategorized.", |     "deleteCategoriesQuestion": "Delete Categories?", | ||||||
|  |     "categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.", | ||||||
|     "addCategory": "Add Category", |     "addCategory": "Add Category", | ||||||
|     "label": "Label", |     "label": "Label", | ||||||
|  |     "language": "Language", | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Too many requests (rate limited) - try again in {} minute", |         "one": "Too many requests (rate limited) - try again in {} minute", | ||||||
|         "other": "Too many requests (rate limited) - try again in {} minutes" |         "other": "Too many requests (rate limited) - try again in {} minutes" | ||||||
|   | |||||||
| @@ -28,13 +28,13 @@ | |||||||
|     "noDescription": "Nincs leírás", |     "noDescription": "Nincs leírás", | ||||||
|     "cancel": "Mégse", |     "cancel": "Mégse", | ||||||
|     "continue": "Tovább", |     "continue": "Tovább", | ||||||
|     "requiredInBrackets": "(Kötlező)", |     "requiredInBrackets": "(Kötelező)", | ||||||
|     "dropdownNoOptsError": "HIBA: A LEDOBÁST LEGALÁBB EGY OPCIÓHOZ KELL RENDELNI", |     "dropdownNoOptsError": "HIBA: A LEDOBÁST LEGALÁBB EGY OPCIÓHOZ KELL RENDELNI", | ||||||
|     "colour": "Szín", |     "colour": "Szín", | ||||||
|     "githubStarredRepos": "GitHub Csillagos Repo-k", |     "githubStarredRepos": "GitHub Csillagos Repo-k", | ||||||
|     "uname": "Felh.név", |     "uname": "Felh.név", | ||||||
|     "wrongArgNum": "Rossz számú argumentumot adott meg", |     "wrongArgNum": "Rossz számú argumentumot adott meg", | ||||||
|     "xIsTrackOnly": "{} csak nyomkövethető", |     "xIsTrackOnly": "A(z) {} csak nyomkövethető", | ||||||
|     "source": "Forrás", |     "source": "Forrás", | ||||||
|     "app": "App", |     "app": "App", | ||||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", |     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", | ||||||
| @@ -65,11 +65,11 @@ | |||||||
|     "estimateInBrackets": "(Becslés)", |     "estimateInBrackets": "(Becslés)", | ||||||
|     "selectAll": "Mindet kiválaszt", |     "selectAll": "Mindet kiválaszt", | ||||||
|     "deselectN": "Törölje {} kijelölését", |     "deselectN": "Törölje {} kijelölését", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{} el lesz távolítva az Obtainiumból, de továbbra is telepítve marad az eszközön.", |     "xWillBeRemovedButRemainInstalled": "A(z) {} el lesz távolítva az Obtainiumból, de továbbra is telepítve marad az eszközön.", | ||||||
|     "removeSelectedAppsQuestion": "Eltávolítja a kiválasztott appokat?", |     "removeSelectedAppsQuestion": "Eltávolítja a kiválasztott appokat?", | ||||||
|     "removeSelectedApps": "Távolítsa el a kiválasztott appokat", |     "removeSelectedApps": "Távolítsa el a kiválasztott appokat", | ||||||
|     "updateX": "Frissítés: {}", |     "updateX": "Frissítés: {}", | ||||||
|     "installX": "Telepítés {}", |     "installX": "Telepítés: {}", | ||||||
|     "markXTrackOnlyAsUpdated": "Jelölje meg: {}\n(Csak nyomon követhető)\nas Frissítve", |     "markXTrackOnlyAsUpdated": "Jelölje meg: {}\n(Csak nyomon követhető)\nas Frissítve", | ||||||
|     "changeX": "Változás {}", |     "changeX": "Változás {}", | ||||||
|     "installUpdateApps": "Appok telepítése/frissítése", |     "installUpdateApps": "Appok telepítése/frissítése", | ||||||
| @@ -118,7 +118,7 @@ | |||||||
|     "selectURLs": "Kiválasztott URL-ek", |     "selectURLs": "Kiválasztott URL-ek", | ||||||
|     "pick": "Válasszon", |     "pick": "Válasszon", | ||||||
|     "theme": "Téma", |     "theme": "Téma", | ||||||
|     "dark": "Söét", |     "dark": "Sötét", | ||||||
|     "light": "Világos", |     "light": "Világos", | ||||||
|     "followSystem": "Rendszer szerint", |     "followSystem": "Rendszer szerint", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
| @@ -184,7 +184,7 @@ | |||||||
|     "appIdOrName": "App ID vagy név", |     "appIdOrName": "App ID vagy név", | ||||||
|     "appWithIdOrNameNotFound": "Nem található app ezzel az azonosítóval vagy névvel", |     "appWithIdOrNameNotFound": "Nem található app ezzel az azonosítóval vagy névvel", | ||||||
|     "reposHaveMultipleApps": "A repók több alkalmazást is tartalmazhatnak", |     "reposHaveMultipleApps": "A repók több alkalmazást is tartalmazhatnak", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Harmadik fél Repo", |     "fdroidThirdPartyRepo": "F-Droid Harmadik-fél Repo", | ||||||
|     "steam": "Steam", |     "steam": "Steam", | ||||||
|     "steamMobile": "Steam Mobile", |     "steamMobile": "Steam Mobile", | ||||||
|     "steamChat": "Steam Chat", |     "steamChat": "Steam Chat", | ||||||
|   | |||||||
| @@ -203,10 +203,12 @@ | |||||||
|     "categories": "Categories", |     "categories": "Categories", | ||||||
|     "category": "Category", |     "category": "Category", | ||||||
|     "noCategory": "No Category", |     "noCategory": "No Category", | ||||||
|     "deleteCategoryQuestion": "Delete Category?", |     "noCategories": "No Categories", | ||||||
|     "categoryDeleteWarning": "All Apps in {} will be set to uncategorized.", |     "deleteCategoriesQuestion": "Delete Categories?", | ||||||
|  |     "categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.", | ||||||
|     "addCategory": "Add Category", |     "addCategory": "Add Category", | ||||||
|     "label": "Label", |     "label": "Label", | ||||||
|  |     "language": "Language", | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Troppe richieste (traffico limitato) - riprova tra {} minuto", |         "one": "Troppe richieste (traffico limitato) - riprova tra {} minuto", | ||||||
|         "other": "Troppe richieste (traffico limitato) - riprova tra {} minuti" |         "other": "Troppe richieste (traffico limitato) - riprova tra {} minuti" | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ | |||||||
|     "invalidRegEx": "無効な正規表現", |     "invalidRegEx": "無効な正規表現", | ||||||
|     "noDescription": "説明はありません", |     "noDescription": "説明はありません", | ||||||
|     "cancel": "キャンセル", |     "cancel": "キャンセル", | ||||||
|     "continue": "続ける", |     "continue": "続行", | ||||||
|     "requiredInBrackets": "(必須)", |     "requiredInBrackets": "(必須)", | ||||||
|     "dropdownNoOptsError": "エラー: ドロップダウンには、少なくとも1つのオプションが必要です", |     "dropdownNoOptsError": "エラー: ドロップダウンには、少なくとも1つのオプションが必要です", | ||||||
|     "colour": "カラー", |     "colour": "カラー", | ||||||
| @@ -64,7 +64,7 @@ | |||||||
|     "notInstalled": "未インストール", |     "notInstalled": "未インストール", | ||||||
|     "estimateInBrackets": "(推定)", |     "estimateInBrackets": "(推定)", | ||||||
|     "selectAll": "すべて選択", |     "selectAll": "すべて選択", | ||||||
|     "deselectN": "{}件を選択解除", |     "deselectN": "{}件の選択を解除", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{}はObtainiumから削除されますが、デバイスにはインストールされたままです。", |     "xWillBeRemovedButRemainInstalled": "{}はObtainiumから削除されますが、デバイスにはインストールされたままです。", | ||||||
|     "removeSelectedAppsQuestion": "選択したアプリを削除しますか?", |     "removeSelectedAppsQuestion": "選択したアプリを削除しますか?", | ||||||
|     "removeSelectedApps": "選択したアプリを削除する", |     "removeSelectedApps": "選択したアプリを削除する", | ||||||
| @@ -135,7 +135,7 @@ | |||||||
|     "appearance": "外観", |     "appearance": "外観", | ||||||
|     "showWebInAppView": "アプリビューにソースウェブページを表示する", |     "showWebInAppView": "アプリビューにソースウェブページを表示する", | ||||||
|     "pinUpdates": "アップデートがあるアプリをトップに固定する", |     "pinUpdates": "アップデートがあるアプリをトップに固定する", | ||||||
|     "updates": "更新", |     "updates": "アップデート", | ||||||
|     "sourceSpecific": "Github アクセストークン", |     "sourceSpecific": "Github アクセストークン", | ||||||
|     "appSource": "アプリのソース", |     "appSource": "アプリのソース", | ||||||
|     "noLogs": "ログはありません", |     "noLogs": "ログはありません", | ||||||
| @@ -144,7 +144,7 @@ | |||||||
|     "share": "共有", |     "share": "共有", | ||||||
|     "appNotFound": "アプリが見つかりません", |     "appNotFound": "アプリが見つかりません", | ||||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-エクスポート", |     "obtainiumExportHyphenatedLowercase": "obtainium-エクスポート", | ||||||
|     "pickAnAPK": "APKを選ぶ", |     "pickAnAPK": "APKを選択", | ||||||
|     "appHasMoreThanOnePackage": "{}は複数のパッケージが存在します: ", |     "appHasMoreThanOnePackage": "{}は複数のパッケージが存在します: ", | ||||||
|     "deviceSupportsXArch": "お使いのデバイスは{} CPUアーキテクチャに対応しています。", |     "deviceSupportsXArch": "お使いのデバイスは{} CPUアーキテクチャに対応しています。", | ||||||
|     "deviceSupportsFollowingArchs": "お使いのデバイスは、以下のCPUアーキテクチャをサポートしています:", |     "deviceSupportsFollowingArchs": "お使いのデバイスは、以下のCPUアーキテクチャをサポートしています:", | ||||||
| @@ -203,10 +203,12 @@ | |||||||
|     "categories": "カテゴリ", |     "categories": "カテゴリ", | ||||||
|     "category": "カテゴリ", |     "category": "カテゴリ", | ||||||
|     "noCategory": "カテゴリなし", |     "noCategory": "カテゴリなし", | ||||||
|     "deleteCategoryQuestion": "カテゴリを削除しますか?", |     "noCategories": "カテゴリなし", | ||||||
|     "categoryDeleteWarning": "「{}」内のすべてのアプリは未分類に設定されます。", |     "deleteCategoriesQuestion": "カテゴリを削除しますか?", | ||||||
|  |     "categoryDeleteWarning": "削除されたカテゴリ内のアプリは未分類に設定されます。", | ||||||
|     "addCategory": "カテゴリを追加", |     "addCategory": "カテゴリを追加", | ||||||
|     "label": "ラベル", |     "label": "ラベル", | ||||||
|  |     "language": "言語", | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "リクエストが多すぎます(レート制限)- {}分後に再試行してください", |         "one": "リクエストが多すぎます(レート制限)- {}分後に再試行してください", | ||||||
|         "other": "リクエストが多すぎます(レート制限)- {}分後に再試行してください" |         "other": "リクエストが多すぎます(レート制限)- {}分後に再試行してください" | ||||||
|   | |||||||
| @@ -203,10 +203,12 @@ | |||||||
|     "categories": "Categories", |     "categories": "Categories", | ||||||
|     "category": "Category", |     "category": "Category", | ||||||
|     "noCategory": "No Category", |     "noCategory": "No Category", | ||||||
|     "deleteCategoryQuestion": "Delete Category?", |     "noCategories": "No Categories", | ||||||
|     "categoryDeleteWarning": "All Apps in {} will be set to uncategorized.", |     "deleteCategoriesQuestion": "Delete Categories?", | ||||||
|  |     "categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.", | ||||||
|     "addCategory": "Add Category", |     "addCategory": "Add Category", | ||||||
|     "label": "Label", |     "label": "Label", | ||||||
|  |     "language": "Language", | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "请求过多 (API 限制) - 在 {} 分钟后重试", |         "one": "请求过多 (API 限制) - 在 {} 分钟后重试", | ||||||
|         "other": "请求过多 (API 限制) - 在 {} 分钟后重试" |         "other": "请求过多 (API 限制) - 在 {} 分钟后重试" | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ class FDroidRepo extends AppSource { | |||||||
|   @override |   @override | ||||||
|   String standardizeURL(String url) { |   String standardizeURL(String url) { | ||||||
|     RegExp standardUrlRegExp = |     RegExp standardUrlRegExp = | ||||||
|         RegExp('^https?://.+/fdroid/(repo(/|\\?)|repo\$)'); |         RegExp('^https?://.+/fdroid/([^/]+(/|\\?)|[^/]+\$)'); | ||||||
|     RegExpMatch? match = standardUrlRegExp.firstMatch(url.toLowerCase()); |     RegExpMatch? match = standardUrlRegExp.firstMatch(url.toLowerCase()); | ||||||
|     if (match == null) { |     if (match == null) { | ||||||
|       throw InvalidURLError(name); |       throw InvalidURLError(name); | ||||||
|   | |||||||
| @@ -1,5 +1,9 @@ | |||||||
|  | import 'dart:math'; | ||||||
|  |  | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
|  | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
|  |  | ||||||
| abstract class GeneratedFormItem { | abstract class GeneratedFormItem { | ||||||
|   late String key; |   late String key; | ||||||
| @@ -82,6 +86,33 @@ class GeneratedFormSwitch extends GeneratedFormItem { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class GeneratedFormTagInput extends GeneratedFormItem { | ||||||
|  |   late MapEntry<String, String>? deleteConfirmationMessage; | ||||||
|  |   late bool singleSelect; | ||||||
|  |   late WrapAlignment alignment; | ||||||
|  |   late String emptyMessage; | ||||||
|  |   GeneratedFormTagInput(String key, | ||||||
|  |       {String label = 'Input', | ||||||
|  |       List<Widget> belowWidgets = const [], | ||||||
|  |       Map<String, MapEntry<int, bool>> defaultValue = const {}, | ||||||
|  |       List<String? Function(Map<String, MapEntry<int, bool>> value)> | ||||||
|  |           additionalValidators = const [], | ||||||
|  |       this.deleteConfirmationMessage, | ||||||
|  |       this.singleSelect = false, | ||||||
|  |       this.alignment = WrapAlignment.start, | ||||||
|  |       this.emptyMessage = 'Input'}) | ||||||
|  |       : super(key, | ||||||
|  |             label: label, | ||||||
|  |             belowWidgets: belowWidgets, | ||||||
|  |             defaultValue: defaultValue, | ||||||
|  |             additionalValidators: additionalValidators); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, MapEntry<int, bool>> ensureType(val) { | ||||||
|  |     return val is Map<String, MapEntry<int, bool>> ? val : {}; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| typedef OnValueChanges = void Function( | typedef OnValueChanges = void Function( | ||||||
|     Map<String, dynamic> values, bool valid, bool isBuilding); |     Map<String, dynamic> values, bool valid, bool isBuilding); | ||||||
|  |  | ||||||
| @@ -120,6 +151,21 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|     widget.onValueChanges(returnValues, valid, isBuilding); |     widget.onValueChanges(returnValues, valid, isBuilding); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Generates a random light color | ||||||
|  | // Courtesy of ChatGPT 😭 (with a bugfix 🥳) | ||||||
|  |   Color generateRandomLightColor() { | ||||||
|  |     // Create a random number generator | ||||||
|  |     final Random random = Random(); | ||||||
|  |  | ||||||
|  |     // Generate random hue, saturation, and value values | ||||||
|  |     final double hue = random.nextDouble() * 360; | ||||||
|  |     final double saturation = 0.5 + random.nextDouble() * 0.5; | ||||||
|  |     final double value = 0.9 + random.nextDouble() * 0.1; | ||||||
|  |  | ||||||
|  |     // Create a HSV color with the random values | ||||||
|  |     return HSVColor.fromAHSV(1.0, hue, saturation, value).toColor(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   void initState() { |   void initState() { | ||||||
|     super.initState(); |     super.initState(); | ||||||
| @@ -212,6 +258,158 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|                   }) |                   }) | ||||||
|             ], |             ], | ||||||
|           ); |           ); | ||||||
|  |         } else if (widget.items[r][e] is GeneratedFormTagInput) { | ||||||
|  |           formInputs[r][e] = Wrap( | ||||||
|  |             alignment: (widget.items[r][e] as GeneratedFormTagInput).alignment, | ||||||
|  |             crossAxisAlignment: WrapCrossAlignment.center, | ||||||
|  |             children: [ | ||||||
|  |               (values[widget.items[r][e].key] | ||||||
|  |                               as Map<String, MapEntry<int, bool>>?) | ||||||
|  |                           ?.isEmpty == | ||||||
|  |                       true | ||||||
|  |                   ? Text( | ||||||
|  |                       (widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                           .emptyMessage, | ||||||
|  |                       style: const TextStyle(fontWeight: FontWeight.bold), | ||||||
|  |                     ) | ||||||
|  |                   : const SizedBox.shrink(), | ||||||
|  |               ...(values[widget.items[r][e].key] | ||||||
|  |                           as Map<String, MapEntry<int, bool>>?) | ||||||
|  |                       ?.entries | ||||||
|  |                       .map((e2) { | ||||||
|  |                     return Padding( | ||||||
|  |                         padding: const EdgeInsets.symmetric(horizontal: 4), | ||||||
|  |                         child: ChoiceChip( | ||||||
|  |                           label: Text(e2.key), | ||||||
|  |                           backgroundColor: Color(e2.value.key).withAlpha(50), | ||||||
|  |                           selectedColor: Color(e2.value.key), | ||||||
|  |                           visualDensity: VisualDensity.compact, | ||||||
|  |                           selected: e2.value.value, | ||||||
|  |                           onSelected: (value) { | ||||||
|  |                             setState(() { | ||||||
|  |                               (values[widget.items[r][e].key] as Map<String, | ||||||
|  |                                       MapEntry<int, bool>>)[e2.key] = | ||||||
|  |                                   MapEntry( | ||||||
|  |                                       (values[widget.items[r][e].key] as Map< | ||||||
|  |                                               String, | ||||||
|  |                                               MapEntry<int, bool>>)[e2.key]! | ||||||
|  |                                           .key, | ||||||
|  |                                       value); | ||||||
|  |                               if ((widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                                       .singleSelect && | ||||||
|  |                                   value == true) { | ||||||
|  |                                 for (var key in (values[widget.items[r][e].key] | ||||||
|  |                                         as Map<String, MapEntry<int, bool>>) | ||||||
|  |                                     .keys) { | ||||||
|  |                                   if (key != e2.key) { | ||||||
|  |                                     (values[widget.items[r][e].key] as Map< | ||||||
|  |                                         String, | ||||||
|  |                                         MapEntry<int, | ||||||
|  |                                             bool>>)[key] = MapEntry( | ||||||
|  |                                         (values[widget.items[r][e].key] as Map< | ||||||
|  |                                                 String, | ||||||
|  |                                                 MapEntry<int, bool>>)[key]! | ||||||
|  |                                             .key, | ||||||
|  |                                         false); | ||||||
|  |                                   } | ||||||
|  |                                 } | ||||||
|  |                               } | ||||||
|  |                               someValueChanged(); | ||||||
|  |                             }); | ||||||
|  |                           }, | ||||||
|  |                         )); | ||||||
|  |                   }) ?? | ||||||
|  |                   [const SizedBox.shrink()], | ||||||
|  |               (values[widget.items[r][e].key] | ||||||
|  |                               as Map<String, MapEntry<int, bool>>?) | ||||||
|  |                           ?.values | ||||||
|  |                           .where((e) => e.value) | ||||||
|  |                           .isNotEmpty == | ||||||
|  |                       true | ||||||
|  |                   ? Padding( | ||||||
|  |                       padding: const EdgeInsets.symmetric(horizontal: 4), | ||||||
|  |                       child: IconButton( | ||||||
|  |                         onPressed: () { | ||||||
|  |                           fn() { | ||||||
|  |                             setState(() { | ||||||
|  |                               var temp = values[widget.items[r][e].key] | ||||||
|  |                                   as Map<String, MapEntry<int, bool>>; | ||||||
|  |                               temp.removeWhere((key, value) => value.value); | ||||||
|  |                               values[widget.items[r][e].key] = temp; | ||||||
|  |                               someValueChanged(); | ||||||
|  |                             }); | ||||||
|  |                           } | ||||||
|  |  | ||||||
|  |                           if ((widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                                   .deleteConfirmationMessage != | ||||||
|  |                               null) { | ||||||
|  |                             var message = | ||||||
|  |                                 (widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                                     .deleteConfirmationMessage!; | ||||||
|  |                             showDialog<Map<String, dynamic>?>( | ||||||
|  |                                 context: context, | ||||||
|  |                                 builder: (BuildContext ctx) { | ||||||
|  |                                   return GeneratedFormModal( | ||||||
|  |                                       title: message.key, | ||||||
|  |                                       message: message.value, | ||||||
|  |                                       items: const []); | ||||||
|  |                                 }).then((value) { | ||||||
|  |                               if (value != null) { | ||||||
|  |                                 fn(); | ||||||
|  |                               } | ||||||
|  |                             }); | ||||||
|  |                           } else { | ||||||
|  |                             fn(); | ||||||
|  |                           } | ||||||
|  |                         }, | ||||||
|  |                         icon: const Icon(Icons.remove), | ||||||
|  |                         visualDensity: VisualDensity.compact, | ||||||
|  |                         tooltip: tr('remove'), | ||||||
|  |                       )) | ||||||
|  |                   : const SizedBox.shrink(), | ||||||
|  |               Padding( | ||||||
|  |                   padding: const EdgeInsets.symmetric(horizontal: 4), | ||||||
|  |                   child: IconButton( | ||||||
|  |                     onPressed: () { | ||||||
|  |                       showDialog<Map<String, dynamic>?>( | ||||||
|  |                           context: context, | ||||||
|  |                           builder: (BuildContext ctx) { | ||||||
|  |                             return GeneratedFormModal( | ||||||
|  |                                 title: widget.items[r][e].label, | ||||||
|  |                                 items: [ | ||||||
|  |                                   [ | ||||||
|  |                                     GeneratedFormTextField('label', | ||||||
|  |                                         label: tr('label')) | ||||||
|  |                                   ] | ||||||
|  |                                 ]); | ||||||
|  |                           }).then((value) { | ||||||
|  |                         String? label = value?['label']; | ||||||
|  |                         if (label != null) { | ||||||
|  |                           setState(() { | ||||||
|  |                             var temp = values[widget.items[r][e].key] | ||||||
|  |                                 as Map<String, MapEntry<int, bool>>?; | ||||||
|  |                             temp ??= {}; | ||||||
|  |                             var singleSelect = | ||||||
|  |                                 (widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                                     .singleSelect; | ||||||
|  |                             var someSelected = temp.entries | ||||||
|  |                                 .where((element) => element.value.value) | ||||||
|  |                                 .isNotEmpty; | ||||||
|  |                             temp[label] = MapEntry( | ||||||
|  |                                 generateRandomLightColor().value, | ||||||
|  |                                 !(someSelected && singleSelect)); | ||||||
|  |                             values[widget.items[r][e].key] = temp; | ||||||
|  |                             someValueChanged(); | ||||||
|  |                           }); | ||||||
|  |                         } | ||||||
|  |                       }); | ||||||
|  |                     }, | ||||||
|  |                     icon: const Icon(Icons.add), | ||||||
|  |                     visualDensity: VisualDensity.compact, | ||||||
|  |                     tooltip: tr('add'), | ||||||
|  |                   )), | ||||||
|  |             ], | ||||||
|  |           ); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -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.9.2'; | const String currentVersion = '0.9.4'; | ||||||
| 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 | ||||||
|  |  | ||||||
| @@ -43,12 +43,16 @@ final globalNavigatorKey = GlobalKey<NavigatorState>(); | |||||||
| Future<void> loadTranslations() async { | Future<void> loadTranslations() async { | ||||||
|   // See easy_localization/issues/210 |   // See easy_localization/issues/210 | ||||||
|   await EasyLocalizationController.initEasyLocation(); |   await EasyLocalizationController.initEasyLocation(); | ||||||
|  |   var s = SettingsProvider(); | ||||||
|  |   await s.initializeSettings(); | ||||||
|  |   var forceLocale = s.forcedLocale; | ||||||
|   final controller = EasyLocalizationController( |   final controller = EasyLocalizationController( | ||||||
|     saveLocale: true, |     saveLocale: true, | ||||||
|  |     forceLocale: forceLocale != null ? Locale(forceLocale) : null, | ||||||
|     fallbackLocale: fallbackLocale, |     fallbackLocale: fallbackLocale, | ||||||
|     supportedLocales: supportedLocales, |     supportedLocales: supportedLocales, | ||||||
|     assetLoader: const RootBundleAssetLoader(), |     assetLoader: const RootBundleAssetLoader(), | ||||||
|     useOnlyLangCode: false, |     useOnlyLangCode: true, | ||||||
|     useFallbackTranslations: true, |     useFallbackTranslations: true, | ||||||
|     path: localeDir, |     path: localeDir, | ||||||
|     onLoadError: (FlutterError e) { |     onLoadError: (FlutterError e) { | ||||||
| @@ -160,6 +164,7 @@ void main() async { | |||||||
|         supportedLocales: supportedLocales, |         supportedLocales: supportedLocales, | ||||||
|         path: localeDir, |         path: localeDir, | ||||||
|         fallbackLocale: fallbackLocale, |         fallbackLocale: fallbackLocale, | ||||||
|  |         useOnlyLangCode: true, | ||||||
|         child: const Obtainium()), |         child: const Obtainium()), | ||||||
|   )); |   )); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import 'package:obtainium/components/generated_form.dart'; | |||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
|  | import 'package:obtainium/pages/settings.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| @@ -152,49 +153,22 @@ class _AppPageState extends State<AppPage> { | |||||||
|                               fontStyle: FontStyle.italic, fontSize: 12), |                               fontStyle: FontStyle.italic, fontSize: 12), | ||||||
|                         ), |                         ), | ||||||
|                         const SizedBox( |                         const SizedBox( | ||||||
|                           height: 32, |                           height: 48, | ||||||
|                         ), |                         ), | ||||||
|                         app?.app.category != null |                         CategoryEditorSelector( | ||||||
|                             ? Chip( |                             alignment: WrapAlignment.center, | ||||||
|                                 label: Text(app!.app.category!), |                             singleSelect: true, | ||||||
|                                 backgroundColor: |                             preselected: app?.app.category != null | ||||||
|                                     Color(categories[app.app.category!] ?? 0x0), |                                 ? {app!.app.category!} | ||||||
|                                 onDeleted: () { |                                 : {}, | ||||||
|                                   app.app.category = null; |                             onSelected: (categories) { | ||||||
|                                   appsProvider.saveApps([app.app]); |                               if (app != null) { | ||||||
|                                 }, |                                 app.app.category = categories.isNotEmpty | ||||||
|                                 visualDensity: VisualDensity.compact, |                                     ? categories[0] | ||||||
|                               ) |                                     : null; | ||||||
|                             : Row( |                                 appsProvider.saveApps([app.app]); | ||||||
|                                 mainAxisAlignment: MainAxisAlignment.center, |                               } | ||||||
|                                 children: [ |                             }) | ||||||
|                                     TextButton( |  | ||||||
|                                         onPressed: () { |  | ||||||
|                                           showDialog<Map<String, dynamic>?>( |  | ||||||
|                                               context: context, |  | ||||||
|                                               builder: (BuildContext ctx) { |  | ||||||
|                                                 return GeneratedFormModal( |  | ||||||
|                                                     title: 'Pick a Category', |  | ||||||
|                                                     items: [ |  | ||||||
|                                                       [ |  | ||||||
|                                                         settingsProvider |  | ||||||
|                                                             .getCategoryFormItem() |  | ||||||
|                                                       ] |  | ||||||
|                                                     ]); |  | ||||||
|                                               }).then((value) { |  | ||||||
|                                             if (value != null && app != null) { |  | ||||||
|                                               String? cat = (value['category'] |  | ||||||
|                                                           ?.isNotEmpty ?? |  | ||||||
|                                                       false) |  | ||||||
|                                                   ? value['category'] |  | ||||||
|                                                   : null; |  | ||||||
|                                               app.app.category = cat; |  | ||||||
|                                               appsProvider.saveApps([app.app]); |  | ||||||
|                                             } |  | ||||||
|                                           }); |  | ||||||
|                                         }, |  | ||||||
|                                         child: Text(tr('categorize'))) |  | ||||||
|                                   ]) |  | ||||||
|                       ], |                       ], | ||||||
|                     )), |                     )), | ||||||
|                   ], |                   ], | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import 'package:obtainium/components/custom_app_bar.dart'; | |||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/main.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| import 'package:obtainium/providers/logs_provider.dart'; | import 'package:obtainium/providers/logs_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| @@ -41,7 +42,6 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); |     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |     SourceProvider sourceProvider = SourceProvider(); | ||||||
|     AppsProvider appsProvider = context.read<AppsProvider>(); |  | ||||||
|     if (settingsProvider.prefs == null) { |     if (settingsProvider.prefs == null) { | ||||||
|       settingsProvider.initializeSettings(); |       settingsProvider.initializeSettings(); | ||||||
|     } |     } | ||||||
| @@ -130,6 +130,28 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  |     var localeDropdown = DropdownButtonFormField( | ||||||
|  |         decoration: InputDecoration(labelText: tr('language')), | ||||||
|  |         value: settingsProvider.forcedLocale, | ||||||
|  |         items: [ | ||||||
|  |           DropdownMenuItem( | ||||||
|  |             value: null, | ||||||
|  |             child: Text(tr('followSystem')), | ||||||
|  |           ), | ||||||
|  |           ...supportedLocales.map((e) => DropdownMenuItem( | ||||||
|  |                 value: e.toLanguageTag(), | ||||||
|  |                 child: Text(e.toLanguageTag().toUpperCase()), | ||||||
|  |               )) | ||||||
|  |         ], | ||||||
|  |         onChanged: (value) { | ||||||
|  |           settingsProvider.forcedLocale = value; | ||||||
|  |           if (value != null) { | ||||||
|  |             context.setLocale(Locale(value)); | ||||||
|  |           } else { | ||||||
|  |             context.resetLocale(); | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|     var intervalDropdown = DropdownButtonFormField( |     var intervalDropdown = DropdownButtonFormField( | ||||||
|         decoration: InputDecoration(labelText: tr('bgUpdateCheckInterval')), |         decoration: InputDecoration(labelText: tr('bgUpdateCheckInterval')), | ||||||
|         value: settingsProvider.updateInterval, |         value: settingsProvider.updateInterval, | ||||||
| @@ -178,8 +200,6 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|       height: 16, |       height: 16, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     var categories = settingsProvider.categories; |  | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|         backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|         body: CustomScrollView(slivers: <Widget>[ |         body: CustomScrollView(slivers: <Widget>[ | ||||||
| @@ -213,6 +233,8 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|                             height16, |                             height16, | ||||||
|  |                             localeDropdown, | ||||||
|  |                             height16, | ||||||
|                             Row( |                             Row( | ||||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|                               children: [ |                               children: [ | ||||||
| @@ -264,85 +286,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                   color: Theme.of(context).colorScheme.primary), |                                   color: Theme.of(context).colorScheme.primary), | ||||||
|                             ), |                             ), | ||||||
|                             height16, |                             height16, | ||||||
|                             Wrap( |                             const CategoryEditorSelector() | ||||||
|                               children: [ |  | ||||||
|                                 ...categories.entries.toList().map((e) { |  | ||||||
|                                   return Padding( |  | ||||||
|                                       padding: const EdgeInsets.symmetric( |  | ||||||
|                                           horizontal: 4), |  | ||||||
|                                       child: Chip( |  | ||||||
|                                         label: Text(e.key), |  | ||||||
|                                         backgroundColor: Color(e.value), |  | ||||||
|                                         visualDensity: VisualDensity.compact, |  | ||||||
|                                         onDeleted: () { |  | ||||||
|                                           showDialog<Map<String, dynamic>?>( |  | ||||||
|                                               context: context, |  | ||||||
|                                               builder: (BuildContext ctx) { |  | ||||||
|                                                 return GeneratedFormModal( |  | ||||||
|                                                     title: tr( |  | ||||||
|                                                         'deleteCategoryQuestion'), |  | ||||||
|                                                     message: tr( |  | ||||||
|                                                         'categoryDeleteWarning', |  | ||||||
|                                                         args: [e.key]), |  | ||||||
|                                                     items: []); |  | ||||||
|                                               }).then((value) { |  | ||||||
|                                             if (value != null) { |  | ||||||
|                                               setState(() { |  | ||||||
|                                                 categories.remove(e.key); |  | ||||||
|                                                 settingsProvider.categories = |  | ||||||
|                                                     categories; |  | ||||||
|                                               }); |  | ||||||
|                                               appsProvider.saveApps(appsProvider |  | ||||||
|                                                   .apps.values |  | ||||||
|                                                   .where((element) => |  | ||||||
|                                                       element.app.category == |  | ||||||
|                                                       e.key) |  | ||||||
|                                                   .map((e) { |  | ||||||
|                                                 var a = e.app; |  | ||||||
|                                                 a.category = null; |  | ||||||
|                                                 return a; |  | ||||||
|                                               }).toList()); |  | ||||||
|                                             } |  | ||||||
|                                           }); |  | ||||||
|                                         }, |  | ||||||
|                                       )); |  | ||||||
|                                 }), |  | ||||||
|                                 Padding( |  | ||||||
|                                     padding: const EdgeInsets.symmetric( |  | ||||||
|                                         horizontal: 4), |  | ||||||
|                                     child: IconButton( |  | ||||||
|                                       onPressed: () { |  | ||||||
|                                         showDialog<Map<String, dynamic>?>( |  | ||||||
|                                             context: context, |  | ||||||
|                                             builder: (BuildContext ctx) { |  | ||||||
|                                               return GeneratedFormModal( |  | ||||||
|                                                   title: tr('addCategory'), |  | ||||||
|                                                   items: [ |  | ||||||
|                                                     [ |  | ||||||
|                                                       GeneratedFormTextField( |  | ||||||
|                                                           'label', |  | ||||||
|                                                           label: tr('label')) |  | ||||||
|                                                     ] |  | ||||||
|                                                   ]); |  | ||||||
|                                             }).then((value) { |  | ||||||
|                                           String? label = value?['label']; |  | ||||||
|                                           if (label != null) { |  | ||||||
|                                             setState(() { |  | ||||||
|                                               categories[label] = |  | ||||||
|                                                   generateRandomLightColor() |  | ||||||
|                                                       .value; |  | ||||||
|                                               settingsProvider.categories = |  | ||||||
|                                                   categories; |  | ||||||
|                                             }); |  | ||||||
|                                           } |  | ||||||
|                                         }); |  | ||||||
|                                       }, |  | ||||||
|                                       icon: const Icon(Icons.add), |  | ||||||
|                                       visualDensity: VisualDensity.compact, |  | ||||||
|                                       tooltip: tr('add'), |  | ||||||
|                                     )) |  | ||||||
|                               ], |  | ||||||
|                             ) |  | ||||||
|                           ], |                           ], | ||||||
|                         ))), |                         ))), | ||||||
|           SliverToBoxAdapter( |           SliverToBoxAdapter( | ||||||
| @@ -457,3 +401,59 @@ class _LogsDialogState extends State<LogsDialog> { | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class CategoryEditorSelector extends StatefulWidget { | ||||||
|  |   final void Function(List<String> categories)? onSelected; | ||||||
|  |   final bool singleSelect; | ||||||
|  |   final Set<String> preselected; | ||||||
|  |   final WrapAlignment alignment; | ||||||
|  |   const CategoryEditorSelector( | ||||||
|  |       {super.key, | ||||||
|  |       this.onSelected, | ||||||
|  |       this.singleSelect = false, | ||||||
|  |       this.preselected = const {}, | ||||||
|  |       this.alignment = WrapAlignment.start}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   State<CategoryEditorSelector> createState() => _CategoryEditorSelectorState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _CategoryEditorSelectorState extends State<CategoryEditorSelector> { | ||||||
|  |   Map<String, MapEntry<int, bool>> storedValues = {}; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|  |     storedValues = settingsProvider.categories.map((key, value) => MapEntry( | ||||||
|  |         key, | ||||||
|  |         MapEntry(value, | ||||||
|  |             storedValues[key]?.value ?? widget.preselected.contains(key)))); | ||||||
|  |     return GeneratedForm( | ||||||
|  |         items: [ | ||||||
|  |           [ | ||||||
|  |             GeneratedFormTagInput('categories', | ||||||
|  |                 label: tr('category'), | ||||||
|  |                 emptyMessage: tr('noCategories'), | ||||||
|  |                 defaultValue: storedValues, | ||||||
|  |                 alignment: widget.alignment, | ||||||
|  |                 deleteConfirmationMessage: MapEntry( | ||||||
|  |                     tr('deleteCategoriesQuestion'), | ||||||
|  |                     tr('categoryDeleteWarning')), | ||||||
|  |                 singleSelect: widget.singleSelect) | ||||||
|  |           ] | ||||||
|  |         ], | ||||||
|  |         onValueChanges: ((values, valid, isBuilding) { | ||||||
|  |           if (!isBuilding) { | ||||||
|  |             storedValues = | ||||||
|  |                 values['categories'] as Map<String, MapEntry<int, bool>>; | ||||||
|  |             settingsProvider.categories = | ||||||
|  |                 storedValues.map((key, value) => MapEntry(key, value.key)); | ||||||
|  |             if (widget.onSelected != null) { | ||||||
|  |               widget.onSelected!(storedValues.keys | ||||||
|  |                   .where((k) => storedValues[k]!.value) | ||||||
|  |                   .toList()); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         })); | ||||||
|  |   } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import 'package:flutter/material.dart'; | |||||||
| import 'package:fluttertoast/fluttertoast.dart'; | import 'package:fluttertoast/fluttertoast.dart'; | ||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
|  | import 'package:obtainium/main.dart'; | ||||||
| import 'package:permission_handler/permission_handler.dart'; | import 'package:permission_handler/permission_handler.dart'; | ||||||
| import 'package:shared_preferences/shared_preferences.dart'; | import 'package:shared_preferences/shared_preferences.dart'; | ||||||
|  |  | ||||||
| @@ -153,6 +154,7 @@ class SettingsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|   set categories(Map<String, int> cats) { |   set categories(Map<String, int> cats) { | ||||||
|     prefs?.setString('categories', jsonEncode(cats)); |     prefs?.setString('categories', jsonEncode(cats)); | ||||||
|  |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getCategoryFormItem({String initCategory = ''}) => GeneratedFormDropdown( |   getCategoryFormItem({String initCategory = ''}) => GeneratedFormDropdown( | ||||||
| @@ -163,4 +165,24 @@ class SettingsProvider with ChangeNotifier { | |||||||
|         ...categories.entries.map((e) => MapEntry(e.key, e.key)).toList() |         ...categories.entries.map((e) => MapEntry(e.key, e.key)).toList() | ||||||
|       ], |       ], | ||||||
|       defaultValue: initCategory); |       defaultValue: initCategory); | ||||||
|  |  | ||||||
|  |   String? get forcedLocale { | ||||||
|  |     var fl = prefs?.getString('forcedLocale'); | ||||||
|  |     return supportedLocales | ||||||
|  |             .where((element) => element.toLanguageTag() == fl) | ||||||
|  |             .isNotEmpty | ||||||
|  |         ? fl | ||||||
|  |         : null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set forcedLocale(String? fl) { | ||||||
|  |     if (fl == null) { | ||||||
|  |       prefs?.remove('forcedLocale'); | ||||||
|  |     } else if (supportedLocales | ||||||
|  |         .where((element) => element.toLanguageTag() == fl) | ||||||
|  |         .isNotEmpty) { | ||||||
|  |       prefs?.setString('forcedLocale', fl); | ||||||
|  |     } | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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.9.2+90 # When changing this, update the tag in main() accordingly | version: 0.9.4+92 # 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