Compare commits
171 Commits
v0.8.18-be
...
v0.10.8-be
Author | SHA1 | Date | |
---|---|---|---|
1be38d361f | |||
32c40ae7b3 | |||
07223d81c7 | |||
78baee7265 | |||
348c33dfe9 | |||
c408d70ae6 | |||
3ae4e7cc8a | |||
dab0f2bb72 | |||
4baf6bcd3b | |||
db4517aa13 | |||
55d4d1f978 | |||
f89ac5965f | |||
d5ebaa161f | |||
a4c014a8bf | |||
bbaa42fb01 | |||
4fe311bc03 | |||
ea68b97ff7 | |||
6e0f6b528e | |||
a2c227931e | |||
15ad3bb439 | |||
b03d7fba1a | |||
31c491d7c5 | |||
71c80f11f5 | |||
eef4d33431 | |||
d56342e907 | |||
c72c0fdb57 | |||
ffe29009ed | |||
60e3b68ebd | |||
ee4d0f259f | |||
0ecfbef0a0 | |||
1b60e75ca7 | |||
abcfa389e8 | |||
a64bd67ef1 | |||
4252c2711b | |||
52913b0450 | |||
427b0ed8d2 | |||
a85d6d4f08 | |||
05f712603c | |||
fa2a80e34c | |||
f43e5a2ff1 | |||
b72aa8273e | |||
520f186e4a | |||
e1e97672cf | |||
1494bcd013 | |||
3457a0a12f | |||
b165400a6e | |||
c47bf937f1 | |||
2e19a8c04c | |||
05d4da86ec | |||
e9d1b04d54 | |||
cff5334c25 | |||
a55346fc22 | |||
885df678e5 | |||
bf7b0c5702 | |||
2972da4609 | |||
b8567af98e | |||
ea62c68b40 | |||
08a5af0449 | |||
36f327c16e | |||
768213cb34 | |||
e888fb7120 | |||
1fb68dd674 | |||
5c4bb8f84c | |||
1c8e759494 | |||
081c2a07d2 | |||
02751fe8fa | |||
95f3362a84 | |||
b68cf5a1be | |||
4eb7499591 | |||
98fafe2aa4 | |||
9bac74aadd | |||
0a93117bf0 | |||
451cc41c45 | |||
3b449d0982 | |||
1863f55372 | |||
0c4b8ac79d | |||
e287087753 | |||
82bcc46d42 | |||
1f26188ec6 | |||
794c3e1a81 | |||
16369b4adf | |||
8f16f745be | |||
8ddeb3d776 | |||
21cf9c98d9 | |||
358f910d19 | |||
7a3d74bd05 | |||
6f27f64699 | |||
3341fecb68 | |||
d3bce63ca4 | |||
8aa8b6b698 | |||
3d6c9bbf98 | |||
7af0a8628c | |||
4573ce6bcf | |||
e29d38fa32 | |||
dc82431235 | |||
424b0028bf | |||
46fba9e0a4 | |||
b40be7569b | |||
a173be11eb | |||
0c97b25d99 | |||
f836fd20d8 | |||
2f6917592d | |||
b864fef3ad | |||
8e487592b3 | |||
e9a44746a5 | |||
9123737bf3 | |||
12f70951c2 | |||
c1d56f89f0 | |||
4dfd29f5de | |||
226cfa25e0 | |||
4e0c655538 | |||
45a23e9025 | |||
1e5aa0999a | |||
beeec356e5 | |||
01fa9a2e96 | |||
0da7a36f1a | |||
ed2a4e674f | |||
0f6a683faa | |||
fa4d46b622 | |||
a3f9947f28 | |||
6977858b99 | |||
2ff6acb701 | |||
0c2d6ce84d | |||
9072862862 | |||
3cbaac2f5d | |||
0f8871efcb | |||
ee216cbbba | |||
ebe5b79dc5 | |||
60014c864c | |||
070b6033bd | |||
626bebbe5a | |||
118460ccb9 | |||
26f953dbb0 | |||
99d7595f2d | |||
e2f99c5e71 | |||
1f582d239b | |||
5e6b00718e | |||
56594e6b19 | |||
bbcc3ff9b3 | |||
ee66c53320 | |||
b7d581f8b0 | |||
ead63ba21d | |||
c69404363f | |||
99d0bd2461 | |||
54efda3eea | |||
d76d68329c | |||
b151eb27e1 | |||
6a21045e5b | |||
6aedd9ce37 | |||
f319639a99 | |||
92e6798809 | |||
9a129d41df | |||
0c2654a226 | |||
afc8e41171 | |||
1fe9e4f91e | |||
dbd6dec0a6 | |||
d068db2a57 | |||
dd5c5fd2bc | |||
ac9dadd9d0 | |||
bb0540b644 | |||
819334021a | |||
8ece0bbef9 | |||
6a41283e74 | |||
e6d5c7db3e | |||
d4c016d8ee | |||
63034dd3f9 | |||
67b986de93 | |||
aafe4bc515 | |||
e524335900 | |||
77751fa03f | |||
8fac67c9e9 |
19
README.md
@ -1,4 +1,4 @@
|
|||||||
#  Obtainium
|
#  Obtainium
|
||||||
|
|
||||||
Get Android App Updates Directly From the Source.
|
Get Android App Updates Directly From the Source.
|
||||||
|
|
||||||
@ -9,14 +9,27 @@ Motivation: [Side Of Burritos - You should use this instead of F-Droid | How to
|
|||||||
Currently supported App sources:
|
Currently supported App sources:
|
||||||
- [GitHub](https://github.com/)
|
- [GitHub](https://github.com/)
|
||||||
- [GitLab](https://gitlab.com/)
|
- [GitLab](https://gitlab.com/)
|
||||||
|
- [Codeberg](https://codeberg.org/)
|
||||||
- [F-Droid](https://f-droid.org/)
|
- [F-Droid](https://f-droid.org/)
|
||||||
- [IzzyOnDroid](https://android.izzysoft.de/)
|
- [IzzyOnDroid](https://android.izzysoft.de/)
|
||||||
- [Mullvad](https://mullvad.net/en/)
|
- [Mullvad](https://mullvad.net/en/)
|
||||||
- [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)
|
||||||
|
- "HTML" (Fallback)
|
||||||
|
- Any other URL that returns an HTML page with links to APK files (if multiple, the last file alphabetically is picked)
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
[<img src="https://github.com/machiav3lli/oandbackupx/blob/034b226cea5c1b30eb4f6a6f313e4dadcbb0ece4/badge_github.png"
|
||||||
|
alt="Get it on GitHub"
|
||||||
|
height="80">](https://github.com/ImranR98/Obtainium/releases)
|
||||||
|
[<img src="https://gitlab.com/IzzyOnDroid/repo/-/raw/master/assets/IzzyOnDroid.png"
|
||||||
|
alt="Get it on IzzyOnDroid"
|
||||||
|
height="80">](https://apt.izzysoft.de/fdroid/index/apk/dev.imranr.obtainium)
|
||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
- App installs happen asynchronously and the success/failure of an install cannot be determined directly. This results in install statuses and versions sometimes being out of sync with the OS until the next launch or until the problem is manually corrected.
|
- App installs happen asynchronously and the success/failure of an install cannot be determined directly. This results in install statuses and versions sometimes being out of sync with the OS until the next launch or until the problem is manually corrected.
|
||||||
@ -27,4 +40,4 @@ Currently supported App sources:
|
|||||||
|
|
||||||
| <img src="./assets/screenshots/1.apps.png" alt="Apps Page" /> | <img src="./assets/screenshots/2.dark_theme.png" alt="Dark Theme" /> | <img src="./assets/screenshots/3.material_you.png" alt="Material You" /> |
|
| <img src="./assets/screenshots/1.apps.png" alt="Apps Page" /> | <img src="./assets/screenshots/2.dark_theme.png" alt="Dark Theme" /> | <img src="./assets/screenshots/3.material_you.png" alt="Material You" /> |
|
||||||
| ------------------------------------------------------ | ----------------------------------------------------------------------- | -------------------------------------------------------------------- |
|
| ------------------------------------------------------ | ----------------------------------------------------------------------- | -------------------------------------------------------------------- |
|
||||||
| <img src="./assets/screenshots/4.app.png" alt="App Page" /> | <img src="./assets/screenshots/5.apk_picker.png" alt="Multiple APK Support" /> | <img src="./assets/screenshots/6.apk_install.png" alt="App Installation" /> |
|
| <img src="./assets/screenshots/4.app.png" alt="App Page" /> | <img src="./assets/screenshots/5.app_opts.png" alt="App Options" /> | <img src="./assets/screenshots/6.app_webview.png" alt="App Web View" /> |
|
||||||
|
@ -3,7 +3,8 @@
|
|||||||
<application
|
<application
|
||||||
android:label="Obtainium"
|
android:label="Obtainium"
|
||||||
android:name="${applicationName}"
|
android:name="${applicationName}"
|
||||||
android:icon="@mipmap/ic_launcher">
|
android:icon="@mipmap/ic_launcher"
|
||||||
|
android:requestLegacyExternalStorage="true">
|
||||||
<activity
|
<activity
|
||||||
android:name=".MainActivity"
|
android:name=".MainActivity"
|
||||||
android:exported="true"
|
android:exported="true"
|
||||||
@ -51,4 +52,8 @@
|
|||||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
|
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
|
||||||
<uses-permission android:name="android.permission.WAKE_LOCK"/>
|
<uses-permission android:name="android.permission.WAKE_LOCK"/>
|
||||||
<uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/>
|
<uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/>
|
||||||
|
<uses-permission android:name="android.permission.REQUEST_DELETE_PACKAGES" />
|
||||||
|
<uses-permission
|
||||||
|
android:name="android.permission.WRITE_EXTERNAL_STORAGE"
|
||||||
|
android:maxSdkVersion="29"/>
|
||||||
</manifest>
|
</manifest>
|
Before Width: | Height: | Size: 7.6 KiB After Width: | Height: | Size: 8.1 KiB |
@ -2,4 +2,5 @@
|
|||||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
<background android:drawable="@color/ic_launcher_background"/>
|
<background android:drawable="@color/ic_launcher_background"/>
|
||||||
<foreground android:drawable="@drawable/ic_launcher_foreground"/>
|
<foreground android:drawable="@drawable/ic_launcher_foreground"/>
|
||||||
|
<monochrome android:drawable="@drawable/ic_launcher_foreground"/>
|
||||||
</adaptive-icon>
|
</adaptive-icon>
|
||||||
|
BIN
assets/graphics/icon_small.png
Normal file
After Width: | Height: | Size: 7.6 KiB |
Before Width: | Height: | Size: 228 KiB After Width: | Height: | Size: 234 KiB |
Before Width: | Height: | Size: 162 KiB After Width: | Height: | Size: 238 KiB |
Before Width: | Height: | Size: 170 KiB After Width: | Height: | Size: 140 KiB |
Before Width: | Height: | Size: 146 KiB After Width: | Height: | Size: 139 KiB |
Before Width: | Height: | Size: 188 KiB |
BIN
assets/screenshots/5.app_opts.png
Normal file
After Width: | Height: | Size: 118 KiB |
Before Width: | Height: | Size: 192 KiB |
BIN
assets/screenshots/6.app_webview.png
Normal file
After Width: | Height: | Size: 262 KiB |
263
assets/translations/de.json
Normal file
@ -0,0 +1,263 @@
|
|||||||
|
{
|
||||||
|
"invalidURLForSource": "Keine gültige {} App-URL",
|
||||||
|
"noReleaseFound": "Keine passende Version gefunden",
|
||||||
|
"noVersionFound": "Release-Version nicht ermittelbar",
|
||||||
|
"urlMatchesNoSource": "URL stimmt mit keiner bekannten Quelle überein",
|
||||||
|
"cantInstallOlderVersion": "Installation einer älteren App-Version nicht möglich",
|
||||||
|
"functionNotImplemented": "Diese Klasse hat diese Funktion nicht implementiert",
|
||||||
|
"placeholder": "Platzhalter",
|
||||||
|
"someErrors": "Es traten einige Fehler auf",
|
||||||
|
"unexpectedError": "Unerwarteter Fehler",
|
||||||
|
"ok": "Okay",
|
||||||
|
"and": "und",
|
||||||
|
"startedBgUpdateTask": "Hintergrundaktualisierungsprüfung gestartet",
|
||||||
|
"bgUpdateIgnoreAfterIs": "Hintergrundaktualisierung 'ignoreAfter' ist {}",
|
||||||
|
"startedActualBGUpdateCheck": "Überprüfung der Hintergrundaktualisierung gestartet",
|
||||||
|
"bgUpdateTaskFinished": "Hintergrundaktualisierungsprüfung abgeschlossen",
|
||||||
|
"firstRun": "Dies ist der erste Start von Obtainium überhaupt",
|
||||||
|
"settingUpdateCheckIntervalTo": "Aktualisierungsintervall auf {} stellen",
|
||||||
|
"githubPATLabel": "GitHub Personal Access Token (Erhöht das Ratenlimit)",
|
||||||
|
"githubPATHint": "PAT muss in diesem Format sein: Benutzername:Token",
|
||||||
|
"githubPATFormat": "Benutzername:Token",
|
||||||
|
"githubPATLinkText": "Über GitHub PATs",
|
||||||
|
"includePrereleases": "Vorabversionen einbeziehen",
|
||||||
|
"fallbackToOlderReleases": "Fallback auf ältere Versionen",
|
||||||
|
"filterReleaseTitlesByRegEx": "Release-Titel nach regulärem Ausdruck\nfiltern",
|
||||||
|
"invalidRegEx": "Ungültiger regulärer Ausdruck",
|
||||||
|
"noDescription": "Keine Beschreibung",
|
||||||
|
"cancel": "Abbrechen",
|
||||||
|
"continue": "Weiter",
|
||||||
|
"requiredInBrackets": "(Benötigt)",
|
||||||
|
"dropdownNoOptsError": "FEHLER: DROPDOWN MUSS MINDESTENS EINE OPTION HABEN",
|
||||||
|
"colour": "Farbe",
|
||||||
|
"githubStarredRepos": "GitHub Starred Repos",
|
||||||
|
"uname": "Benutzername",
|
||||||
|
"wrongArgNum": "Falsche Anzahl von Argumenten übermittelt",
|
||||||
|
"xIsTrackOnly": "{} ist nur zur Nachverfolgung",
|
||||||
|
"source": "Quelle",
|
||||||
|
"app": "App",
|
||||||
|
"appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind 'Nur Nachverfolgen'.",
|
||||||
|
"youPickedTrackOnly": "Sie haben die Option 'Nur Nachverfolgen' gewählt.",
|
||||||
|
"trackOnlyAppDescription": "Die App wird auf Updates überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.",
|
||||||
|
"cancelled": "Abgebrochen",
|
||||||
|
"appAlreadyAdded": "App bereits hinzugefügt",
|
||||||
|
"alreadyUpToDateQuestion": "App bereits auf dem neuesten Stand?",
|
||||||
|
"addApp": "App hinzufügen",
|
||||||
|
"appSourceURL": "Quell-URL der App",
|
||||||
|
"error": "Fehler",
|
||||||
|
"add": "Hinzufügen",
|
||||||
|
"searchSomeSourcesLabel": "Suche (nur bestimmte Quellen)",
|
||||||
|
"search": "Suchen",
|
||||||
|
"additionalOptsFor": "Zusatzoptionen für {}",
|
||||||
|
"supportedSourcesBelow": "Unterstützte Quellen:",
|
||||||
|
"trackOnlyInBrackets": "(Nur Nachverfolgen)",
|
||||||
|
"searchableInBrackets": "(Durchsuchbar)",
|
||||||
|
"appsString": "Apps",
|
||||||
|
"noApps": "Keine Apps",
|
||||||
|
"noAppsForFilter": "Keine Apps für ausgewählten Filter",
|
||||||
|
"byX": "Von {}",
|
||||||
|
"percentProgress": "Fortschritt: {}%",
|
||||||
|
"pleaseWait": "Bitte warten",
|
||||||
|
"updateAvailable": "Aktualisierung verfügbar",
|
||||||
|
"estimateInBracketsShort": "(ca.)",
|
||||||
|
"notInstalled": "Nicht installiert",
|
||||||
|
"estimateInBrackets": "(Ungefähr)",
|
||||||
|
"selectAll": "Alle auswählen",
|
||||||
|
"deselectN": "{} abgewählt",
|
||||||
|
"xWillBeRemovedButRemainInstalled": "{} wird aus Obtainium entfernt, bleibt aber auf dem Gerät installiert.",
|
||||||
|
"removeSelectedAppsQuestion": "Ausgewählte Apps entfernen?",
|
||||||
|
"removeSelectedApps": "Ausgewählte Apps entfernen",
|
||||||
|
"updateX": "Aktualisiere {}",
|
||||||
|
"installX": "Installiere {}",
|
||||||
|
"markXTrackOnlyAsUpdated": "Markiere {}\n(Nur Nachverfolgen)\nals aktualisiert",
|
||||||
|
"changeX": "Ändern {}",
|
||||||
|
"installUpdateApps": "Apps installieren/aktualisieren",
|
||||||
|
"installUpdateSelectedApps": "Ausgewählte Apps installieren/aktualisieren",
|
||||||
|
"markXSelectedAppsAsUpdated": "Markiere {} ausgewählte Apps als aktuell?",
|
||||||
|
"no": "Nein",
|
||||||
|
"yes": "Ja",
|
||||||
|
"markSelectedAppsUpdated": "Markiere ausgewählte Apps als aktuell",
|
||||||
|
"pinToTop": "Oben anheften",
|
||||||
|
"unpinFromTop": "'Oben anheften' aufheben",
|
||||||
|
"resetInstallStatusForSelectedAppsQuestion": "Installationsstatus für ausgewählte Apps zurücksetzen?",
|
||||||
|
"installStatusOfXWillBeResetExplanation": "Der Installationsstatus der ausgewählten Apps wird zurückgesetzt. Dies kann hilfreich sein, wenn die in Obtainium angezeigte App-Version aufgrund fehlgeschlagener Aktualisierungen oder anderer Probleme falsch ist.",
|
||||||
|
"shareSelectedAppURLs": "Ausgewählte App-URLs teilen",
|
||||||
|
"resetInstallStatus": "Installationsstatus zurücksetzen",
|
||||||
|
"more": "Mehr",
|
||||||
|
"removeOutdatedFilter": "App-Filter 'Nicht aktuell' entfernen",
|
||||||
|
"showOutdatedOnly": "Nur nicht aktuelle Apps anzeigen",
|
||||||
|
"filter": "Filter",
|
||||||
|
"filterActive": "Filter *",
|
||||||
|
"filterApps": "Apps filtern",
|
||||||
|
"appName": "App Name",
|
||||||
|
"author": "Autor",
|
||||||
|
"upToDateApps": "Apps mit aktueller Version",
|
||||||
|
"nonInstalledApps": "Nicht installierte Apps",
|
||||||
|
"importExport": "Import/Export",
|
||||||
|
"settings": "Einstellungen",
|
||||||
|
"exportedTo": "Exportiert zu {}",
|
||||||
|
"obtainiumExport": "Obtainium Export",
|
||||||
|
"invalidInput": "Ungültige Eingabe",
|
||||||
|
"importedX": "Importiert {}",
|
||||||
|
"obtainiumImport": "Obtainium Import",
|
||||||
|
"importFromURLList": "Importieren aus URL-Liste",
|
||||||
|
"searchQuery": "Suchanfrage",
|
||||||
|
"appURLList": "App URL-Liste",
|
||||||
|
"line": "Linie",
|
||||||
|
"searchX": "Suche {}",
|
||||||
|
"noResults": "Keine Ergebnisse gefunden",
|
||||||
|
"importX": "Import {}",
|
||||||
|
"importedAppsIdDisclaimer": "Importierte Apps werden möglicherweise fälschlicherweise als \"Nicht installiert\" angezeigt. Um dies zu beheben, installieren Sie sie erneut über Obtainium. Dies hat keine Auswirkungen auf App-Daten. Es betrifft nur URL- und Drittanbieter-Importmethoden.",
|
||||||
|
"importErrors": "Importfehler",
|
||||||
|
"importedXOfYApps": "{} von {} Apps importiert.",
|
||||||
|
"followingURLsHadErrors": "Bei folgenden URLs traten Fehler auf:",
|
||||||
|
"okay": "Okay",
|
||||||
|
"selectURL": "URL auswählen",
|
||||||
|
"selectURLs": "URLs auswählen",
|
||||||
|
"pick": "Auswählen",
|
||||||
|
"theme": "Theme",
|
||||||
|
"dark": "Dunkel",
|
||||||
|
"light": "Hell",
|
||||||
|
"followSystem": "System folgen",
|
||||||
|
"obtainium": "Obtainium",
|
||||||
|
"materialYou": "Material You",
|
||||||
|
"appSortBy": "App sortieren nach",
|
||||||
|
"authorName": "Autor/Name",
|
||||||
|
"nameAuthor": "Name/Autor",
|
||||||
|
"asAdded": "Wie hinzugefügt",
|
||||||
|
"appSortOrder": "App Sortierung nach",
|
||||||
|
"ascending": "Aufsteigend",
|
||||||
|
"descending": "Absteigend",
|
||||||
|
"bgUpdateCheckInterval": "Prüfintervall für Hintergrundaktualisierung",
|
||||||
|
"neverManualOnly": "Nie - nur manuell",
|
||||||
|
"appearance": "Aussehen",
|
||||||
|
"showWebInAppView": "Quellwebseite in der App-Ansicht anzeigen",
|
||||||
|
"pinUpdates": "Apps mit Aktualisierungen oben anheften",
|
||||||
|
"updates": "Aktualisierungen",
|
||||||
|
"sourceSpecific": "Quellenspezifisch",
|
||||||
|
"appSource": "App-Quelle",
|
||||||
|
"noLogs": "Keine Protokolle",
|
||||||
|
"appLogs": "App Protokolle",
|
||||||
|
"close": "Schließen",
|
||||||
|
"share": "Teilen",
|
||||||
|
"appNotFound": "App nicht gefunden",
|
||||||
|
"obtainiumExportHyphenatedLowercase": "obtainium-export",
|
||||||
|
"pickAnAPK": "APK auswählen",
|
||||||
|
"appHasMoreThanOnePackage": "{} verfügt über mehr als ein Paket:",
|
||||||
|
"deviceSupportsXArch": "Ihr Gerät unterstützt die CPU-Architektur {}.",
|
||||||
|
"deviceSupportsFollowingArchs": "Ihr Gerät unterstützt die folgenden CPU-Architekturen:",
|
||||||
|
"warning": "Warnung",
|
||||||
|
"sourceIsXButPackageFromYPrompt": "Die App-Quelle ist '{}', aber das Release-Paket stammt von '{}'. Fortfahren?",
|
||||||
|
"updatesAvailable": "Aktualisierungen verfügbar",
|
||||||
|
"updatesAvailableNotifDescription": "Benachrichtigt den Nutzer, dass Aktualisierungen für eine oder mehrere von Obtainium verfolgte Apps verfügbar sind",
|
||||||
|
"noNewUpdates": "Keine neuen Aktualisierungen.",
|
||||||
|
"xHasAnUpdate": "{} hat eine Aktualisierung.",
|
||||||
|
"appsUpdated": "Apps aktualisiert",
|
||||||
|
"appsUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps im Hintergrund durchgeführt wurden",
|
||||||
|
"xWasUpdatedToY": "{} wurde auf {} aktualisiert.",
|
||||||
|
"errorCheckingUpdates": "Fehler beim Prüfen auf Aktualisierungen",
|
||||||
|
"errorCheckingUpdatesNotifDescription": "Eine Benachrichtigung, die angezeigt wird, wenn die Prüfung der Hintergrundaktualisierung fehlschlägt",
|
||||||
|
"appsRemoved": "Apps entfernt",
|
||||||
|
"appsRemovedNotifDescription": "Benachrichtigt den Benutzer, dass eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden",
|
||||||
|
"xWasRemovedDueToErrorY": "{} wurde aufgrund des folgenden Fehlers entfernt: {}",
|
||||||
|
"completeAppInstallation": "App Installation abschließen",
|
||||||
|
"obtainiumMustBeOpenToInstallApps": "Obtainium muss geöffnet sein, um Apps zu installieren",
|
||||||
|
"completeAppInstallationNotifDescription": "Aufforderung an den Benutzer, zu Obtainium zurückzukehren, um die Installation einer App abzuschließen",
|
||||||
|
"checkingForUpdates": "Nach Aktualisierungen suchen",
|
||||||
|
"checkingForUpdatesNotifDescription": "Vorübergehende Benachrichtigung, die bei der Suche nach Aktualisierungen angezeigt wird",
|
||||||
|
"pleaseAllowInstallPerm": "Bitte erlauben Sie Obtainium die Installation von Apps",
|
||||||
|
"trackOnly": "Nur Nachverfolgen",
|
||||||
|
"errorWithHttpStatusCode": "Fehler {}",
|
||||||
|
"versionCorrectionDisabled": "Versionskorrektur deaktiviert (Plugin scheint nicht zu funktionieren)",
|
||||||
|
"unknown": "Unbekannt",
|
||||||
|
"none": "Keine",
|
||||||
|
"never": "Nie",
|
||||||
|
"latestVersionX": "Neueste Version: {}",
|
||||||
|
"installedVersionX": "Installierte Version: {}",
|
||||||
|
"lastUpdateCheckX": "Letzte Aktualisierungsprüfung: {}",
|
||||||
|
"remove": "Entfernen",
|
||||||
|
"yesMarkUpdated": "Ja, als aktualisiert markieren",
|
||||||
|
"fdroid": "F-Droid",
|
||||||
|
"appIdOrName": "App ID oder Name",
|
||||||
|
"appWithIdOrNameNotFound": "Es wurde keine App mit dieser ID oder diesem Namen gefunden",
|
||||||
|
"reposHaveMultipleApps": "Repos können mehrere Apps enthalten",
|
||||||
|
"fdroidThirdPartyRepo": "F-Droid Third-Party Repo",
|
||||||
|
"steam": "Steam",
|
||||||
|
"steamMobile": "Steam Mobile",
|
||||||
|
"steamChat": "Steam Chat",
|
||||||
|
"install": "Installieren",
|
||||||
|
"markInstalled": "Als Installiert markieren",
|
||||||
|
"update": "Aktualisieren",
|
||||||
|
"markUpdated": "Als Aktuell markieren",
|
||||||
|
"additionalOptions": "Zusätzliche Optionen",
|
||||||
|
"disableVersionDetection": "Versionsermittlung deaktivieren",
|
||||||
|
"noVersionDetectionExplanation": "Diese Option sollte nur für Apps verwendet werden, bei denen die Versionserkennung nicht korrekt funktioniert.",
|
||||||
|
"downloadingX": "Lade {} herunter",
|
||||||
|
"downloadNotifDescription": "Benachrichtigt den Nutzer über den Fortschritt beim Herunterladen einer App",
|
||||||
|
"noAPKFound": "Keine APK gefunden",
|
||||||
|
"noVersionDetection": "Keine Versionserkennung",
|
||||||
|
"categorize": "Kategorisieren",
|
||||||
|
"categories": "Kategorien",
|
||||||
|
"category": "Kategorie",
|
||||||
|
"noCategory": "Keine Kategorie",
|
||||||
|
"noCategories": "Keine Kategorien",
|
||||||
|
"deleteCategoriesQuestion": "Kategorien löschen?",
|
||||||
|
"categoryDeleteWarning": "Alle Apps in gelöschten Kategorien werden auf nicht kategorisiert gesetzt.",
|
||||||
|
"addCategory": "Kategorie hinzufügen",
|
||||||
|
"label": "Bezeichnung",
|
||||||
|
"language": "Sprache",
|
||||||
|
"storagePermissionDenied": "Storage permission denied",
|
||||||
|
"selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.",
|
||||||
|
"filterAPKsByRegEx": "Filter APKs by Regular Expression",
|
||||||
|
"removeFromObtainium": "Remove from Obtainium",
|
||||||
|
"uninstallFromDevice": "Uninstall from Device",
|
||||||
|
"onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.",
|
||||||
|
"removeAppQuestion": {
|
||||||
|
"one": "App entfernen?",
|
||||||
|
"other": "App entfernen?"
|
||||||
|
},
|
||||||
|
"tooManyRequestsTryAgainInMinutes": {
|
||||||
|
"one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut",
|
||||||
|
"other": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minuten erneut"
|
||||||
|
},
|
||||||
|
"bgUpdateGotErrorRetryInMinutes": {
|
||||||
|
"one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant",
|
||||||
|
"other": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minuten geplant"
|
||||||
|
},
|
||||||
|
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
||||||
|
"one": "Hintergrundaktualisierungsprüfung fand {} Aktualisierung - benachrichtigt den Benutzer, falls erforderlich",
|
||||||
|
"other": "Hintergrundaktualisierungsprüfung fand {} Aktualisierungen - benachrichtigt den Benutzer, falls erforderlich"
|
||||||
|
},
|
||||||
|
"apps": {
|
||||||
|
"one": "{} App",
|
||||||
|
"other": "{} Apps"
|
||||||
|
},
|
||||||
|
"url": {
|
||||||
|
"one": "{} URL",
|
||||||
|
"other": "{} URLs"
|
||||||
|
},
|
||||||
|
"minute": {
|
||||||
|
"one": "{} Minute",
|
||||||
|
"other": "{} Minutes"
|
||||||
|
},
|
||||||
|
"hour": {
|
||||||
|
"one": "{} Stunde",
|
||||||
|
"other": "{} Stunden"
|
||||||
|
},
|
||||||
|
"day": {
|
||||||
|
"one": "{} Tag",
|
||||||
|
"other": "{} Tage"
|
||||||
|
},
|
||||||
|
"clearedNLogsBeforeXAfterY": {
|
||||||
|
"one": "{n} Protokoll gelöscht (vorher = {vorher}, nachher = {nachher})",
|
||||||
|
"other": "{n} Protokolle gelöscht (vorher = {vorher}, nachher = {nachher})"
|
||||||
|
},
|
||||||
|
"xAndNMoreUpdatesAvailable": {
|
||||||
|
"one": "{} und 1 weitere App haben Aktualisierungen.",
|
||||||
|
"other": "{} und {} weitere Apps haben Aktualisierungen."
|
||||||
|
},
|
||||||
|
"xAndNMoreUpdatesInstalled": {
|
||||||
|
"one": "{} und 1 weitere Anwendung wurden aktualisiert.",
|
||||||
|
"other": "{} und {} weitere Anwendungen wurden aktualisiert."
|
||||||
|
}
|
||||||
|
}
|
@ -4,7 +4,6 @@
|
|||||||
"noVersionFound": "Could not determine release version",
|
"noVersionFound": "Could not determine release version",
|
||||||
"urlMatchesNoSource": "URL does not match a known source",
|
"urlMatchesNoSource": "URL does not match a known source",
|
||||||
"cantInstallOlderVersion": "Cannot install an older version of an App",
|
"cantInstallOlderVersion": "Cannot install an older version of an App",
|
||||||
"appIdMismatch": "Downloaded package ID does not match existing App ID",
|
|
||||||
"functionNotImplemented": "This class has not implemented this function",
|
"functionNotImplemented": "This class has not implemented this function",
|
||||||
"placeholder": "Placeholder",
|
"placeholder": "Placeholder",
|
||||||
"someErrors": "Some Errors Occurred",
|
"someErrors": "Some Errors Occurred",
|
||||||
@ -37,7 +36,7 @@
|
|||||||
"xIsTrackOnly": "{} is Track-Only",
|
"xIsTrackOnly": "{} is Track-Only",
|
||||||
"source": "Source",
|
"source": "Source",
|
||||||
"app": "App",
|
"app": "App",
|
||||||
"appsFromSourceAreTrackOnly": "Apps from this source are 'Track-Only'.' ",
|
"appsFromSourceAreTrackOnly": "Apps from this source are 'Track-Only'.",
|
||||||
"youPickedTrackOnly": "You have selected the 'Track-Only' option.",
|
"youPickedTrackOnly": "You have selected the 'Track-Only' option.",
|
||||||
"trackOnlyAppDescription": "The App will be tracked for updates, but Obtainium will not be able to download or install it.",
|
"trackOnlyAppDescription": "The App will be tracked for updates, but Obtainium will not be able to download or install it.",
|
||||||
"cancelled": "Cancelled",
|
"cancelled": "Cancelled",
|
||||||
@ -74,7 +73,6 @@
|
|||||||
"changeX": "Change {}",
|
"changeX": "Change {}",
|
||||||
"installUpdateApps": "Install/Update Apps",
|
"installUpdateApps": "Install/Update Apps",
|
||||||
"installUpdateSelectedApps": "Install/Update Selected Apps",
|
"installUpdateSelectedApps": "Install/Update Selected Apps",
|
||||||
"onlyWorksWithNonEVDApps": "Only works for Apps whose install status cannot be automatically detected (uncommon).",
|
|
||||||
"markXSelectedAppsAsUpdated": "Mark {} Selected Apps as Updated?",
|
"markXSelectedAppsAsUpdated": "Mark {} Selected Apps as Updated?",
|
||||||
"no": "No",
|
"no": "No",
|
||||||
"yes": "Yes",
|
"yes": "Yes",
|
||||||
@ -135,7 +133,7 @@
|
|||||||
"appearance": "Appearance",
|
"appearance": "Appearance",
|
||||||
"showWebInAppView": "Show Source Webpage in App View",
|
"showWebInAppView": "Show Source Webpage in App View",
|
||||||
"pinUpdates": "Pin Updates to Top of Apps View",
|
"pinUpdates": "Pin Updates to Top of Apps View",
|
||||||
"updates": "Updated",
|
"updates": "Updates",
|
||||||
"sourceSpecific": "Source-Specific",
|
"sourceSpecific": "Source-Specific",
|
||||||
"appSource": "App Source",
|
"appSource": "App Source",
|
||||||
"noLogs": "No Logs",
|
"noLogs": "No Logs",
|
||||||
@ -178,7 +176,6 @@
|
|||||||
"installedVersionX": "Installed Version: {}",
|
"installedVersionX": "Installed Version: {}",
|
||||||
"lastUpdateCheckX": "Last Update Check: {}",
|
"lastUpdateCheckX": "Last Update Check: {}",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"removeAppQuestion": "Remove App?",
|
|
||||||
"yesMarkUpdated": "Yes, Mark as Updated",
|
"yesMarkUpdated": "Yes, Mark as Updated",
|
||||||
"fdroid": "F-Droid",
|
"fdroid": "F-Droid",
|
||||||
"appIdOrName": "App ID or Name",
|
"appIdOrName": "App ID or Name",
|
||||||
@ -188,6 +185,37 @@
|
|||||||
"steam": "Steam",
|
"steam": "Steam",
|
||||||
"steamMobile": "Steam Mobile",
|
"steamMobile": "Steam Mobile",
|
||||||
"steamChat": "Steam Chat",
|
"steamChat": "Steam Chat",
|
||||||
|
"install": "Install",
|
||||||
|
"markInstalled": "Mark Installed",
|
||||||
|
"update": "Update",
|
||||||
|
"markUpdated": "Mark Updated",
|
||||||
|
"additionalOptions": "Additional Options",
|
||||||
|
"disableVersionDetection": "Disable Version Detection",
|
||||||
|
"noVersionDetectionExplanation": "This option should only be used for Apps where version detection does not work correctly.",
|
||||||
|
"downloadingX": "Downloading {}",
|
||||||
|
"downloadNotifDescription": "Notifies the user of the progress in downloading an App",
|
||||||
|
"noAPKFound": "No APK found",
|
||||||
|
"noVersionDetection": "No version detection",
|
||||||
|
"categorize": "Categorize",
|
||||||
|
"categories": "Categories",
|
||||||
|
"category": "Category",
|
||||||
|
"noCategory": "No Category",
|
||||||
|
"noCategories": "No Categories",
|
||||||
|
"deleteCategoriesQuestion": "Delete Categories?",
|
||||||
|
"categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.",
|
||||||
|
"addCategory": "Add Category",
|
||||||
|
"label": "Label",
|
||||||
|
"language": "Language",
|
||||||
|
"storagePermissionDenied": "Storage permission denied",
|
||||||
|
"selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.",
|
||||||
|
"filterAPKsByRegEx": "Filter APKs by Regular Expression",
|
||||||
|
"removeFromObtainium": "Remove from Obtainium",
|
||||||
|
"uninstallFromDevice": "Uninstall from Device",
|
||||||
|
"onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.",
|
||||||
|
"removeAppQuestion": {
|
||||||
|
"one": "Remove App?",
|
||||||
|
"other": "Remove Apps?"
|
||||||
|
},
|
||||||
"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"
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
"noVersionFound": "Nem sikerült meghatározni a kiadás verzióját",
|
"noVersionFound": "Nem sikerült meghatározni a kiadás verzióját",
|
||||||
"urlMatchesNoSource": "Az URL nem egyezik ismert forrással",
|
"urlMatchesNoSource": "Az URL nem egyezik ismert forrással",
|
||||||
"cantInstallOlderVersion": "Nem telepíthető egy app régebbi verziója",
|
"cantInstallOlderVersion": "Nem telepíthető egy app régebbi verziója",
|
||||||
"appIdMismatch": "A letöltött csomagazonosító nem egyezik a meglévő app azonosítóval",
|
|
||||||
"functionNotImplemented": "Ez az osztály nem valósította meg ezt a függvényt",
|
"functionNotImplemented": "Ez az osztály nem valósította meg ezt a függvényt",
|
||||||
"placeholder": "Helykitöltő",
|
"placeholder": "Helykitöltő",
|
||||||
"someErrors": "Néhány hiba történt",
|
"someErrors": "Néhány hiba történt",
|
||||||
@ -13,10 +12,10 @@
|
|||||||
"and": "és",
|
"and": "és",
|
||||||
"startedBgUpdateTask": "Háttérfrissítés ellenőrzési feladat elindítva",
|
"startedBgUpdateTask": "Háttérfrissítés ellenőrzési feladat elindítva",
|
||||||
"bgUpdateIgnoreAfterIs": "Háttérfrissítés ignoreAfter a következő: {}",
|
"bgUpdateIgnoreAfterIs": "Háttérfrissítés ignoreAfter a következő: {}",
|
||||||
"startedActualBGUpdateCheck": "Elkezdődött a tényleges BG frissítés ellenőrzése",
|
"startedActualBGUpdateCheck": "Elkezdődött a tényleges háttérfrissítés ellenőrzése",
|
||||||
"bgUpdateTaskFinished": "A háttérfrissítés ellenőrzési feladat befejeződött",
|
"bgUpdateTaskFinished": "A háttérfrissítés ellenőrzési feladat befejeződött",
|
||||||
"firstRun": "Ez az Obtainium első futása",
|
"firstRun": "Ez az Obtainium első futása",
|
||||||
"settingUpdateCheckIntervalTo": "A frissítési intervallum beállítása a erre: {}",
|
"settingUpdateCheckIntervalTo": "A frissítési intervallum beállítása erre: {}",
|
||||||
"githubPATLabel": "GitHub személyes hozzáférési token (megnöveli a díjkorlátot)",
|
"githubPATLabel": "GitHub személyes hozzáférési token (megnöveli a díjkorlátot)",
|
||||||
"githubPATHint": "A PAT-nak a következő formátumban kell lennie: felhasználónév:token",
|
"githubPATHint": "A PAT-nak a következő formátumban kell lennie: felhasználónév:token",
|
||||||
"githubPATFormat": "felhasználónév:token",
|
"githubPATFormat": "felhasználónév:token",
|
||||||
@ -28,16 +27,16 @@
|
|||||||
"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ÓBAN KELL RENDELNI",
|
"dropdownNoOptsError": "HIBA: A LEDOBÁST LEGALÁBB EGY OPCIÓHOZ KELL RENDELNI",
|
||||||
"colour": "Szín",
|
"colour": "Szín",
|
||||||
"githubStarredRepos": "GitHub Csillagozott 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 nyomon kö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'.",
|
||||||
"youPickedTrackOnly": "A 'Csak követés' opciót választotta.",
|
"youPickedTrackOnly": "A 'Csak követés' opciót választotta.",
|
||||||
"trackOnlyAppDescription": "Az alkalmazás frissítéseit nyomon követi, de az Obtainium nem tudja letölteni vagy telepíteni.",
|
"trackOnlyAppDescription": "Az alkalmazás frissítéseit nyomon követi, de az Obtainium nem tudja letölteni vagy telepíteni.",
|
||||||
"cancelled": "Törölve",
|
"cancelled": "Törölve",
|
||||||
@ -56,25 +55,24 @@
|
|||||||
"appsString": "Appok",
|
"appsString": "Appok",
|
||||||
"noApps": "Nincs App",
|
"noApps": "Nincs App",
|
||||||
"noAppsForFilter": "Nincsenek appok a szűrőhöz",
|
"noAppsForFilter": "Nincsenek appok a szűrőhöz",
|
||||||
"byX": "By {}",
|
"byX": "{} által",
|
||||||
"percentProgress": "Folyamat: {}%",
|
"percentProgress": "Folyamat: {}%",
|
||||||
"pleaseWait": "Kis türelmet",
|
"pleaseWait": "Kis türelmet",
|
||||||
"updateAvailable": "Frissítés elérhető",
|
"updateAvailable": "Frissítés érhető el",
|
||||||
"estimateInBracketsShort": "(Becsült)",
|
"estimateInBracketsShort": "(Becsült)",
|
||||||
"notInstalled": "Nem telepített",
|
"notInstalled": "Nem telepített",
|
||||||
"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ő)\nmint Frissített",
|
||||||
"changeX": "Változás {}",
|
"changeX": "Változás {}",
|
||||||
"installUpdateApps": "Appok telepítése/frissítése",
|
"installUpdateApps": "Appok telepítése/frissítése",
|
||||||
"installUpdateSelectedApps": "Telepítse/frissítse a kiválasztott appokat",
|
"installUpdateSelectedApps": "Telepítse/frissítse a kiválasztott appokat",
|
||||||
"onlyWorksWithNonEVDApps": "Csak azoknál az alkalmazásoknál működik, amelyek telepítési állapota nem észlelhető automatikusan (nem gyakori).",
|
|
||||||
"markXSelectedAppsAsUpdated": "Megjelöl {} kiválasztott alkalmazást frissítettként?",
|
"markXSelectedAppsAsUpdated": "Megjelöl {} kiválasztott alkalmazást frissítettként?",
|
||||||
"no": "Nem",
|
"no": "Nem",
|
||||||
"yes": "Igen",
|
"yes": "Igen",
|
||||||
@ -86,8 +84,8 @@
|
|||||||
"shareSelectedAppURLs": "Ossza meg a kiválasztott app URL címeit",
|
"shareSelectedAppURLs": "Ossza meg a kiválasztott app URL címeit",
|
||||||
"resetInstallStatus": "Telepítési állapot visszaállítása",
|
"resetInstallStatus": "Telepítési állapot visszaállítása",
|
||||||
"more": "További",
|
"more": "További",
|
||||||
"removeOutdatedFilter": "Távolítsa el az elavult alkalmazásszűrőt",
|
"removeOutdatedFilter": "Távolítsa el az elavult app szűrőt",
|
||||||
"showOutdatedOnly": "Csak az elavult alkalmazások megjelenítése",
|
"showOutdatedOnly": "Csak az elavult appok megjelenítése",
|
||||||
"filter": "Szűrő",
|
"filter": "Szűrő",
|
||||||
"filterActive": "Szűrő *",
|
"filterActive": "Szűrő *",
|
||||||
"filterApps": "Appok szűrése",
|
"filterApps": "Appok szűrése",
|
||||||
@ -118,7 +116,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",
|
||||||
@ -126,16 +124,16 @@
|
|||||||
"appSortBy": "App rendezés...",
|
"appSortBy": "App rendezés...",
|
||||||
"authorName": "Szerző/Név",
|
"authorName": "Szerző/Név",
|
||||||
"nameAuthor": "Név/Szerző",
|
"nameAuthor": "Név/Szerző",
|
||||||
"asAdded": "Mint hozzáadott",
|
"asAdded": "Mint Hozzáadott",
|
||||||
"appSortOrder": "Appok rendezése",
|
"appSortOrder": "Appok rendezése",
|
||||||
"ascending": "Emelkedő",
|
"ascending": "Emelkedő",
|
||||||
"descending": "Csökkenő",
|
"descending": "Csökkenő",
|
||||||
"bgUpdateCheckInterval": "Háttérfrissítés ellenőrzési időköz",
|
"bgUpdateCheckInterval": "Háttérfrissítés ellenőrzés időköze",
|
||||||
"neverManualOnly": "Soha – csak manuális",
|
"neverManualOnly": "Soha – csak manuális",
|
||||||
"appearance": "Megjelenés",
|
"appearance": "Megjelenés",
|
||||||
"showWebInAppView": "Forrás webhely megjelenítése az Alkalmazásnézetben",
|
"showWebInAppView": "Forrás megjelenítése az Appok nézetben",
|
||||||
"pinUpdates": "Rögzítse a frissítéseket az Appok teteje nézethez",
|
"pinUpdates": "Frissítések kitűzése az App nézet tetejére",
|
||||||
"updates": "Frissítve",
|
"updates": "Frissítések",
|
||||||
"sourceSpecific": "Forrás-specifikus",
|
"sourceSpecific": "Forrás-specifikus",
|
||||||
"appSource": "App forrás",
|
"appSource": "App forrás",
|
||||||
"noLogs": "Nincsenek naplók",
|
"noLogs": "Nincsenek naplók",
|
||||||
@ -145,7 +143,7 @@
|
|||||||
"appNotFound": "App nem található",
|
"appNotFound": "App nem található",
|
||||||
"obtainiumExportHyphenatedLowercase": "obtainium-export",
|
"obtainiumExportHyphenatedLowercase": "obtainium-export",
|
||||||
"pickAnAPK": "Válasszon egy APK-t",
|
"pickAnAPK": "Válasszon egy APK-t",
|
||||||
"appHasMoreThanOnePackage": "{} egynél több csomaggal rendelkezik:",
|
"appHasMoreThanOnePackage": "A(z) {} egynél több csomaggal rendelkezik:",
|
||||||
"deviceSupportsXArch": "Eszköze támogatja a {} CPU architektúrát.",
|
"deviceSupportsXArch": "Eszköze támogatja a {} CPU architektúrát.",
|
||||||
"deviceSupportsFollowingArchs": "Az eszköze a következő CPU architektúrákat támogatja:",
|
"deviceSupportsFollowingArchs": "Az eszköze a következő CPU architektúrákat támogatja:",
|
||||||
"warning": "Figyelem",
|
"warning": "Figyelem",
|
||||||
@ -153,16 +151,16 @@
|
|||||||
"updatesAvailable": "Frissítések érhetők el",
|
"updatesAvailable": "Frissítések érhetők el",
|
||||||
"updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy frissítések állnak rendelkezésre egy vagy több, az Obtainium által nyomon követett alkalmazáshoz",
|
"updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy frissítések állnak rendelkezésre egy vagy több, az Obtainium által nyomon követett alkalmazáshoz",
|
||||||
"noNewUpdates": "Nincsenek új frissítések.",
|
"noNewUpdates": "Nincsenek új frissítések.",
|
||||||
"xHasAnUpdate": "{} frissítést kapott.",
|
"xHasAnUpdate": "A(z) {} frissítést kapott.",
|
||||||
"appsUpdated": "Alkalmazások frissítve",
|
"appsUpdated": "Alkalmazások frissítve",
|
||||||
"appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több app frissítése történt a háttérben",
|
"appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy/több app frissítése megtörtént a háttérben",
|
||||||
"xWasUpdatedToY": "{} frissítve a következőre: {}.",
|
"xWasUpdatedToY": "{} frissítve a következőre: {}.",
|
||||||
"errorCheckingUpdates": "Hiba a frissítések keresésekor",
|
"errorCheckingUpdates": "Hiba a frissítések keresésekor",
|
||||||
"errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérbeli frissítések ellenőrzése sikertelen",
|
"errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérbeli frissítések ellenőrzése sikertelen",
|
||||||
"appsRemoved": "Alkalmazások eltávolítva",
|
"appsRemoved": "Alkalmazások eltávolítva",
|
||||||
"appsRemovedNotifDescription": "Értesíti a felhasználót egy vagy több alkalmazás eltávolításáról a betöltésük során fellépő hibák miatt",
|
"appsRemovedNotifDescription": "Értesíti a felhasználót egy vagy több alkalmazás eltávolításáról a betöltésük során fellépő hibák miatt",
|
||||||
"xWasRemovedDueToErrorY": "A(z) {} a következő hiba miatt lett eltávolítva: {}",
|
"xWasRemovedDueToErrorY": "A(z) {} a következő hiba miatt lett eltávolítva: {}",
|
||||||
"completeAppInstallation": "Teljes alkalmazástelepítés",
|
"completeAppInstallation": "Teljes app telepítés",
|
||||||
"obtainiumMustBeOpenToInstallApps": "Az Obtainiumnak megnyitva kell lennie az alkalmazások telepítéséhez",
|
"obtainiumMustBeOpenToInstallApps": "Az Obtainiumnak megnyitva kell lennie az alkalmazások telepítéséhez",
|
||||||
"completeAppInstallationNotifDescription": "Megkéri a felhasználót, hogy térjen vissza az Obtainiumhoz, hogy befejezze az alkalmazás telepítését",
|
"completeAppInstallationNotifDescription": "Megkéri a felhasználót, hogy térjen vissza az Obtainiumhoz, hogy befejezze az alkalmazás telepítését",
|
||||||
"checkingForUpdates": "Frissítések keresése",
|
"checkingForUpdates": "Frissítések keresése",
|
||||||
@ -178,16 +176,45 @@
|
|||||||
"installedVersionX": "Telepített verzió: {}",
|
"installedVersionX": "Telepített verzió: {}",
|
||||||
"lastUpdateCheckX": "Frissítés ellenőrizve: {}",
|
"lastUpdateCheckX": "Frissítés ellenőrizve: {}",
|
||||||
"remove": "Eltávolítás",
|
"remove": "Eltávolítás",
|
||||||
"removeAppQuestion": "Eltávolítja az alkalmazást?",
|
|
||||||
"yesMarkUpdated": "Igen, megjelölés frissítettként",
|
"yesMarkUpdated": "Igen, megjelölés frissítettként",
|
||||||
"fdroid": "F-Droid",
|
"fdroid": "F-Droid",
|
||||||
"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",
|
||||||
|
"install": "Telepít",
|
||||||
|
"markInstalled": "Telepítettnek jelöl",
|
||||||
|
"update": "Frissít",
|
||||||
|
"markUpdated": "Frissítettnek jelöl",
|
||||||
|
"additionalOptions": "További lehetőségek",
|
||||||
|
"disableVersionDetection": "Verzióérzékelés letiltása",
|
||||||
|
"noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően.",
|
||||||
|
"downloadingX": "{} letöltés",
|
||||||
|
"downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról",
|
||||||
|
"noAPKFound": "Nem található APK",
|
||||||
|
"noVersionDetection": "Nincs verzió érzékelés",
|
||||||
|
"categorize": "Kategorizálás",
|
||||||
|
"categories": "Kategóriák",
|
||||||
|
"category": "Kategória",
|
||||||
|
"noCategory": "Nincs kategória",
|
||||||
|
"deleteCategoryQuestion": "Törli a kategóriát?",
|
||||||
|
"categoryDeleteWarning": "A(z) {} összes app kategorizálatlan állapotba kerül.",
|
||||||
|
"addCategory": "Új kategória",
|
||||||
|
"label": "Címke",
|
||||||
|
"language": "Nyelv",
|
||||||
|
"storagePermissionDenied": "Tárhely engedély megtagadva",
|
||||||
|
"selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.",
|
||||||
|
"filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel",
|
||||||
|
"removeFromObtainium": "Eltávolítás az Obtainiumból",
|
||||||
|
"uninstallFromDevice": "Eltávolítás a készülékről",
|
||||||
|
"onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzióérzékelés le van tiltva.",
|
||||||
|
"removeAppQuestion": {
|
||||||
|
"one": "Eltávolítja az alkalmazást?",
|
||||||
|
"other": "Eltávolítja az alkalmazást?"
|
||||||
|
},
|
||||||
"tooManyRequestsTryAgainInMinutes": {
|
"tooManyRequestsTryAgainInMinutes": {
|
||||||
"one": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva",
|
"one": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva",
|
||||||
"other": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva"
|
"other": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva"
|
||||||
@ -232,4 +259,4 @@
|
|||||||
"one": "A(z) {} és 1 további alkalmazás frissítve.",
|
"one": "A(z) {} és 1 további alkalmazás frissítve.",
|
||||||
"other": "{} és további {} alkalmazás frissítve."
|
"other": "{} és további {} alkalmazás frissítve."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
"noVersionFound": "Impossibile determinare la versione della release",
|
"noVersionFound": "Impossibile determinare la versione della release",
|
||||||
"urlMatchesNoSource": "L'URL non corrisponde ad alcuna fonte conosciuta",
|
"urlMatchesNoSource": "L'URL non corrisponde ad alcuna fonte conosciuta",
|
||||||
"cantInstallOlderVersion": "Impossibile installare una versione precedente di un'App",
|
"cantInstallOlderVersion": "Impossibile installare una versione precedente di un'App",
|
||||||
"appIdMismatch": "L'ID del pacchetto scaricato non corrisponde all'ID dell'App esistente",
|
|
||||||
"functionNotImplemented": "Questa classe non ha implementato questa funzione",
|
"functionNotImplemented": "Questa classe non ha implementato questa funzione",
|
||||||
"placeholder": "Segnaposto",
|
"placeholder": "Segnaposto",
|
||||||
"someErrors": "Si sono verificati degli errori",
|
"someErrors": "Si sono verificati degli errori",
|
||||||
@ -16,38 +15,38 @@
|
|||||||
"startedActualBGUpdateCheck": "Avviato il controllo effettivo degli aggiornamenti in background",
|
"startedActualBGUpdateCheck": "Avviato il controllo effettivo degli aggiornamenti in background",
|
||||||
"bgUpdateTaskFinished": "Terminata l'attività di controllo degli aggiornamenti in background",
|
"bgUpdateTaskFinished": "Terminata l'attività di controllo degli aggiornamenti in background",
|
||||||
"firstRun": "Questo è il primo avvio di sempre di Obtainium",
|
"firstRun": "Questo è il primo avvio di sempre di Obtainium",
|
||||||
"settingUpdateCheckIntervalTo": "Imposta l'intervallo di aggiornamento a {}",
|
"settingUpdateCheckIntervalTo": "Fissato intervallo di aggiornamento a {}",
|
||||||
"githubPATLabel": "GitHub Personal Access Token (aumenta il limite di traffico)",
|
"githubPATLabel": "GitHub Personal Access Token (diminuisce limite di traffico)",
|
||||||
"githubPATHint": "PAT deve seguire questo formato: username:token",
|
"githubPATHint": "PAT deve seguire questo formato: username:token",
|
||||||
"githubPATFormat": "username:token",
|
"githubPATFormat": "username:token",
|
||||||
"githubPATLinkText": "Informazioni su GitHub PAT",
|
"githubPATLinkText": "Informazioni su GitHub PAT",
|
||||||
"includePrereleases": "Includi prerelease",
|
"includePrereleases": "Includi prerelease",
|
||||||
"fallbackToOlderReleases": "Ripiega su release datate",
|
"fallbackToOlderReleases": "Ripiega su release precedenti",
|
||||||
"filterReleaseTitlesByRegEx": "Filtra le release con le espressioni regolari",
|
"filterReleaseTitlesByRegEx": "Filtra release con espressioni regolari",
|
||||||
"invalidRegEx": "Espressione regolare invalida",
|
"invalidRegEx": "Espressione regolare non valida",
|
||||||
"noDescription": "Descrizione assente",
|
"noDescription": "Descrizione assente",
|
||||||
"cancel": "Annulla",
|
"cancel": "Annulla",
|
||||||
"continue": "Continua",
|
"continue": "Continua",
|
||||||
"requiredInBrackets": "(Richiesto)",
|
"requiredInBrackets": "(richiesto)",
|
||||||
"dropdownNoOptsError": "ERRORE: LA TENDINA DEVE AVERE ALMENO UN'OPZIONE",
|
"dropdownNoOptsError": "ERRORE: LA TENDINA DEVE AVERE ALMENO UN'OPZIONE",
|
||||||
"colour": "Colore",
|
"colour": "Colore",
|
||||||
"githubStarredRepos": "i repository stellati da GitHub",
|
"githubStarredRepos": "repository stellati da GitHub",
|
||||||
"uname": "Username",
|
"uname": "Username",
|
||||||
"wrongArgNum": "Numero di argomenti forniti errato",
|
"wrongArgNum": "Numero di argomenti forniti errato",
|
||||||
"xIsTrackOnly": "{} è Solo-Monitoraggio",
|
"xIsTrackOnly": "{} è in modalità Solo-Monitoraggio",
|
||||||
"source": "Fonte",
|
"source": "Fonte",
|
||||||
"app": "App",
|
"app": "App",
|
||||||
"appsFromSourceAreTrackOnly": "Le App da questa fonte sono in modalità 'Solo-Monitoraggio'.",
|
"appsFromSourceAreTrackOnly": "Le App da questa fonte sono in modalità 'Solo-Monitoraggio'.",
|
||||||
"youPickedTrackOnly": "Hai selezionato l'opzione 'Solo-Monitoraggio'.",
|
"youPickedTrackOnly": "È stata selezionata l'opzione 'Solo-Monitoraggio'.",
|
||||||
"trackOnlyAppDescription": "L'App sarà monitorata per gli aggiornamenti, ma Obtainium non sarà in grado di scaricarli o di installarli.",
|
"trackOnlyAppDescription": "L'App sarà monitorata per gli aggiornamenti, ma Obtainium non sarà in grado di scaricarli o di installarli.",
|
||||||
"cancelled": "Annullato",
|
"cancelled": "Annullato",
|
||||||
"appAlreadyAdded": "App già aggiunta",
|
"appAlreadyAdded": "App già aggiunta",
|
||||||
"alreadyUpToDateQuestion": "App già aggiornata?",
|
"alreadyUpToDateQuestion": "L'App è già aggiornata?",
|
||||||
"addApp": "Aggiungi App",
|
"addApp": "Aggiungi App",
|
||||||
"appSourceURL": "URL della fonte dell'App",
|
"appSourceURL": "URL della fonte dell'App",
|
||||||
"error": "Errore",
|
"error": "Errore",
|
||||||
"add": "Aggiungi",
|
"add": "Aggiungi",
|
||||||
"searchSomeSourcesLabel": "Cerca (disponibile solo per alcune fonti)",
|
"searchSomeSourcesLabel": "Cerca (solo per alcune fonti)",
|
||||||
"search": "Cerca",
|
"search": "Cerca",
|
||||||
"additionalOptsFor": "Opzioni aggiuntive per {}",
|
"additionalOptsFor": "Opzioni aggiuntive per {}",
|
||||||
"supportedSourcesBelow": "Fonti supportate:",
|
"supportedSourcesBelow": "Fonti supportate:",
|
||||||
@ -60,21 +59,20 @@
|
|||||||
"percentProgress": "Progresso: {}%",
|
"percentProgress": "Progresso: {}%",
|
||||||
"pleaseWait": "Attendere prego",
|
"pleaseWait": "Attendere prego",
|
||||||
"updateAvailable": "Aggiornamento disponibile",
|
"updateAvailable": "Aggiornamento disponibile",
|
||||||
"estimateInBracketsShort": "(Prev.)",
|
"estimateInBracketsShort": "(prev.)",
|
||||||
"notInstalled": "Non installato",
|
"notInstalled": "Non installato",
|
||||||
"estimateInBrackets": "(Previsto)",
|
"estimateInBrackets": "(previsto)",
|
||||||
"selectAll": "Seleziona tutto",
|
"selectAll": "Seleziona tutto",
|
||||||
"deselectN": "Deseleziona {}",
|
"deselectN": "Deseleziona {}",
|
||||||
"xWillBeRemovedButRemainInstalled": "{} sarà rimosso da Obtainium ma resterà installato sul dispositivo.",
|
"xWillBeRemovedButRemainInstalled": "Verà effettuata la rimozione di {}, ma non la disinstallazione.",
|
||||||
"removeSelectedAppsQuestion": "Rimuovere le App selezionate?",
|
"removeSelectedAppsQuestion": "Rimuovere le App selezionate?",
|
||||||
"removeSelectedApps": "Rimuovi le App selezionate",
|
"removeSelectedApps": "Rimuovi le App selezionate",
|
||||||
"updateX": "Aggiorna {}",
|
"updateX": "Aggiorna {}",
|
||||||
"installX": "Installa {}",
|
"installX": "Installa {}",
|
||||||
"markXTrackOnlyAsUpdated": "Contrassegna {}\n(Solo-Monitoraggio)\ncome aggiornato",
|
"markXTrackOnlyAsUpdated": "Contrassegna {}\n(Solo-Monitoraggio)\ncome aggiornato",
|
||||||
"changeX": "modifica {}",
|
"changeX": "Modifica {}",
|
||||||
"installUpdateApps": "Installa/Aggiorna le App",
|
"installUpdateApps": "Installa/Aggiorna App",
|
||||||
"installUpdateSelectedApps": "Installa/Aggiornale le App selezionate",
|
"installUpdateSelectedApps": "Installa/Aggiorna le App selezionate",
|
||||||
"onlyWorksWithNonEVDApps": "Funziona solo per le App il cui stato d'installazione non può essere rilevato automaticamente (inconsueto).",
|
|
||||||
"markXSelectedAppsAsUpdated": "Contrassegnare le {} App selezionate come aggiornate?",
|
"markXSelectedAppsAsUpdated": "Contrassegnare le {} App selezionate come aggiornate?",
|
||||||
"no": "No",
|
"no": "No",
|
||||||
"yes": "Sì",
|
"yes": "Sì",
|
||||||
@ -86,8 +84,8 @@
|
|||||||
"shareSelectedAppURLs": "Condividi gli URL delle App selezionate",
|
"shareSelectedAppURLs": "Condividi gli URL delle App selezionate",
|
||||||
"resetInstallStatus": "Ripristina lo stato d'installazione",
|
"resetInstallStatus": "Ripristina lo stato d'installazione",
|
||||||
"more": "Di più",
|
"more": "Di più",
|
||||||
"removeOutdatedFilter": "Rimuovi il filtro per le App datate",
|
"removeOutdatedFilter": "Rimuovi il filtro per le App non aggiornate",
|
||||||
"showOutdatedOnly": "Mostra solo le App datate",
|
"showOutdatedOnly": "Mostra solo le App non aggiornate",
|
||||||
"filter": "Filtri",
|
"filter": "Filtri",
|
||||||
"filterActive": "Filtri *",
|
"filterActive": "Filtri *",
|
||||||
"filterApps": "Filtra App",
|
"filterApps": "Filtra App",
|
||||||
@ -95,7 +93,7 @@
|
|||||||
"author": "Autore",
|
"author": "Autore",
|
||||||
"upToDateApps": "App aggiornate",
|
"upToDateApps": "App aggiornate",
|
||||||
"nonInstalledApps": "App non installate",
|
"nonInstalledApps": "App non installate",
|
||||||
"importExport": "Importa/Esporta",
|
"importExport": "Importa - Esporta",
|
||||||
"settings": "Impostazioni",
|
"settings": "Impostazioni",
|
||||||
"exportedTo": "Esportato in {}",
|
"exportedTo": "Esportato in {}",
|
||||||
"obtainiumExport": "Esporta da Obtainium",
|
"obtainiumExport": "Esporta da Obtainium",
|
||||||
@ -106,7 +104,7 @@
|
|||||||
"searchQuery": "Stringa di ricerca",
|
"searchQuery": "Stringa di ricerca",
|
||||||
"appURLList": "Lista di URL delle App",
|
"appURLList": "Lista di URL delle App",
|
||||||
"line": "Linea",
|
"line": "Linea",
|
||||||
"searchX": "Cerca {}",
|
"searchX": "Cerca su {}",
|
||||||
"noResults": "Nessun risultato trovato",
|
"noResults": "Nessun risultato trovato",
|
||||||
"importX": "Importa {}",
|
"importX": "Importa {}",
|
||||||
"importedAppsIdDisclaimer": "Le App importate potrebbero essere visualizzate erroneamente come \"Non installate\".\nPer risolvere il problema, reinstallale con Obtainium.\nQuesto non dovrebbe influire sui dati delle App.\n\nRiguarda solo l'URL e i metodi di importazione di terze parti.",
|
"importedAppsIdDisclaimer": "Le App importate potrebbero essere visualizzate erroneamente come \"Non installate\".\nPer risolvere il problema, reinstallale con Obtainium.\nQuesto non dovrebbe influire sui dati delle App.\n\nRiguarda solo l'URL e i metodi di importazione di terze parti.",
|
||||||
@ -120,7 +118,7 @@
|
|||||||
"theme": "Tema",
|
"theme": "Tema",
|
||||||
"dark": "Scuro",
|
"dark": "Scuro",
|
||||||
"light": "Chiaro",
|
"light": "Chiaro",
|
||||||
"followSystem": "Segui il sistema",
|
"followSystem": "Segui sistema",
|
||||||
"obtainium": "Obtainium",
|
"obtainium": "Obtainium",
|
||||||
"materialYou": "Material You",
|
"materialYou": "Material You",
|
||||||
"appSortBy": "App ordinate per",
|
"appSortBy": "App ordinate per",
|
||||||
@ -133,10 +131,10 @@
|
|||||||
"bgUpdateCheckInterval": "Intervallo di controllo degli aggiornamenti in background",
|
"bgUpdateCheckInterval": "Intervallo di controllo degli aggiornamenti in background",
|
||||||
"neverManualOnly": "Mai - Solo manuale",
|
"neverManualOnly": "Mai - Solo manuale",
|
||||||
"appearance": "Aspetto",
|
"appearance": "Aspetto",
|
||||||
"showWebInAppView": "Mostra la pagina web dell'App se selezionata",
|
"showWebInAppView": "Mostra pagina web dell'App se selezionata",
|
||||||
"pinUpdates": "Fissa in alto gli aggiornamenti disponibili nella pagina delle App",
|
"pinUpdates": "Fissa aggiornamenti disponibili in alto",
|
||||||
"updates": "Aggiornato",
|
"updates": "Aggiornamenti",
|
||||||
"sourceSpecific": "Specifico per la fonte",
|
"sourceSpecific": "Specifiche per la fonte",
|
||||||
"appSource": "Sorgente dell'App",
|
"appSource": "Sorgente dell'App",
|
||||||
"noLogs": "Nessun log",
|
"noLogs": "Nessun log",
|
||||||
"appLogs": "Log dell'App",
|
"appLogs": "Log dell'App",
|
||||||
@ -146,21 +144,21 @@
|
|||||||
"obtainiumExportHyphenatedLowercase": "esportazione-obtainium",
|
"obtainiumExportHyphenatedLowercase": "esportazione-obtainium",
|
||||||
"pickAnAPK": "Seleziona un APK",
|
"pickAnAPK": "Seleziona un APK",
|
||||||
"appHasMoreThanOnePackage": "{} offre più di un pacchetto:",
|
"appHasMoreThanOnePackage": "{} offre più di un pacchetto:",
|
||||||
"deviceSupportsXArch": "Il tuo dispositivo supporta l'architettura {} della CPU.",
|
"deviceSupportsXArch": "Il dispositivo in uso supporta l'architettura {} della CPU.",
|
||||||
"deviceSupportsFollowingArchs": "Il tuo dispositivo supporta le seguenti architetture della CPU:",
|
"deviceSupportsFollowingArchs": "Il dispositivo in uso supporta le seguenti architetture della CPU:",
|
||||||
"warning": "Attenzione",
|
"warning": "Attenzione",
|
||||||
"sourceIsXButPackageFromYPrompt": "L'origine dell'App è '{}' ma il pacchetto della release proviene da '{}'. Continuare?",
|
"sourceIsXButPackageFromYPrompt": "L'origine dell'App è '{}' ma il pacchetto della release proviene da '{}'. Continuare?",
|
||||||
"updatesAvailable": "Aggiornamenti disponibili",
|
"updatesAvailable": "Aggiornamenti disponibili",
|
||||||
"updatesAvailableNotifDescription": "Avvisa l'utente che sono disponibili gli aggiornamenti di una o più App monitorate da Obtainium",
|
"updatesAvailableNotifDescription": "Notifica all'utente che sono disponibili gli aggiornamenti di una o più App monitorate da Obtainium",
|
||||||
"noNewUpdates": "Nessun nuovo aggiornamento.",
|
"noNewUpdates": "Nessun nuovo aggiornamento.",
|
||||||
"xHasAnUpdate": "{} è stato aggiornato.",
|
"xHasAnUpdate": "Aggiornamento disponibile per {}",
|
||||||
"appsUpdated": "App aggiornate",
|
"appsUpdated": "App aggiornate",
|
||||||
"appsUpdatedNotifDescription": "Avvisa l'utente che una o più App sono state aggiornate in background",
|
"appsUpdatedNotifDescription": "Notifica all'utente che una o più App sono state aggiornate in background",
|
||||||
"xWasUpdatedToY": "{} è stato aggiornato a {}.",
|
"xWasUpdatedToY": "{} è stato aggiornato a {}.",
|
||||||
"errorCheckingUpdates": "Controllo degli errori per gli aggiornamenti",
|
"errorCheckingUpdates": "Controllo degli errori per gli aggiornamenti",
|
||||||
"errorCheckingUpdatesNotifDescription": "Una notifica che mostra quando il controllo degli aggiornamenti in background fallisce",
|
"errorCheckingUpdatesNotifDescription": "Una notifica che mostra quando il controllo degli aggiornamenti in background fallisce",
|
||||||
"appsRemoved": "App rimosse",
|
"appsRemoved": "App rimosse",
|
||||||
"appsRemovedNotifDescription": "Avvisa l'utente che una o più App sono state rimosse a causa di errori durante il caricamento",
|
"appsRemovedNotifDescription": "Notifica all'utente che una o più App sono state rimosse a causa di errori durante il caricamento",
|
||||||
"xWasRemovedDueToErrorY": "{} è stata rimosso a causa di questo errore: {}",
|
"xWasRemovedDueToErrorY": "{} è stata rimosso a causa di questo errore: {}",
|
||||||
"completeAppInstallation": "Completa l'installazione dell'App",
|
"completeAppInstallation": "Completa l'installazione dell'App",
|
||||||
"obtainiumMustBeOpenToInstallApps": "Obtainium deve essere aperto per poter installare le App",
|
"obtainiumMustBeOpenToInstallApps": "Obtainium deve essere aperto per poter installare le App",
|
||||||
@ -178,27 +176,57 @@
|
|||||||
"installedVersionX": "Versione installata: {}",
|
"installedVersionX": "Versione installata: {}",
|
||||||
"lastUpdateCheckX": "Ultimo controllo degli aggiornamenti: {}",
|
"lastUpdateCheckX": "Ultimo controllo degli aggiornamenti: {}",
|
||||||
"remove": "Rimuovi",
|
"remove": "Rimuovi",
|
||||||
"removeAppQuestion": "Rimuovere App?",
|
|
||||||
"yesMarkUpdated": "Sì, contrassegna come aggiornato",
|
"yesMarkUpdated": "Sì, contrassegna come aggiornato",
|
||||||
"fdroid": "F-Droid",
|
"fdroid": "F-Droid",
|
||||||
"appIdOrName": "ID o nome dell'App",
|
"appIdOrName": "ID o nome dell'App",
|
||||||
"appWithIdOrNameNotFound": "Non è stata trovata alcuna App con quell'ID o nome",
|
"appWithIdOrNameNotFound": "Non è stata trovata alcuna App con quell'ID o nome",
|
||||||
"reposHaveMultipleApps": "I repository possono contenere più App",
|
"reposHaveMultipleApps": "I repository possono contenere più App",
|
||||||
"fdroidThirdPartyRepo": "Repository di terze parti di F-Droid",
|
"fdroidThirdPartyRepo": "Repository F-Droid di terze parti",
|
||||||
"steam": "Steam",
|
"steam": "Steam",
|
||||||
"steamMobile": "Steam Mobile",
|
"steamMobile": "Steam Mobile",
|
||||||
"steamChat": "Steam Chat",
|
"steamChat": "Steam Chat",
|
||||||
|
"install": "Installa",
|
||||||
|
"markInstalled": "Contrassegna come installato",
|
||||||
|
"update": "Aggiorna",
|
||||||
|
"markUpdated": "Contrassegna come aggiornato",
|
||||||
|
"additionalOptions": "Opzioni aggiuntive",
|
||||||
|
"disableVersionDetection": "Disattiva il rilevamento della versione",
|
||||||
|
"noVersionDetectionExplanation": "Questa opzione dovrebbe essere usata solo per le App la cui versione non viene rilevata correttamente.",
|
||||||
|
"downloadingX": "Scaricamento di {} in corso",
|
||||||
|
"downloadNotifDescription": "Notifica all'utente lo stato di avanzamento del download di un'App",
|
||||||
|
"noAPKFound": "Nessun APK trovato",
|
||||||
|
"noVersionDetection": "Disattiva rilevamento di versione",
|
||||||
|
"categorize": "Aggiungi a categoria",
|
||||||
|
"categories": "Categorie",
|
||||||
|
"category": "Categoria",
|
||||||
|
"noCategory": "Nessuna categoria",
|
||||||
|
"noCategories": "Nessuna categoria",
|
||||||
|
"deleteCategoriesQuestion": "Eliminare le categorie?",
|
||||||
|
"categoryDeleteWarning": "Tutte le App nelle categorie eliminate saranno impostate come non categorizzate.",
|
||||||
|
"addCategory": "Aggiungi categoria",
|
||||||
|
"label": "Etichetta",
|
||||||
|
"language": "Lingua",
|
||||||
|
"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",
|
||||||
|
"removeFromObtainium": "Rimuovi da Obtainium",
|
||||||
|
"uninstallFromDevice": "Disinstalla dal dispositivo",
|
||||||
|
"onlyWorksWithNonVersionDetectApps": "Funziona solo per le App con il rilevamento della versione disattivato.",
|
||||||
|
"removeAppQuestion": {
|
||||||
|
"one": "Rimuovere l'App?",
|
||||||
|
"other": "Rimuovere le App?"
|
||||||
|
},
|
||||||
"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"
|
||||||
},
|
},
|
||||||
"bgUpdateGotErrorRetryInMinutes": {
|
"bgUpdateGotErrorRetryInMinutes": {
|
||||||
"one": "Il controllo degli aggiornamenti in background ha incontrato un {}, ricontrollo tra {} minuto",
|
"one": "Il controllo degli aggiornamenti in background ha incontrato un {}, nuovo tentativo tra {} minuto",
|
||||||
"other": "Il controllo degli aggiornamenti in background ha incontrato un {}, ricontrollo tra {} minuti"
|
"other": "Il controllo degli aggiornamenti in background ha incontrato un {}, nuovo tentativo tra {} minuti"
|
||||||
},
|
},
|
||||||
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
||||||
"one": "Il controllo degli aggiornamenti in background ha trovato {} aggiornamento - avviserà l'utento se necessario",
|
"one": "Il controllo degli aggiornamenti in background ha trovato {} aggiornamento - notificherà l'utente se necessario",
|
||||||
"other": "Il controllo degli aggiornamenti in background ha trovato {} aggiornamenti - avviserà l'utento se necessario"
|
"other": "Il controllo degli aggiornamenti in background ha trovato {} aggiornamenti - notificherà l'utente se necessario"
|
||||||
},
|
},
|
||||||
"apps": {
|
"apps": {
|
||||||
"one": "{} App",
|
"one": "{} App",
|
||||||
|
@ -4,20 +4,19 @@
|
|||||||
"noVersionFound": "リリースバージョンを特定できませんでした",
|
"noVersionFound": "リリースバージョンを特定できませんでした",
|
||||||
"urlMatchesNoSource": "URLが既知のソースと一致しません",
|
"urlMatchesNoSource": "URLが既知のソースと一致しません",
|
||||||
"cantInstallOlderVersion": "旧バージョンのアプリをインストールできません",
|
"cantInstallOlderVersion": "旧バージョンのアプリをインストールできません",
|
||||||
"appIdMismatch": "ダウンロードしたパッケージのIDが既存のApp IDと一致しません",
|
|
||||||
"functionNotImplemented": "このクラスはこの機能を実装していません",
|
"functionNotImplemented": "このクラスはこの機能を実装していません",
|
||||||
"placeholder": "プレースホルダー",
|
"placeholder": "プレースホルダー",
|
||||||
"someErrors": "いくつかのエラーが発生しました",
|
"someErrors": "何らかのエラーが発生しました",
|
||||||
"unexpectedError": "予期せぬエラーが発生しました",
|
"unexpectedError": "予期せぬエラーが発生しました",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"and": "と",
|
"and": "と",
|
||||||
"startedBgUpdateTask": "Started BG update check task",
|
"startedBgUpdateTask": "バックグラウンドのアップデート確認タスクを開始",
|
||||||
"bgUpdateIgnoreAfterIs": "Bg update ignoreAfter is {}",
|
"bgUpdateIgnoreAfterIs": "Bg update ignoreAfter is {}",
|
||||||
"startedActualBGUpdateCheck": "Started actual BG update checking",
|
"startedActualBGUpdateCheck": "実際のバックグラウンドのアップデート確認を開始",
|
||||||
"bgUpdateTaskFinished": "Finished BG update check task",
|
"bgUpdateTaskFinished": "バックグラウンドのアップデート確認タスクを終了",
|
||||||
"firstRun": "This is the first ever run of Obtainium",
|
"firstRun": "これがObtainiumの最初の実行です",
|
||||||
"settingUpdateCheckIntervalTo": "更新間隔を{}に設定する",
|
"settingUpdateCheckIntervalTo": "確認間隔を{}に設定する",
|
||||||
"githubPATLabel": "GitHub パーソナルアクセストークン (レートリミットの引き上げ)",
|
"githubPATLabel": "GitHub パーソナルアクセストークン (レート制限の引き上げ)",
|
||||||
"githubPATHint": "PATは次の形式でなければなりません: ユーザー名:トークン",
|
"githubPATHint": "PATは次の形式でなければなりません: ユーザー名:トークン",
|
||||||
"githubPATFormat": "ユーザー名:トークン",
|
"githubPATFormat": "ユーザー名:トークン",
|
||||||
"githubPATLinkText": "GitHub PATsについて",
|
"githubPATLinkText": "GitHub PATsについて",
|
||||||
@ -27,23 +26,23 @@
|
|||||||
"invalidRegEx": "無効な正規表現",
|
"invalidRegEx": "無効な正規表現",
|
||||||
"noDescription": "説明はありません",
|
"noDescription": "説明はありません",
|
||||||
"cancel": "キャンセル",
|
"cancel": "キャンセル",
|
||||||
"continue": "続ける",
|
"continue": "続行",
|
||||||
"requiredInBrackets": "(必須)",
|
"requiredInBrackets": "(必須)",
|
||||||
"dropdownNoOptsError": "エラー: ドロップダウンには、少なくとも1つのoptが必要です。",
|
"dropdownNoOptsError": "エラー: ドロップダウンには、少なくとも1つのオプションが必要です",
|
||||||
"colour": "カラー",
|
"colour": "カラー",
|
||||||
"githubStarredRepos": "Githubでスターしたリポジトリ",
|
"githubStarredRepos": "Githubでスターしたリポジトリ",
|
||||||
"uname": "ユーザー名",
|
"uname": "ユーザー名",
|
||||||
"wrongArgNum": "提供する引数の数が間違っています",
|
"wrongArgNum": "提供する引数の数が間違っています",
|
||||||
"xIsTrackOnly": "{} は'Track-Only'です",
|
"xIsTrackOnly": "{} は「追跡のみ」です",
|
||||||
"source": "ソース",
|
"source": "ソース",
|
||||||
"app": "アプリ",
|
"app": "アプリ",
|
||||||
"appsFromSourceAreTrackOnly": "このソースからのアプリは'Track-Only'です'。",
|
"appsFromSourceAreTrackOnly": "このソースからのアプリは「追跡のみ」です。",
|
||||||
"youPickedTrackOnly": "'Track-Only'を選択しています",
|
"youPickedTrackOnly": "「追跡のみ」を選択しています",
|
||||||
"trackOnlyAppDescription": "アプリのアップデートは追跡されますが、Obtainiumはアプリのダウンロードやインストールをすることはできません。",
|
"trackOnlyAppDescription": "アプリのアップデートは追跡されますが、Obtainiumはアプリのダウンロードやインストールをすることはできません。",
|
||||||
"cancelled": "キャンセルしました",
|
"cancelled": "キャンセルしました",
|
||||||
"appAlreadyAdded": "アプリはすでに追加されています",
|
"appAlreadyAdded": "アプリはすでに追加されています",
|
||||||
"alreadyUpToDateQuestion": "アプリはすでに最新ですか?",
|
"alreadyUpToDateQuestion": "アプリはすでに最新ですか?",
|
||||||
"addApp": "アプリ追加",
|
"addApp": "アプリの追加",
|
||||||
"appSourceURL": "アプリのソースURL",
|
"appSourceURL": "アプリのソースURL",
|
||||||
"error": "エラー",
|
"error": "エラー",
|
||||||
"add": "追加",
|
"add": "追加",
|
||||||
@ -51,7 +50,7 @@
|
|||||||
"search": "検索",
|
"search": "検索",
|
||||||
"additionalOptsFor": "{}の追加オプション",
|
"additionalOptsFor": "{}の追加オプション",
|
||||||
"supportedSourcesBelow": "対応するソース:",
|
"supportedSourcesBelow": "対応するソース:",
|
||||||
"trackOnlyInBrackets": "(Track-Only)",
|
"trackOnlyInBrackets": "(追跡のみ)",
|
||||||
"searchableInBrackets": "(検索可能)",
|
"searchableInBrackets": "(検索可能)",
|
||||||
"appsString": "アプリ",
|
"appsString": "アプリ",
|
||||||
"noApps": "アプリはありません",
|
"noApps": "アプリはありません",
|
||||||
@ -59,22 +58,21 @@
|
|||||||
"byX": "by {}",
|
"byX": "by {}",
|
||||||
"percentProgress": "ダウンロード中: {}%",
|
"percentProgress": "ダウンロード中: {}%",
|
||||||
"pleaseWait": "しばらくお待ちください",
|
"pleaseWait": "しばらくお待ちください",
|
||||||
"updateAvailable": "アップデートを利用可能",
|
"updateAvailable": "アップデートが利用可能",
|
||||||
"estimateInBracketsShort": "(推定)",
|
"estimateInBracketsShort": "(推定)",
|
||||||
"notInstalled": "未インストール",
|
"notInstalled": "未インストール",
|
||||||
"estimateInBrackets": "(推定)",
|
"estimateInBrackets": "(推定)",
|
||||||
"selectAll": "すべて選択",
|
"selectAll": "すべて選択",
|
||||||
"deselectN": "{}件を選択解除",
|
"deselectN": "{}件の選択を解除",
|
||||||
"xWillBeRemovedButRemainInstalled": "{}はObtainiumから削除されますが、デバイスにはインストールされたままです。",
|
"xWillBeRemovedButRemainInstalled": "{} はObtainiumから削除されますが、デバイスにはインストールされたままです。",
|
||||||
"removeSelectedAppsQuestion": "選択したアプリを削除しますか?",
|
"removeSelectedAppsQuestion": "選択したアプリを削除しますか?",
|
||||||
"removeSelectedApps": "選択したアプリを削除する",
|
"removeSelectedApps": "選択したアプリを削除する",
|
||||||
"updateX": "{}を更新する",
|
"updateX": "{} をアップデートする",
|
||||||
"installX": "{}をインストールする",
|
"installX": "{} をインストールする",
|
||||||
"markXTrackOnlyAsUpdated": "Mark {}\n(Track-Only)\nas Updated",
|
"markXTrackOnlyAsUpdated": "{}\n(追跡のみ)\nをアップデート済みとしてマークする",
|
||||||
"changeX": "{}を変更する",
|
"changeX": "{} を変更する",
|
||||||
"installUpdateApps": "アプリのインストール/アップデート",
|
"installUpdateApps": "アプリのインストール/アップデート",
|
||||||
"installUpdateSelectedApps": "選択したアプリのインストール/アップデート",
|
"installUpdateSelectedApps": "選択したアプリのインストール/アップデート",
|
||||||
"onlyWorksWithNonEVDApps": "インストール状況を自動検出できないアプリ(一般的でないもの)のみ動作します。",
|
|
||||||
"markXSelectedAppsAsUpdated": "{}個の選択したアプリをアップデート済みとしてマークしますか?",
|
"markXSelectedAppsAsUpdated": "{}個の選択したアプリをアップデート済みとしてマークしますか?",
|
||||||
"no": "いいえ",
|
"no": "いいえ",
|
||||||
"yes": "はい",
|
"yes": "はい",
|
||||||
@ -82,12 +80,12 @@
|
|||||||
"pinToTop": "トップに固定",
|
"pinToTop": "トップに固定",
|
||||||
"unpinFromTop": "トップから固定解除",
|
"unpinFromTop": "トップから固定解除",
|
||||||
"resetInstallStatusForSelectedAppsQuestion": "選択したアプリのインストール状態をリセットしますか?",
|
"resetInstallStatusForSelectedAppsQuestion": "選択したアプリのインストール状態をリセットしますか?",
|
||||||
"installStatusOfXWillBeResetExplanation": "選択したアプリのインストール状態がリセットされます。\n\nアップデートに失敗するなどして、Obtainiumに表示されるアプリのバージョンが正しくない場合に役立ちます。",
|
"installStatusOfXWillBeResetExplanation": "選択したアプリのインストール状態がリセットされます。\n\nアップデートに失敗した場合など、Obtainiumに表示されるアプリのバージョンが正しくない場合に有効です。",
|
||||||
"shareSelectedAppURLs": "選択したアプリのURLを共有する",
|
"shareSelectedAppURLs": "選択したアプリのURLを共有する",
|
||||||
"resetInstallStatus": "インストール状態をリセットする",
|
"resetInstallStatus": "インストール状態をリセットする",
|
||||||
"more": "もっと見る",
|
"more": "もっと見る",
|
||||||
"removeOutdatedFilter": "期限切れのアプリフィルターを削除",
|
"removeOutdatedFilter": "アップデートが存在するアプリのフィルターを解除",
|
||||||
"showOutdatedOnly": "期限切れのアプリのみ表示する",
|
"showOutdatedOnly": "アップデートが存在するアプリのみ表示する",
|
||||||
"filter": "フィルター",
|
"filter": "フィルター",
|
||||||
"filterActive": "フィルター *",
|
"filterActive": "フィルター *",
|
||||||
"filterApps": "アプリを絞り込む",
|
"filterApps": "アプリを絞り込む",
|
||||||
@ -98,10 +96,10 @@
|
|||||||
"importExport": "インポート/エクスポート",
|
"importExport": "インポート/エクスポート",
|
||||||
"settings": "設定",
|
"settings": "設定",
|
||||||
"exportedTo": "{}にエクスポートしました",
|
"exportedTo": "{}にエクスポートしました",
|
||||||
"obtainiumExport": "Obtainium Export",
|
"obtainiumExport": "Obtainium エクスポート",
|
||||||
"invalidInput": "無効な入力",
|
"invalidInput": "無効な入力",
|
||||||
"importedX": "{}をインポートしました",
|
"importedX": "{}をインポートしました",
|
||||||
"obtainiumImport": "Obtainium Import",
|
"obtainiumImport": "Obtainium インポート",
|
||||||
"importFromURLList": "URLリストからのインポート",
|
"importFromURLList": "URLリストからのインポート",
|
||||||
"searchQuery": "検索キーワード",
|
"searchQuery": "検索キーワード",
|
||||||
"appURLList": "アプリのURLリスト",
|
"appURLList": "アプリのURLリスト",
|
||||||
@ -109,10 +107,10 @@
|
|||||||
"searchX": "{}で検索",
|
"searchX": "{}で検索",
|
||||||
"noResults": "結果は見つかりませんでした",
|
"noResults": "結果は見つかりませんでした",
|
||||||
"importX": "{}をインポートする",
|
"importX": "{}をインポートする",
|
||||||
"importedAppsIdDisclaimer": "インポートしたアプリが「Not Installed」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響ありません。\n\nURLとサードパーティーのインポートメソッドにのみ影響します。",
|
"importedAppsIdDisclaimer": "インポートしたアプリが「未インストール」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響しません。\n\nURLとサードパーティのインポートメソッドにのみ影響します。",
|
||||||
"importErrors": "インポートエラー",
|
"importErrors": "インポートエラー",
|
||||||
"importedXOfYApps": "{} / {} アプリをインポートしました",
|
"importedXOfYApps": "{} / {} アプリをインポートしました",
|
||||||
"followingURLsHadErrors": "以下のURLでエラーが発生しました。:",
|
"followingURLsHadErrors": "以下のURLでエラーが発生しました:",
|
||||||
"okay": "OK",
|
"okay": "OK",
|
||||||
"selectURL": "URLを選択",
|
"selectURL": "URLを選択",
|
||||||
"selectURLs": "URLを選択",
|
"selectURLs": "URLを選択",
|
||||||
@ -124,18 +122,18 @@
|
|||||||
"obtainium": "Obtainium",
|
"obtainium": "Obtainium",
|
||||||
"materialYou": "Material You",
|
"materialYou": "Material You",
|
||||||
"appSortBy": "アプリの並び方",
|
"appSortBy": "アプリの並び方",
|
||||||
"authorName": "作者/名前",
|
"authorName": "作者名/アプリ名",
|
||||||
"nameAuthor": "名前/作者",
|
"nameAuthor": "アプリ名/作者名",
|
||||||
"asAdded": "追加順",
|
"asAdded": "追加順",
|
||||||
"appSortOrder": "並び順",
|
"appSortOrder": "並び順",
|
||||||
"ascending": "昇順",
|
"ascending": "昇順",
|
||||||
"descending": "下降",
|
"descending": "降順",
|
||||||
"bgUpdateCheckInterval": "バックグラウンド更新の確認間隔",
|
"bgUpdateCheckInterval": "バックグラウンドでのアップデート確認の間隔",
|
||||||
"neverManualOnly": "OFF - 手動のみ",
|
"neverManualOnly": "手動",
|
||||||
"appearance": "外観",
|
"appearance": "外観",
|
||||||
"showWebInAppView": "アプリビューにソースウェブページを表示する",
|
"showWebInAppView": "アプリページにソースのWebページを表示する",
|
||||||
"pinUpdates": "更新があるアプリをトップに固定する",
|
"pinUpdates": "アップデートがあるアプリをトップに固定する",
|
||||||
"updates": "更新",
|
"updates": "アップデート",
|
||||||
"sourceSpecific": "Github アクセストークン",
|
"sourceSpecific": "Github アクセストークン",
|
||||||
"appSource": "アプリのソース",
|
"appSource": "アプリのソース",
|
||||||
"noLogs": "ログはありません",
|
"noLogs": "ログはありません",
|
||||||
@ -143,62 +141,92 @@
|
|||||||
"close": "閉じる",
|
"close": "閉じる",
|
||||||
"share": "共有",
|
"share": "共有",
|
||||||
"appNotFound": "アプリが見つかりません",
|
"appNotFound": "アプリが見つかりません",
|
||||||
"obtainiumExportHyphenatedLowercase": "obtainium-export",
|
"obtainiumExportHyphenatedLowercase": "obtainium-エクスポート",
|
||||||
"pickAnAPK": "APKを選ぶ",
|
"pickAnAPK": "APKを選択",
|
||||||
"appHasMoreThanOnePackage": "{}は複数のパッケージが存在します: ",
|
"appHasMoreThanOnePackage": "{} は複数のパッケージが存在します: ",
|
||||||
"deviceSupportsXArch": "お使いのデバイスは{} CPUアーキテクチャに対応しています。",
|
"deviceSupportsXArch": "お使いのデバイスは {} CPUアーキテクチャに対応しています。",
|
||||||
"deviceSupportsFollowingArchs": "お使いのデバイスは、以下のCPUアーキテクチャをサポートしています。:",
|
"deviceSupportsFollowingArchs": "お使いのデバイスは、以下のCPUアーキテクチャをサポートしています:",
|
||||||
"warning": "警告",
|
"warning": "警告",
|
||||||
"sourceIsXButPackageFromYPrompt": "アプリのソースは'{}'ですが、リリースパッケージは'{}'から来ています。続行しますか?",
|
"sourceIsXButPackageFromYPrompt": "アプリのソースは'{}'ですが、リリースパッケージは'{}'から来ています。続行しますか?",
|
||||||
"updatesAvailable": "更新があります",
|
"updatesAvailable": "アップデートが利用可能",
|
||||||
"updatesAvailableNotifDescription": "Obtainiumが追跡している1つまたは複数のアプリのアップデートが利用可能であることをユーザーに通知します",
|
"updatesAvailableNotifDescription": "Obtainiumが追跡している1つまたは複数のアプリのアップデートが利用可能であることをユーザーに通知する",
|
||||||
"noNewUpdates": "新しいアップデートはありません。",
|
"noNewUpdates": "新しいアップデートはありません",
|
||||||
"xHasAnUpdate": "{}は更新があります。",
|
"xHasAnUpdate": "{} のアップデートが利用可能です",
|
||||||
"appsUpdated": "アプリを更新しました",
|
"appsUpdated": "アプリをアップデートしました",
|
||||||
"appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する",
|
"appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する",
|
||||||
"xWasUpdatedToY": "{}が{}に更新されました。",
|
"xWasUpdatedToY": "{} が {} にアップデートされました",
|
||||||
"errorCheckingUpdates": "アップデート時のエラーチェック",
|
"errorCheckingUpdates": "アップデート確認中のエラー",
|
||||||
"errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデートチェックに失敗した際に表示される通知する",
|
"errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデート確認に失敗した際に表示される通知",
|
||||||
"appsRemoved": "削除されたアプリ",
|
"appsRemoved": "削除されたアプリ",
|
||||||
"appsRemovedNotifDescription": "アプリの読み込み中にエラーが発生したため、1つまたは複数のアプリが削除されたことをユーザーに通知する",
|
"appsRemovedNotifDescription": "アプリの読み込み中にエラーが発生したため、1つまたは複数のアプリが削除されたことをユーザーに通知する",
|
||||||
"xWasRemovedDueToErrorY": "このエラーのため、{}は削除されました: {}",
|
"xWasRemovedDueToErrorY": "このエラーのため、{} は削除されました: {}",
|
||||||
"completeAppInstallation": "アプリのインストールを完了する",
|
"completeAppInstallation": "アプリのインストールを完了する",
|
||||||
"obtainiumMustBeOpenToInstallApps": "アプリをインストールするにはObtainiumが開いている必要があります。",
|
"obtainiumMustBeOpenToInstallApps": "アプリをインストールするにはObtainiumを開いている必要があります。",
|
||||||
"completeAppInstallationNotifDescription": "アプリのインストールを完了するために、Obtainiumに戻る必要があります。",
|
"completeAppInstallationNotifDescription": "アプリのインストールを完了するために、Obtainiumに戻る必要があります。",
|
||||||
"checkingForUpdates": "アップデートの確認",
|
"checkingForUpdates": "アップデートを確認中",
|
||||||
"checkingForUpdatesNotifDescription": "Transient notification that appears when checking for updates",
|
"checkingForUpdatesNotifDescription": "アップデートを確認する際に表示される一時的な通知",
|
||||||
"pleaseAllowInstallPerm": "Obtainiumによるアプリのインストールを許可してください。",
|
"pleaseAllowInstallPerm": "Obtainiumによるアプリのインストールを許可してください。",
|
||||||
"trackOnly": "Track-Only",
|
"trackOnly": "追跡のみ",
|
||||||
"errorWithHttpStatusCode": "エラー {}",
|
"errorWithHttpStatusCode": "エラー {}",
|
||||||
"versionCorrectionDisabled": "バージョン補正無効 (プラグインが動作していません)",
|
"versionCorrectionDisabled": "バージョン補正無効 (プラグインが動作していません)",
|
||||||
"unknown": "不明",
|
"unknown": "不明",
|
||||||
"none": "なし",
|
"none": "なし",
|
||||||
"never": "Never",
|
"never": "しない",
|
||||||
"latestVersionX": "最新版: {}",
|
"latestVersionX": "最新のバージョン: {}",
|
||||||
"installedVersionX": "インストールされたバージョン: {}",
|
"installedVersionX": "インストールされたバージョン: {}",
|
||||||
"lastUpdateCheckX": "最終アップデート確認: {}",
|
"lastUpdateCheckX": "最終アップデート確認: {}",
|
||||||
"remove": "削除",
|
"remove": "削除",
|
||||||
"removeAppQuestion": "アプリを削除しますか?",
|
"yesMarkUpdated": "はい、アップデート済みとしてマークします",
|
||||||
"yesMarkUpdated": "はい、更新済みとしてマーク",
|
|
||||||
"fdroid": "F-Droid",
|
"fdroid": "F-Droid",
|
||||||
"appIdOrName": "アプリIDまたは名前",
|
"appIdOrName": "アプリのIDまたは名前",
|
||||||
"appWithIdOrNameNotFound": "そのIDや名前を持つアプリは見つかりませんでした",
|
"appWithIdOrNameNotFound": "そのIDや名前を持つアプリは見つかりませんでした",
|
||||||
"reposHaveMultipleApps": "レポには複数のAppが含まれることがあります",
|
"reposHaveMultipleApps": "リポジトリには複数のアプリが含まれることがあります",
|
||||||
"fdroidThirdPartyRepo": "F-Droid Third-Party Repo",
|
"fdroidThirdPartyRepo": "F-Droid サードパーティリポジトリ",
|
||||||
"steam": "Steam",
|
"steam": "Steam",
|
||||||
"steamMobile": "Steam Mobile",
|
"steamMobile": "Steam Mobile",
|
||||||
"steamChat": "Steam Chat",
|
"steamChat": "Steam Chat",
|
||||||
|
"install": "インストール",
|
||||||
|
"markInstalled": "インストール済みとしてマークする",
|
||||||
|
"update": "アップデート",
|
||||||
|
"markUpdated": "アップデート済みとしてマークする",
|
||||||
|
"additionalOptions": "追加のオプション",
|
||||||
|
"disableVersionDetection": "バージョン検出を無効にする",
|
||||||
|
"noVersionDetectionExplanation": "このオプションは、バージョン検出が正しく機能しないアプリにのみ使用する必要があります。",
|
||||||
|
"downloadingX": "{} をダウンロード中",
|
||||||
|
"downloadNotifDescription": "アプリのダウンロード状況を通知する",
|
||||||
|
"noAPKFound": "APKが見つかりません",
|
||||||
|
"noVersionDetection": "バージョン検出を行わない",
|
||||||
|
"categorize": "カテゴライズ",
|
||||||
|
"categories": "カテゴリ",
|
||||||
|
"category": "カテゴリ",
|
||||||
|
"noCategory": "カテゴリなし",
|
||||||
|
"noCategories": "カテゴリなし",
|
||||||
|
"deleteCategoriesQuestion": "カテゴリを削除しますか?",
|
||||||
|
"categoryDeleteWarning": "削除されたカテゴリ内のアプリは未分類に設定されます。",
|
||||||
|
"addCategory": "カテゴリを追加",
|
||||||
|
"label": "ラベル",
|
||||||
|
"language": "言語",
|
||||||
|
"storagePermissionDenied": "ストレージ権限が拒否されました",
|
||||||
|
"selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。",
|
||||||
|
"filterAPKsByRegEx": "正規表現でAPKを絞り込む",
|
||||||
|
"removeFromObtainium": "Obtainiumから削除する",
|
||||||
|
"uninstallFromDevice": "デバイスからアンインストールする",
|
||||||
|
"onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。",
|
||||||
|
"removeAppQuestion": {
|
||||||
|
"one": "アプリを削除しますか?",
|
||||||
|
"other": "アプリを削除しますか?"
|
||||||
|
},
|
||||||
"tooManyRequestsTryAgainInMinutes": {
|
"tooManyRequestsTryAgainInMinutes": {
|
||||||
"one": "リクエストが多すぎます(レート制限あり)- {}分後に再試行してください。",
|
"one": "リクエストが多すぎます(レート制限)- {}分後に再試行してください",
|
||||||
"other": "リクエストが多すぎます(レート制限あり)- {}分後に再試行してください。"
|
"other": "リクエストが多すぎます(レート制限)- {}分後に再試行してください"
|
||||||
},
|
},
|
||||||
"bgUpdateGotErrorRetryInMinutes": {
|
"bgUpdateGotErrorRetryInMinutes": {
|
||||||
"one": "BG update checking encountered a {}, will schedule a retry check in {} minute",
|
"one": "バックグラウンドでのアップデート確認で {} の問題が発生, {} 分後に再試行します",
|
||||||
"other": "BG update checking encountered a {}, will schedule a retry check in {} minutes"
|
"other": "バックグラウンドでのアップデート確認で {} の問題が発生, {} 分後に再試行します"
|
||||||
},
|
},
|
||||||
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
||||||
"one": "BG update checking found {} update - will notify user if needed",
|
"one": "バックグラウンドでのアップデート確認で {} 個のアップデートを発見 - 必要に応じてユーザーに通知します",
|
||||||
"other": "BG update checking found {} updates - will notify user if needed"
|
"other": "バックグラウンドでのアップデート確認で {} 個のアップデートを発見 - 必要に応じてユーザーに通知します"
|
||||||
},
|
},
|
||||||
"apps": {
|
"apps": {
|
||||||
"one": "{}個のアプリ",
|
"one": "{}個のアプリ",
|
||||||
@ -225,11 +253,11 @@
|
|||||||
"other": "{n}個のログをクリアしました (前 = {before}, 後 = {after})"
|
"other": "{n}個のログをクリアしました (前 = {before}, 後 = {after})"
|
||||||
},
|
},
|
||||||
"xAndNMoreUpdatesAvailable": {
|
"xAndNMoreUpdatesAvailable": {
|
||||||
"one": "{}と{}のアプリが更新があります。",
|
"one": "{} とさらに {} 個のアプリのアップデートが利用可能です",
|
||||||
"other": "{}と{}のアプリが更新があります。"
|
"other": "{} とさらに {} 個のアプリのアップデートが利用可能です"
|
||||||
},
|
},
|
||||||
"xAndNMoreUpdatesInstalled": {
|
"xAndNMoreUpdatesInstalled": {
|
||||||
"one": "{}と{}のアプリが更新しました。",
|
"one": "{} とさらに {} 個のアプリがアップデートされました",
|
||||||
"other": "{}と{}のアプリが更新しました。"
|
"other": "{} とさらに {} 個のアプリがアップデートされました"
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -4,7 +4,6 @@
|
|||||||
"noVersionFound": "无法确定更新版本",
|
"noVersionFound": "无法确定更新版本",
|
||||||
"urlMatchesNoSource": "URL 与已知来源不符",
|
"urlMatchesNoSource": "URL 与已知来源不符",
|
||||||
"cantInstallOlderVersion": "无法安装旧版应用程序",
|
"cantInstallOlderVersion": "无法安装旧版应用程序",
|
||||||
"appIdMismatch": "下载的软件包名与现有的应用程序包名不一致",
|
|
||||||
"functionNotImplemented": "该类没有实现此功能",
|
"functionNotImplemented": "该类没有实现此功能",
|
||||||
"placeholder": "占位符",
|
"placeholder": "占位符",
|
||||||
"someErrors": "出现了一些错误",
|
"someErrors": "出现了一些错误",
|
||||||
@ -12,7 +11,7 @@
|
|||||||
"ok": "好的",
|
"ok": "好的",
|
||||||
"and": "和",
|
"and": "和",
|
||||||
"startedBgUpdateTask": "开始后台检查更新任务",
|
"startedBgUpdateTask": "开始后台检查更新任务",
|
||||||
"bgUpdateIgnoreAfterIs": "Bg update ignoreAfter is {}",
|
"bgUpdateIgnoreAfterIs": "下次后台更新检查 {}",
|
||||||
"startedActualBGUpdateCheck": "后台检查更新已开始",
|
"startedActualBGUpdateCheck": "后台检查更新已开始",
|
||||||
"bgUpdateTaskFinished": "后台检查更新已完成",
|
"bgUpdateTaskFinished": "后台检查更新已完成",
|
||||||
"firstRun": "这是你第一次运行 Obtainium",
|
"firstRun": "这是你第一次运行 Obtainium",
|
||||||
@ -22,9 +21,9 @@
|
|||||||
"githubPATFormat": "username:token",
|
"githubPATFormat": "username:token",
|
||||||
"githubPATLinkText": "关于 GitHub 个人访问令牌",
|
"githubPATLinkText": "关于 GitHub 个人访问令牌",
|
||||||
"includePrereleases": "包含预发布版",
|
"includePrereleases": "包含预发布版",
|
||||||
"fallbackToOlderReleases": "回落到旧版",
|
"fallbackToOlderReleases": "回退到旧版",
|
||||||
"filterReleaseTitlesByRegEx": "通过正则表达式过滤发布标题",
|
"filterReleaseTitlesByRegEx": "使用正则以过滤发布标题",
|
||||||
"invalidRegEx": "无效的正则表达式",
|
"invalidRegEx": "表达式无效",
|
||||||
"noDescription": "无描述",
|
"noDescription": "无描述",
|
||||||
"cancel": "取消",
|
"cancel": "取消",
|
||||||
"continue": "继续",
|
"continue": "继续",
|
||||||
@ -109,7 +108,7 @@
|
|||||||
"searchX": "搜索 {}",
|
"searchX": "搜索 {}",
|
||||||
"noResults": "无结果",
|
"noResults": "无结果",
|
||||||
"importX": "导入 {}",
|
"importX": "导入 {}",
|
||||||
"importedAppsIdDisclaimer": "导入的应用程序可能不正确地显示为未安装。要解决这个问题,请通过 Obtainium 重新安装它们。",
|
"importedAppsIdDisclaimer": "导入的应用程序可能显示为未安装。要解决这个问题,请通过 Obtainium 重新安装它们。",
|
||||||
"importErrors": "导入错误",
|
"importErrors": "导入错误",
|
||||||
"importedXOfYApps": "{} 中的 {} 个应用已导入",
|
"importedXOfYApps": "{} 中的 {} 个应用已导入",
|
||||||
"followingURLsHadErrors": "以下 URL 有错误:",
|
"followingURLsHadErrors": "以下 URL 有错误:",
|
||||||
@ -134,7 +133,7 @@
|
|||||||
"neverManualOnly": "手动",
|
"neverManualOnly": "手动",
|
||||||
"appearance": "外观",
|
"appearance": "外观",
|
||||||
"showWebInAppView": "在应用来源页显示网页",
|
"showWebInAppView": "在应用来源页显示网页",
|
||||||
"pinUpdates": "将需要更新的应用固定到顶部",
|
"pinUpdates": "需更新的应用置顶",
|
||||||
"updates": "检查间隔",
|
"updates": "检查间隔",
|
||||||
"sourceSpecific": "Github 访问令牌",
|
"sourceSpecific": "Github 访问令牌",
|
||||||
"appSource": "源代码",
|
"appSource": "源代码",
|
||||||
@ -145,9 +144,9 @@
|
|||||||
"appNotFound": "未找到应用",
|
"appNotFound": "未找到应用",
|
||||||
"obtainiumExportHyphenatedLowercase": "obtainium-导出",
|
"obtainiumExportHyphenatedLowercase": "obtainium-导出",
|
||||||
"pickAnAPK": "选择一个安装包",
|
"pickAnAPK": "选择一个安装包",
|
||||||
"appHasMoreThanOnePackage": "{} 有多于一个安装包:",
|
"appHasMoreThanOnePackage": "{} 有多个架构可用:",
|
||||||
"deviceSupportsXArch": "你的设备支持 {} CPU 架构",
|
"deviceSupportsXArch": "你的设备支持 {} 架构",
|
||||||
"deviceSupportsFollowingArchs": "你的设备支持以下 CPU 架构:",
|
"deviceSupportsFollowingArchs": "你的设备支持以下架构:",
|
||||||
"warning": "警告",
|
"warning": "警告",
|
||||||
"sourceIsXButPackageFromYPrompt": "此应用来源是 '{}' 但更新包来自 '{}'。 继续吗?",
|
"sourceIsXButPackageFromYPrompt": "此应用来源是 '{}' 但更新包来自 '{}'。 继续吗?",
|
||||||
"updatesAvailable": "更新可用",
|
"updatesAvailable": "更新可用",
|
||||||
@ -178,7 +177,6 @@
|
|||||||
"installedVersionX": "已安装: {}",
|
"installedVersionX": "已安装: {}",
|
||||||
"lastUpdateCheckX": "最后检查: {}",
|
"lastUpdateCheckX": "最后检查: {}",
|
||||||
"remove": "删除",
|
"remove": "删除",
|
||||||
"removeAppQuestion": "删除应用?",
|
|
||||||
"yesMarkUpdated": "'是的,标为已更新",
|
"yesMarkUpdated": "'是的,标为已更新",
|
||||||
"fdroid": "F-Droid",
|
"fdroid": "F-Droid",
|
||||||
"appIdOrName": "应用 ID 或名称",
|
"appIdOrName": "应用 ID 或名称",
|
||||||
@ -188,6 +186,36 @@
|
|||||||
"steam": "Steam",
|
"steam": "Steam",
|
||||||
"steamMobile": "Steam Mobile",
|
"steamMobile": "Steam Mobile",
|
||||||
"steamChat": "Steam Chat",
|
"steamChat": "Steam Chat",
|
||||||
|
"install": "安装",
|
||||||
|
"markInstalled": "标记为已安装",
|
||||||
|
"update": "更新",
|
||||||
|
"markUpdated": "标记为已更新",
|
||||||
|
"additionalOptions": "附加选项",
|
||||||
|
"disableVersionDetection": "关闭版本检测",
|
||||||
|
"noVersionDetectionExplanation": "此选项应只用于版本检测不能工作的应用程序",
|
||||||
|
"downloadingX": "下载中 {}",
|
||||||
|
"downloadNotifDescription": "通知用户下载进度",
|
||||||
|
"noAPKFound": "未找到安装包",
|
||||||
|
"noVersionDetection": "无版本检测",
|
||||||
|
"categorize": "归档",
|
||||||
|
"categories": "归档",
|
||||||
|
"category": "类别",
|
||||||
|
"noCategory": "无类别",
|
||||||
|
"noCategories": "无类别",
|
||||||
|
"deleteCategoriesQuestion": "删除所有类别?",
|
||||||
|
"categoryDeleteWarning": "所有被删除类别的应用程序将被设置为无类别",
|
||||||
|
"addCategory": "添加类别",
|
||||||
|
"label": "标签",
|
||||||
|
"language": "语言",
|
||||||
|
"storagePermissionDenied": "存储权限已被拒绝",
|
||||||
|
"selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别",
|
||||||
|
"filterAPKsByRegEx": "Filter APKs by Regular Expression",
|
||||||
|
"removeFromObtainium": "Remove from Obtainium",
|
||||||
|
"uninstallFromDevice": "Uninstall from Device",
|
||||||
|
"removeAppQuestion": {
|
||||||
|
"one": "删除应用?",
|
||||||
|
"other": "删除应用?"
|
||||||
|
},
|
||||||
"tooManyRequestsTryAgainInMinutes": {
|
"tooManyRequestsTryAgainInMinutes": {
|
||||||
"one": "请求过多 (API 限制) - 在 {} 分钟后重试",
|
"one": "请求过多 (API 限制) - 在 {} 分钟后重试",
|
||||||
"other": "请求过多 (API 限制) - 在 {} 分钟后重试"
|
"other": "请求过多 (API 限制) - 在 {} 分钟后重试"
|
||||||
|
@ -25,8 +25,9 @@ class APKMirror extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
Response res = await get(Uri.parse('$standardUrl/feed'));
|
Response res = await get(Uri.parse('$standardUrl/feed'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
String? titleString = parse(res.body)
|
String? titleString = parse(res.body)
|
||||||
@ -45,7 +46,7 @@ class APKMirror extends AppSource {
|
|||||||
}
|
}
|
||||||
return APKDetails(version, [], getAppNames(standardUrl));
|
return APKDetails(version, [], getAppNames(standardUrl));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
149
lib/app_sources/codeberg.dart
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
import 'dart:convert';
|
||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
|
import 'package:http/http.dart';
|
||||||
|
import 'package:obtainium/components/generated_form.dart';
|
||||||
|
import 'package:obtainium/custom_errors.dart';
|
||||||
|
import 'package:obtainium/providers/source_provider.dart';
|
||||||
|
|
||||||
|
class Codeberg extends AppSource {
|
||||||
|
Codeberg() {
|
||||||
|
host = 'codeberg.org';
|
||||||
|
|
||||||
|
additionalSourceSpecificSettingFormItems = [];
|
||||||
|
|
||||||
|
additionalSourceAppSpecificSettingFormItems = [
|
||||||
|
[
|
||||||
|
GeneratedFormSwitch('includePrereleases',
|
||||||
|
label: tr('includePrereleases'), defaultValue: false)
|
||||||
|
],
|
||||||
|
[
|
||||||
|
GeneratedFormSwitch('fallbackToOlderReleases',
|
||||||
|
label: tr('fallbackToOlderReleases'), defaultValue: true)
|
||||||
|
],
|
||||||
|
[
|
||||||
|
GeneratedFormTextField('filterReleaseTitlesByRegEx',
|
||||||
|
label: tr('filterReleaseTitlesByRegEx'),
|
||||||
|
required: false,
|
||||||
|
additionalValidators: [
|
||||||
|
(value) {
|
||||||
|
return regExValidator(value);
|
||||||
|
}
|
||||||
|
])
|
||||||
|
]
|
||||||
|
];
|
||||||
|
|
||||||
|
canSearch = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
String standardizeURL(String url) {
|
||||||
|
RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+');
|
||||||
|
RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase());
|
||||||
|
if (match == null) {
|
||||||
|
throw InvalidURLError(name);
|
||||||
|
}
|
||||||
|
return url.substring(0, match.end);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
String? changeLogPageFromStandardUrl(String standardUrl) =>
|
||||||
|
'$standardUrl/releases';
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
|
String standardUrl,
|
||||||
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
|
bool includePrereleases = additionalSettings['includePrereleases'];
|
||||||
|
bool fallbackToOlderReleases =
|
||||||
|
additionalSettings['fallbackToOlderReleases'];
|
||||||
|
String? regexFilter =
|
||||||
|
(additionalSettings['filterReleaseTitlesByRegEx'] as String?)
|
||||||
|
?.isNotEmpty ==
|
||||||
|
true
|
||||||
|
? additionalSettings['filterReleaseTitlesByRegEx']
|
||||||
|
: null;
|
||||||
|
Response res = await get(Uri.parse(
|
||||||
|
'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100'));
|
||||||
|
if (res.statusCode == 200) {
|
||||||
|
var releases = jsonDecode(res.body) as List<dynamic>;
|
||||||
|
|
||||||
|
List<String> getReleaseAPKUrls(dynamic release) =>
|
||||||
|
(release['assets'] as List<dynamic>?)
|
||||||
|
?.map((e) {
|
||||||
|
return e['name'] != null && e['browser_download_url'] != null
|
||||||
|
? MapEntry(e['name'] as String,
|
||||||
|
e['browser_download_url'] as String)
|
||||||
|
: const MapEntry('', '');
|
||||||
|
})
|
||||||
|
.where((element) => element.key.toLowerCase().endsWith('.apk'))
|
||||||
|
.map((e) => e.value)
|
||||||
|
.toList() ??
|
||||||
|
[];
|
||||||
|
|
||||||
|
dynamic targetRelease;
|
||||||
|
|
||||||
|
for (int i = 0; i < releases.length; i++) {
|
||||||
|
if (!fallbackToOlderReleases && i > 0) break;
|
||||||
|
if (!includePrereleases && releases[i]['prerelease'] == true) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (releases[i]['draft'] == true) {
|
||||||
|
// Draft releases not supported
|
||||||
|
}
|
||||||
|
var nameToFilter = releases[i]['name'] as String?;
|
||||||
|
if (nameToFilter == null || nameToFilter.trim().isEmpty) {
|
||||||
|
// Some leave titles empty so tag is used
|
||||||
|
nameToFilter = releases[i]['tag_name'] as String;
|
||||||
|
}
|
||||||
|
if (regexFilter != null &&
|
||||||
|
!RegExp(regexFilter).hasMatch(nameToFilter.trim())) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
var apkUrls = getReleaseAPKUrls(releases[i]);
|
||||||
|
if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
targetRelease = releases[i];
|
||||||
|
targetRelease['apkUrls'] = apkUrls;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (targetRelease == null) {
|
||||||
|
throw NoReleasesError();
|
||||||
|
}
|
||||||
|
String? version = targetRelease['tag_name'];
|
||||||
|
if (version == null) {
|
||||||
|
throw NoVersionError();
|
||||||
|
}
|
||||||
|
return APKDetails(version, targetRelease['apkUrls'] as List<String>,
|
||||||
|
getAppNames(standardUrl));
|
||||||
|
} else {
|
||||||
|
throw getObtainiumHttpError(res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
AppNames getAppNames(String standardUrl) {
|
||||||
|
String temp = standardUrl.substring(standardUrl.indexOf('://') + 3);
|
||||||
|
List<String> names = temp.substring(temp.indexOf('/') + 1).split('/');
|
||||||
|
return AppNames(names[0], names[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<Map<String, String>> search(String query) async {
|
||||||
|
Response res = await get(Uri.parse(
|
||||||
|
'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100'));
|
||||||
|
if (res.statusCode == 200) {
|
||||||
|
Map<String, String> urlsWithDescriptions = {};
|
||||||
|
for (var e in (jsonDecode(res.body)['data'] as List<dynamic>)) {
|
||||||
|
urlsWithDescriptions.addAll({
|
||||||
|
e['html_url'] as String: e['description'] != null
|
||||||
|
? e['description'] as String
|
||||||
|
: tr('noDescription')
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return urlsWithDescriptions;
|
||||||
|
} else {
|
||||||
|
throw getObtainiumHttpError(res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -32,7 +32,7 @@ class FDroid extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String? tryInferringAppId(String standardUrl,
|
String? tryInferringAppId(String standardUrl,
|
||||||
{List<String> additionalData = const []}) {
|
{Map<String, dynamic> additionalSettings = const {}}) {
|
||||||
return Uri.parse(standardUrl).pathSegments.last;
|
return Uri.parse(standardUrl).pathSegments.last;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -54,14 +54,15 @@ class FDroid extends AppSource {
|
|||||||
return APKDetails(latestVersion, apkUrls,
|
return APKDetails(latestVersion, apkUrls,
|
||||||
AppNames(name, Uri.parse(standardUrl).pathSegments.last));
|
AppNames(name, Uri.parse(standardUrl).pathSegments.last));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
String? appId = tryInferringAppId(standardUrl);
|
String? appId = tryInferringAppId(standardUrl);
|
||||||
return getAPKUrlsFromFDroidPackagesAPIResponse(
|
return getAPKUrlsFromFDroidPackagesAPIResponse(
|
||||||
await get(Uri.parse('https://f-droid.org/api/v1/packages/$appId')),
|
await get(Uri.parse('https://f-droid.org/api/v1/packages/$appId')),
|
||||||
|
@ -9,13 +9,12 @@ class FDroidRepo extends AppSource {
|
|||||||
FDroidRepo() {
|
FDroidRepo() {
|
||||||
name = tr('fdroidThirdPartyRepo');
|
name = tr('fdroidThirdPartyRepo');
|
||||||
|
|
||||||
additionalSourceAppSpecificFormItems = [
|
additionalSourceAppSpecificSettingFormItems = [
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField('appIdOrName',
|
||||||
label: tr('appIdOrName'),
|
label: tr('appIdOrName'),
|
||||||
hint: tr('reposHaveMultipleApps'),
|
hint: tr('reposHaveMultipleApps'),
|
||||||
required: true,
|
required: true)
|
||||||
key: 'appIdOrName')
|
|
||||||
]
|
]
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@ -23,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);
|
||||||
@ -33,13 +32,10 @@ class FDroidRepo extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
String? appIdOrName = findGeneratedFormValueByKey(
|
) async {
|
||||||
additionalSourceAppSpecificFormItems
|
String? appIdOrName = additionalSettings['appIdOrName'];
|
||||||
.reduce((value, element) => [...value, ...element]),
|
|
||||||
additionalData,
|
|
||||||
'appIdOrName');
|
|
||||||
if (appIdOrName == null) {
|
if (appIdOrName == null) {
|
||||||
throw NoReleasesError();
|
throw NoReleasesError();
|
||||||
}
|
}
|
||||||
@ -84,7 +80,7 @@ class FDroidRepo extends AppSource {
|
|||||||
.toList();
|
.toList();
|
||||||
return APKDetails(latestVersion, apkUrls, AppNames(authorName, appName));
|
return APKDetails(latestVersion, apkUrls, AppNames(authorName, appName));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,12 +12,10 @@ class GitHub extends AppSource {
|
|||||||
GitHub() {
|
GitHub() {
|
||||||
host = 'github.com';
|
host = 'github.com';
|
||||||
|
|
||||||
additionalSourceAppSpecificDefaults = ['true', 'true', ''];
|
|
||||||
|
|
||||||
additionalSourceSpecificSettingFormItems = [
|
additionalSourceSpecificSettingFormItems = [
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField('github-creds',
|
||||||
label: tr('githubPATLabel'),
|
label: tr('githubPATLabel'),
|
||||||
id: 'github-creds',
|
password: true,
|
||||||
required: false,
|
required: false,
|
||||||
additionalValidators: [
|
additionalValidators: [
|
||||||
(value) {
|
(value) {
|
||||||
@ -52,31 +50,22 @@ class GitHub extends AppSource {
|
|||||||
])
|
])
|
||||||
];
|
];
|
||||||
|
|
||||||
additionalSourceAppSpecificFormItems = [
|
additionalSourceAppSpecificSettingFormItems = [
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormSwitch('includePrereleases',
|
||||||
label: tr('includePrereleases'), type: FormItemType.bool)
|
label: tr('includePrereleases'), defaultValue: false)
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormSwitch('fallbackToOlderReleases',
|
||||||
label: tr('fallbackToOlderReleases'), type: FormItemType.bool)
|
label: tr('fallbackToOlderReleases'), defaultValue: true)
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField('filterReleaseTitlesByRegEx',
|
||||||
label: tr('filterReleaseTitlesByRegEx'),
|
label: tr('filterReleaseTitlesByRegEx'),
|
||||||
type: FormItemType.string,
|
|
||||||
required: false,
|
required: false,
|
||||||
additionalValidators: [
|
additionalValidators: [
|
||||||
(value) {
|
(value) {
|
||||||
if (value == null || value.isEmpty) {
|
return regExValidator(value);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
RegExp(value);
|
|
||||||
} catch (e) {
|
|
||||||
return tr('invalidRegEx');
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
])
|
])
|
||||||
]
|
]
|
||||||
@ -99,7 +88,7 @@ class GitHub extends AppSource {
|
|||||||
SettingsProvider settingsProvider = SettingsProvider();
|
SettingsProvider settingsProvider = SettingsProvider();
|
||||||
await settingsProvider.initializeSettings();
|
await settingsProvider.initializeSettings();
|
||||||
String? creds = settingsProvider
|
String? creds = settingsProvider
|
||||||
.getSettingString(additionalSourceSpecificSettingFormItems[0].id);
|
.getSettingString(additionalSourceSpecificSettingFormItems[0].key);
|
||||||
return creds != null && creds.isNotEmpty ? '$creds@' : '';
|
return creds != null && creds.isNotEmpty ? '$creds@' : '';
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -109,17 +98,20 @@ class GitHub extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
var includePrereleases =
|
) async {
|
||||||
additionalData.isNotEmpty && additionalData[0] == 'true';
|
bool includePrereleases = additionalSettings['includePrereleases'];
|
||||||
var fallbackToOlderReleases =
|
bool fallbackToOlderReleases =
|
||||||
additionalData.length >= 2 && additionalData[1] == 'true';
|
additionalSettings['fallbackToOlderReleases'];
|
||||||
var regexFilter = additionalData.length >= 3 && additionalData[2].isNotEmpty
|
String? regexFilter =
|
||||||
? additionalData[2]
|
(additionalSettings['filterReleaseTitlesByRegEx'] as String?)
|
||||||
: null;
|
?.isNotEmpty ==
|
||||||
|
true
|
||||||
|
? additionalSettings['filterReleaseTitlesByRegEx']
|
||||||
|
: null;
|
||||||
Response res = await get(Uri.parse(
|
Response res = await get(Uri.parse(
|
||||||
'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases'));
|
'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
var releases = jsonDecode(res.body) as List<dynamic>;
|
var releases = jsonDecode(res.body) as List<dynamic>;
|
||||||
|
|
||||||
@ -141,14 +133,17 @@ class GitHub extends AppSource {
|
|||||||
if (!includePrereleases && releases[i]['prerelease'] == true) {
|
if (!includePrereleases && releases[i]['prerelease'] == true) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
var nameToFilter = releases[i]['name'] as String?;
|
||||||
|
if (nameToFilter == null || nameToFilter.trim().isEmpty) {
|
||||||
|
// Some leave titles empty so tag is used
|
||||||
|
nameToFilter = releases[i]['tag_name'] as String;
|
||||||
|
}
|
||||||
if (regexFilter != null &&
|
if (regexFilter != null &&
|
||||||
!RegExp(regexFilter)
|
!RegExp(regexFilter).hasMatch(nameToFilter.trim())) {
|
||||||
.hasMatch((releases[i]['name'] as String).trim())) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
var apkUrls = getReleaseAPKUrls(releases[i]);
|
var apkUrls = getReleaseAPKUrls(releases[i]);
|
||||||
if (apkUrls.isEmpty && !trackOnly) {
|
if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
targetRelease = releases[i];
|
targetRelease = releases[i];
|
||||||
|
@ -25,8 +25,9 @@ class GitLab extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom'));
|
Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
var standardUri = Uri.parse(standardUrl);
|
var standardUri = Uri.parse(standardUrl);
|
||||||
@ -58,7 +59,7 @@ class GitLab extends AppSource {
|
|||||||
}
|
}
|
||||||
return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl));
|
return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
47
lib/app_sources/html.dart
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
|
import 'package:html/parser.dart';
|
||||||
|
import 'package:http/http.dart';
|
||||||
|
import 'package:obtainium/custom_errors.dart';
|
||||||
|
import 'package:obtainium/providers/source_provider.dart';
|
||||||
|
|
||||||
|
class HTML extends AppSource {
|
||||||
|
@override
|
||||||
|
String standardizeURL(String url) {
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
String? changeLogPageFromStandardUrl(String standardUrl) => null;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
|
String standardUrl,
|
||||||
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
|
var uri = Uri.parse(standardUrl);
|
||||||
|
Response res = await get(uri);
|
||||||
|
if (res.statusCode == 200) {
|
||||||
|
List<String> links = parse(res.body)
|
||||||
|
.querySelectorAll('a')
|
||||||
|
.map((element) => element.attributes['href'] ?? '')
|
||||||
|
.where((element) => element.toLowerCase().endsWith('.apk'))
|
||||||
|
.toList();
|
||||||
|
links.sort((a, b) => a.split('/').last.compareTo(b.split('/').last));
|
||||||
|
if (links.isEmpty) {
|
||||||
|
throw NoReleasesError();
|
||||||
|
}
|
||||||
|
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
|
||||||
|
: '${uri.origin}/$e')
|
||||||
|
.toList();
|
||||||
|
return APKDetails(version, apkUrls, AppNames(uri.host, tr('app')));
|
||||||
|
} else {
|
||||||
|
throw getObtainiumHttpError(res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -23,14 +23,15 @@ class IzzyOnDroid extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String? tryInferringAppId(String standardUrl,
|
String? tryInferringAppId(String standardUrl,
|
||||||
{List<String> additionalData = const []}) {
|
{Map<String, dynamic> additionalSettings = const {}}) {
|
||||||
return FDroid().tryInferringAppId(standardUrl);
|
return FDroid().tryInferringAppId(standardUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
String? appId = tryInferringAppId(standardUrl);
|
String? appId = tryInferringAppId(standardUrl);
|
||||||
return FDroid().getAPKUrlsFromFDroidPackagesAPIResponse(
|
return FDroid().getAPKUrlsFromFDroidPackagesAPIResponse(
|
||||||
await get(
|
await get(
|
||||||
|
@ -24,8 +24,9 @@ class Mullvad extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
Response res = await get(Uri.parse('$standardUrl/en/download/android'));
|
Response res = await get(Uri.parse('$standardUrl/en/download/android'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
var version = parse(res.body)
|
var version = parse(res.body)
|
||||||
@ -42,7 +43,7 @@ class Mullvad extends AppSource {
|
|||||||
['https://mullvad.net/download/app/apk/latest'],
|
['https://mullvad.net/download/app/apk/latest'],
|
||||||
AppNames(name, 'Mullvad-VPN'));
|
AppNames(name, 'Mullvad-VPN'));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,8 +18,9 @@ class Signal extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
Response res =
|
Response res =
|
||||||
await get(Uri.parse('https://updates.$host/android/latest.json'));
|
await get(Uri.parse('https://updates.$host/android/latest.json'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
@ -32,7 +33,7 @@ class Signal extends AppSource {
|
|||||||
}
|
}
|
||||||
return APKDetails(version, apkUrls, AppNames(name, 'Signal'));
|
return APKDetails(version, apkUrls, AppNames(name, 'Signal'));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -23,8 +23,9 @@ class SourceForge extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
Response res = await get(Uri.parse('$standardUrl/rss?path=/'));
|
Response res = await get(Uri.parse('$standardUrl/rss?path=/'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
var parsedHtml = parse(res.body);
|
var parsedHtml = parse(res.body);
|
||||||
@ -56,7 +57,7 @@ class SourceForge extends AppSource {
|
|||||||
AppNames(
|
AppNames(
|
||||||
name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1)));
|
name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1)));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,14 +9,8 @@ class SteamMobile extends AppSource {
|
|||||||
SteamMobile() {
|
SteamMobile() {
|
||||||
host = 'store.steampowered.com';
|
host = 'store.steampowered.com';
|
||||||
name = tr('steam');
|
name = tr('steam');
|
||||||
additionalSourceAppSpecificFormItems = [
|
additionalSourceAppSpecificSettingFormItems = [
|
||||||
[
|
[GeneratedFormDropdown('app', apks.entries.toList(), label: tr('app'))]
|
||||||
GeneratedFormItem(
|
|
||||||
label: tr('app'),
|
|
||||||
key: 'app',
|
|
||||||
required: true,
|
|
||||||
opts: apks.entries.toList())
|
|
||||||
]
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -32,15 +26,12 @@ class SteamMobile extends AppSource {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl,
|
||||||
{bool trackOnly = false}) async {
|
Map<String, dynamic> additionalSettings,
|
||||||
|
) async {
|
||||||
Response res = await get(Uri.parse('https://$host/mobile'));
|
Response res = await get(Uri.parse('https://$host/mobile'));
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
var apkNamePrefix = findGeneratedFormValueByKey(
|
var apkNamePrefix = additionalSettings['app'] as String?;
|
||||||
additionalSourceAppSpecificFormItems
|
|
||||||
.reduce((value, element) => [...value, ...element]),
|
|
||||||
additionalData,
|
|
||||||
'app');
|
|
||||||
if (apkNamePrefix == null) {
|
if (apkNamePrefix == null) {
|
||||||
throw NoReleasesError();
|
throw NoReleasesError();
|
||||||
}
|
}
|
||||||
@ -63,7 +54,7 @@ class SteamMobile extends AppSource {
|
|||||||
var apkUrls = [links[0]];
|
var apkUrls = [links[0]];
|
||||||
return APKDetails(version, apkUrls, AppNames(name, apks[apkNamePrefix]!));
|
return APKDetails(version, apkUrls, AppNames(name, apks[apkNamePrefix]!));
|
||||||
} else {
|
} else {
|
||||||
throw NoReleasesError();
|
throw getObtainiumHttpError(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,121 +1,215 @@
|
|||||||
|
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';
|
||||||
|
|
||||||
enum FormItemType { string, bool }
|
abstract class GeneratedFormItem {
|
||||||
|
|
||||||
typedef OnValueChanges = void Function(
|
|
||||||
List<String> values, bool valid, bool isBuilding);
|
|
||||||
|
|
||||||
class GeneratedFormItem {
|
|
||||||
late String key;
|
late String key;
|
||||||
late String label;
|
late String label;
|
||||||
late FormItemType type;
|
late List<Widget> belowWidgets;
|
||||||
|
late dynamic defaultValue;
|
||||||
|
List<dynamic> additionalValidators;
|
||||||
|
dynamic ensureType(dynamic val);
|
||||||
|
|
||||||
|
GeneratedFormItem(this.key,
|
||||||
|
{this.label = 'Input',
|
||||||
|
this.belowWidgets = const [],
|
||||||
|
this.defaultValue,
|
||||||
|
this.additionalValidators = const []});
|
||||||
|
}
|
||||||
|
|
||||||
|
class GeneratedFormTextField extends GeneratedFormItem {
|
||||||
late bool required;
|
late bool required;
|
||||||
late int max;
|
late int max;
|
||||||
late List<String? Function(String? value)> additionalValidators;
|
|
||||||
late String id;
|
|
||||||
late List<Widget> belowWidgets;
|
|
||||||
late String? hint;
|
late String? hint;
|
||||||
late List<MapEntry<String, String>>? opts;
|
late bool password;
|
||||||
|
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField(String key,
|
||||||
{this.label = 'Input',
|
{String label = 'Input',
|
||||||
this.type = FormItemType.string,
|
List<Widget> belowWidgets = const [],
|
||||||
|
String defaultValue = '',
|
||||||
|
List<String? Function(String? value)> additionalValidators = const [],
|
||||||
this.required = true,
|
this.required = true,
|
||||||
this.max = 1,
|
this.max = 1,
|
||||||
this.additionalValidators = const [],
|
|
||||||
this.id = 'input',
|
|
||||||
this.belowWidgets = const [],
|
|
||||||
this.hint,
|
this.hint,
|
||||||
this.opts,
|
this.password = false})
|
||||||
this.key = 'default'}) {
|
: super(key,
|
||||||
if (type != FormItemType.string) {
|
label: label,
|
||||||
required = false;
|
belowWidgets: belowWidgets,
|
||||||
}
|
defaultValue: defaultValue,
|
||||||
|
additionalValidators: additionalValidators);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String ensureType(val) {
|
||||||
|
return val.toString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class GeneratedFormDropdown extends GeneratedFormItem {
|
||||||
|
late List<MapEntry<String, String>>? opts;
|
||||||
|
|
||||||
|
GeneratedFormDropdown(
|
||||||
|
String key,
|
||||||
|
this.opts, {
|
||||||
|
String label = 'Input',
|
||||||
|
List<Widget> belowWidgets = const [],
|
||||||
|
String defaultValue = '',
|
||||||
|
List<String? Function(String? value)> additionalValidators = const [],
|
||||||
|
}) : super(key,
|
||||||
|
label: label,
|
||||||
|
belowWidgets: belowWidgets,
|
||||||
|
defaultValue: defaultValue,
|
||||||
|
additionalValidators: additionalValidators);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String ensureType(val) {
|
||||||
|
return val.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class GeneratedFormSwitch extends GeneratedFormItem {
|
||||||
|
GeneratedFormSwitch(
|
||||||
|
String key, {
|
||||||
|
String label = 'Input',
|
||||||
|
List<Widget> belowWidgets = const [],
|
||||||
|
bool defaultValue = false,
|
||||||
|
List<String? Function(bool value)> additionalValidators = const [],
|
||||||
|
}) : super(key,
|
||||||
|
label: label,
|
||||||
|
belowWidgets: belowWidgets,
|
||||||
|
defaultValue: defaultValue,
|
||||||
|
additionalValidators: additionalValidators);
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool ensureType(val) {
|
||||||
|
return val == true || val == 'true';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class GeneratedFormTagInput extends GeneratedFormItem {
|
||||||
|
late MapEntry<String, String>? deleteConfirmationMessage;
|
||||||
|
late bool singleSelect;
|
||||||
|
late WrapAlignment alignment;
|
||||||
|
late String emptyMessage;
|
||||||
|
late bool showLabelWhenNotEmpty;
|
||||||
|
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',
|
||||||
|
this.showLabelWhenNotEmpty = true})
|
||||||
|
: 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(
|
||||||
|
Map<String, dynamic> values, bool valid, bool isBuilding);
|
||||||
|
|
||||||
class GeneratedForm extends StatefulWidget {
|
class GeneratedForm extends StatefulWidget {
|
||||||
const GeneratedForm(
|
const GeneratedForm(
|
||||||
{super.key,
|
{super.key, required this.items, required this.onValueChanges});
|
||||||
required this.items,
|
|
||||||
required this.onValueChanges,
|
|
||||||
required this.defaultValues});
|
|
||||||
|
|
||||||
final List<List<GeneratedFormItem>> items;
|
final List<List<GeneratedFormItem>> items;
|
||||||
final OnValueChanges onValueChanges;
|
final OnValueChanges onValueChanges;
|
||||||
final List<String> defaultValues;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<GeneratedForm> createState() => _GeneratedFormState();
|
State<GeneratedForm> createState() => _GeneratedFormState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 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();
|
||||||
|
}
|
||||||
|
|
||||||
class _GeneratedFormState extends State<GeneratedForm> {
|
class _GeneratedFormState extends State<GeneratedForm> {
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
late List<List<String>> values;
|
Map<String, dynamic> values = {};
|
||||||
late List<List<Widget>> formInputs;
|
late List<List<Widget>> formInputs;
|
||||||
List<List<Widget>> rows = [];
|
List<List<Widget>> rows = [];
|
||||||
|
String? initKey;
|
||||||
|
|
||||||
// If any value changes, call this to update the parent with value and validity
|
// If any value changes, call this to update the parent with value and validity
|
||||||
void someValueChanged({bool isBuilding = false}) {
|
void someValueChanged({bool isBuilding = false}) {
|
||||||
List<String> returnValues = [];
|
Map<String, dynamic> returnValues = values;
|
||||||
var valid = true;
|
var valid = true;
|
||||||
for (int r = 0; r < values.length; r++) {
|
for (int r = 0; r < widget.items.length; r++) {
|
||||||
for (int i = 0; i < values[r].length; i++) {
|
for (int i = 0; i < widget.items[r].length; i++) {
|
||||||
returnValues.add(values[r][i]);
|
|
||||||
if (formInputs[r][i] is TextFormField) {
|
if (formInputs[r][i] is TextFormField) {
|
||||||
valid = valid &&
|
var fieldState =
|
||||||
((formInputs[r][i].key as GlobalKey<FormFieldState>)
|
(formInputs[r][i].key as GlobalKey<FormFieldState>).currentState;
|
||||||
.currentState
|
if (fieldState != null) {
|
||||||
?.isValid ??
|
valid = valid && fieldState.isValid;
|
||||||
false);
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
widget.onValueChanges(returnValues, valid, isBuilding);
|
widget.onValueChanges(returnValues, valid, isBuilding);
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
initForm() {
|
||||||
void initState() {
|
initKey = widget.key.toString();
|
||||||
super.initState();
|
|
||||||
|
|
||||||
// Initialize form values as all empty
|
// Initialize form values as all empty
|
||||||
int j = 0;
|
values.clear();
|
||||||
values = widget.items
|
for (var row in widget.items) {
|
||||||
.map((row) => row.map((e) {
|
for (var e in row) {
|
||||||
return j < widget.defaultValues.length
|
values[e.key] = e.defaultValue;
|
||||||
? widget.defaultValues[j++]
|
}
|
||||||
: e.opts != null
|
}
|
||||||
? e.opts!.first.key
|
|
||||||
: '';
|
|
||||||
}).toList())
|
|
||||||
.toList();
|
|
||||||
|
|
||||||
// Dynamically create form inputs
|
// Dynamically create form inputs
|
||||||
formInputs = widget.items.asMap().entries.map((row) {
|
formInputs = widget.items.asMap().entries.map((row) {
|
||||||
return row.value.asMap().entries.map((e) {
|
return row.value.asMap().entries.map((e) {
|
||||||
if (e.value.type == FormItemType.string && e.value.opts == null) {
|
var formItem = e.value;
|
||||||
|
if (formItem is GeneratedFormTextField) {
|
||||||
final formFieldKey = GlobalKey<FormFieldState>();
|
final formFieldKey = GlobalKey<FormFieldState>();
|
||||||
return TextFormField(
|
return TextFormField(
|
||||||
|
obscureText: formItem.password,
|
||||||
|
autocorrect: !formItem.password,
|
||||||
|
enableSuggestions: !formItem.password,
|
||||||
key: formFieldKey,
|
key: formFieldKey,
|
||||||
initialValue: values[row.key][e.key],
|
initialValue: values[formItem.key],
|
||||||
autovalidateMode: AutovalidateMode.onUserInteraction,
|
autovalidateMode: AutovalidateMode.onUserInteraction,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
setState(() {
|
setState(() {
|
||||||
values[row.key][e.key] = value;
|
values[formItem.key] = value;
|
||||||
someValueChanged();
|
someValueChanged();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
helperText: e.value.label + (e.value.required ? ' *' : ''),
|
helperText: formItem.label + (formItem.required ? ' *' : ''),
|
||||||
hintText: e.value.hint),
|
hintText: formItem.hint),
|
||||||
minLines: e.value.max <= 1 ? null : e.value.max,
|
minLines: formItem.max <= 1 ? null : formItem.max,
|
||||||
maxLines: e.value.max <= 1 ? 1 : e.value.max,
|
maxLines: formItem.max <= 1 ? 1 : formItem.max,
|
||||||
validator: (value) {
|
validator: (value) {
|
||||||
if (e.value.required && (value == null || value.trim().isEmpty)) {
|
if (formItem.required &&
|
||||||
return '${e.value.label} ${tr('requiredInBrackets')}';
|
(value == null || value.trim().isEmpty)) {
|
||||||
|
return '${formItem.label} ${tr('requiredInBrackets')}';
|
||||||
}
|
}
|
||||||
for (var validator in e.value.additionalValidators) {
|
for (var validator in formItem.additionalValidators) {
|
||||||
String? result = validator(value);
|
String? result = validator(value);
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
return result;
|
return result;
|
||||||
@ -124,21 +218,20 @@ class _GeneratedFormState extends State<GeneratedForm> {
|
|||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
} else if (e.value.type == FormItemType.string &&
|
} else if (formItem is GeneratedFormDropdown) {
|
||||||
e.value.opts != null) {
|
if (formItem.opts!.isEmpty) {
|
||||||
if (e.value.opts!.isEmpty) {
|
|
||||||
return Text(tr('dropdownNoOptsError'));
|
return Text(tr('dropdownNoOptsError'));
|
||||||
}
|
}
|
||||||
return DropdownButtonFormField(
|
return DropdownButtonFormField(
|
||||||
decoration: InputDecoration(labelText: e.value.label),
|
decoration: InputDecoration(labelText: formItem.label),
|
||||||
value: values[row.key][e.key],
|
value: values[formItem.key],
|
||||||
items: e.value.opts!
|
items: formItem.opts!
|
||||||
.map((e) =>
|
.map((e2) =>
|
||||||
DropdownMenuItem(value: e.key, child: Text(e.value)))
|
DropdownMenuItem(value: e2.key, child: Text(e2.value)))
|
||||||
.toList(),
|
.toList(),
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
setState(() {
|
setState(() {
|
||||||
values[row.key][e.key] = value ?? e.value.opts!.first.key;
|
values[formItem.key] = value ?? formItem.opts!.first.key;
|
||||||
someValueChanged();
|
someValueChanged();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -150,25 +243,214 @@ class _GeneratedFormState extends State<GeneratedForm> {
|
|||||||
someValueChanged(isBuilding: true);
|
someValueChanged(isBuilding: true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
initForm();
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
if (widget.key.toString() != initKey) {
|
||||||
|
initForm();
|
||||||
|
}
|
||||||
for (var r = 0; r < formInputs.length; r++) {
|
for (var r = 0; r < formInputs.length; r++) {
|
||||||
for (var e = 0; e < formInputs[r].length; e++) {
|
for (var e = 0; e < formInputs[r].length; e++) {
|
||||||
if (widget.items[r][e].type == FormItemType.bool) {
|
if (widget.items[r][e] is GeneratedFormSwitch) {
|
||||||
formInputs[r][e] = Row(
|
formInputs[r][e] = Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Text(widget.items[r][e].label),
|
Text(widget.items[r][e].label),
|
||||||
Switch(
|
Switch(
|
||||||
value: values[r][e] == 'true',
|
value: values[widget.items[r][e].key],
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
setState(() {
|
setState(() {
|
||||||
values[r][e] = value ? 'true' : '';
|
values[widget.items[r][e].key] = value;
|
||||||
someValueChanged();
|
someValueChanged();
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
} else if (widget.items[r][e] is GeneratedFormTagInput) {
|
||||||
|
formInputs[r][e] =
|
||||||
|
Column(crossAxisAlignment: CrossAxisAlignment.stretch, children: [
|
||||||
|
if ((values[widget.items[r][e].key]
|
||||||
|
as Map<String, MapEntry<int, bool>>?)
|
||||||
|
?.isNotEmpty ==
|
||||||
|
true &&
|
||||||
|
(widget.items[r][e] as GeneratedFormTagInput)
|
||||||
|
.showLabelWhenNotEmpty)
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment:
|
||||||
|
(widget.items[r][e] as GeneratedFormTagInput).alignment ==
|
||||||
|
WrapAlignment.center
|
||||||
|
? CrossAxisAlignment.center
|
||||||
|
: CrossAxisAlignment.stretch,
|
||||||
|
children: [
|
||||||
|
Text(widget.items[r][e].label),
|
||||||
|
const SizedBox(
|
||||||
|
height: 8,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
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,
|
||||||
|
)
|
||||||
|
: 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 ??= {};
|
||||||
|
if (temp[label] == null) {
|
||||||
|
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'),
|
||||||
|
)),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -178,9 +460,8 @@ class _GeneratedFormState extends State<GeneratedForm> {
|
|||||||
if (rowInputs.key > 0) {
|
if (rowInputs.key > 0) {
|
||||||
rows.add([
|
rows.add([
|
||||||
SizedBox(
|
SizedBox(
|
||||||
height: widget.items[rowInputs.key][0].type == FormItemType.bool &&
|
height: widget.items[rowInputs.key][0] is GeneratedFormSwitch &&
|
||||||
widget.items[rowInputs.key - 1][0].type ==
|
widget.items[rowInputs.key - 1][0] is! GeneratedFormSwitch
|
||||||
FormItemType.string
|
|
||||||
? 25
|
? 25
|
||||||
: 8,
|
: 8,
|
||||||
)
|
)
|
||||||
@ -217,18 +498,3 @@ class _GeneratedFormState extends State<GeneratedForm> {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
String? findGeneratedFormValueByKey(
|
|
||||||
List<GeneratedFormItem> items, List<String> values, String key) {
|
|
||||||
var foundIndex = -1;
|
|
||||||
for (var i = 0; i < items.length; i++) {
|
|
||||||
if (items[i].key == key) {
|
|
||||||
foundIndex = i;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (foundIndex >= 0 && foundIndex < values.length) {
|
|
||||||
return values[foundIndex];
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
@ -8,28 +8,29 @@ class GeneratedFormModal extends StatefulWidget {
|
|||||||
{super.key,
|
{super.key,
|
||||||
required this.title,
|
required this.title,
|
||||||
required this.items,
|
required this.items,
|
||||||
required this.defaultValues,
|
|
||||||
this.initValid = false,
|
this.initValid = false,
|
||||||
this.message = ''});
|
this.message = '',
|
||||||
|
this.additionalWidgets = const [],
|
||||||
|
this.singleNullReturnButton});
|
||||||
|
|
||||||
final String title;
|
final String title;
|
||||||
final String message;
|
final String message;
|
||||||
final List<List<GeneratedFormItem>> items;
|
final List<List<GeneratedFormItem>> items;
|
||||||
final List<String> defaultValues;
|
|
||||||
final bool initValid;
|
final bool initValid;
|
||||||
|
final List<Widget> additionalWidgets;
|
||||||
|
final String? singleNullReturnButton;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<GeneratedFormModal> createState() => _GeneratedFormModalState();
|
State<GeneratedFormModal> createState() => _GeneratedFormModalState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _GeneratedFormModalState extends State<GeneratedFormModal> {
|
class _GeneratedFormModalState extends State<GeneratedFormModal> {
|
||||||
List<String> values = [];
|
Map<String, dynamic> values = {};
|
||||||
bool valid = false;
|
bool valid = false;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
values = widget.defaultValues;
|
|
||||||
valid = widget.initValid || widget.items.isEmpty;
|
valid = widget.initValid || widget.items.isEmpty;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -57,25 +58,29 @@ class _GeneratedFormModalState extends State<GeneratedFormModal> {
|
|||||||
this.valid = valid;
|
this.valid = valid;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}),
|
||||||
defaultValues: widget.defaultValues)
|
if (widget.additionalWidgets.isNotEmpty) ...widget.additionalWidgets
|
||||||
]),
|
]),
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
Navigator.of(context).pop(null);
|
Navigator.of(context).pop(null);
|
||||||
},
|
},
|
||||||
child: Text(tr('cancel'))),
|
child: Text(widget.singleNullReturnButton == null
|
||||||
TextButton(
|
? tr('cancel')
|
||||||
onPressed: !valid
|
: widget.singleNullReturnButton!)),
|
||||||
? null
|
widget.singleNullReturnButton == null
|
||||||
: () {
|
? TextButton(
|
||||||
if (valid) {
|
onPressed: !valid
|
||||||
HapticFeedback.selectionClick();
|
? null
|
||||||
Navigator.of(context).pop(values);
|
: () {
|
||||||
}
|
if (valid) {
|
||||||
},
|
HapticFeedback.selectionClick();
|
||||||
child: Text(tr('continue')))
|
Navigator.of(context).pop(values);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Text(tr('continue')))
|
||||||
|
: const SizedBox.shrink()
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -13,13 +13,10 @@ class ObtainiumError {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class RateLimitError {
|
class RateLimitError extends ObtainiumError {
|
||||||
late int remainingMinutes;
|
late int remainingMinutes;
|
||||||
RateLimitError(this.remainingMinutes);
|
RateLimitError(this.remainingMinutes)
|
||||||
|
: super(plural('tooManyRequestsTryAgainInMinutes', remainingMinutes));
|
||||||
@override
|
|
||||||
String toString() =>
|
|
||||||
plural('tooManyRequestsTryAgainInMinutes', remainingMinutes);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class InvalidURLError extends ObtainiumError {
|
class InvalidURLError extends ObtainiumError {
|
||||||
@ -32,7 +29,7 @@ class NoReleasesError extends ObtainiumError {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class NoAPKError extends ObtainiumError {
|
class NoAPKError extends ObtainiumError {
|
||||||
NoAPKError() : super(tr('noReleaseFound'));
|
NoAPKError() : super(tr('noAPKFound'));
|
||||||
}
|
}
|
||||||
|
|
||||||
class NoVersionError extends ObtainiumError {
|
class NoVersionError extends ObtainiumError {
|
||||||
@ -47,10 +44,6 @@ class DowngradeError extends ObtainiumError {
|
|||||||
DowngradeError() : super(tr('cantInstallOlderVersion'));
|
DowngradeError() : super(tr('cantInstallOlderVersion'));
|
||||||
}
|
}
|
||||||
|
|
||||||
class IDChangedError extends ObtainiumError {
|
|
||||||
IDChangedError() : super(tr('appIdMismatch'));
|
|
||||||
}
|
|
||||||
|
|
||||||
class NotImplementedError extends ObtainiumError {
|
class NotImplementedError extends ObtainiumError {
|
||||||
NotImplementedError() : super(tr('functionNotImplemented'));
|
NotImplementedError() : super(tr('functionNotImplemented'));
|
||||||
}
|
}
|
||||||
|
@ -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.8.18';
|
const String currentVersion = '0.10.8';
|
||||||
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
|
||||||
|
|
||||||
@ -32,7 +32,8 @@ const supportedLocales = [
|
|||||||
Locale('zh'),
|
Locale('zh'),
|
||||||
Locale('it'),
|
Locale('it'),
|
||||||
Locale('ja'),
|
Locale('ja'),
|
||||||
Locale('hu')
|
Locale('hu'),
|
||||||
|
Locale('de')
|
||||||
];
|
];
|
||||||
const fallbackLocale = Locale('en');
|
const fallbackLocale = Locale('en');
|
||||||
const localeDir = 'assets/translations';
|
const localeDir = 'assets/translations';
|
||||||
@ -42,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) {
|
||||||
@ -159,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()),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
@ -199,9 +205,8 @@ class _ObtainiumState extends State<Obtainium> {
|
|||||||
currentReleaseTag,
|
currentReleaseTag,
|
||||||
[],
|
[],
|
||||||
0,
|
0,
|
||||||
['true'],
|
{'includePrereleases': true},
|
||||||
null,
|
null,
|
||||||
false,
|
|
||||||
false)
|
false)
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,7 @@ import 'package:obtainium/custom_errors.dart';
|
|||||||
import 'package:obtainium/main.dart';
|
import 'package:obtainium/main.dart';
|
||||||
import 'package:obtainium/pages/app.dart';
|
import 'package:obtainium/pages/app.dart';
|
||||||
import 'package:obtainium/pages/import_export.dart';
|
import 'package:obtainium/pages/import_export.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';
|
||||||
@ -23,39 +24,42 @@ class AddAppPage extends StatefulWidget {
|
|||||||
|
|
||||||
class _AddAppPageState extends State<AddAppPage> {
|
class _AddAppPageState extends State<AddAppPage> {
|
||||||
bool gettingAppInfo = false;
|
bool gettingAppInfo = false;
|
||||||
|
bool searching = false;
|
||||||
|
|
||||||
String userInput = '';
|
String userInput = '';
|
||||||
String searchQuery = '';
|
String searchQuery = '';
|
||||||
AppSource? pickedSource;
|
AppSource? pickedSource;
|
||||||
List<String> sourceSpecificAdditionalData = [];
|
Map<String, dynamic> additionalSettings = {};
|
||||||
bool sourceSpecificDataIsValid = true;
|
bool additionalSettingsValid = true;
|
||||||
List<String> otherAdditionalData = [];
|
List<String> pickedCategories = [];
|
||||||
bool otherAdditionalDataIsValid = true;
|
int searchnum = 0;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
SourceProvider sourceProvider = SourceProvider();
|
SourceProvider sourceProvider = SourceProvider();
|
||||||
AppsProvider appsProvider = context.read<AppsProvider>();
|
AppsProvider appsProvider = context.read<AppsProvider>();
|
||||||
|
|
||||||
changeUserInput(String input, bool valid, bool isBuilding) {
|
bool doingSomething = gettingAppInfo || searching;
|
||||||
userInput = input;
|
|
||||||
fn() {
|
|
||||||
var source = valid ? sourceProvider.getSource(userInput) : null;
|
|
||||||
if (pickedSource.runtimeType != source.runtimeType) {
|
|
||||||
pickedSource = source;
|
|
||||||
sourceSpecificAdditionalData =
|
|
||||||
source != null ? source.additionalSourceAppSpecificDefaults : [];
|
|
||||||
sourceSpecificDataIsValid = source != null
|
|
||||||
? !sourceProvider.ifSourceAppsRequireAdditionalData(source)
|
|
||||||
: true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isBuilding) {
|
changeUserInput(String input, bool valid, bool isBuilding,
|
||||||
fn();
|
{bool isSearch = false}) {
|
||||||
} else {
|
userInput = input;
|
||||||
|
if (!isBuilding) {
|
||||||
setState(() {
|
setState(() {
|
||||||
fn();
|
if (isSearch) {
|
||||||
|
searchnum++;
|
||||||
|
}
|
||||||
|
var source = valid ? sourceProvider.getSource(userInput) : null;
|
||||||
|
if (pickedSource.runtimeType != source.runtimeType) {
|
||||||
|
pickedSource = source;
|
||||||
|
additionalSettings = source != null
|
||||||
|
? getDefaultValuesFromFormItems(
|
||||||
|
source.combinedAppSpecificSettingFormItems)
|
||||||
|
: {};
|
||||||
|
additionalSettingsValid = source != null
|
||||||
|
? !sourceProvider.ifRequiredAppSpecificSettingsExist(source)
|
||||||
|
: true;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -66,13 +70,12 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
});
|
});
|
||||||
var settingsProvider = context.read<SettingsProvider>();
|
var settingsProvider = context.read<SettingsProvider>();
|
||||||
() async {
|
() async {
|
||||||
var userPickedTrackOnly = findGeneratedFormValueByKey(
|
var userPickedTrackOnly = additionalSettings['trackOnly'] == true;
|
||||||
pickedSource!.additionalAppSpecificSourceAgnosticFormItems,
|
var userPickedNoVersionDetection =
|
||||||
otherAdditionalData,
|
additionalSettings['noVersionDetection'] == true;
|
||||||
'trackOnlyFormItemKey') ==
|
|
||||||
'true';
|
|
||||||
var cont = true;
|
var cont = true;
|
||||||
if ((userPickedTrackOnly || pickedSource!.enforceTrackOnly) &&
|
if ((userPickedTrackOnly || pickedSource!.enforceTrackOnly) &&
|
||||||
|
// ignore: use_build_context_synchronously
|
||||||
await showDialog(
|
await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext ctx) {
|
builder: (BuildContext ctx) {
|
||||||
@ -83,7 +86,6 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
: tr('app')
|
: tr('app')
|
||||||
]),
|
]),
|
||||||
items: const [],
|
items: const [],
|
||||||
defaultValues: const [],
|
|
||||||
message:
|
message:
|
||||||
'${pickedSource!.enforceTrackOnly ? tr('appsFromSourceAreTrackOnly') : tr('youPickedTrackOnly')}\n\n${tr('trackOnlyAppDescription')}',
|
'${pickedSource!.enforceTrackOnly ? tr('appsFromSourceAreTrackOnly') : tr('youPickedTrackOnly')}\n\n${tr('trackOnlyAppDescription')}',
|
||||||
);
|
);
|
||||||
@ -91,17 +93,33 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
null) {
|
null) {
|
||||||
cont = false;
|
cont = false;
|
||||||
}
|
}
|
||||||
|
if (userPickedNoVersionDetection &&
|
||||||
|
// ignore: use_build_context_synchronously
|
||||||
|
await showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (BuildContext ctx) {
|
||||||
|
return GeneratedFormModal(
|
||||||
|
title: tr('disableVersionDetection'),
|
||||||
|
items: const [],
|
||||||
|
message: tr('noVersionDetectionExplanation'),
|
||||||
|
);
|
||||||
|
}) ==
|
||||||
|
null) {
|
||||||
|
cont = false;
|
||||||
|
}
|
||||||
if (cont) {
|
if (cont) {
|
||||||
HapticFeedback.selectionClick();
|
HapticFeedback.selectionClick();
|
||||||
var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly;
|
var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly;
|
||||||
App app = await sourceProvider.getApp(
|
App app = await sourceProvider.getApp(
|
||||||
pickedSource!, userInput, sourceSpecificAdditionalData,
|
pickedSource!, userInput, additionalSettings,
|
||||||
trackOnly: trackOnly);
|
trackOnlyOverride: trackOnly,
|
||||||
|
noVersionDetectionOverride: userPickedNoVersionDetection);
|
||||||
if (!trackOnly) {
|
if (!trackOnly) {
|
||||||
await settingsProvider.getInstallPermission();
|
await settingsProvider.getInstallPermission();
|
||||||
}
|
}
|
||||||
// Only download the APK here if you need to for the package ID
|
// Only download the APK here if you need to for the package ID
|
||||||
if (sourceProvider.isTempId(app.id) && !app.trackOnly) {
|
if (sourceProvider.isTempId(app.id) &&
|
||||||
|
app.additionalSettings['trackOnly'] != true) {
|
||||||
// ignore: use_build_context_synchronously
|
// ignore: use_build_context_synchronously
|
||||||
var apkUrl = await appsProvider.confirmApkUrl(app, context);
|
var apkUrl = await appsProvider.confirmApkUrl(app, context);
|
||||||
if (apkUrl == null) {
|
if (apkUrl == null) {
|
||||||
@ -116,9 +134,10 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
if (appsProvider.apps.containsKey(app.id)) {
|
if (appsProvider.apps.containsKey(app.id)) {
|
||||||
throw ObtainiumError(tr('appAlreadyAdded'));
|
throw ObtainiumError(tr('appAlreadyAdded'));
|
||||||
}
|
}
|
||||||
if (app.trackOnly) {
|
if (app.additionalSettings['trackOnly'] == true) {
|
||||||
app.installedVersion = app.latestVersion;
|
app.installedVersion = app.latestVersion;
|
||||||
}
|
}
|
||||||
|
app.categories = pickedCategories;
|
||||||
await appsProvider.saveApps([app]);
|
await appsProvider.saveApps([app]);
|
||||||
|
|
||||||
return app;
|
return app;
|
||||||
@ -155,10 +174,12 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: GeneratedForm(
|
child: GeneratedForm(
|
||||||
|
key: Key(searchnum.toString()),
|
||||||
items: [
|
items: [
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField('appSourceURL',
|
||||||
label: tr('appSourceURL'),
|
label: tr('appSourceURL'),
|
||||||
|
defaultValue: userInput,
|
||||||
additionalValidators: [
|
additionalValidators: [
|
||||||
(value) {
|
(value) {
|
||||||
try {
|
try {
|
||||||
@ -180,26 +201,21 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
]
|
]
|
||||||
],
|
],
|
||||||
onValueChanges: (values, valid, isBuilding) {
|
onValueChanges: (values, valid, isBuilding) {
|
||||||
changeUserInput(
|
changeUserInput(values['appSourceURL']!,
|
||||||
values[0], valid, isBuilding);
|
valid, isBuilding);
|
||||||
},
|
})),
|
||||||
defaultValues: const [])),
|
|
||||||
const SizedBox(
|
const SizedBox(
|
||||||
width: 16,
|
width: 16,
|
||||||
),
|
),
|
||||||
gettingAppInfo
|
gettingAppInfo
|
||||||
? const CircularProgressIndicator()
|
? const CircularProgressIndicator()
|
||||||
: ElevatedButton(
|
: ElevatedButton(
|
||||||
onPressed: gettingAppInfo ||
|
onPressed: doingSomething ||
|
||||||
pickedSource == null ||
|
pickedSource == null ||
|
||||||
(pickedSource!
|
(pickedSource!
|
||||||
.additionalSourceAppSpecificFormItems
|
.combinedAppSpecificSettingFormItems
|
||||||
.isNotEmpty &&
|
.isNotEmpty &&
|
||||||
!sourceSpecificDataIsValid) ||
|
!additionalSettingsValid)
|
||||||
(pickedSource!
|
|
||||||
.additionalAppSpecificSourceAgnosticDefaults
|
|
||||||
.isNotEmpty &&
|
|
||||||
!otherAdditionalDataIsValid)
|
|
||||||
? null
|
? null
|
||||||
: addApp,
|
: addApp,
|
||||||
child: Text(tr('add')))
|
child: Text(tr('add')))
|
||||||
@ -224,27 +240,33 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
child: GeneratedForm(
|
child: GeneratedForm(
|
||||||
items: [
|
items: [
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField(
|
||||||
|
'searchSomeSources',
|
||||||
label: tr('searchSomeSourcesLabel'),
|
label: tr('searchSomeSourcesLabel'),
|
||||||
required: false),
|
required: false),
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
onValueChanges: (values, valid, isBuilding) {
|
onValueChanges: (values, valid, isBuilding) {
|
||||||
if (values.isNotEmpty && valid) {
|
if (values.isNotEmpty &&
|
||||||
|
valid &&
|
||||||
|
!isBuilding) {
|
||||||
setState(() {
|
setState(() {
|
||||||
searchQuery = values[0].trim();
|
searchQuery =
|
||||||
|
values['searchSomeSources']!.trim();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}),
|
||||||
defaultValues: const ['']),
|
|
||||||
),
|
),
|
||||||
const SizedBox(
|
const SizedBox(
|
||||||
width: 16,
|
width: 16,
|
||||||
),
|
),
|
||||||
ElevatedButton(
|
ElevatedButton(
|
||||||
onPressed: searchQuery.isEmpty || gettingAppInfo
|
onPressed: searchQuery.isEmpty || doingSomething
|
||||||
? null
|
? null
|
||||||
: () {
|
: () {
|
||||||
|
setState(() {
|
||||||
|
searching = true;
|
||||||
|
});
|
||||||
Future.wait(sourceProvider.sources
|
Future.wait(sourceProvider.sources
|
||||||
.where((e) => e.canSearch)
|
.where((e) => e.canSearch)
|
||||||
.map((e) =>
|
.map((e) =>
|
||||||
@ -281,26 +303,21 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
if (selectedUrls != null &&
|
if (selectedUrls != null &&
|
||||||
selectedUrls.isNotEmpty) {
|
selectedUrls.isNotEmpty) {
|
||||||
changeUserInput(
|
changeUserInput(
|
||||||
selectedUrls[0], true, true);
|
selectedUrls[0], true, false,
|
||||||
addApp(resetUserInputAfter: true);
|
isSearch: true);
|
||||||
}
|
}
|
||||||
}).catchError((e) {
|
}).catchError((e) {
|
||||||
showError(e, context);
|
showError(e, context);
|
||||||
|
}).whenComplete(() {
|
||||||
|
setState(() {
|
||||||
|
searching = false;
|
||||||
|
});
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
child: Text(tr('search')))
|
child: Text(tr('search')))
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
if (pickedSource != null &&
|
if (pickedSource != null)
|
||||||
(pickedSource!.additionalSourceAppSpecificDefaults
|
|
||||||
.isNotEmpty ||
|
|
||||||
pickedSource!
|
|
||||||
.additionalAppSpecificSourceAgnosticFormItems
|
|
||||||
.where((e) => pickedSource!.enforceTrackOnly
|
|
||||||
? e.key != 'trackOnlyFormItemKey'
|
|
||||||
: true)
|
|
||||||
.map((e) => [e])
|
|
||||||
.isNotEmpty))
|
|
||||||
Column(
|
Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
children: [
|
children: [
|
||||||
@ -316,52 +333,30 @@ class _AddAppPageState extends State<AddAppPage> {
|
|||||||
const SizedBox(
|
const SizedBox(
|
||||||
height: 16,
|
height: 16,
|
||||||
),
|
),
|
||||||
if (pickedSource!
|
|
||||||
.additionalSourceAppSpecificFormItems
|
|
||||||
.isNotEmpty)
|
|
||||||
GeneratedForm(
|
|
||||||
items: pickedSource!
|
|
||||||
.additionalSourceAppSpecificFormItems,
|
|
||||||
onValueChanges: (values, valid, isBuilding) {
|
|
||||||
if (isBuilding) {
|
|
||||||
sourceSpecificAdditionalData = values;
|
|
||||||
sourceSpecificDataIsValid = valid;
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
sourceSpecificAdditionalData = values;
|
|
||||||
sourceSpecificDataIsValid = valid;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
defaultValues: pickedSource!
|
|
||||||
.additionalSourceAppSpecificDefaults),
|
|
||||||
if (pickedSource!
|
|
||||||
.additionalAppSpecificSourceAgnosticDefaults
|
|
||||||
.isNotEmpty)
|
|
||||||
const SizedBox(
|
|
||||||
height: 8,
|
|
||||||
),
|
|
||||||
GeneratedForm(
|
GeneratedForm(
|
||||||
|
key: Key(pickedSource.runtimeType.toString()),
|
||||||
items: pickedSource!
|
items: pickedSource!
|
||||||
.additionalAppSpecificSourceAgnosticFormItems
|
.combinedAppSpecificSettingFormItems,
|
||||||
.where((e) => pickedSource!.enforceTrackOnly
|
|
||||||
? e.key != 'trackOnlyFormItemKey'
|
|
||||||
: true)
|
|
||||||
.map((e) => [e])
|
|
||||||
.toList(),
|
|
||||||
onValueChanges: (values, valid, isBuilding) {
|
onValueChanges: (values, valid, isBuilding) {
|
||||||
if (isBuilding) {
|
if (!isBuilding) {
|
||||||
otherAdditionalData = values;
|
|
||||||
otherAdditionalDataIsValid = valid;
|
|
||||||
} else {
|
|
||||||
setState(() {
|
setState(() {
|
||||||
otherAdditionalData = values;
|
additionalSettings = values;
|
||||||
otherAdditionalDataIsValid = valid;
|
additionalSettingsValid = valid;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}),
|
||||||
defaultValues: pickedSource!
|
Column(
|
||||||
.additionalAppSpecificSourceAgnosticDefaults),
|
children: [
|
||||||
|
const SizedBox(
|
||||||
|
height: 16,
|
||||||
|
),
|
||||||
|
CategoryEditorSelector(
|
||||||
|
alignment: WrapAlignment.start,
|
||||||
|
onSelected: (categories) {
|
||||||
|
pickedCategories = categories;
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
else
|
else
|
||||||
|
@ -4,6 +4,7 @@ import 'package:flutter/services.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';
|
||||||
@ -40,98 +41,139 @@ class _AppPageState extends State<AppPage> {
|
|||||||
prevApp = app;
|
prevApp = app;
|
||||||
getUpdate(app.app.id);
|
getUpdate(app.app.id);
|
||||||
}
|
}
|
||||||
|
var trackOnly = app?.app.additionalSettings['trackOnly'] == true;
|
||||||
|
var noVersionDetection =
|
||||||
|
app?.app.additionalSettings['noVersionDetection'] == true;
|
||||||
|
|
||||||
|
var infoColumn = Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
|
children: [
|
||||||
|
GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
if (app?.app.url != null) {
|
||||||
|
launchUrlString(app?.app.url ?? '',
|
||||||
|
mode: LaunchMode.externalApplication);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Text(
|
||||||
|
app?.app.url ?? '',
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: const TextStyle(
|
||||||
|
decoration: TextDecoration.underline,
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
fontSize: 12),
|
||||||
|
)),
|
||||||
|
const SizedBox(
|
||||||
|
height: 32,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
tr('latestVersionX', args: [app?.app.latestVersion ?? tr('unknown')]),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: Theme.of(context).textTheme.bodyLarge,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
'${tr('installedVersionX', args: [
|
||||||
|
app?.app.installedVersion ?? tr('none')
|
||||||
|
])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [
|
||||||
|
tr('app')
|
||||||
|
])}' : ''}',
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: Theme.of(context).textTheme.bodyLarge,
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 32,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
tr('lastUpdateCheckX', args: [
|
||||||
|
app?.app.lastUpdateCheck == null
|
||||||
|
? tr('never')
|
||||||
|
: '\n${app?.app.lastUpdateCheck?.toLocal()}'
|
||||||
|
]),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 48,
|
||||||
|
),
|
||||||
|
CategoryEditorSelector(
|
||||||
|
alignment: WrapAlignment.center,
|
||||||
|
preselected:
|
||||||
|
app?.app.categories != null ? app!.app.categories.toSet() : {},
|
||||||
|
onSelected: (categories) {
|
||||||
|
if (app != null) {
|
||||||
|
app.app.categories = categories;
|
||||||
|
appsProvider.saveApps([app.app]);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
var fullInfoColumn = Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
|
children: [
|
||||||
|
const SizedBox(height: 150),
|
||||||
|
app?.installedInfo != null
|
||||||
|
? Row(mainAxisAlignment: MainAxisAlignment.center, children: [
|
||||||
|
Image.memory(
|
||||||
|
app!.installedInfo!.icon!,
|
||||||
|
height: 150,
|
||||||
|
gaplessPlayback: true,
|
||||||
|
)
|
||||||
|
])
|
||||||
|
: Container(),
|
||||||
|
const SizedBox(
|
||||||
|
height: 25,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
app?.installedInfo?.name ?? app?.app.name ?? tr('app'),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: Theme.of(context).textTheme.displayLarge,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
tr('byX', args: [app?.app.author ?? tr('unknown')]),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: Theme.of(context).textTheme.headlineMedium,
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 32,
|
||||||
|
),
|
||||||
|
infoColumn,
|
||||||
|
const SizedBox(height: 150)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: settingsProvider.showAppWebpage ? AppBar() : null,
|
appBar: settingsProvider.showAppWebpage ? AppBar() : null,
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||||
body: RefreshIndicator(
|
body: RefreshIndicator(
|
||||||
child: settingsProvider.showAppWebpage
|
child: settingsProvider.showAppWebpage
|
||||||
? WebView(
|
? app != null
|
||||||
backgroundColor: Theme.of(context).colorScheme.background,
|
? WebViewWidget(
|
||||||
initialUrl: app?.app.url,
|
controller: WebViewController()
|
||||||
javascriptMode: JavascriptMode.unrestricted,
|
..setJavaScriptMode(JavaScriptMode.unrestricted)
|
||||||
)
|
..setBackgroundColor(
|
||||||
: CustomScrollView(
|
Theme.of(context).colorScheme.background)
|
||||||
slivers: [
|
..setJavaScriptMode(JavaScriptMode.unrestricted)
|
||||||
SliverFillRemaining(
|
..setNavigationDelegate(
|
||||||
child: Column(
|
NavigationDelegate(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
onWebResourceError: (WebResourceError error) {
|
||||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
if (error.isForMainFrame == true) {
|
||||||
children: [
|
showError(
|
||||||
app?.installedInfo != null
|
ObtainiumError(error.description,
|
||||||
? Row(
|
unexpected: true),
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
context);
|
||||||
children: [
|
|
||||||
Image.memory(
|
|
||||||
app!.installedInfo!.icon!,
|
|
||||||
height: 150,
|
|
||||||
gaplessPlayback: true,
|
|
||||||
)
|
|
||||||
])
|
|
||||||
: Container(),
|
|
||||||
const SizedBox(
|
|
||||||
height: 25,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
app?.installedInfo?.name ?? app?.app.name ?? 'App',
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: Theme.of(context).textTheme.displayLarge,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
tr('byX', args: [app?.app.author ?? tr('unknown')]),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: Theme.of(context).textTheme.headlineMedium,
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 32,
|
|
||||||
),
|
|
||||||
GestureDetector(
|
|
||||||
onTap: () {
|
|
||||||
if (app?.app.url != null) {
|
|
||||||
launchUrlString(app?.app.url ?? '',
|
|
||||||
mode: LaunchMode.externalApplication);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
child: Text(
|
),
|
||||||
app?.app.url ?? '',
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
decoration: TextDecoration.underline,
|
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
fontSize: 12),
|
|
||||||
)),
|
|
||||||
const SizedBox(
|
|
||||||
height: 32,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
tr('latestVersionX',
|
|
||||||
args: [app?.app.latestVersion ?? tr('unknown')]),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: Theme.of(context).textTheme.bodyLarge,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
'${tr('installedVersionX', args: [
|
|
||||||
app?.app.installedVersion ?? tr('none')
|
|
||||||
])}${app?.app.trackOnly == true ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [
|
|
||||||
tr('app')
|
|
||||||
])}' : ''}',
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: Theme.of(context).textTheme.bodyLarge,
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 32,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
tr('lastUpdateCheckX', args: [
|
|
||||||
app?.app.lastUpdateCheck == null
|
|
||||||
? tr('never')
|
|
||||||
: '\n${app?.app.lastUpdateCheck?.toLocal()}'
|
|
||||||
]),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontStyle: FontStyle.italic, fontSize: 12),
|
|
||||||
)
|
)
|
||||||
],
|
..loadRequest(Uri.parse(app.app.url)))
|
||||||
)),
|
: Container()
|
||||||
|
: CustomScrollView(
|
||||||
|
slivers: [
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: Column(children: [fullInfoColumn])),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
onRefresh: () async {
|
onRefresh: () async {
|
||||||
@ -150,8 +192,9 @@ class _AppPageState extends State<AppPage> {
|
|||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
children: [
|
children: [
|
||||||
if (app?.app.installedVersion != null &&
|
if (noVersionDetection &&
|
||||||
app?.app.trackOnly == false &&
|
!trackOnly &&
|
||||||
|
app?.app.installedVersion != null &&
|
||||||
app?.app.installedVersion != app?.app.latestVersion)
|
app?.app.installedVersion != app?.app.latestVersion)
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: app?.downloadProgress != null
|
onPressed: app?.downloadProgress != null
|
||||||
@ -163,13 +206,6 @@ class _AppPageState extends State<AppPage> {
|
|||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
title: Text(tr(
|
title: Text(tr(
|
||||||
'alreadyUpToDateQuestion')),
|
'alreadyUpToDateQuestion')),
|
||||||
content: Text(
|
|
||||||
tr('onlyWorksWithNonEVDApps'),
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight:
|
|
||||||
FontWeight.bold,
|
|
||||||
fontStyle:
|
|
||||||
FontStyle.italic)),
|
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
@ -199,30 +235,49 @@ class _AppPageState extends State<AppPage> {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
tooltip: 'Mark as Updated',
|
tooltip: tr('markUpdated'),
|
||||||
icon: const Icon(Icons.done)),
|
icon: const Icon(Icons.done)),
|
||||||
if (source != null &&
|
if (source != null &&
|
||||||
source.additionalSourceAppSpecificFormItems
|
source
|
||||||
.isNotEmpty)
|
.combinedAppSpecificSettingFormItems.isNotEmpty)
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: app?.downloadProgress != null
|
onPressed: app?.downloadProgress != null
|
||||||
? null
|
? null
|
||||||
: () {
|
: () {
|
||||||
showDialog<List<String>>(
|
showDialog<Map<String, dynamic>?>(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext ctx) {
|
builder: (BuildContext ctx) {
|
||||||
|
var items = source
|
||||||
|
.combinedAppSpecificSettingFormItems
|
||||||
|
.map((row) {
|
||||||
|
row.map((e) {
|
||||||
|
if (app?.app.additionalSettings[
|
||||||
|
e.key] !=
|
||||||
|
null) {
|
||||||
|
e.defaultValue = app?.app
|
||||||
|
.additionalSettings[
|
||||||
|
e.key];
|
||||||
|
}
|
||||||
|
return e;
|
||||||
|
}).toList();
|
||||||
|
return row;
|
||||||
|
}).toList();
|
||||||
return GeneratedFormModal(
|
return GeneratedFormModal(
|
||||||
title: 'Additional Options',
|
title: tr('additionalOptions'),
|
||||||
items: source
|
items: items,
|
||||||
.additionalSourceAppSpecificFormItems,
|
);
|
||||||
defaultValues: app != null
|
|
||||||
? app.app.additionalData
|
|
||||||
: source
|
|
||||||
.additionalSourceAppSpecificDefaults);
|
|
||||||
}).then((values) {
|
}).then((values) {
|
||||||
if (app != null && values != null) {
|
if (app != null && values != null) {
|
||||||
var changedApp = app.app;
|
var changedApp = app.app;
|
||||||
changedApp.additionalData = values;
|
changedApp.additionalSettings =
|
||||||
|
values;
|
||||||
|
if (source.enforceTrackOnly) {
|
||||||
|
changedApp.additionalSettings[
|
||||||
|
'trackOnly'] = true;
|
||||||
|
showError(
|
||||||
|
tr('appsFromSourceAreTrackOnly'),
|
||||||
|
context);
|
||||||
|
}
|
||||||
appsProvider.saveApps(
|
appsProvider.saveApps(
|
||||||
[changedApp]).then((value) {
|
[changedApp]).then((value) {
|
||||||
getUpdate(changedApp.id);
|
getUpdate(changedApp.id);
|
||||||
@ -230,11 +285,44 @@ class _AppPageState extends State<AppPage> {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
tooltip: 'Additional Options',
|
tooltip: tr('additionalOptions'),
|
||||||
icon: const Icon(Icons.settings)),
|
icon: const Icon(Icons.edit)),
|
||||||
|
if (app != null && app.installedInfo != null)
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
appsProvider.openAppSettings(app.app.id);
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.settings),
|
||||||
|
tooltip: tr('settings'),
|
||||||
|
),
|
||||||
|
if (app != null && settingsProvider.showAppWebpage)
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (BuildContext ctx) {
|
||||||
|
return AlertDialog(
|
||||||
|
scrollable: true,
|
||||||
|
content: infoColumn,
|
||||||
|
title: Text(
|
||||||
|
'${app.app.name} ${tr('byX', args: [
|
||||||
|
app.app.author
|
||||||
|
])}'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
child: Text(tr('continue')))
|
||||||
|
],
|
||||||
|
);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.more_horiz),
|
||||||
|
tooltip: tr('more')),
|
||||||
const SizedBox(width: 16.0),
|
const SizedBox(width: 16.0),
|
||||||
Expanded(
|
Expanded(
|
||||||
child: ElevatedButton(
|
child: TextButton(
|
||||||
onPressed: (app?.app.installedVersion == null ||
|
onPressed: (app?.app.installedVersion == null ||
|
||||||
app?.app.installedVersion !=
|
app?.app.installedVersion !=
|
||||||
app?.app.latestVersion) &&
|
app?.app.latestVersion) &&
|
||||||
@ -242,7 +330,9 @@ class _AppPageState extends State<AppPage> {
|
|||||||
? () {
|
? () {
|
||||||
HapticFeedback.heavyImpact();
|
HapticFeedback.heavyImpact();
|
||||||
() async {
|
() async {
|
||||||
if (app?.app.trackOnly != true) {
|
if (app?.app.additionalSettings[
|
||||||
|
'trackOnly'] !=
|
||||||
|
true) {
|
||||||
await settingsProvider
|
await settingsProvider
|
||||||
.getInstallPermission();
|
.getInstallPermission();
|
||||||
}
|
}
|
||||||
@ -257,6 +347,8 @@ class _AppPageState extends State<AppPage> {
|
|||||||
if (res.isNotEmpty && mounted) {
|
if (res.isNotEmpty && mounted) {
|
||||||
Navigator.of(context).pop();
|
Navigator.of(context).pop();
|
||||||
}
|
}
|
||||||
|
}).catchError((e) {
|
||||||
|
showError(e, context);
|
||||||
});
|
});
|
||||||
}).catchError((e) {
|
}).catchError((e) {
|
||||||
showError(e, context);
|
showError(e, context);
|
||||||
@ -264,51 +356,24 @@ class _AppPageState extends State<AppPage> {
|
|||||||
}
|
}
|
||||||
: null,
|
: null,
|
||||||
child: Text(app?.app.installedVersion == null
|
child: Text(app?.app.installedVersion == null
|
||||||
? app?.app.trackOnly == false
|
? !trackOnly
|
||||||
? 'Install'
|
? tr('install')
|
||||||
: 'Mark Installed'
|
: tr('markInstalled')
|
||||||
: app?.app.trackOnly == false
|
: !trackOnly
|
||||||
? 'Update'
|
? tr('update')
|
||||||
: 'Mark Updated'))),
|
: tr('markUpdated')))),
|
||||||
const SizedBox(width: 16.0),
|
const SizedBox(width: 16.0),
|
||||||
ElevatedButton(
|
Expanded(
|
||||||
|
child: TextButton(
|
||||||
onPressed: app?.downloadProgress != null
|
onPressed: app?.downloadProgress != null
|
||||||
? null
|
? null
|
||||||
: () {
|
: () {
|
||||||
showDialog(
|
appsProvider.removeAppsWithModal(
|
||||||
context: context,
|
context, [app!.app]).then((value) {
|
||||||
builder: (BuildContext ctx) {
|
if (value == true) {
|
||||||
return AlertDialog(
|
Navigator.of(context).pop();
|
||||||
title: Text(tr('removeAppQuestion')),
|
}
|
||||||
content: Text(tr(
|
});
|
||||||
'xWillBeRemovedButRemainInstalled',
|
|
||||||
args: [
|
|
||||||
app?.installedInfo?.name ??
|
|
||||||
app?.app.name ??
|
|
||||||
tr('app')
|
|
||||||
])),
|
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
HapticFeedback
|
|
||||||
.selectionClick();
|
|
||||||
appsProvider.removeApps(
|
|
||||||
[app!.app.id]).then((_) {
|
|
||||||
int count = 0;
|
|
||||||
Navigator.of(context)
|
|
||||||
.popUntil((_) =>
|
|
||||||
count++ >= 2);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
child: Text(tr('remove'))),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: Text(tr('cancel')))
|
|
||||||
],
|
|
||||||
);
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
foregroundColor:
|
foregroundColor:
|
||||||
@ -316,7 +381,7 @@ class _AppPageState extends State<AppPage> {
|
|||||||
surfaceTintColor:
|
surfaceTintColor:
|
||||||
Theme.of(context).colorScheme.error),
|
Theme.of(context).colorScheme.error),
|
||||||
child: Text(tr('remove')),
|
child: Text(tr('remove')),
|
||||||
),
|
)),
|
||||||
])),
|
])),
|
||||||
if (app?.downloadProgress != null)
|
if (app?.downloadProgress != null)
|
||||||
Padding(
|
Padding(
|
||||||
|
1209
lib/pages/apps.dart
@ -63,21 +63,29 @@ class _HomePageState extends State<HomePage> {
|
|||||||
.map((e) =>
|
.map((e) =>
|
||||||
NavigationDestination(icon: Icon(e.icon), label: e.title))
|
NavigationDestination(icon: Icon(e.icon), label: e.title))
|
||||||
.toList(),
|
.toList(),
|
||||||
onDestinationSelected: (int index) {
|
onDestinationSelected: (int index) async {
|
||||||
HapticFeedback.selectionClick();
|
HapticFeedback.selectionClick();
|
||||||
setState(() {
|
if (index == 0) {
|
||||||
if (index == 0) {
|
while ((pages[0].widget.key as GlobalKey<AppsPageState>)
|
||||||
|
.currentState !=
|
||||||
|
null) {
|
||||||
|
// Avoid duplicate GlobalKey error
|
||||||
|
await Future.delayed(const Duration(microseconds: 1));
|
||||||
|
}
|
||||||
|
setState(() {
|
||||||
selectedIndexHistory.clear();
|
selectedIndexHistory.clear();
|
||||||
} else if (selectedIndexHistory.isEmpty ||
|
});
|
||||||
(selectedIndexHistory.isNotEmpty &&
|
} else if (selectedIndexHistory.isEmpty ||
|
||||||
selectedIndexHistory.last != index)) {
|
(selectedIndexHistory.isNotEmpty &&
|
||||||
|
selectedIndexHistory.last != index)) {
|
||||||
|
setState(() {
|
||||||
int existingInd = selectedIndexHistory.indexOf(index);
|
int existingInd = selectedIndexHistory.indexOf(index);
|
||||||
if (existingInd >= 0) {
|
if (existingInd >= 0) {
|
||||||
selectedIndexHistory.removeAt(existingInd);
|
selectedIndexHistory.removeAt(existingInd);
|
||||||
}
|
}
|
||||||
selectedIndexHistory.add(index);
|
selectedIndexHistory.add(index);
|
||||||
}
|
});
|
||||||
});
|
}
|
||||||
},
|
},
|
||||||
selectedIndex:
|
selectedIndex:
|
||||||
selectedIndexHistory.isEmpty ? 0 : selectedIndexHistory.last,
|
selectedIndexHistory.isEmpty ? 0 : selectedIndexHistory.last,
|
||||||
|
@ -9,6 +9,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/providers/apps_provider.dart';
|
import 'package:obtainium/providers/apps_provider.dart';
|
||||||
|
import 'package:obtainium/providers/settings_provider.dart';
|
||||||
import 'package:obtainium/providers/source_provider.dart';
|
import 'package:obtainium/providers/source_provider.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
import 'package:file_picker/file_picker.dart';
|
import 'package:file_picker/file_picker.dart';
|
||||||
@ -28,6 +29,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
SourceProvider sourceProvider = SourceProvider();
|
SourceProvider sourceProvider = SourceProvider();
|
||||||
var appsProvider = context.read<AppsProvider>();
|
var appsProvider = context.read<AppsProvider>();
|
||||||
|
var settingsProvider = context.read<SettingsProvider>();
|
||||||
var outlineButtonStyle = ButtonStyle(
|
var outlineButtonStyle = ButtonStyle(
|
||||||
shape: MaterialStateProperty.all(
|
shape: MaterialStateProperty.all(
|
||||||
StadiumBorder(
|
StadiumBorder(
|
||||||
@ -66,6 +68,8 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
showError(
|
showError(
|
||||||
tr('exportedTo', args: [path]),
|
tr('exportedTo', args: [path]),
|
||||||
context);
|
context);
|
||||||
|
}).catchError((e) {
|
||||||
|
showError(e, context);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
child: Text(tr('obtainiumExport')))),
|
child: Text(tr('obtainiumExport')))),
|
||||||
@ -98,6 +102,21 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
appsProvider
|
appsProvider
|
||||||
.importApps(data)
|
.importApps(data)
|
||||||
.then((value) {
|
.then((value) {
|
||||||
|
var cats =
|
||||||
|
settingsProvider.categories;
|
||||||
|
appsProvider.apps
|
||||||
|
.forEach((key, value) {
|
||||||
|
for (var c
|
||||||
|
in value.app.categories) {
|
||||||
|
if (!cats.containsKey(c)) {
|
||||||
|
cats[c] =
|
||||||
|
generateRandomLightColor()
|
||||||
|
.value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
settingsProvider.categories =
|
||||||
|
cats;
|
||||||
showError(
|
showError(
|
||||||
tr('importedX', args: [
|
tr('importedX', args: [
|
||||||
plural('apps', value)
|
plural('apps', value)
|
||||||
@ -138,18 +157,19 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
onPressed: importInProgress
|
onPressed: importInProgress
|
||||||
? null
|
? null
|
||||||
: () {
|
: () {
|
||||||
showDialog(
|
showDialog<Map<String, dynamic>?>(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext ctx) {
|
builder: (BuildContext ctx) {
|
||||||
return GeneratedFormModal(
|
return GeneratedFormModal(
|
||||||
title: tr('importFromURLList'),
|
title: tr('importFromURLList'),
|
||||||
items: [
|
items: [
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField(
|
||||||
|
'appURLList',
|
||||||
label: tr('appURLList'),
|
label: tr('appURLList'),
|
||||||
max: 7,
|
max: 7,
|
||||||
additionalValidators: [
|
additionalValidators: [
|
||||||
(String? value) {
|
(dynamic value) {
|
||||||
if (value != null &&
|
if (value != null &&
|
||||||
value.isNotEmpty) {
|
value.isNotEmpty) {
|
||||||
var lines = value
|
var lines = value
|
||||||
@ -172,12 +192,12 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
])
|
])
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
defaultValues: const [],
|
|
||||||
);
|
);
|
||||||
}).then((values) {
|
}).then((values) {
|
||||||
if (values != null) {
|
if (values != null) {
|
||||||
var urls =
|
var urls =
|
||||||
(values[0] as String).split('\n');
|
(values['appURLList'] as String)
|
||||||
|
.split('\n');
|
||||||
setState(() {
|
setState(() {
|
||||||
importInProgress = true;
|
importInProgress = true;
|
||||||
});
|
});
|
||||||
@ -225,7 +245,8 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
: () {
|
: () {
|
||||||
() async {
|
() async {
|
||||||
var values = await showDialog<
|
var values = await showDialog<
|
||||||
List<String>>(
|
Map<String,
|
||||||
|
dynamic>?>(
|
||||||
context: context,
|
context: context,
|
||||||
builder:
|
builder:
|
||||||
(BuildContext ctx) {
|
(BuildContext ctx) {
|
||||||
@ -236,22 +257,26 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
]),
|
]),
|
||||||
items: [
|
items: [
|
||||||
[
|
[
|
||||||
GeneratedFormItem(
|
GeneratedFormTextField(
|
||||||
|
'searchQuery',
|
||||||
label: tr(
|
label: tr(
|
||||||
'searchQuery'))
|
'searchQuery'))
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
defaultValues: const [],
|
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
if (values != null &&
|
if (values != null &&
|
||||||
values[0].isNotEmpty) {
|
(values['searchQuery']
|
||||||
|
as String?)
|
||||||
|
?.isNotEmpty ==
|
||||||
|
true) {
|
||||||
setState(() {
|
setState(() {
|
||||||
importInProgress = true;
|
importInProgress = true;
|
||||||
});
|
});
|
||||||
var urlsWithDescriptions =
|
var urlsWithDescriptions =
|
||||||
await source
|
await source.search(
|
||||||
.search(values[0]);
|
values['searchQuery']
|
||||||
|
as String);
|
||||||
if (urlsWithDescriptions
|
if (urlsWithDescriptions
|
||||||
.isNotEmpty) {
|
.isNotEmpty) {
|
||||||
var selectedUrls =
|
var selectedUrls =
|
||||||
@ -332,7 +357,9 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
? null
|
? null
|
||||||
: () {
|
: () {
|
||||||
() async {
|
() async {
|
||||||
var values = await showDialog(
|
var values = await showDialog<
|
||||||
|
Map<String,
|
||||||
|
dynamic>?>(
|
||||||
context: context,
|
context: context,
|
||||||
builder:
|
builder:
|
||||||
(BuildContext ctx) {
|
(BuildContext ctx) {
|
||||||
@ -346,10 +373,10 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
.requiredArgs
|
.requiredArgs
|
||||||
.map(
|
.map(
|
||||||
(e) => [
|
(e) => [
|
||||||
GeneratedFormItem(label: e)
|
GeneratedFormTextField(e,
|
||||||
|
label: e)
|
||||||
])
|
])
|
||||||
.toList(),
|
.toList(),
|
||||||
defaultValues: const [],
|
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
if (values != null) {
|
if (values != null) {
|
||||||
@ -359,7 +386,10 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
var urlsWithDescriptions =
|
var urlsWithDescriptions =
|
||||||
await source
|
await source
|
||||||
.getUrlsWithDescriptions(
|
.getUrlsWithDescriptions(
|
||||||
values);
|
values.values
|
||||||
|
.map((e) =>
|
||||||
|
e.toString())
|
||||||
|
.toList());
|
||||||
var selectedUrls =
|
var selectedUrls =
|
||||||
await showDialog<
|
await showDialog<
|
||||||
List<String>?>(
|
List<String>?>(
|
||||||
@ -534,18 +564,22 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> {
|
|||||||
widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')),
|
widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')),
|
||||||
content: Column(children: [
|
content: Column(children: [
|
||||||
...urlWithDescriptionSelections.keys.map((urlWithD) {
|
...urlWithDescriptionSelections.keys.map((urlWithD) {
|
||||||
|
select(bool? value) {
|
||||||
|
setState(() {
|
||||||
|
value ??= false;
|
||||||
|
if (value! && widget.onlyOneSelectionAllowed) {
|
||||||
|
selectOnlyOne(urlWithD.key);
|
||||||
|
} else {
|
||||||
|
urlWithDescriptionSelections[urlWithD] = value!;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return Row(children: [
|
return Row(children: [
|
||||||
Checkbox(
|
Checkbox(
|
||||||
value: urlWithDescriptionSelections[urlWithD],
|
value: urlWithDescriptionSelections[urlWithD],
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
setState(() {
|
select(value);
|
||||||
value ??= false;
|
|
||||||
if (value! && widget.onlyOneSelectionAllowed) {
|
|
||||||
selectOnlyOne(urlWithD.key);
|
|
||||||
} else {
|
|
||||||
urlWithDescriptionSelections[urlWithD] = value!;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}),
|
}),
|
||||||
const SizedBox(
|
const SizedBox(
|
||||||
width: 8,
|
width: 8,
|
||||||
@ -569,12 +603,17 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> {
|
|||||||
const TextStyle(decoration: TextDecoration.underline),
|
const TextStyle(decoration: TextDecoration.underline),
|
||||||
textAlign: TextAlign.start,
|
textAlign: TextAlign.start,
|
||||||
)),
|
)),
|
||||||
Text(
|
GestureDetector(
|
||||||
urlWithD.value.length > 128
|
onTap: () {
|
||||||
? '${urlWithD.value.substring(0, 128)}...'
|
select(!(urlWithDescriptionSelections[urlWithD] ?? false));
|
||||||
: urlWithD.value,
|
},
|
||||||
style: const TextStyle(
|
child: Text(
|
||||||
fontStyle: FontStyle.italic, fontSize: 12),
|
urlWithD.value.length > 128
|
||||||
|
? '${urlWithD.value.substring(0, 128)}...'
|
||||||
|
: urlWithD.value,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontStyle: FontStyle.italic, fontSize: 12),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(
|
const SizedBox(
|
||||||
height: 8,
|
height: 8,
|
||||||
|
@ -1,8 +1,11 @@
|
|||||||
|
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/custom_app_bar.dart';
|
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/custom_errors.dart';
|
import 'package:obtainium/custom_errors.dart';
|
||||||
|
import 'package:obtainium/main.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';
|
||||||
import 'package:obtainium/providers/source_provider.dart';
|
import 'package:obtainium/providers/source_provider.dart';
|
||||||
@ -17,6 +20,21 @@ class SettingsPage extends StatefulWidget {
|
|||||||
State<SettingsPage> createState() => _SettingsPageState();
|
State<SettingsPage> createState() => _SettingsPageState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 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();
|
||||||
|
}
|
||||||
|
|
||||||
class _SettingsPageState extends State<SettingsPage> {
|
class _SettingsPageState extends State<SettingsPage> {
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@ -110,6 +128,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,
|
||||||
@ -138,21 +178,17 @@ class _SettingsPageState extends State<SettingsPage> {
|
|||||||
var sourceSpecificFields = sourceProvider.sources.map((e) {
|
var sourceSpecificFields = sourceProvider.sources.map((e) {
|
||||||
if (e.additionalSourceSpecificSettingFormItems.isNotEmpty) {
|
if (e.additionalSourceSpecificSettingFormItems.isNotEmpty) {
|
||||||
return GeneratedForm(
|
return GeneratedForm(
|
||||||
items: e.additionalSourceSpecificSettingFormItems
|
items: e.additionalSourceSpecificSettingFormItems.map((e) {
|
||||||
.map((e) => [e])
|
e.defaultValue = settingsProvider.getSettingString(e.key);
|
||||||
.toList(),
|
return [e];
|
||||||
|
}).toList(),
|
||||||
onValueChanges: (values, valid, isBuilding) {
|
onValueChanges: (values, valid, isBuilding) {
|
||||||
if (valid) {
|
if (valid && !isBuilding) {
|
||||||
for (var i = 0; i < values.length; i++) {
|
values.forEach((key, value) {
|
||||||
settingsProvider.setSettingString(
|
settingsProvider.setSettingString(key, value);
|
||||||
e.additionalSourceSpecificSettingFormItems[i].id,
|
});
|
||||||
values[i]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
});
|
||||||
defaultValues: e.additionalSourceSpecificSettingFormItems.map((e) {
|
|
||||||
return settingsProvider.getSettingString(e.id) ?? '';
|
|
||||||
}).toList());
|
|
||||||
} else {
|
} else {
|
||||||
return Container();
|
return Container();
|
||||||
}
|
}
|
||||||
@ -195,6 +231,8 @@ class _SettingsPageState extends State<SettingsPage> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
height16,
|
height16,
|
||||||
|
localeDropdown,
|
||||||
|
height16,
|
||||||
Row(
|
Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
@ -237,6 +275,18 @@ class _SettingsPageState extends State<SettingsPage> {
|
|||||||
color: Theme.of(context).colorScheme.primary),
|
color: Theme.of(context).colorScheme.primary),
|
||||||
),
|
),
|
||||||
...sourceSpecificFields,
|
...sourceSpecificFields,
|
||||||
|
const Divider(
|
||||||
|
height: 48,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
tr('categories'),
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context).colorScheme.primary),
|
||||||
|
),
|
||||||
|
height16,
|
||||||
|
const CategoryEditorSelector(
|
||||||
|
showLabelWhenNotEmpty: false,
|
||||||
|
)
|
||||||
],
|
],
|
||||||
))),
|
))),
|
||||||
SliverToBoxAdapter(
|
SliverToBoxAdapter(
|
||||||
@ -351,3 +401,62 @@ 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;
|
||||||
|
final bool showLabelWhenNotEmpty;
|
||||||
|
const CategoryEditorSelector(
|
||||||
|
{super.key,
|
||||||
|
this.onSelected,
|
||||||
|
this.singleSelect = false,
|
||||||
|
this.preselected = const {},
|
||||||
|
this.alignment = WrapAlignment.start,
|
||||||
|
this.showLabelWhenNotEmpty = true});
|
||||||
|
|
||||||
|
@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('categories'),
|
||||||
|
emptyMessage: tr('noCategories'),
|
||||||
|
defaultValue: storedValues,
|
||||||
|
alignment: widget.alignment,
|
||||||
|
deleteConfirmationMessage: MapEntry(
|
||||||
|
tr('deleteCategoriesQuestion'),
|
||||||
|
tr('categoryDeleteWarning')),
|
||||||
|
singleSelect: widget.singleSelect,
|
||||||
|
showLabelWhenNotEmpty: widget.showLabelWhenNotEmpty)
|
||||||
|
]
|
||||||
|
],
|
||||||
|
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());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -5,6 +5,7 @@ import 'dart:async';
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:android_intent_plus/flag.dart';
|
||||||
import 'package:device_info_plus/device_info_plus.dart';
|
import 'package:device_info_plus/device_info_plus.dart';
|
||||||
import 'package:easy_localization/easy_localization.dart';
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
@ -12,16 +13,20 @@ import 'package:flutter/services.dart';
|
|||||||
import 'package:install_plugin_v2/install_plugin_v2.dart';
|
import 'package:install_plugin_v2/install_plugin_v2.dart';
|
||||||
import 'package:installed_apps/app_info.dart';
|
import 'package:installed_apps/app_info.dart';
|
||||||
import 'package:installed_apps/installed_apps.dart';
|
import 'package:installed_apps/installed_apps.dart';
|
||||||
|
import 'package:obtainium/components/generated_form.dart';
|
||||||
|
import 'package:obtainium/components/generated_form_modal.dart';
|
||||||
import 'package:obtainium/custom_errors.dart';
|
import 'package:obtainium/custom_errors.dart';
|
||||||
import 'package:obtainium/providers/logs_provider.dart';
|
import 'package:obtainium/providers/logs_provider.dart';
|
||||||
import 'package:obtainium/providers/notifications_provider.dart';
|
import 'package:obtainium/providers/notifications_provider.dart';
|
||||||
import 'package:obtainium/providers/settings_provider.dart';
|
import 'package:obtainium/providers/settings_provider.dart';
|
||||||
import 'package:package_archive_info/package_archive_info.dart';
|
import 'package:package_archive_info/package_archive_info.dart';
|
||||||
|
import 'package:permission_handler/permission_handler.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:flutter_fgbg/flutter_fgbg.dart';
|
import 'package:flutter_fgbg/flutter_fgbg.dart';
|
||||||
import 'package:obtainium/providers/source_provider.dart';
|
import 'package:obtainium/providers/source_provider.dart';
|
||||||
import 'package:http/http.dart';
|
import 'package:http/http.dart';
|
||||||
|
import 'package:android_intent_plus/android_intent.dart';
|
||||||
|
|
||||||
class AppInMemory {
|
class AppInMemory {
|
||||||
late App app;
|
late App app;
|
||||||
@ -174,12 +179,9 @@ class AppsProvider with ChangeNotifier {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If the APK package ID is different from the App ID, it is either new (using a placeholder ID) or the ID has changed
|
// If the APK package ID is different from the App ID, it is either new (using a placeholder ID) or the ID has changed
|
||||||
// The former case should be handled (give the App its real ID), the latter is a security issue
|
// In either case, the app should be given the new ID
|
||||||
var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path);
|
var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path);
|
||||||
if (app.id != newInfo.packageName) {
|
if (app.id != newInfo.packageName) {
|
||||||
if (apps[app.id] != null && !SourceProvider().isTempId(app.id)) {
|
|
||||||
throw IDChangedError();
|
|
||||||
}
|
|
||||||
var originalAppId = app.id;
|
var originalAppId = app.id;
|
||||||
app.id = newInfo.packageName;
|
app.id = newInfo.packageName;
|
||||||
downloadedFile = downloadedFile.renameSync(
|
downloadedFile = downloadedFile.renameSync(
|
||||||
@ -246,9 +248,10 @@ class AppsProvider with ChangeNotifier {
|
|||||||
!(await canDowngradeApps())) {
|
!(await canDowngradeApps())) {
|
||||||
throw DowngradeError();
|
throw DowngradeError();
|
||||||
}
|
}
|
||||||
if (appInfo == null ||
|
await InstallPlugin.installApk(file.file.path, obtainiumId);
|
||||||
int.parse(newInfo.buildNumber) > appInfo.versionCode!) {
|
if (file.appId == obtainiumId) {
|
||||||
await InstallPlugin.installApk(file.file.path, 'dev.imranr.obtainium');
|
// Obtainium prompt should be lowest
|
||||||
|
await Future.delayed(const Duration(milliseconds: 500));
|
||||||
}
|
}
|
||||||
apps[file.appId]!.app.installedVersion =
|
apps[file.appId]!.app.installedVersion =
|
||||||
apps[file.appId]!.app.latestVersion;
|
apps[file.appId]!.app.latestVersion;
|
||||||
@ -257,6 +260,15 @@ class AppsProvider with ChangeNotifier {
|
|||||||
attemptToCorrectInstallStatus: false);
|
attemptToCorrectInstallStatus: false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void uninstallApp(String appId) async {
|
||||||
|
var intent = AndroidIntent(
|
||||||
|
action: 'android.intent.action.DELETE',
|
||||||
|
data: 'package:$appId',
|
||||||
|
flags: <int>[Flag.FLAG_ACTIVITY_NEW_TASK],
|
||||||
|
package: 'vnd.android.package-archive');
|
||||||
|
await intent.launch();
|
||||||
|
}
|
||||||
|
|
||||||
Future<String?> confirmApkUrl(App app, BuildContext? context) async {
|
Future<String?> confirmApkUrl(App app, BuildContext? context) async {
|
||||||
// If the App has more than one APK, the user should pick one (if context provided)
|
// If the App has more than one APK, the user should pick one (if context provided)
|
||||||
String? apkUrl = app.apkUrls[app.preferredApkIndex];
|
String? apkUrl = app.apkUrls[app.preferredApkIndex];
|
||||||
@ -264,6 +276,7 @@ class AppsProvider with ChangeNotifier {
|
|||||||
List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis;
|
List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis;
|
||||||
|
|
||||||
if (app.apkUrls.length > 1 && context != null) {
|
if (app.apkUrls.length > 1 && context != null) {
|
||||||
|
// ignore: use_build_context_synchronously
|
||||||
apkUrl = await showDialog(
|
apkUrl = await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext ctx) {
|
builder: (BuildContext ctx) {
|
||||||
@ -283,6 +296,7 @@ class AppsProvider with ChangeNotifier {
|
|||||||
if (apkUrl != null &&
|
if (apkUrl != null &&
|
||||||
getHost(apkUrl) != getHost(app.url) &&
|
getHost(apkUrl) != getHost(app.url) &&
|
||||||
context != null) {
|
context != null) {
|
||||||
|
// ignore: use_build_context_synchronously
|
||||||
if (await showDialog(
|
if (await showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext ctx) {
|
builder: (BuildContext ctx) {
|
||||||
@ -313,7 +327,8 @@ class AppsProvider with ChangeNotifier {
|
|||||||
throw ObtainiumError(tr('appNotFound'));
|
throw ObtainiumError(tr('appNotFound'));
|
||||||
}
|
}
|
||||||
String? apkUrl;
|
String? apkUrl;
|
||||||
if (!apps[id]!.app.trackOnly) {
|
var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true;
|
||||||
|
if (!trackOnly) {
|
||||||
apkUrl = await confirmApkUrl(apps[id]!.app, context);
|
apkUrl = await confirmApkUrl(apps[id]!.app, context);
|
||||||
}
|
}
|
||||||
if (apkUrl != null) {
|
if (apkUrl != null) {
|
||||||
@ -326,7 +341,7 @@ class AppsProvider with ChangeNotifier {
|
|||||||
appsToInstall.add(id);
|
appsToInstall.add(id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (apps[id]!.app.trackOnly) {
|
if (trackOnly) {
|
||||||
trackOnlyAppsToUpdate.add(id);
|
trackOnlyAppsToUpdate.add(id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -439,9 +454,6 @@ class AppsProvider with ChangeNotifier {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
//
|
//
|
||||||
}
|
}
|
||||||
if (!res) {
|
|
||||||
logs.add(tr('versionCorrectionDisabled'));
|
|
||||||
}
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -451,9 +463,10 @@ class AppsProvider with ChangeNotifier {
|
|||||||
// Don't save changes, just return the object if changes were made (else null)
|
// Don't save changes, just return the object if changes were made (else null)
|
||||||
App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) {
|
App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) {
|
||||||
var modded = false;
|
var modded = false;
|
||||||
if (installedInfo == null &&
|
var trackOnly = app.additionalSettings['trackOnly'] == true;
|
||||||
app.installedVersion != null &&
|
var noVersionDetection =
|
||||||
!app.trackOnly) {
|
app.additionalSettings['noVersionDetection'] == true;
|
||||||
|
if (installedInfo == null && app.installedVersion != null && !trackOnly) {
|
||||||
app.installedVersion = null;
|
app.installedVersion = null;
|
||||||
modded = true;
|
modded = true;
|
||||||
} else if (installedInfo?.versionName != null &&
|
} else if (installedInfo?.versionName != null &&
|
||||||
@ -461,7 +474,8 @@ class AppsProvider with ChangeNotifier {
|
|||||||
app.installedVersion = installedInfo!.versionName;
|
app.installedVersion = installedInfo!.versionName;
|
||||||
modded = true;
|
modded = true;
|
||||||
} else if (installedInfo?.versionName != null &&
|
} else if (installedInfo?.versionName != null &&
|
||||||
installedInfo!.versionName != app.installedVersion) {
|
installedInfo!.versionName != app.installedVersion &&
|
||||||
|
!noVersionDetection) {
|
||||||
String? correctedInstalledVersion = reconcileRealAndInternalVersions(
|
String? correctedInstalledVersion = reconcileRealAndInternalVersions(
|
||||||
installedInfo.versionName!, app.installedVersion!);
|
installedInfo.versionName!, app.installedVersion!);
|
||||||
if (correctedInstalledVersion != null) {
|
if (correctedInstalledVersion != null) {
|
||||||
@ -470,7 +484,8 @@ class AppsProvider with ChangeNotifier {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (app.installedVersion != null &&
|
if (app.installedVersion != null &&
|
||||||
app.installedVersion != app.latestVersion) {
|
app.installedVersion != app.latestVersion &&
|
||||||
|
!noVersionDetection) {
|
||||||
app.installedVersion = reconcileRealAndInternalVersions(
|
app.installedVersion = reconcileRealAndInternalVersions(
|
||||||
app.installedVersion!, app.latestVersion,
|
app.installedVersion!, app.latestVersion,
|
||||||
matchMode: true) ??
|
matchMode: true) ??
|
||||||
@ -617,18 +632,65 @@ class AppsProvider with ChangeNotifier {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<bool> removeAppsWithModal(BuildContext context, List<App> apps) async {
|
||||||
|
var showUninstallOption =
|
||||||
|
apps.where((a) => a.installedVersion != null).isNotEmpty;
|
||||||
|
var values = await showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (BuildContext ctx) {
|
||||||
|
return GeneratedFormModal(
|
||||||
|
title: plural('removeAppQuestion', apps.length),
|
||||||
|
items: !showUninstallOption
|
||||||
|
? []
|
||||||
|
: [
|
||||||
|
[
|
||||||
|
GeneratedFormSwitch('rmAppEntry',
|
||||||
|
label: tr('removeFromObtainium'), defaultValue: true)
|
||||||
|
],
|
||||||
|
[
|
||||||
|
GeneratedFormSwitch('uninstallApp',
|
||||||
|
label: tr('uninstallFromDevice'))
|
||||||
|
]
|
||||||
|
],
|
||||||
|
initValid: true,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
if (values != null) {
|
||||||
|
bool uninstall = values['uninstallApp'] == true && showUninstallOption;
|
||||||
|
bool remove = values['rmAppEntry'] == true || !showUninstallOption;
|
||||||
|
if (uninstall) {
|
||||||
|
for (var i = 0; i < apps.length; i++) {
|
||||||
|
if (apps[i].installedVersion != null) {
|
||||||
|
uninstallApp(apps[i].id);
|
||||||
|
apps[i].installedVersion = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await saveApps(apps, attemptToCorrectInstallStatus: false);
|
||||||
|
}
|
||||||
|
if (remove) {
|
||||||
|
await removeApps(apps.map((e) => e.id).toList());
|
||||||
|
}
|
||||||
|
return uninstall || remove;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> openAppSettings(String appId) async {
|
||||||
|
final AndroidIntent intent = AndroidIntent(
|
||||||
|
action: 'action_application_details_settings',
|
||||||
|
data: 'package:$appId',
|
||||||
|
);
|
||||||
|
await intent.launch();
|
||||||
|
}
|
||||||
|
|
||||||
Future<App?> checkUpdate(String appId) async {
|
Future<App?> checkUpdate(String appId) async {
|
||||||
App? currentApp = apps[appId]!.app;
|
App? currentApp = apps[appId]!.app;
|
||||||
SourceProvider sourceProvider = SourceProvider();
|
SourceProvider sourceProvider = SourceProvider();
|
||||||
App newApp = await sourceProvider.getApp(
|
App newApp = await sourceProvider.getApp(
|
||||||
sourceProvider.getSource(currentApp.url),
|
sourceProvider.getSource(currentApp.url),
|
||||||
currentApp.url,
|
currentApp.url,
|
||||||
currentApp.additionalData,
|
currentApp.additionalSettings,
|
||||||
name: currentApp.name,
|
currentApp: currentApp);
|
||||||
id: currentApp.id,
|
|
||||||
pinned: currentApp.pinned,
|
|
||||||
trackOnly: currentApp.trackOnly,
|
|
||||||
installedVersion: currentApp.installedVersion);
|
|
||||||
if (currentApp.preferredApkIndex < newApp.apkUrls.length) {
|
if (currentApp.preferredApkIndex < newApp.apkUrls.length) {
|
||||||
newApp.preferredApkIndex = currentApp.preferredApkIndex;
|
newApp.preferredApkIndex = currentApp.preferredApkIndex;
|
||||||
}
|
}
|
||||||
@ -706,6 +768,14 @@ class AppsProvider with ChangeNotifier {
|
|||||||
exportDir = await getExternalStorageDirectory();
|
exportDir = await getExternalStorageDirectory();
|
||||||
path = exportDir!.path;
|
path = exportDir!.path;
|
||||||
}
|
}
|
||||||
|
if ((await DeviceInfoPlugin().androidInfo).version.sdkInt <= 29) {
|
||||||
|
if (await Permission.storage.isDenied) {
|
||||||
|
await Permission.storage.request();
|
||||||
|
}
|
||||||
|
if (await Permission.storage.isDenied) {
|
||||||
|
throw ObtainiumError(tr('storagePermissionDenied'));
|
||||||
|
}
|
||||||
|
}
|
||||||
File export = File(
|
File export = File(
|
||||||
'${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json');
|
'${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json');
|
||||||
export.writeAsStringSync(
|
export.writeAsStringSync(
|
||||||
|
@ -80,11 +80,11 @@ class DownloadNotification extends ObtainiumNotification {
|
|||||||
DownloadNotification(String appName, int progPercent)
|
DownloadNotification(String appName, int progPercent)
|
||||||
: super(
|
: super(
|
||||||
appName.hashCode,
|
appName.hashCode,
|
||||||
'Downloading $appName',
|
tr('downloadingX', args: [appName]),
|
||||||
'',
|
'',
|
||||||
'APP_DOWNLOADING',
|
'APP_DOWNLOADING',
|
||||||
'Downloading App',
|
tr('downloadingX', args: [tr('app')]),
|
||||||
'Notifies the user of the progress in downloading an App',
|
tr('downloadNotifDescription'),
|
||||||
Importance.low,
|
Importance.low,
|
||||||
onlyAlertOnce: true,
|
onlyAlertOnce: true,
|
||||||
progPercent: progPercent);
|
progPercent: progPercent);
|
||||||
|
@ -1,9 +1,13 @@
|
|||||||
// Exposes functions used to save/load app settings
|
// Exposes functions used to save/load app settings
|
||||||
|
|
||||||
|
import 'dart:convert';
|
||||||
|
|
||||||
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: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/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';
|
||||||
|
|
||||||
@ -144,4 +148,35 @@ class SettingsProvider with ChangeNotifier {
|
|||||||
prefs?.setString(settingId, value);
|
prefs?.setString(settingId, value);
|
||||||
notifyListeners();
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Map<String, int> get categories =>
|
||||||
|
Map<String, int>.from(jsonDecode(prefs?.getString('categories') ?? '{}'));
|
||||||
|
|
||||||
|
set categories(Map<String, int> cats) {
|
||||||
|
prefs?.setString('categories', jsonEncode(cats));
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool setEqual(Set<String> a, Set<String> b) =>
|
||||||
|
a.length == b.length && a.union(b).length == a.length;
|
||||||
}
|
}
|
||||||
|
@ -7,11 +7,13 @@ import 'package:easy_localization/easy_localization.dart';
|
|||||||
import 'package:html/dom.dart';
|
import 'package:html/dom.dart';
|
||||||
import 'package:http/http.dart';
|
import 'package:http/http.dart';
|
||||||
import 'package:obtainium/app_sources/apkmirror.dart';
|
import 'package:obtainium/app_sources/apkmirror.dart';
|
||||||
|
import 'package:obtainium/app_sources/codeberg.dart';
|
||||||
import 'package:obtainium/app_sources/fdroid.dart';
|
import 'package:obtainium/app_sources/fdroid.dart';
|
||||||
import 'package:obtainium/app_sources/fdroidrepo.dart';
|
import 'package:obtainium/app_sources/fdroidrepo.dart';
|
||||||
import 'package:obtainium/app_sources/github.dart';
|
import 'package:obtainium/app_sources/github.dart';
|
||||||
import 'package:obtainium/app_sources/gitlab.dart';
|
import 'package:obtainium/app_sources/gitlab.dart';
|
||||||
import 'package:obtainium/app_sources/izzyondroid.dart';
|
import 'package:obtainium/app_sources/izzyondroid.dart';
|
||||||
|
import 'package:obtainium/app_sources/html.dart';
|
||||||
import 'package:obtainium/app_sources/mullvad.dart';
|
import 'package:obtainium/app_sources/mullvad.dart';
|
||||||
import 'package:obtainium/app_sources/signal.dart';
|
import 'package:obtainium/app_sources/signal.dart';
|
||||||
import 'package:obtainium/app_sources/sourceforge.dart';
|
import 'package:obtainium/app_sources/sourceforge.dart';
|
||||||
@ -44,10 +46,10 @@ class App {
|
|||||||
late String latestVersion;
|
late String latestVersion;
|
||||||
List<String> apkUrls = [];
|
List<String> apkUrls = [];
|
||||||
late int preferredApkIndex;
|
late int preferredApkIndex;
|
||||||
late List<String> additionalData;
|
late Map<String, dynamic> additionalSettings;
|
||||||
late DateTime? lastUpdateCheck;
|
late DateTime? lastUpdateCheck;
|
||||||
bool pinned = false;
|
bool pinned = false;
|
||||||
bool trackOnly = false;
|
List<String> categories;
|
||||||
App(
|
App(
|
||||||
this.id,
|
this.id,
|
||||||
this.url,
|
this.url,
|
||||||
@ -57,39 +59,83 @@ class App {
|
|||||||
this.latestVersion,
|
this.latestVersion,
|
||||||
this.apkUrls,
|
this.apkUrls,
|
||||||
this.preferredApkIndex,
|
this.preferredApkIndex,
|
||||||
this.additionalData,
|
this.additionalSettings,
|
||||||
this.lastUpdateCheck,
|
this.lastUpdateCheck,
|
||||||
this.pinned,
|
this.pinned,
|
||||||
this.trackOnly);
|
{this.categories = const []});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() {
|
String toString() {
|
||||||
return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALDATA: ${additionalData.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned';
|
return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALSETTINGS: ${additionalSettings.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned';
|
||||||
}
|
}
|
||||||
|
|
||||||
factory App.fromJson(Map<String, dynamic> json) => App(
|
factory App.fromJson(Map<String, dynamic> json) {
|
||||||
json['id'] as String,
|
var source = SourceProvider().getSource(json['url']);
|
||||||
json['url'] as String,
|
var formItems = source.combinedAppSpecificSettingFormItems
|
||||||
json['author'] as String,
|
.reduce((value, element) => [...value, ...element]);
|
||||||
json['name'] as String,
|
Map<String, dynamic> additionalSettings =
|
||||||
json['installedVersion'] == null
|
getDefaultValuesFromFormItems([formItems]);
|
||||||
? null
|
if (json['additionalSettings'] != null) {
|
||||||
: json['installedVersion'] as String,
|
additionalSettings.addEntries(
|
||||||
json['latestVersion'] as String,
|
Map<String, dynamic>.from(jsonDecode(json['additionalSettings']))
|
||||||
json['apkUrls'] == null
|
.entries);
|
||||||
? []
|
}
|
||||||
: List<String>.from(jsonDecode(json['apkUrls'])),
|
// If needed, migrate old-style additionalData to newer-style additionalSettings (V1)
|
||||||
json['preferredApkIndex'] == null ? 0 : json['preferredApkIndex'] as int,
|
if (json['additionalData'] != null) {
|
||||||
json['additionalData'] == null
|
List<String> temp = List<String>.from(jsonDecode(json['additionalData']));
|
||||||
? SourceProvider()
|
temp.asMap().forEach((i, value) {
|
||||||
.getSource(json['url'])
|
if (i < formItems.length) {
|
||||||
.additionalSourceAppSpecificDefaults
|
if (formItems[i] is GeneratedFormSwitch) {
|
||||||
: List<String>.from(jsonDecode(json['additionalData'])),
|
additionalSettings[formItems[i].key] = value == 'true';
|
||||||
json['lastUpdateCheck'] == null
|
} else {
|
||||||
? null
|
additionalSettings[formItems[i].key] = value;
|
||||||
: DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']),
|
}
|
||||||
json['pinned'] ?? false,
|
}
|
||||||
json['trackOnly'] ?? false);
|
});
|
||||||
|
additionalSettings['trackOnly'] =
|
||||||
|
json['trackOnly'] == 'true' || json['trackOnly'] == true;
|
||||||
|
additionalSettings['noVersionDetection'] =
|
||||||
|
json['noVersionDetection'] == 'true' || json['trackOnly'] == true;
|
||||||
|
}
|
||||||
|
// Ensure additionalSettings are correctly typed
|
||||||
|
for (var item in formItems) {
|
||||||
|
if (additionalSettings[item.key] != null) {
|
||||||
|
additionalSettings[item.key] =
|
||||||
|
item.ensureType(additionalSettings[item.key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
int preferredApkIndex = json['preferredApkIndex'] == null
|
||||||
|
? 0
|
||||||
|
: json['preferredApkIndex'] as int;
|
||||||
|
if (preferredApkIndex < 0) {
|
||||||
|
preferredApkIndex = 0;
|
||||||
|
}
|
||||||
|
return App(
|
||||||
|
json['id'] as String,
|
||||||
|
json['url'] as String,
|
||||||
|
json['author'] as String,
|
||||||
|
json['name'] as String,
|
||||||
|
json['installedVersion'] == null
|
||||||
|
? null
|
||||||
|
: json['installedVersion'] as String,
|
||||||
|
json['latestVersion'] as String,
|
||||||
|
json['apkUrls'] == null
|
||||||
|
? []
|
||||||
|
: List<String>.from(jsonDecode(json['apkUrls'])),
|
||||||
|
preferredApkIndex,
|
||||||
|
additionalSettings,
|
||||||
|
json['lastUpdateCheck'] == null
|
||||||
|
? null
|
||||||
|
: DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']),
|
||||||
|
json['pinned'] ?? false,
|
||||||
|
categories: json['categories'] != null
|
||||||
|
? (json['categories'] as List<dynamic>)
|
||||||
|
.map((e) => e.toString())
|
||||||
|
.toList()
|
||||||
|
: json['category'] != null
|
||||||
|
? [json['category'] as String]
|
||||||
|
: []);
|
||||||
|
}
|
||||||
|
|
||||||
Map<String, dynamic> toJson() => {
|
Map<String, dynamic> toJson() => {
|
||||||
'id': id,
|
'id': id,
|
||||||
@ -100,15 +146,19 @@ class App {
|
|||||||
'latestVersion': latestVersion,
|
'latestVersion': latestVersion,
|
||||||
'apkUrls': jsonEncode(apkUrls),
|
'apkUrls': jsonEncode(apkUrls),
|
||||||
'preferredApkIndex': preferredApkIndex,
|
'preferredApkIndex': preferredApkIndex,
|
||||||
'additionalData': jsonEncode(additionalData),
|
'additionalSettings': jsonEncode(additionalSettings),
|
||||||
'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch,
|
'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch,
|
||||||
'pinned': pinned,
|
'pinned': pinned,
|
||||||
'trackOnly': trackOnly
|
'categories': categories
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure the input is starts with HTTPS and has no WWW
|
// Ensure the input is starts with HTTPS and has no WWW
|
||||||
preStandardizeUrl(String url) {
|
preStandardizeUrl(String url) {
|
||||||
|
var firstDotIndex = url.indexOf('.');
|
||||||
|
if (!(firstDotIndex >= 0 && firstDotIndex != url.length - 1)) {
|
||||||
|
throw UnsupportedURLError();
|
||||||
|
}
|
||||||
if (url.toLowerCase().indexOf('http://') != 0 &&
|
if (url.toLowerCase().indexOf('http://') != 0 &&
|
||||||
url.toLowerCase().indexOf('https://') != 0) {
|
url.toLowerCase().indexOf('https://') != 0) {
|
||||||
url = 'https://$url';
|
url = 'https://$url';
|
||||||
@ -124,7 +174,7 @@ preStandardizeUrl(String url) {
|
|||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
const String noAPKFound = 'No APK found';
|
String noAPKFound = tr('noAPKFound');
|
||||||
|
|
||||||
List<String> getLinksFromParsedHTML(
|
List<String> getLinksFromParsedHTML(
|
||||||
Document dom, RegExp hrefPattern, String prependToLinks) =>
|
Document dom, RegExp hrefPattern, String prependToLinks) =>
|
||||||
@ -137,6 +187,13 @@ List<String> getLinksFromParsedHTML(
|
|||||||
.map((e) => '$prependToLinks${e.attributes['href']!}')
|
.map((e) => '$prependToLinks${e.attributes['href']!}')
|
||||||
.toList();
|
.toList();
|
||||||
|
|
||||||
|
Map<String, dynamic> getDefaultValuesFromFormItems(
|
||||||
|
List<List<GeneratedFormItem>> items) {
|
||||||
|
return Map.fromEntries(items
|
||||||
|
.map((row) => row.map((el) => MapEntry(el.key, el.defaultValue ?? '')))
|
||||||
|
.reduce((value, element) => [...value, ...element]));
|
||||||
|
}
|
||||||
|
|
||||||
class AppSource {
|
class AppSource {
|
||||||
String? host;
|
String? host;
|
||||||
late String name;
|
late String name;
|
||||||
@ -151,23 +208,45 @@ class AppSource {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<APKDetails> getLatestAPKDetails(
|
Future<APKDetails> getLatestAPKDetails(
|
||||||
String standardUrl, List<String> additionalData,
|
String standardUrl, Map<String, dynamic> additionalSettings) {
|
||||||
{bool trackOnly = false}) {
|
|
||||||
throw NotImplementedError();
|
throw NotImplementedError();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Different Sources may need different kinds of additional data for Apps
|
// Different Sources may need different kinds of additional data for Apps
|
||||||
List<List<GeneratedFormItem>> additionalSourceAppSpecificFormItems = [];
|
List<List<GeneratedFormItem>> additionalSourceAppSpecificSettingFormItems =
|
||||||
List<String> additionalSourceAppSpecificDefaults = [];
|
[];
|
||||||
|
|
||||||
// Some additional data may be needed for Apps regardless of Source
|
// Some additional data may be needed for Apps regardless of Source
|
||||||
final List<GeneratedFormItem> additionalAppSpecificSourceAgnosticFormItems = [
|
final List<List<GeneratedFormItem>>
|
||||||
GeneratedFormItem(
|
additionalAppSpecificSourceAgnosticSettingFormItems = [
|
||||||
|
[
|
||||||
|
GeneratedFormSwitch(
|
||||||
|
'trackOnly',
|
||||||
label: tr('trackOnly'),
|
label: tr('trackOnly'),
|
||||||
type: FormItemType.bool,
|
)
|
||||||
key: 'trackOnlyFormItemKey')
|
],
|
||||||
|
[
|
||||||
|
GeneratedFormSwitch('noVersionDetection', label: tr('noVersionDetection'))
|
||||||
|
],
|
||||||
|
[
|
||||||
|
GeneratedFormTextField('apkFilterRegEx',
|
||||||
|
label: tr('filterAPKsByRegEx'),
|
||||||
|
required: false,
|
||||||
|
additionalValidators: [
|
||||||
|
(value) {
|
||||||
|
return regExValidator(value);
|
||||||
|
}
|
||||||
|
])
|
||||||
|
]
|
||||||
];
|
];
|
||||||
final List<String> additionalAppSpecificSourceAgnosticDefaults = [''];
|
|
||||||
|
// Previous 2 variables combined into one at runtime for convenient usage
|
||||||
|
List<List<GeneratedFormItem>> get combinedAppSpecificSettingFormItems {
|
||||||
|
return [
|
||||||
|
...additionalSourceAppSpecificSettingFormItems,
|
||||||
|
...additionalAppSpecificSourceAgnosticSettingFormItems
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
// Some Sources may have additional settings at the Source level (not specific to Apps) - these use SettingsProvider
|
// Some Sources may have additional settings at the Source level (not specific to Apps) - these use SettingsProvider
|
||||||
List<GeneratedFormItem> additionalSourceSpecificSettingFormItems = [];
|
List<GeneratedFormItem> additionalSourceSpecificSettingFormItems = [];
|
||||||
@ -186,7 +265,7 @@ class AppSource {
|
|||||||
}
|
}
|
||||||
|
|
||||||
String? tryInferringAppId(String standardUrl,
|
String? tryInferringAppId(String standardUrl,
|
||||||
{List<String> additionalData = const []}) {
|
{Map<String, dynamic> additionalSettings = const {}}) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -202,11 +281,24 @@ abstract class MassAppUrlSource {
|
|||||||
Future<Map<String, String>> getUrlsWithDescriptions(List<String> args);
|
Future<Map<String, String>> getUrlsWithDescriptions(List<String> args);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
regExValidator(String? value) {
|
||||||
|
if (value == null || value.isEmpty) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
RegExp(value);
|
||||||
|
} catch (e) {
|
||||||
|
return tr('invalidRegEx');
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
class SourceProvider {
|
class SourceProvider {
|
||||||
// Add more source classes here so they are available via the service
|
// Add more source classes here so they are available via the service
|
||||||
List<AppSource> sources = [
|
List<AppSource> sources = [
|
||||||
GitHub(),
|
GitHub(),
|
||||||
GitLab(),
|
GitLab(),
|
||||||
|
Codeberg(),
|
||||||
FDroid(),
|
FDroid(),
|
||||||
IzzyOnDroid(),
|
IzzyOnDroid(),
|
||||||
Mullvad(),
|
Mullvad(),
|
||||||
@ -214,7 +306,8 @@ class SourceProvider {
|
|||||||
SourceForge(),
|
SourceForge(),
|
||||||
APKMirror(),
|
APKMirror(),
|
||||||
FDroidRepo(),
|
FDroidRepo(),
|
||||||
SteamMobile()
|
SteamMobile(),
|
||||||
|
HTML() // This should ALWAYS be the last option as they are tried in order
|
||||||
];
|
];
|
||||||
|
|
||||||
// Add more mass url source classes here so they are available via the service
|
// Add more mass url source classes here so they are available via the service
|
||||||
@ -246,10 +339,10 @@ class SourceProvider {
|
|||||||
return source;
|
return source;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ifSourceAppsRequireAdditionalData(AppSource source) {
|
bool ifRequiredAppSpecificSettingsExist(AppSource source) {
|
||||||
for (var row in source.additionalSourceAppSpecificFormItems) {
|
for (var row in source.combinedAppSpecificSettingFormItems) {
|
||||||
for (var element in row) {
|
for (var element in row) {
|
||||||
if (element.required && element.opts == null) {
|
if (element is GeneratedFormTextField && element.required) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -274,37 +367,53 @@ class SourceProvider {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<App> getApp(AppSource source, String url, List<String> additionalData,
|
Future<App> getApp(
|
||||||
{String name = '',
|
AppSource source,
|
||||||
String? id,
|
String url,
|
||||||
bool pinned = false,
|
Map<String, dynamic> additionalSettings, {
|
||||||
bool trackOnly = false,
|
App? currentApp,
|
||||||
String? installedVersion}) async {
|
bool trackOnlyOverride = false,
|
||||||
|
noVersionDetectionOverride = false,
|
||||||
|
}) async {
|
||||||
|
if (trackOnlyOverride || source.enforceTrackOnly) {
|
||||||
|
additionalSettings['trackOnly'] = true;
|
||||||
|
}
|
||||||
|
if (noVersionDetectionOverride) {
|
||||||
|
additionalSettings['noVersionDetection'] = true;
|
||||||
|
}
|
||||||
|
var trackOnly = additionalSettings['trackOnly'] == true;
|
||||||
String standardUrl = source.standardizeURL(preStandardizeUrl(url));
|
String standardUrl = source.standardizeURL(preStandardizeUrl(url));
|
||||||
APKDetails apk = await source
|
APKDetails apk =
|
||||||
.getLatestAPKDetails(standardUrl, additionalData, trackOnly: trackOnly);
|
await source.getLatestAPKDetails(standardUrl, additionalSettings);
|
||||||
|
if (additionalSettings['apkFilterRegEx'] != null) {
|
||||||
|
var reg = RegExp(additionalSettings['apkFilterRegEx']);
|
||||||
|
apk.apkUrls =
|
||||||
|
apk.apkUrls.where((element) => reg.hasMatch(element)).toList();
|
||||||
|
}
|
||||||
if (apk.apkUrls.isEmpty && !trackOnly) {
|
if (apk.apkUrls.isEmpty && !trackOnly) {
|
||||||
throw NoAPKError();
|
throw NoAPKError();
|
||||||
}
|
}
|
||||||
String apkVersion = apk.version.replaceAll('/', '-');
|
String apkVersion = apk.version.replaceAll('/', '-');
|
||||||
|
var name = currentApp?.name.trim() ??
|
||||||
|
apk.names.name[0].toUpperCase() + apk.names.name.substring(1);
|
||||||
return App(
|
return App(
|
||||||
id ??
|
currentApp?.id ??
|
||||||
source.tryInferringAppId(standardUrl,
|
source.tryInferringAppId(standardUrl,
|
||||||
additionalData: additionalData) ??
|
additionalSettings: additionalSettings) ??
|
||||||
generateTempID(apk.names, source),
|
generateTempID(apk.names, source),
|
||||||
standardUrl,
|
standardUrl,
|
||||||
apk.names.author[0].toUpperCase() + apk.names.author.substring(1),
|
apk.names.author[0].toUpperCase() + apk.names.author.substring(1),
|
||||||
name.trim().isNotEmpty
|
name.trim().isNotEmpty
|
||||||
? name
|
? name
|
||||||
: apk.names.name[0].toUpperCase() + apk.names.name.substring(1),
|
: apk.names.name[0].toUpperCase() + apk.names.name.substring(1),
|
||||||
installedVersion,
|
currentApp?.installedVersion,
|
||||||
apkVersion,
|
apkVersion,
|
||||||
apk.apkUrls,
|
apk.apkUrls,
|
||||||
apk.apkUrls.length - 1,
|
apk.apkUrls.length - 1 >= 0 ? apk.apkUrls.length - 1 : 0,
|
||||||
additionalData,
|
additionalSettings,
|
||||||
DateTime.now(),
|
DateTime.now(),
|
||||||
pinned,
|
currentApp?.pinned ?? false,
|
||||||
trackOnly);
|
categories: currentApp?.categories ?? const []);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns errors in [results, errors] instead of throwing them
|
// Returns errors in [results, errors] instead of throwing them
|
||||||
@ -316,7 +425,10 @@ class SourceProvider {
|
|||||||
try {
|
try {
|
||||||
var source = getSource(url);
|
var source = getSource(url);
|
||||||
apps.add(await getApp(
|
apps.add(await getApp(
|
||||||
source, url, source.additionalSourceAppSpecificDefaults));
|
source,
|
||||||
|
url,
|
||||||
|
getDefaultValuesFromFormItems(
|
||||||
|
source.combinedAppSpecificSettingFormItems)));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
errors.addAll(<String, dynamic>{url: e});
|
errors.addAll(<String, dynamic>{url: e});
|
||||||
}
|
}
|
||||||
|
453
pubspec.lock
@ -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.8.18+82 # When changing this, update the tag in main() accordingly
|
version: 0.10.8+114 # When changing this, update the tag in main() accordingly
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
sdk: '>=2.18.2 <3.0.0'
|
sdk: '>=2.18.2 <3.0.0'
|
||||||
@ -38,10 +38,10 @@ dependencies:
|
|||||||
cupertino_icons: ^1.0.5
|
cupertino_icons: ^1.0.5
|
||||||
path_provider: ^2.0.11
|
path_provider: ^2.0.11
|
||||||
flutter_fgbg: ^0.2.0 # Try removing reliance on this
|
flutter_fgbg: ^0.2.0 # Try removing reliance on this
|
||||||
flutter_local_notifications: ^12.0.0
|
flutter_local_notifications: ^13.0.0
|
||||||
provider: ^6.0.3
|
provider: ^6.0.3
|
||||||
http: ^0.13.5
|
http: ^0.13.5
|
||||||
webview_flutter: ^3.0.4
|
webview_flutter: ^4.0.0
|
||||||
dynamic_color: ^1.5.4
|
dynamic_color: ^1.5.4
|
||||||
html: ^0.15.0
|
html: ^0.15.0
|
||||||
shared_preferences: ^2.0.15
|
shared_preferences: ^2.0.15
|
||||||
@ -58,6 +58,7 @@ dependencies:
|
|||||||
android_alarm_manager_plus: ^2.1.0
|
android_alarm_manager_plus: ^2.1.0
|
||||||
sqflite: ^2.2.0+3
|
sqflite: ^2.2.0+3
|
||||||
easy_localization: ^3.0.1
|
easy_localization: ^3.0.1
|
||||||
|
android_intent_plus: ^3.1.5
|
||||||
|
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
|