mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-22 18:33:45 +02:00 
			
		
		
		
	Compare commits
	
		
			98 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 18ccd01186 | ||
|  | c84f51b6ce | ||
|  | 48958748f6 | ||
|  | 966b1ee6e3 | ||
|  | 7432ee867f | ||
|  | 3ebbd2a4a5 | ||
|  | 82207a1b3b | ||
|  | 2ec5c7db11 | ||
|  | ab949af700 | ||
|  | 5c9a35c4f0 | ||
|  | 8651f58744 | ||
|  | d93798a8df | ||
|  | 0822f991ff | ||
|  | fde63a0f05 | ||
|  | 523e1151b2 | ||
|  | d5d6825ed9 | ||
|  | 79c4d3b9fe | ||
|  | d6b99b903f | ||
|  | 335752ee7c | ||
|  | e5afe75213 | ||
|  | 0df1c1c512 | ||
|  | bd2ac73b9f | ||
|  | 26b5a2d976 | ||
|  | c0968e8991 | ||
|  | 243b1eb2ac | ||
|  | c6c61d1c83 | ||
|  | f207967411 | ||
|  | 29bb5cc8cd | ||
|  | dbbc2ff19c | ||
|  | c012b016f9 | ||
|  | 22815b8209 | ||
|  | 619d4f1a51 | ||
|  | 50e8929763 | ||
|  | 0e2a0b65ec | ||
|  | 5b79f399d1 | ||
|  | 2961d5ac17 | ||
|  | 4af4160aaa | ||
|  | 327f73cc9e | ||
|  | e82170fec6 | ||
|  | 8922b1c048 | ||
|  | e9550c6ff0 | ||
|  | 890c3682c4 | ||
|  | a2c38968e1 | ||
|  | a9c3ee4c54 | ||
|  | 36db226024 | ||
|  | 6fbdf62afa | ||
|  | 9344ebbb06 | ||
|  | 51b66ab983 | ||
|  | 7f2a9b6fa5 | ||
|  | ac6bc4d786 | ||
|  | acd30b516c | ||
|  | 5bc9234101 | ||
|  | 80d15f5232 | ||
|  | eb7c2a9fc4 | ||
|  | 40175468b2 | ||
|  | 814b5a71bd | ||
|  | 38a1e43116 | ||
|  | 5064b78c79 | ||
|  | 767350e4e3 | ||
|  | 4caae51904 | ||
|  | 83616b7a79 | ||
|  | 52b2d0868c | ||
|  | 94f629859e | ||
|  | 773d3455ae | ||
|  | 919ab16528 | ||
|  | 268b89eaf6 | ||
|  | 9c9a264e39 | ||
|  | e1e612455a | ||
|  | e191d75300 | ||
|  | fce2a2f15c | ||
|  | dd8acd9451 | ||
|  | 2d5676f13d | ||
|  | dfbf9e925c | ||
|  | e44f77a68a | ||
|  | cc373c8d7f | ||
|  | 2ad88bf3ca | ||
|  | b1f9375bb1 | ||
|  | 0e14e17236 | ||
|  | fc541837ef | ||
|  | da0b1d0684 | ||
|  | 3fd89e8567 | ||
|  | 8cab348eed | ||
|  | 6bd821985f | ||
|  | 1f2efe435f | ||
|  | 8276a809a5 | ||
|  | 5fe1a8a370 | ||
|  | 58710093ba | ||
|  | ac6f8c456d | ||
|  | 905461c242 | ||
|  | e3fcf6e0b5 | ||
|  | 903fad5158 | ||
|  | 1de93e827a | ||
|  | a5040aa0c1 | ||
|  | 9f7c4e23d5 | ||
|  | 876f67aacb | ||
|  | 9d5ce75e27 | ||
|  | c1c06b3f9c | ||
|  | 5e0333c4c8 | 
							
								
								
									
										2
									
								
								.flutter
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								.flutter
									
									
									
									
									
								
							 Submodule .flutter updated: 2663184aa7...dec2ee5c1f
									
								
							
							
								
								
									
										39
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,9 +6,6 @@ on: | ||||
|       beta: | ||||
|         type: boolean | ||||
|         description: Is beta? | ||||
|       draft: | ||||
|         type: boolean | ||||
|         description: Is draft? | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
| @@ -18,6 +15,9 @@ jobs: | ||||
|        | ||||
|       - uses: actions/checkout@v3 | ||||
|       - uses: subosito/flutter-action@v2 | ||||
|         with: | ||||
|           channel: stable | ||||
|           flutter-version: 3.24.5 | ||||
|       - uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           distribution: 'temurin' # See 'Supported distributions' for available options | ||||
| @@ -28,13 +28,6 @@ jobs: | ||||
|         run: | | ||||
|           flutter doctor -v | ||||
|  | ||||
|       - name: Import GPG key | ||||
|         id: import_pgp_key | ||||
|         uses: crazy-max/ghaction-import-gpg@v6 | ||||
|         with: | ||||
|           gpg_private_key: ${{ secrets.PGP_KEY_BASE64 }} | ||||
|           passphrase: ${{ secrets.PGP_PASSPHRASE }} | ||||
|  | ||||
|       - name: Check submodule | ||||
|         id: check_submodule | ||||
|         run: | | ||||
| @@ -57,24 +50,13 @@ jobs: | ||||
|           for file in build/app/outputs/flutter-apk/app-*normal*.apk*; do mv "$file" "${file//-normal/}"; done | ||||
|           flutter build apk --flavor fdroid -t lib/main_fdroid.dart && flutter build apk --split-per-abi --flavor fdroid -t lib/main_fdroid.dart | ||||
|           rm ./build/app/outputs/flutter-apk/*.sha1 | ||||
|           cp ./sign.sh ./build/app/outputs/flutter-apk/ | ||||
|           ls -l ./build/app/outputs/flutter-apk/ | ||||
|  | ||||
|       - name: Sign APKs | ||||
|         env: | ||||
|           KEYSTORE_BASE64: ${{ secrets.KEYSTORE_BASE64 }} | ||||
|           KEYSTORE_PASSWORD: ${{ secrets.KEYSTORE_PASSWORD }} | ||||
|           PGP_PASSPHRASE: ${{ secrets.PGP_PASSPHRASE }} | ||||
|         run: | | ||||
|           echo "${KEYSTORE_BASE64}" | base64 -d > apksign.keystore | ||||
|           for apk in ./build/app/outputs/flutter-apk/*-release*.apk; do | ||||
|             unsignedFn=${apk/-release/-unsigned} | ||||
|             mv "$apk" "$unsignedFn" | ||||
|             ${ANDROID_HOME}/build-tools/$(ls ${ANDROID_HOME}/build-tools/ | tail -1)/apksigner sign --ks apksign.keystore --ks-pass pass:"${KEYSTORE_PASSWORD}" --out "${apk}" "${unsignedFn}" | ||||
|             sha256sum ${apk} | cut -d " " -f 1 > "$apk".sha256 | ||||
|             gpg --batch  --pinentry-mode loopback --passphrase "${PGP_PASSPHRASE}" --sign --detach-sig "$apk".sha256 | ||||
|           done | ||||
|           rm apksign.keystore | ||||
|           PGP_KEY_FINGERPRINT="${{ steps.import_pgp_key.outputs.fingerprint }}" | ||||
|       - name: Save Unsigned APKs as Action Artifacts | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           path: build/app/outputs/flutter-apk/* | ||||
|  | ||||
|       - name: Create Tag | ||||
|         uses: mathieudutour/github-tag-action@v6.1 | ||||
| @@ -83,12 +65,11 @@ jobs: | ||||
|           custom_tag: "${{ steps.extract_version.outputs.tag }}" | ||||
|           tag_prefix: "" | ||||
|        | ||||
|       - name: Create Release And Upload APKs | ||||
|       - name: Create Draft Release | ||||
|         uses: ncipollo/release-action@v1 | ||||
|         with: | ||||
|           token: ${{ secrets.GH_ACCESS_TOKEN }} | ||||
|           tag: "${{ steps.extract_version.outputs.tag }}" | ||||
|           prerelease: "${{ steps.extract_version.outputs.beta }}" | ||||
|           draft: "${{ inputs.draft }}" | ||||
|           artifacts: ./build/app/outputs/flutter-apk/*-release*.apk* | ||||
|           draft: "true" | ||||
|           generateReleaseNotes: true | ||||
|   | ||||
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @@ -7,10 +7,12 @@ Get Android app updates straight from the source. | ||||
| Obtainium allows you to install and update apps directly from their releases pages, and receive notifications when new releases are made available. | ||||
|  | ||||
| More info: | ||||
| - [Obtainium/wiki](https://github.com/ImranR98/Obtainium/wiki) | ||||
| - [Obtainium Wiki](https://wiki.obtainium.imranr.dev/) ([repository](https://github.com/ImranR98/Obtainium-Wiki)) | ||||
| - [AppVerifier](https://github.com/soupslurpr/AppVerifier) - App verification tool (recommended, integrates with Obtainium) | ||||
| - [apps.obtainium.imranr.dev](https://apps.obtainium.imranr.dev/) - Crowdsourced app configurations | ||||
| - [apps.obtainium.imranr.dev](https://apps.obtainium.imranr.dev/) - Crowdsourced app configurations ([repository](https://github.com/ImranR98/apps.obtainium.imranr.dev)) | ||||
| - [Side Of Burritos - You should use this instead of F-Droid | How to use app RSS feed](https://youtu.be/FFz57zNR_M0) - Original motivation for this app | ||||
| - [Website](https://obtainium.imranr.dev) ([repository](https://github.com/ImranR98/obtainium.imranr.dev)) | ||||
| - [Source code](https://github.com/ImranR98/Obtainium) | ||||
|  | ||||
| Currently supported App sources: | ||||
| - Open Source - General: | ||||
| @@ -29,11 +31,7 @@ Currently supported App sources: | ||||
|   - [Tencent App Store](https://sj.qq.com/) | ||||
|   - Jenkins Jobs | ||||
|   - [APKMirror](https://apkmirror.com/) (Track-Only) | ||||
| - Open Source - App-Specific: | ||||
|   - [Signal](https://signal.org/) | ||||
|   - [VLC](https://videolan.org/) | ||||
| - Other - App-Specific: | ||||
|   - [WhatsApp](https://whatsapp.com) | ||||
|   - [Telegram App](https://telegram.org) | ||||
|   - [Neutron Code](https://neutroncode.com) | ||||
| - Direct APK Link | ||||
| @@ -59,7 +57,11 @@ Or, contribute some configurations to the website by creating a PR at [this repo | ||||
|     alt="Get it on F-Droid" | ||||
|     height="80">](https://f-droid.org/packages/dev.imranr.obtainium.fdroid/) | ||||
|       | ||||
| [PGP Public Key](https://keyserver.ubuntu.com/pks/lookup?search=contact%40imranr.dev&fingerprint=on&op=index) | ||||
| Verification info: | ||||
| - Package ID: `dev.imranr.obtainium` | ||||
| - SHA-256 Hash of Signing Certificate: `B3:53:60:1F:6A:1D:5F:D6:60:3A:E2:F5:0B:E8:0C:F3:01:36:7B:86:B6:AB:8B:1F:66:24:3D:A9:6C:D5:73:62` | ||||
|   - Note: The above signature is also valid for the F-Droid flavour of Obtainium, thanks to [reproducible builds](https://f-droid.org/docs/Reproducible_Builds/). | ||||
| - [PGP Public Key](https://keyserver.ubuntu.com/pks/lookup?search=contact%40imranr.dev&fingerprint=on&op=index) (to verify APK hashes) | ||||
|  | ||||
| ## Limitations | ||||
| - For some sources, data is gathered using Web scraping and can easily break due to changes in website design. In such cases, more reliable methods may be unavailable. | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Nije pronađena aplikacija s tim ID-om ili imenom", | ||||
|     "reposHaveMultipleApps": "Repo-i mogu sadržavati više aplikacija", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Repo treće strane", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Razgovor na Steamu (chat)", | ||||
|     "install": "Instaliraj", | ||||
|     "markInstalled": "Označi kao instalirano", | ||||
|     "update": "Nadogradi", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Želite li ukloniti aplikaciju?", | ||||
|         "other": "Želite li ukloniti aplikacije?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Žádná aplikace s tímto ID nebo názvem nebyla nalezena", | ||||
|     "reposHaveMultipleApps": "Repozitáře mohou obsahovat více aplikací", | ||||
|     "fdroidThirdPartyRepo": "F-Droid repozitář třetí strany", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Nainstalujte", | ||||
|     "markInstalled": "Označit jako nainstalovaný", | ||||
|     "update": "Aktualizovat", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Konfigurace aplikací s využitím crowdsourcingu", | ||||
|     "allowInsecure": "Povolení nezabezpečených požadavků HTTP", | ||||
|     "stayOneVersionBehind": "Zůstaňte o jednu verzi pozadu za nejnovější", | ||||
|     "refreshBeforeDownload": "Obnovení údajů o aplikaci před stažením", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Odstranit Apku?", | ||||
|         "other": "Odstranit Apky?" | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
|     "invalidURLForSource": "Ikke et gyldigt {} App-URL", | ||||
|     "invalidURLForSource": "Ikke et gyldigt {} app-URL", | ||||
|     "noReleaseFound": "Kunne ikke finde en passende udgivelse", | ||||
|     "noVersionFound": "Kunne ikke afgøre udgivelsesversion", | ||||
|     "urlMatchesNoSource": "URL'en matcher ikke en kendt kilde", | ||||
| @@ -28,7 +28,7 @@ | ||||
|     "githubStarredRepos": "Stjernemarkeret GitHub-repos", | ||||
|     "uname": "Brugernavn", | ||||
|     "wrongArgNum": "Forkert antal argumenter angivet", | ||||
|     "xIsTrackOnly": "{} kan kun følges", | ||||
|     "xIsTrackOnly": "{} er 'Følg Kun'", | ||||
|     "source": "Kilde", | ||||
|     "app": "App", | ||||
|     "appsFromSourceAreTrackOnly": "Apps fra denne kilde er 'Følg Kun'.", | ||||
| @@ -36,14 +36,14 @@ | ||||
|     "trackOnlyAppDescription": "Appen tjekkes for opdateringer, men Obtainium kan ikke hente eller installere den.", | ||||
|     "cancelled": "Annulleret", | ||||
|     "appAlreadyAdded": "Appen er allerede tilføjet", | ||||
|     "alreadyUpToDateQuestion": "Appen er allerede opdateret?", | ||||
|     "alreadyUpToDateQuestion": "Er appen allerede opdateret?", | ||||
|     "addApp": "Tilføj app", | ||||
|     "appSourceURL": "URL til app-kilde", | ||||
|     "error": "Fejl", | ||||
|     "add": "Tilføj", | ||||
|     "searchSomeSourcesLabel": "Søg (kun visse kilder)", | ||||
|     "search": "Søg", | ||||
|     "additionalOptsFor": "Yderligere indstillinger for {}", | ||||
|     "additionalOptsFor": "Flere indstillinger for {}", | ||||
|     "supportedSources": "Understøttede kilder", | ||||
|     "trackOnlyInBrackets": "(Følg Kun)", | ||||
|     "searchableInBrackets": "(Kan Søges)", | ||||
| @@ -51,7 +51,7 @@ | ||||
|     "noApps": "Ingen apps", | ||||
|     "noAppsForFilter": "Ingen apps til filter", | ||||
|     "byX": "Af {}", | ||||
|     "percentProgress": "Fremskridt: {}%", | ||||
|     "percentProgress": "Hentning: {}%", | ||||
|     "pleaseWait": "Vent venligst", | ||||
|     "updateAvailable": "Opdatering tilgængelig", | ||||
|     "notInstalled": "Ikke installeret", | ||||
| @@ -74,14 +74,14 @@ | ||||
|     "pinToTop": "Fastgør til toppen", | ||||
|     "unpinFromTop": "Frigør fra toppen", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Nulstil installationsstatus for valgte apps?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Installationsstatus for alle valgte apps nulstilles.\n\nDette kan hjælpe, når den app-version, der vises i Obtainium, er forkert grundet mislykkede opdateringer eller andre problemer.", | ||||
|     "installStatusOfXWillBeResetExplanation": "Installationsstatus for alle valgte apps nulstilles.\n\nDette kan hjælpe, når en forkert app-version vises i Obtainium grundet mislykkede opdateringer eller andre problemer.", | ||||
|     "customLinkMessage": "Disse links virker på enheder med Obtainium installeret", | ||||
|     "shareAppConfigLinks": "Del app-konfiguration som HTML-link", | ||||
|     "shareSelectedAppURLs": "Del valgte app-URL'er", | ||||
|     "resetInstallStatus": "Nulstil installationsstatus", | ||||
|     "more": "Mere", | ||||
|     "removeOutdatedFilter": "Fjern forældet app-filter", | ||||
|     "showOutdatedOnly": "Vis kun forældet apps", | ||||
|     "showOutdatedOnly": "Vis kun forældede apps", | ||||
|     "filter": "Filtrer", | ||||
|     "filterApps": "Filtrer Apps", | ||||
|     "appName": "Appnavn", | ||||
| @@ -114,7 +114,7 @@ | ||||
|     "light": "Lys", | ||||
|     "followSystem": "Følg system", | ||||
|     "followSystemThemeExplanation": "Det er kun muligt at følge systemtemaet ved brug af tredjepartsapplikationer", | ||||
|     "useBlackTheme": "Brug rent sort, mørkt tema", | ||||
|     "useBlackTheme": "Brug rent sort mørkt tema", | ||||
|     "appSortBy": "Sortér apps efter:", | ||||
|     "authorName": "Udvikler/Navn", | ||||
|     "nameAuthor": "Navn/Udvikler", | ||||
| @@ -125,8 +125,8 @@ | ||||
|     "bgUpdateCheckInterval": "Kontrolinterval for baggrundsopdatering", | ||||
|     "neverManualOnly": "Aldrig - Kun manuelt", | ||||
|     "appearance": "Udseende", | ||||
|     "showWebInAppView": "Vis kildewebsiden i appvisning", | ||||
|     "pinUpdates": "Fastgør opdateringer til toppen af appvisning", | ||||
|     "showWebInAppView": "Vis kildewebsiden i app-visning", | ||||
|     "pinUpdates": "Fastgør opdateringer øverst i app-visning", | ||||
|     "updates": "Opdateringer", | ||||
|     "sourceSpecific": "Kildespecifik", | ||||
|     "appSource": "App-kilde", | ||||
| @@ -138,12 +138,12 @@ | ||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-eksport", | ||||
|     "pickAnAPK": "Vælg en APK", | ||||
|     "appHasMoreThanOnePackage": "{} har mere end én pakke:", | ||||
|     "deviceSupportsXArch": "Din enhed understøtter {} CPU-arkitekturen.", | ||||
|     "deviceSupportsXArch": "Din enhed understøtter CPU-arkitekturen {}.", | ||||
|     "deviceSupportsFollowingArchs": "Din enhed understøtter følgende CPU-arkitekturer:", | ||||
|     "warning": "Advarsel", | ||||
|     "sourceIsXButPackageFromYPrompt": "App-kilden er '{}', men udgivelsespakken kommer fra '{}'. Fortsæt?", | ||||
|     "updatesAvailable": "Opdateringer tilgængelige", | ||||
|     "updatesAvailableNotifDescription": "Underretter brugeren om, at opdateringer er tilgængelige for en eller flere apps, der spores af Obtainium", | ||||
|     "updatesAvailableNotifDescription": "Underretter brugeren om tilgængelige opdateringer for en eller flere apps, som Obtainium følger", | ||||
|     "noNewUpdates": "Ingen nye opdateringer.", | ||||
|     "xHasAnUpdate": "{} har en opdatering.", | ||||
|     "appsUpdated": "Apps opdateret", | ||||
| @@ -152,19 +152,19 @@ | ||||
|     "xWasUpdatedToY": "{} blev opdateret til {}.", | ||||
|     "xWasNotUpdatedToY": "Kunne ikke opdatere {} til {}.", | ||||
|     "errorCheckingUpdates": "Fejl ved tjek for opdateringer", | ||||
|     "errorCheckingUpdatesNotifDescription": "En meddelelse, der vises, når opdateringstjek i baggrunden mislykkes", | ||||
|     "errorCheckingUpdatesNotifDescription": "En meddelelse, der vises, opdateringstjek i baggrunden mislykkes", | ||||
|     "appsRemoved": "Apps fjernet", | ||||
|     "appsRemovedNotifDescription": "Underretter brugeren om, at en eller flere apps blev fjernet grundet fejl under indlæsning af dem", | ||||
|     "xWasRemovedDueToErrorY": "{} blev fjernet grundet denne fejl: {}", | ||||
|     "completeAppInstallation": "Færdiggør app-installation", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium skal være åben for at installere apps", | ||||
|     "completeAppInstallationNotifDescription": "Beder brugeren om at vende tilbage til Obtainium for at afslutte installationen af en app", | ||||
|     "completeAppInstallationNotifDescription": "Beder brugeren om at gå tilbage til Obtainium for at færdiggøre installation af en app", | ||||
|     "checkingForUpdates": "Tjekker for opdateringer", | ||||
|     "checkingForUpdatesNotifDescription": "Kortvarig meddelelse, der vises ved tjek for opdateringer", | ||||
|     "checkingForUpdatesNotifDescription": "Kortvarig meddelelse ved tjek for opdateringer", | ||||
|     "pleaseAllowInstallPerm": "Tillad venligst Obtainium at installere apps", | ||||
|     "trackOnly": "Følg Kun", | ||||
|     "errorWithHttpStatusCode": "Fejl {}", | ||||
|     "versionCorrectionDisabled": "Versionskorrigering deaktiveret (plugin ser ikke ud til at virke)", | ||||
|     "versionCorrectionDisabled": "Versionskorrektion deaktiveret (plugin ser ikke ud til at virke)", | ||||
|     "unknown": "Ukendt", | ||||
|     "none": "Ingen", | ||||
|     "never": "Aldrig", | ||||
| @@ -174,25 +174,23 @@ | ||||
|     "remove": "Fjern", | ||||
|     "yesMarkUpdated": "Ja, markér som opdateret", | ||||
|     "fdroid": "F-Droid Officiel", | ||||
|     "appIdOrName": "App-ID eller navn", | ||||
|     "appIdOrName": "App-ID eller -navn", | ||||
|     "appId": "App-ID", | ||||
|     "appWithIdOrNameNotFound": "Ingen app med det ID eller navn blev fundet", | ||||
|     "reposHaveMultipleApps": "Repos kan indeholde flere apps", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Tredjeparts-repo", | ||||
|     "steamMobile": "Steam Mobil", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Installer", | ||||
|     "markInstalled": "Markér som installeret", | ||||
|     "update": "Opdater", | ||||
|     "markUpdated": "Markér som opdateret", | ||||
|     "additionalOptions": "Yderligere indstillinger", | ||||
|     "additionalOptions": "Flere indstillinger", | ||||
|     "disableVersionDetection": "Deaktivér versionsregistrering", | ||||
|     "noVersionDetectionExplanation": "Denne indstilling bør kun bruges til apps, hvor versionsregistrering ikke virker korrekt.", | ||||
|     "downloadingX": "Henter {}", | ||||
|     "downloadX": "Hent {}", | ||||
|     "downloadedX": "Hentede {}", | ||||
|     "releaseAsset": "Udgivelsesressource", | ||||
|     "downloadNotifDescription": "Underretter brugeren om fremskridt i hentning af en app", | ||||
|     "downloadNotifDescription": "Underretter brugeren om status på hentning af en app", | ||||
|     "noAPKFound": "Ingen APK fundet", | ||||
|     "noVersionDetection": "Ingen versionsregistrering", | ||||
|     "categorize": "Kategoriser", | ||||
| @@ -224,10 +222,10 @@ | ||||
|     "groupByCategory": "Gruppér efter kategori", | ||||
|     "autoApkFilterByArch": "Forsøg at filtrere APK'er efter CPU-arkitektur, hvis muligt", | ||||
|     "overrideSource": "Tilsidesæt kilde", | ||||
|     "dontShowAgain": "Vis ikke denne igen", | ||||
|     "dontShowAgain": "Vis ikke igen", | ||||
|     "dontShowTrackOnlyWarnings": "Vis ikke 'Følg Kun'-advarsler", | ||||
|     "dontShowAPKOriginWarnings": "Vis ikke advarsler om APK-oprindelse", | ||||
|     "moveNonInstalledAppsToBottom": "Flyt ikke-installerede apps til bunden af appvisning", | ||||
|     "moveNonInstalledAppsToBottom": "Flyt ikke-installerede apps nederst i app-visning", | ||||
|     "gitlabPATLabel": "GitLab Personlig Adgangstoken", | ||||
|     "about": "Om", | ||||
|     "requiresCredentialsInSettings": "{} kræver yderligere legitimation (i Indstillinger)", | ||||
| @@ -240,44 +238,44 @@ | ||||
|     "reversePageTransitions": "Omvendte sideovergangsanimationer", | ||||
|     "minStarCount": "Minimum antal stjerner", | ||||
|     "addInfoBelow": "Tilføj denne info nedenfor.", | ||||
|     "addInfoInSettings": "Tilføj denne info i indstillingerne.", | ||||
|     "githubSourceNote": "GitHubs hastighedsbegrænsning kan undgås med en API-nøgle.", | ||||
|     "addInfoInSettings": "Tilføj denne info i Indstillinger.", | ||||
|     "githubSourceNote": "GitHub's hastighedsbegrænsning kan undgås med en API-nøgle.", | ||||
|     "sortByLastLinkSegment": "Sortér kun efter det sidste segment af linket", | ||||
|     "filterReleaseNotesByRegEx": "Filtrer udgivelsesnoter efter regulært udtryk", | ||||
|     "customLinkFilterRegex": "Brugerdefineret APK-linkfilter efter regulært udtryk (standard '.apk$')", | ||||
|     "appsPossiblyUpdated": "App-opdateringer forsøgt", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Underretter brugeren om, at opdateringer til en eller flere apps potentielt blev udført i baggrunden", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Underretter brugeren om, at opdateringer til en eller flere apps muligvis blev udført i baggrunden", | ||||
|     "xWasPossiblyUpdatedToY": "{} er muligvis blevet opdateret til {}.", | ||||
|     "enableBackgroundUpdates": "Aktivér baggrundsopdateringer", | ||||
|     "backgroundUpdateReqsExplanation": "Baggrundsopdateringer er muligvis ikke mulige for alle apps.", | ||||
|     "backgroundUpdateLimitsExplanation": "Om en baggrundsinstallation er vellykket, kan kun afgøres, når Obtainium åbnes.", | ||||
|     "backgroundUpdateReqsExplanation": "Baggrundsopdateringer er måske ikke mulige for alle apps.", | ||||
|     "backgroundUpdateLimitsExplanation": "En vellykket baggrundsinstallation kan kun afgøres, når Obtainium åbnes.", | ||||
|     "verifyLatestTag": "Verificer 'seneste'-tagget", | ||||
|     "intermediateLinkRegex": "Filtrer efter et 'mellemliggende' link at besøge", | ||||
|     "filterByLinkText": "Filtrer links efter linktekst", | ||||
|     "intermediateLinkNotFound": "Mellemliggende link ikke fundet", | ||||
|     "intermediateLink": "Mellemliggende link", | ||||
|     "exemptFromBackgroundUpdates": "Undtag fra baggrundsopdateringer (hvis aktiveret)", | ||||
|     "bgUpdatesOnWiFiOnly": "Deaktivér baggrundsopdateringer, når du ikke er på WiFi", | ||||
|     "bgUpdatesWhileChargingOnly": "Deaktiver baggrundsopdateringer, når der ikke oplades", | ||||
|     "bgUpdatesOnWiFiOnly": "Deaktiver baggrundsopdateringer, når du ikke er på Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Deaktiver baggrundsopdateringer, når du ikke oplader", | ||||
|     "autoSelectHighestVersionCode": "Auto-vælg højeste versionKode af APK", | ||||
|     "versionExtractionRegEx": "RegEx for versionsstrengsudtrækning", | ||||
|     "trimVersionString": "Trim versionsstrengen med RegEx", | ||||
|     "matchGroupToUseForX": "Matchgruppe til brug for \"{}\"", | ||||
|     "matchGroupToUse": "Match gruppe til brug til RegEx for versionsstrengsudtrækning", | ||||
|     "trimVersionString": "Trim versionsstreng med RegEx", | ||||
|     "matchGroupToUseForX": "Match-gruppe til brug for \"{}\"", | ||||
|     "matchGroupToUse": "RegEx-matchgruppe til brug for versionsstrengsudtrækning", | ||||
|     "highlightTouchTargets": "Fremhæv mindre åbenlyse berøringsmål", | ||||
|     "pickExportDir": "Vælg eksportmappe", | ||||
|     "autoExportOnChanges": "Auto-eksport ved ændringer", | ||||
|     "autoExportOnChanges": "Auto-eksportér ved ændringer", | ||||
|     "includeSettings": "Inkluder indstillinger", | ||||
|     "filterVersionsByRegEx": "Filtrer versioner efter regulært udtryk", | ||||
|     "trySelectingSuggestedVersionCode": "Forsøg at vælge den foreslåede versionKode af APK", | ||||
|     "dontSortReleasesList": "Behold udgivelsesordre fra API", | ||||
|     "dontSortReleasesList": "Behold udgivelsesrækkefølge fra API", | ||||
|     "reverseSort": "Omvendt sortering", | ||||
|     "takeFirstLink": "Tag første link", | ||||
|     "skipSort": "Spring sortering over", | ||||
|     "debugMenu": "Fejlfindingsmenu", | ||||
|     "bgTaskStarted": "Baggrundsopgave startet - tjek logfiler.", | ||||
|     "runBgCheckNow": "Kør baggrundsopdateringstjek nu", | ||||
|     "versionExtractWholePage": "Anvend RegEx for versionsstrengsudtrækning for hele siden", | ||||
|     "versionExtractWholePage": "Anvend RegEx til versionsstrengsudtrækning på hele siden", | ||||
|     "installing": "Installerer", | ||||
|     "skipUpdateNotifications": "Spring opdateringsmeddelelser over", | ||||
|     "updatesAvailableNotifChannel": "Opdateringer tilgængelige", | ||||
| @@ -288,7 +286,7 @@ | ||||
|     "downloadingXNotifChannel": "Henter {}", | ||||
|     "completeAppInstallationNotifChannel": "Færdiggør app-installation", | ||||
|     "checkingForUpdatesNotifChannel": "Tjekker for opdateringer", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Tjek kun installeret og 'Følg Kun'-apps for opdateringer", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Tjek kun installerede og 'Følg Kun'-apps for opdateringer", | ||||
|     "supportFixedAPKURL": "Understøt fikserede APK-URL'er", | ||||
|     "selectX": "Vælg {}", | ||||
|     "parallelDownloads": "Tillad samtidige overførsler", | ||||
| @@ -298,8 +296,8 @@ | ||||
|     "shizukuOldAndroidWithADB": "Shizuku kører på Android <8.1 med ADB. Opdater Android eller brug Sui i stedet", | ||||
|     "shizukuPretendToBeGooglePlay": "Indstil Google Play som installationskilde (hvis Shizuku bruges)", | ||||
|     "useSystemFont": "Brug systemskrifttype", | ||||
|     "useVersionCodeAsOSVersion": "Brug app versionKode som OS-registreret version", | ||||
|     "requestHeader": "Anmod overskrift", | ||||
|     "useVersionCodeAsOSVersion": "Brug app-versionKode som OS-registreret version", | ||||
|     "requestHeader": "Anmodningsheader", | ||||
|     "useLatestAssetDateAsReleaseDate": "Brug seneste ressourceupload som udgivelsesdato", | ||||
|     "defaultPseudoVersioningMethod": "Standard pseudo-versioneringsmetode", | ||||
|     "partialAPKHash": "Delvis APK-hash", | ||||
| @@ -313,12 +311,13 @@ | ||||
|     "selfHostedNote": "Rullemenuen \"{}\" kan bruges til at nå selvhostede/brugerdefinerede instanser af enhver kilde.", | ||||
|     "badDownload": "APK'en kunne ikke analyseres (inkompatibel eller delvis hentning)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Del nye apps med AppVerifier (hvis tilgængelig)", | ||||
|     "appVerifierInstructionToast": "Del til AppVerifier, og vend tilbage hertil, når du er klar.", | ||||
|     "appVerifierInstructionToast": "Del til AppVerifier, og vend tilbage, når du er klar.", | ||||
|     "wiki": "Hjælp/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourcede app-konfigurationer (brug på egen risiko)", | ||||
|     "crowdsourcedConfigsShort": "Crowdsourcede app-konfigurationer", | ||||
|     "allowInsecure": "Tillad usikre HTTP-anmodninger", | ||||
|     "stayOneVersionBehind": "Vær en version bagud i forhold til den nyeste", | ||||
|     "stayOneVersionBehind": "Forbliv én version bagud den seneste", | ||||
|     "refreshBeforeDownload": "Opdater app-detaljer før download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Fjern app?", | ||||
|         "other": "Fjern apps?" | ||||
| @@ -357,7 +356,7 @@ | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "Ryddet {n} log (før = {before}, efter = {after})", | ||||
|         "other": "Ryddet {n} logs (før = {before}, efter = {after})" | ||||
|         "other": "Ryddede {n} logs (før = {before}, efter = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} og 1 anden app har opdateringer.", | ||||
|   | ||||
| @@ -114,7 +114,7 @@ | ||||
|     "light": "Hell", | ||||
|     "followSystem": "System folgen", | ||||
|     "followSystemThemeExplanation": "Das Folgen des Systemthemes ist unter Android < 10 nur mit Hilfe von Drittanbieterapps möglich", | ||||
|     "useBlackTheme": "Pure Black Dark Theme verwenden", | ||||
|     "useBlackTheme": "Rein schwarzen Hintergrund verwenden", | ||||
|     "appSortBy": "App sortieren nach", | ||||
|     "authorName": "Autor/Name", | ||||
|     "nameAuthor": "Name/Autor", | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Es wurde keine App mit dieser ID oder diesem Namen gefunden", | ||||
|     "reposHaveMultipleApps": "Repos können mehrere Apps enthalten", | ||||
|     "fdroidThirdPartyRepo": "F-Droid-Drittanbieter-Repo", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam-Chat", | ||||
|     "install": "Installieren", | ||||
|     "markInstalled": "Als installiert markieren", | ||||
|     "update": "Aktualisieren", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App-Konfigurationen", | ||||
|     "allowInsecure": "Unsichere HTTP-Anfragen zulassen", | ||||
|     "stayOneVersionBehind": "Eine Version hinter der neuesten Version bleiben", | ||||
|     "refreshBeforeDownload": "App-Details vor dem Download aktualisieren", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App entfernen?", | ||||
|         "other": "Apps entfernen?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Neniu apo estas trovita kun tiu identigilo aŭ nomo", | ||||
|     "reposHaveMultipleApps": "Deponejoj povas enhavi plurajn apojn", | ||||
|     "fdroidThirdPartyRepo": "Tria deponejo de F-Droid", | ||||
|     "steamMobile": "Telefona Steam", | ||||
|     "steamChat": "Steam Babilejo", | ||||
|     "install": "Instali", | ||||
|     "markInstalled": "Marki kiel instalita", | ||||
|     "update": "Ĝisdatigi", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Forigi la aplikaĵon?", | ||||
|         "other": "Forigi la aplikaĵojn?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "No App was found with that ID or Name", | ||||
|     "reposHaveMultipleApps": "Repos may contain multiple Apps", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Third-Party Repo", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Install", | ||||
|     "markInstalled": "Mark Installed", | ||||
|     "update": "Update", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Remove App?", | ||||
|         "other": "Remove Apps?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "No se han encontrado aplicaciones con ese ID o nombre", | ||||
|     "reposHaveMultipleApps": "Los repositorios pueden contener varias aplicaciones", | ||||
|     "fdroidThirdPartyRepo": "Repositorio de terceros F-Droid", | ||||
|     "steamMobile": "Steam para móviles", | ||||
|     "steamChat": "Chat de Steam", | ||||
|     "install": "Instalar", | ||||
|     "markInstalled": "Marcar como instalada", | ||||
|     "update": "Actualizar", | ||||
| @@ -260,10 +258,10 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin WiFi", | ||||
|     "bgUpdatesWhileChargingOnly": "Desactiva las actualizaciones en segundo plano cuando no estés cargando", | ||||
|     "autoSelectHighestVersionCode": "Auto selección del paquete APK con versión más reciente", | ||||
|     "versionExtractionRegEx": "Versión de extracción regex", | ||||
|     "versionExtractionRegEx": "Versión de extracción RegEx", | ||||
|     "trimVersionString": "Recortar cadena de versión con RegEx", | ||||
|     "matchGroupToUseForX": "Grupo de coincidencia a utilizar para \"{}\"", | ||||
|     "matchGroupToUse": "Grupo a usar para versión de extracción regex", | ||||
|     "matchGroupToUse": "Grupo a usar para versión de extracción RegEx", | ||||
|     "highlightTouchTargets": "Resaltar objetivos menos obvios", | ||||
|     "pickExportDir": "Directorio para exportar", | ||||
|     "autoExportOnChanges": "Auto exportar cuando haya cambios", | ||||
| @@ -277,7 +275,7 @@ | ||||
|     "debugMenu": "Menu Depurar", | ||||
|     "bgTaskStarted": "Iniciada tarea en segundo plano; revise los registros.", | ||||
|     "runBgCheckNow": "Ejecutar verficiación de actualizaciones en segundo plano", | ||||
|     "versionExtractWholePage": "Aplicar la versión de extracción regex a la página entera", | ||||
|     "versionExtractWholePage": "Aplicar la versión de extracción RegEx a la página entera", | ||||
|     "installing": "Instalando", | ||||
|     "skipUpdateNotifications": "No notificar sobre actualizaciones", | ||||
|     "updatesAvailableNotifChannel": "Actualizaciones disponibles", | ||||
| @@ -312,13 +310,14 @@ | ||||
|     "note": "Nota", | ||||
|     "selfHostedNote": "El desplegable «{}» puede usarse para acceder a instancias autoalojadas/personalizadas de cualquier fuente.", | ||||
|     "badDownload": "No se ha podido analizar el APK (incompatible o descarga parcial)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Compartir aplicaciones nuevas con AppVerifier (si está disponible)", | ||||
|     "appVerifierInstructionToast": "Comparta con AppVerifier y vuelva aquí cuando esté listo.", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Compartir aplicaciones nuevas con AppVerifier (si está instalado)", | ||||
|     "appVerifierInstructionToast": "Compartir con AppVerifier y vuelver aquí cuando esté listo.", | ||||
|     "wiki": "Ayuda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (uso bajo su propia responsabilidad)", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use bajo su responsabilidad)", | ||||
|     "crowdsourcedConfigsShort": "Configuración de aplicaciones por crowdsourcing", | ||||
|     "allowInsecure": "Permitir peticiones HTTP inseguras", | ||||
|     "stayOneVersionBehind": "Mantenerse una versión por detrás de la última", | ||||
|     "refreshBeforeDownload": "Actualiza los datos de la aplicación antes de descargarla", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "¿Eliminar aplicación?", | ||||
|         "other": "¿Eliminar aplicaciones?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "هیچ برنامه ای با آن شناسه یا نام یافت نشد", | ||||
|     "reposHaveMultipleApps": "مخازن ممکن است شامل چندین برنامه باشد", | ||||
|     "fdroidThirdPartyRepo": "مخازن شخص ثالث F-Droid", | ||||
|     "steamMobile": "استیم موبایل", | ||||
|     "steamChat": "چت استیم", | ||||
|     "install": "نصب", | ||||
|     "markInstalled": "علامت گذاری به عنوان نصب شده", | ||||
|     "update": "به روز رسانی", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "درخواست های HTTP ناامن را مجاز کنید", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "برنامه حذف شود؟", | ||||
|         "other": "برنامه ها حذف شوند؟" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Aucune application n'a été trouvée avec cet identifiant ou ce nom", | ||||
|     "reposHaveMultipleApps": "Les dépôts peuvent contenir plusieurs applications", | ||||
|     "fdroidThirdPartyRepo": "Dépôt tiers F-Droid", | ||||
|     "steamMobile": "Application mobile Steam", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Installer", | ||||
|     "markInstalled": "Marquer comme étant installé", | ||||
|     "update": "Mettre à jour", | ||||
| @@ -315,10 +313,11 @@ | ||||
|     "beforeNewInstallsShareToAppVerifier": "Partager les nouvelles applications avec AppVerifier (si disponible)", | ||||
|     "appVerifierInstructionToast": "Partagez avec AppVerifier, puis revenez ici lorsque tout est prêt.", | ||||
|     "wiki": "Aide/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurations d'applications par la communauté (à utiliser à vos risques et périls)", | ||||
|     "crowdsourcedConfigsShort": "Configurations d'applications par la communauté", | ||||
|     "crowdsourcedConfigsLabel": "Configuration d'applis communautaire (à utiliser à vos risques et périls)", | ||||
|     "crowdsourcedConfigsShort": "Applis communautaires", | ||||
|     "allowInsecure": "Autoriser les requêtes HTTP non sécurisées", | ||||
|     "stayOneVersionBehind": "Rester à une version de la dernière", | ||||
|     "refreshBeforeDownload": "Actualiser les détails de l'application avant de la télécharger", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Supprimer l'application ?", | ||||
|         "other": "Supprimer les applications ?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Nem található alkalmazás ezzel az azonosítóval vagy névvel", | ||||
|     "reposHaveMultipleApps": "A tárolók több alkalmazást is tartalmazhatnak", | ||||
|     "fdroidThirdPartyRepo": "F-Droid harmadik féltől származó tároló", | ||||
|     "steamMobile": "Steam Mobil", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Telepítés", | ||||
|     "markInstalled": "Telepítettnek jelölés", | ||||
|     "update": "Frissítés", | ||||
| @@ -215,7 +213,7 @@ | ||||
|     "releaseDateAsVersion": "Használja a kiadás dátumát verzió-karakterláncként", | ||||
|     "releaseTitleAsVersion": "Használja a kiadás címét verzió-karakterláncként", | ||||
|     "releaseDateAsVersionExplanation": "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, de elérhető a kiadás dátuma.", | ||||
|     "changes": "Változások", | ||||
|     "changes": "Változáslista", | ||||
|     "releaseDate": "Kiadás dátuma", | ||||
|     "importFromURLsInFile": "Importálás fájlban található webcímből (pl. OPML)", | ||||
|     "versionDetectionExplanation": "A verzió-karakterlánc egyeztetése az rendszer által érzékelt verzióval", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "Nem biztonságos HTTP-kérések engedélyezése", | ||||
|     "stayOneVersionBehind": "Maradjon egy verzióval a legújabb mögött", | ||||
|     "refreshBeforeDownload": "Az alkalmazás adatainak frissítése a letöltés előtt", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Eltávolítja az alkalmazást?", | ||||
|         "other": "Eltávolítja az alkalmazásokat?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Tidak ada aplikasi yang ditemukan dengan ID atau nama tersebut", | ||||
|     "reposHaveMultipleApps": "Repositori dapat berisi beberapa aplikasi", | ||||
|     "fdroidThirdPartyRepo": "Repositori pihak ketiga F-Droid", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Obrolan Steam", | ||||
|     "install": "Pasang", | ||||
|     "markInstalled": "Tandai terpasang", | ||||
|     "update": "perbarui", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Konfigurasi Aplikasi Crowdsourced", | ||||
|     "allowInsecure": "Izinkan permintaan HTTP yang tidak aman", | ||||
|     "stayOneVersionBehind": "Tetap satu versi di belakang versi terbaru", | ||||
|     "refreshBeforeDownload": "Segarkan detail aplikasi sebelum mengunduh", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Hapus aplikasi?", | ||||
|         "other": "Hapus aplikasi?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Non è stata trovata alcuna app con quell'ID o nome", | ||||
|     "reposHaveMultipleApps": "I repository possono contenere più app", | ||||
|     "fdroidThirdPartyRepo": "Repository F-Droid di terze parti", | ||||
|     "steamMobile": "Mobile a vapore", | ||||
|     "steamChat": "Chat di vapore", | ||||
|     "install": "Installa", | ||||
|     "markInstalled": "Contrassegna come installata", | ||||
|     "update": "Aggiorna", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Configurazioni di app in crowdsourcing", | ||||
|     "allowInsecure": "Consentire le richieste HTTP non sicure", | ||||
|     "stayOneVersionBehind": "Rimanere una versione indietro rispetto alla più recente", | ||||
|     "refreshBeforeDownload": "Aggiornare i dettagli dell'app prima del download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Rimuovere l'app?", | ||||
|         "other": "Rimuovere le app?" | ||||
|   | ||||
| @@ -147,21 +147,21 @@ | ||||
|     "noNewUpdates": "新しいアップデートはありません", | ||||
|     "xHasAnUpdate": "{} のアップデートが利用可能です。", | ||||
|     "appsUpdated": "アプリをアップデートしました", | ||||
|     "appsNotUpdated": "アプリケーションの更新に失敗", | ||||
|     "appsNotUpdated": "アプリのアップデートに失敗しました", | ||||
|     "appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する", | ||||
|     "xWasUpdatedToY": "{} が {} にアップデートされました", | ||||
|     "xWasNotUpdatedToY": "への更新に失敗しました。", | ||||
|     "xWasNotUpdatedToY": "{} の {} へのアップデートに失敗しました", | ||||
|     "errorCheckingUpdates": "アップデート確認中のエラー", | ||||
|     "errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデート確認に失敗した際に表示される通知", | ||||
|     "appsRemoved": "削除されたアプリ", | ||||
|     "appsRemovedNotifDescription": "アプリの読み込み中にエラーが発生したため、1つまたは複数のアプリが削除されたことをユーザーに通知する", | ||||
|     "xWasRemovedDueToErrorY": "このエラーのため、{} は削除されました: {}", | ||||
|     "completeAppInstallation": "アプリのインストールを完了する", | ||||
|     "obtainiumMustBeOpenToInstallApps": "アプリをインストールするにはObtainiumが開いている必要があります", | ||||
|     "obtainiumMustBeOpenToInstallApps": "アプリをインストールするにはObtainiumを開く必要があります", | ||||
|     "completeAppInstallationNotifDescription": "アプリのインストールを完了するために、Obtainiumに戻る必要があります", | ||||
|     "checkingForUpdates": "アップデートを確認中", | ||||
|     "checkingForUpdatesNotifDescription": "アップデートを確認する際に表示される一時的な通知", | ||||
|     "pleaseAllowInstallPerm": "Obtainiumによるアプリのインストールを許可してください。", | ||||
|     "pleaseAllowInstallPerm": "Obtainiumによるアプリのインストールを許可してください", | ||||
|     "trackOnly": "追跡のみ", | ||||
|     "errorWithHttpStatusCode": "エラー {}", | ||||
|     "versionCorrectionDisabled": "バージョン補正無効 (プラグインが動作していません)", | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "そのIDや名前を持つアプリは見つかりませんでした", | ||||
|     "reposHaveMultipleApps": "リポジトリには複数のアプリが含まれることがあります", | ||||
|     "fdroidThirdPartyRepo": "F-Droid サードパーティリポジトリ", | ||||
|     "steamMobile": "Steamモバイル", | ||||
|     "steamChat": "Steamチャット", | ||||
|     "install": "インストール", | ||||
|     "markInstalled": "インストール済みとしてマークする", | ||||
|     "update": "アップデート", | ||||
| @@ -213,7 +211,7 @@ | ||||
|     "uninstallFromDevice": "デバイスからアンインストールする", | ||||
|     "onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。", | ||||
|     "releaseDateAsVersion": "リリース日をバージョンとして使用する", | ||||
|     "releaseTitleAsVersion": "リリースタイトルをバージョン文字列として使用", | ||||
|     "releaseTitleAsVersion": "リリースタイトルをバージョンとして使用する", | ||||
|     "releaseDateAsVersionExplanation": "このオプションは、バージョン検出が正しく機能しないアプリで、リリース日が利用可能な場合にのみ使用する必要があります。", | ||||
|     "changes": "変更点", | ||||
|     "releaseDate": "リリース日", | ||||
| @@ -258,7 +256,7 @@ | ||||
|     "intermediateLink": "中間リンク", | ||||
|     "exemptFromBackgroundUpdates": "バックグラウンドアップデートを行わない (有効な場合)", | ||||
|     "bgUpdatesOnWiFiOnly": "WiFiを使用していない場合、バックグラウンドアップデートを無効にする", | ||||
|     "bgUpdatesWhileChargingOnly": "非充電時にバックグラウンド更新を無効にする", | ||||
|     "bgUpdatesWhileChargingOnly": "非充電時にバックグラウンドアップデートを無効にする", | ||||
|     "autoSelectHighestVersionCode": "最も高いバージョンコードのAPKを自動で選択する", | ||||
|     "versionExtractionRegEx": "バージョン抽出の正規表現", | ||||
|     "trimVersionString": "正規表現でバージョン文字列をトリムする", | ||||
| @@ -318,7 +316,8 @@ | ||||
|     "crowdsourcedConfigsLabel": "クラウドソーシングによるアプリの設定(利用は自己責任で)", | ||||
|     "crowdsourcedConfigsShort": "クラウドソーシングによるアプリの設定", | ||||
|     "allowInsecure": "安全でないHTTPリクエストを許可する", | ||||
|     "stayOneVersionBehind": "最新バージョンから1つ遅れ", | ||||
|     "stayOneVersionBehind": "最新のバージョンから1つ前のものを使用する", | ||||
|     "refreshBeforeDownload": "ダウンロード前にアプリの詳細を更新する", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "アプリを削除しますか?", | ||||
|         "other": "アプリを削除しますか?" | ||||
| @@ -360,16 +359,16 @@ | ||||
|         "other": "{n} 個のログをクリアしました (前 = {before}, 後 = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} とさらに {} 個のアプリのアップデートが利用可能です。", | ||||
|         "one": "{} とさらに 1 個のアプリのアップデートが利用可能です。", | ||||
|         "other": "{} とさらに {} 個のアプリのアップデートが利用可能です。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} とさらに {} 個のアプリがアップデートされました。", | ||||
|         "one": "{} とさらに 1 個のアプリがアップデートされました。", | ||||
|         "other": "{} とさらに {} 個のアプリがアップデートされました。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "更新に失敗しました。", | ||||
|         "other": "アプリのアップデートに失敗しました。" | ||||
|         "one": "{} とさらに 1 個のアプリのアップデートに失敗しました。", | ||||
|         "other": "{} とさらに {} 個のアプリのアップデートに失敗しました。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} とさらに 1 個のアプリがアップデートされた可能性があります。", | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Er is geen app gevonden met dat ID of die naam", | ||||
|     "reposHaveMultipleApps": "Repositories kunnen meerdere apps bevatten", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Repository voor derden", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Installeren", | ||||
|     "markInstalled": "Als geïnstalleerd markeren", | ||||
|     "update": "Bijwerken", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "App-configuraties door menigte", | ||||
|     "allowInsecure": "Onveilige HTTP-verzoeken toestaan", | ||||
|     "stayOneVersionBehind": "Blijf een versie achter op de nieuwste", | ||||
|     "refreshBeforeDownload": "Vernieuw app details voor download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App verwijderen?", | ||||
|         "other": "Apps verwijderen?" | ||||
|   | ||||
| @@ -6,7 +6,7 @@ | ||||
|     "cantInstallOlderVersion": "Nie można zainstalować starszej wersji aplikacji", | ||||
|     "appIdMismatch": "Pobrane ID pakietu nie pasuje do istniejącego ID aplikacji", | ||||
|     "functionNotImplemented": "Ta klasa nie zaimplementowała tej funkcji", | ||||
|     "placeholder": "Zbiornik", | ||||
|     "placeholder": "Placeholder", | ||||
|     "someErrors": "Wystąpiły pewne błędy", | ||||
|     "unexpectedError": "Nieoczekiwany błąd", | ||||
|     "ok": "Okej", | ||||
| @@ -22,9 +22,9 @@ | ||||
|     "requiredInBrackets": "(Wymagane)", | ||||
|     "dropdownNoOptsError": "BŁĄD: LISTA ROZWIJANA MUSI MIEĆ CO NAJMNIEJ JEDNĄ OPCJĘ", | ||||
|     "colour": "Kolor", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Niestandardowe", | ||||
|     "useMaterialYou": "Używaj materiałów", | ||||
|     "standard": "Domyślny", | ||||
|     "custom": "Własny", | ||||
|     "useMaterialYou": "Material You", | ||||
|     "githubStarredRepos": "Repozytoria GitHub oznaczone gwiazdką", | ||||
|     "uname": "Nazwa użytkownika", | ||||
|     "wrongArgNum": "Nieprawidłowa liczba podanych argumentów", | ||||
| @@ -55,7 +55,7 @@ | ||||
|     "pleaseWait": "Proszę czekać", | ||||
|     "updateAvailable": "Dostępna aktualizacja", | ||||
|     "notInstalled": "Nie zainstalowano", | ||||
|     "pseudoVersion": "pseudowersja", | ||||
|     "pseudoVersion": "pseudo-wersja", | ||||
|     "selectAll": "Zaznacz wszystkie", | ||||
|     "deselectX": "Odznacz {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} zostanie usunięty z Obtainium, ale pozostanie zainstalowany na urządzeniu.", | ||||
| @@ -76,7 +76,7 @@ | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Zresetować status instalacji dla wybranych aplikacji?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Stan instalacji wybranych aplikacji zostanie zresetowany.\n\nMoże być to pomocne, gdy wersja aplikacji wyświetlana w Obtainium jest nieprawidłowa z powodu nieudanych aktualizacji lub innych problemów.", | ||||
|     "customLinkMessage": "Te linki działają na urządzeniach z zainstalowanym Obtainium", | ||||
|     "shareAppConfigLinks": "Udostępnij konfigurację aplikacji jako link HTML", | ||||
|     "shareAppConfigLinks": "Udostępnij konfigurację aplikacji w formie linku", | ||||
|     "shareSelectedAppURLs": "Udostępnij wybrane adresy URL aplikacji", | ||||
|     "resetInstallStatus": "Zresetuj stan instalacji", | ||||
|     "more": "Więcej", | ||||
| @@ -150,7 +150,7 @@ | ||||
|     "appsNotUpdated": "Nie udało się zaktualizować aplikacji", | ||||
|     "appsUpdatedNotifDescription": "Informuje, gdy co najmniej jedna aplikacja została zaktualizowana w tle", | ||||
|     "xWasUpdatedToY": "{} zaktualizowano do {}.", | ||||
|     "xWasNotUpdatedToY": "Nie udało się zaktualizować {} do {}.", | ||||
|     "xWasNotUpdatedToY": "Błąd aktualizacji {} do {}.", | ||||
|     "errorCheckingUpdates": "Błąd sprawdzania aktualizacji", | ||||
|     "errorCheckingUpdatesNotifDescription": "Jest wyświetlane, gdy sprawdzanie aktualizacji w tle nie powiedzie się", | ||||
|     "appsRemoved": "Usunięte aplikacje", | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Nie znaleziono aplikacji o tym identyfikatorze lub nazwie", | ||||
|     "reposHaveMultipleApps": "Repozytoria mogą zawierać wiele aplikacji", | ||||
|     "fdroidThirdPartyRepo": "Zewnętrzne repo F-Droid", | ||||
|     "steamMobile": "Mobilny Steam", | ||||
|     "steamChat": "Czat Steam", | ||||
|     "install": "Instaluj", | ||||
|     "markInstalled": "Oznacz jako zainstalowane", | ||||
|     "update": "Zaktualizuj", | ||||
| @@ -191,7 +189,7 @@ | ||||
|     "downloadingX": "Pobieranie {}", | ||||
|     "downloadX": "Pobierz {}", | ||||
|     "downloadedX": "Pobrano {}", | ||||
|     "releaseAsset": "Release Asset", | ||||
|     "releaseAsset": "Wydany pakiet", | ||||
|     "downloadNotifDescription": "Informuje o postępach w pobieraniu aplikacji", | ||||
|     "noAPKFound": "Nie znaleziono pakietu APK", | ||||
|     "noVersionDetection": "Bez wykrywania wersji", | ||||
| @@ -213,12 +211,12 @@ | ||||
|     "uninstallFromDevice": "Odinstaluj z urządzenia", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Działa tylko w przypadku aplikacji z wyłączonym wykrywaniem wersji.", | ||||
|     "releaseDateAsVersion": "Użyj daty wydania jako wersji", | ||||
|     "releaseTitleAsVersion": "Użyj tytułu wydania jako ciągu znaków wersji", | ||||
|     "releaseTitleAsVersion": "Używaj nazwy wydania jako ciągu wersji", | ||||
|     "releaseDateAsVersionExplanation": "Opcja ta powinna być używana tylko w przypadku aplikacji, w których wykrywanie wersji nie działa poprawnie, ale dostępna jest data wydania.", | ||||
|     "changes": "Zmiany", | ||||
|     "releaseDate": "Data wydania", | ||||
|     "importFromURLsInFile": "Importuj z adresów URL w pliku (typu OPML)", | ||||
|     "versionDetectionExplanation": "Uzgodnij ciąg wersji z wersją wykrytą w systemie operacyjnym", | ||||
|     "versionDetectionExplanation": "Uzgodnij ciąg wersji z wersją wykrytą przez system operacyjny", | ||||
|     "versionDetection": "Wykrywanie wersji", | ||||
|     "standardVersionDetection": "Standardowe wykrywanie wersji", | ||||
|     "groupByCategory": "Grupuj według kategorii", | ||||
| @@ -252,16 +250,16 @@ | ||||
|     "backgroundUpdateReqsExplanation": "Aktualizacje w tle mogą nie być możliwe dla wszystkich aplikacji.", | ||||
|     "backgroundUpdateLimitsExplanation": "Powodzenie instalacji w tle można określić dopiero po otwarciu Obtainium.", | ||||
|     "verifyLatestTag": "Zweryfikuj najnowszy tag", | ||||
|     "intermediateLinkRegex": "Filtruj link \"pośredni\" do odwiedzenia", | ||||
|     "intermediateLinkRegex": "Filtr linków \"pośrednich\" do odwiedzenia w pierwszej kolejności", | ||||
|     "filterByLinkText": "Filtruj linki według tekstu linku", | ||||
|     "intermediateLinkNotFound": "Nie znaleziono linku pośredniego", | ||||
|     "intermediateLink": "Łącze pośrednie", | ||||
|     "intermediateLink": "Link pośredni", | ||||
|     "exemptFromBackgroundUpdates": "Wyklucz z uaktualnień w tle (jeśli są włączone)", | ||||
|     "bgUpdatesOnWiFiOnly": "Wyłącz aktualizacje w tle, gdy nie ma połączenia z Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Wyłącz aktualizacje w tle, gdy urządzenie nie jest ładowane", | ||||
|     "autoSelectHighestVersionCode": "Automatycznie wybierz najwyższy kod wersji APK", | ||||
|     "versionExtractionRegEx": "Wyrażenie regularne wyodrębniające wersję", | ||||
|     "trimVersionString": "Przycinanie łańcucha wersji za pomocą RegEx", | ||||
|     "trimVersionString": "Przytnij ciąg wersji za pomocą RegEx", | ||||
|     "matchGroupToUseForX": "Dopasuj grupę do użycia dla \"{}\"", | ||||
|     "matchGroupToUse": "Dopasuj grupę do użycia dla wyrażenia regularnego wyodrębniania wersji", | ||||
|     "highlightTouchTargets": "Wyróżnij mniej oczywiste elementy dotykowe", | ||||
| @@ -272,7 +270,7 @@ | ||||
|     "trySelectingSuggestedVersionCode": "Spróbuj wybierać sugerowany kod wersji APK", | ||||
|     "dontSortReleasesList": "Utrzymaj kolejność wydań z interfejsu API", | ||||
|     "reverseSort": "Odwrotne sortowanie", | ||||
|     "takeFirstLink": "Weź pierwszy link", | ||||
|     "takeFirstLink": "Wykorzystaj pierwszy link", | ||||
|     "skipSort": "Pomiń sortowanie", | ||||
|     "debugMenu": "Menu debugowania", | ||||
|     "bgTaskStarted": "Uruchomiono zadanie w tle - sprawdź logi.", | ||||
| @@ -285,40 +283,41 @@ | ||||
|     "appsPossiblyUpdatedNotifChannel": "Informuj o próbach aktualizacji", | ||||
|     "errorCheckingUpdatesNotifChannel": "Błędy sprawdzania aktualizacji", | ||||
|     "appsRemovedNotifChannel": "Usunięte aplikacje", | ||||
|     "downloadingXNotifChannel": "Pobieranie aplikacji", | ||||
|     "downloadingXNotifChannel": "Pobieranie {}", | ||||
|     "completeAppInstallationNotifChannel": "Ukończenie instalacji aplikacji", | ||||
|     "checkingForUpdatesNotifChannel": "Sprawdzanie dostępności aktualizacji", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Sprawdzaj tylko zainstalowane i obserwowane aplikacje pod kątem aktualizacji", | ||||
|     "supportFixedAPKURL": "Obsługuj stałe adresy URL APK", | ||||
|     "selectX": "Wybierz {}", | ||||
|     "parallelDownloads": "Zezwól na pobieranie równoległe", | ||||
|     "parallelDownloads": "Zezwalaj na równoległe pobierania", | ||||
|     "useShizuku": "Użyj Shizuku lub Sui, aby zainstalować", | ||||
|     "shizukuBinderNotFound": "Shizuku is not running", | ||||
|     "shizukuBinderNotFound": "Usługa Shizuku nie działa", | ||||
|     "shizukuOld": "Stara wersja Shizuku (<11) - zaktualizuj ją", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku działa na Androidzie < 8.1 z ADB - zaktualizuj Androida lub użyj zamiast tego Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Ustaw Google Play jako źródło instalacji (jeśli używana jest aplikacja Shizuku).", | ||||
|     "useSystemFont": "Użyj czcionki systemowej", | ||||
|     "useVersionCodeAsOSVersion": "Użyj kodu wersji aplikacji jako wersji wykrytej przez system operacyjny", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku przez ADB działa na Androidzie 8.1+. Zaktualizuj Androida lub użyj zamiast tego Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Ustaw Google Play jako źródło instalacji (jeśli używana jest aplikacja Shizuku)", | ||||
|     "useSystemFont": "Czcionka systemowa", | ||||
|     "useVersionCodeAsOSVersion": "Użyj versionCode aplikacji jako wersji wykrytej przez system operacyjny", | ||||
|     "requestHeader": "Nagłówek żądania", | ||||
|     "useLatestAssetDateAsReleaseDate": "Użyj najnowszego przesłanego zasobu jako daty wydania", | ||||
|     "defaultPseudoVersioningMethod": "Domyślna metoda pseudowersji", | ||||
|     "partialAPKHash": "Częściowy skrót APK", | ||||
|     "APKLinkHash": "Skrót łącza APK", | ||||
|     "directAPKLink": "Bezpośredni link APK", | ||||
|     "pseudoVersionInUse": "Pseudowersja jest w użyciu", | ||||
|     "defaultPseudoVersioningMethod": "Domyślna metoda pseudo-wersji", | ||||
|     "partialAPKHash": "Częściowy Hash pliku apk", | ||||
|     "APKLinkHash": "Link Hash pliku apk", | ||||
|     "directAPKLink": "Bezpośredni link do pliku apk", | ||||
|     "pseudoVersionInUse": "Pseudo-wersja jest w użyciu", | ||||
|     "installed": "Zainstalowano", | ||||
|     "latest": "Najnowszy", | ||||
|     "latest": "Najnowsza", | ||||
|     "invertRegEx": "Odwróć wyrażenie regularne", | ||||
|     "note": "Uwaga", | ||||
|     "selfHostedNote": "Lista rozwijana \"{}\" może być używana do uzyskiwania dostępu do samodzielnie hostowanych / niestandardowych instancji dowolnego źródła.", | ||||
|     "badDownload": "Nie można przeanalizować pliku APK (niekompatybilny lub częściowo pobrany).", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Udostępnianie nowych aplikacji za pomocą AppVerifier (jeśli dostępne)", | ||||
|     "appVerifierInstructionToast": "Udostępnij w AppVerifier, a następnie wróć tutaj, gdy będziesz gotowy.", | ||||
|     "selfHostedNote": "Wybierz \"{}\", aby uzyskać dostęp do samodzielnie hostowanych lub niestandardowych instancji dowolnego źródła.", | ||||
|     "badDownload": "Nie można przeanalizować pliku apk (jest niekompatybilny lub częściowo pobrany)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Udostępnij nowe aplikacje za pomocą weryfikatora aplikacji (jeśli jest dostępny)", | ||||
|     "appVerifierInstructionToast": "Udostępnij do weryfikatora aplikacji, a następnie wróć tutaj.", | ||||
|     "wiki": "Pomoc/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfiguracje aplikacji pochodzące z crowdsourcingu (korzystanie na własne ryzyko)", | ||||
|     "crowdsourcedConfigsShort": "Konfiguracje aplikacji pochodzące z crowdsourcingu", | ||||
|     "crowdsourcedConfigsLabel": "Baza konfiguracji", | ||||
|     "crowdsourcedConfigsShort": "Baza konfiguracji", | ||||
|     "allowInsecure": "Zezwalaj na niezabezpieczone żądania HTTP", | ||||
|     "stayOneVersionBehind": "Pozostań jedną wersję za najnowszą", | ||||
|     "stayOneVersionBehind": "Pozostań jedną wersję w tyle za najnowszą", | ||||
|     "refreshBeforeDownload": "Odśwież szczegóły aplikacji przed pobraniem", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Usunąć aplikację?", | ||||
|         "few": "Usunąć aplikacje?", | ||||
| @@ -392,8 +391,10 @@ | ||||
|         "other": "{} i {} inne apki zostały zaktualizowane." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Nie udało się zaktualizować {} i 1 innej aplikacji.", | ||||
|         "other": "Nie udało się zaktualizować {} i {} więcej aplikacji." | ||||
|         "one": "Błąd aktualizacji {} i 1 innej apki.", | ||||
|         "few": "Błąd aktualizacji {} i {} innych apek.", | ||||
|         "many": "Błąd aktualizacji {} i {} innych apek.", | ||||
|         "other": "Błąd aktualizacji {} i {} innych apek." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} i 1 inna apka mogły zostać zaktualizowane.", | ||||
| @@ -402,7 +403,9 @@ | ||||
|         "other": "{} i {} inne apki mogły zostać zaktualizowane." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK" | ||||
|         "one": "{} apk", | ||||
|         "few": "{} apki", | ||||
|         "many": "{} apek", | ||||
|         "other": "{} apki" | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Nenhum aplicativo foi encontrado com esse ID ou nome", | ||||
|     "reposHaveMultipleApps": "Repositórios podem conter múltiplos aplicativos", | ||||
|     "fdroidThirdPartyRepo": "Repositórios de terceiros F-Droid", | ||||
|     "steamMobile": "Steam para celular", | ||||
|     "steamChat": "Chat do Steam", | ||||
|     "install": "Instalar", | ||||
|     "markInstalled": "Marcar instalado", | ||||
|     "update": "Atualizar", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Configurações de aplicações com base em crowdsourcing", | ||||
|     "allowInsecure": "Permitir pedidos HTTP inseguros", | ||||
|     "stayOneVersionBehind": "Manter-se uma versão atrás da mais recente", | ||||
|     "refreshBeforeDownload": "Atualizar os detalhes da aplicação antes da transferência", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Remover aplicativo?", | ||||
|         "other": "Remover aplicativos?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Приложение с таким ID или названием не было найдено", | ||||
|     "reposHaveMultipleApps": "В хранилище несколько приложений", | ||||
|     "fdroidThirdPartyRepo": "Сторонние репозитории F-Droid", | ||||
|     "steamMobile": "Приложение Steam", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Установить", | ||||
|     "markInstalled": "Пометить как установленное", | ||||
|     "update": "Обновить", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Конфиги приложений с помощью краудсорсинга", | ||||
|     "allowInsecure": "Разрешить небезопасные HTTP-запросы", | ||||
|     "stayOneVersionBehind": "Не отставайте от последней версии", | ||||
|     "refreshBeforeDownload": "Обновляйте информацию о приложении перед загрузкой", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Удалить приложение?", | ||||
|         "other": "Удалить приложения?" | ||||
|   | ||||
| @@ -12,7 +12,8 @@ const neverAutoTranslate = { | ||||
|     root: ['*'], | ||||
|     obtainiumExportHyphenatedLowercase: ['*'], | ||||
|     theme: ['de'], | ||||
|     appId: ['de'] | ||||
|     appId: ['de'], | ||||
|     placeholder: ['pl'] | ||||
| } | ||||
|  | ||||
| const translateText = async (text, targetLang, authKey) => { | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Ingen App funnen med det namnet eller ID", | ||||
|     "reposHaveMultipleApps": "Förråd kan innehålla flera ApparR", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Tredjeparts Förråd", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Installera", | ||||
|     "markInstalled": "Märk Installerad", | ||||
|     "update": "Uppdatera", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Appkonfigurationer med hjälp av crowdsourcing", | ||||
|     "allowInsecure": "Tillåt osäkra HTTP-förfrågningar", | ||||
|     "stayOneVersionBehind": "Håll dig en version bakom den senaste", | ||||
|     "refreshBeforeDownload": "Uppdatera appdetaljerna före nedladdning", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Ta Bort App?", | ||||
|         "other": "Ta Bort Appar?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Bu kimlik veya ada sahip bir uygulama bulunamadı", | ||||
|     "reposHaveMultipleApps": "Depolar birden fazla uygulama içerebilir", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Üçüncü Parti Depo", | ||||
|     "steamMobile": "Steam Mobil", | ||||
|     "steamChat": "Steam Sohbet", | ||||
|     "install": "Yükle", | ||||
|     "markInstalled": "Yüklendi olarak İşaretle", | ||||
|     "update": "Güncelle", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Kitle Kaynaklı Uygulama Yapılandırmaları", | ||||
|     "allowInsecure": "Güvensiz HTTP isteklerine izin ver", | ||||
|     "stayOneVersionBehind": "En son sürümün bir sürüm gerisinde kalın", | ||||
|     "refreshBeforeDownload": "İndirmeden önce uygulama ayrıntılarını yenileyin", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Uygulamayı Kaldır?", | ||||
|         "other": "Uygulamaları Kaldır?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Застосунок з таким ідентифікатором або назвою не знайдено", | ||||
|     "reposHaveMultipleApps": "Сховища можуть містити кілька застосунків", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Стороннє сховище", | ||||
|     "steamMobile": "Мобільний Steam", | ||||
|     "steamChat": "Чат Steam", | ||||
|     "install": "Встановити", | ||||
|     "markInstalled": "Позначити як встановлене", | ||||
|     "update": "Оновити", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Налаштування краудсорсингових додатків", | ||||
|     "allowInsecure": "Дозволити незахищені HTTP-запити", | ||||
|     "stayOneVersionBehind": "Залишайтеся на одну версію актуальнішою", | ||||
|     "refreshBeforeDownload": "Оновіть інформацію про програму перед завантаженням", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Видалити застосунок?", | ||||
|         "other": "Видалити застосунки?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "Không tìm thấy ứng dụng nào có ID hoặc tên đó", | ||||
|     "reposHaveMultipleApps": "Kho có thể chứa nhiều Ứng dụng", | ||||
|     "fdroidThirdPartyRepo": "Kho lưu trữ bên thứ ba F-Droid", | ||||
|     "steamMobile": "Điện thoại di động hơi nước", | ||||
|     "steamChat": "Trò chuyện hơi nước", | ||||
|     "install": "Cài đặt", | ||||
|     "markInstalled": "Đánh dấu là đã cài đặt", | ||||
|     "update": "Cập nhật", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Gỡ ứng dụng?", | ||||
|         "other": "Gỡ ứng dụng?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "找不到具有該 ID 或名稱的應用程式", | ||||
|     "reposHaveMultipleApps": "倉庫可能包含多個應用程式", | ||||
|     "fdroidThirdPartyRepo": "F-Droid 第三方倉庫", | ||||
|     "steamMobile": "Steam 行動版", | ||||
|     "steamChat": "Steam 聊天", | ||||
|     "install": "安裝", | ||||
|     "markInstalled": "標記為已安裝", | ||||
|     "update": "更新", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "移除應用程式?", | ||||
|         "other": "移除應用程式?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "未找到符合此 ID 或名称的应用", | ||||
|     "reposHaveMultipleApps": "存储库中可能包含多个应用", | ||||
|     "fdroidThirdPartyRepo": "F-Droid 第三方存储库", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "安装", | ||||
|     "markInstalled": "标记为已安装", | ||||
|     "update": "更新", | ||||
| @@ -188,10 +186,10 @@ | ||||
|     "additionalOptions": "附加选项", | ||||
|     "disableVersionDetection": "禁用版本检测", | ||||
|     "noVersionDetectionExplanation": "此选项应该仅用于无法进行版本检测的应用。", | ||||
|     "downloadingX": "正在下载“{}”", | ||||
|     "downloadingX": "正在下载 {}", | ||||
|     "downloadX": "下载 {}", | ||||
|     "downloadedX": "下载 {}", | ||||
|     "releaseAsset": "发行版附件", | ||||
|     "downloadedX": "已下载 {}", | ||||
|     "releaseAsset": "发行文件", | ||||
|     "downloadNotifDescription": "提示应用的下载进度", | ||||
|     "noAPKFound": "未找到 APK 文件", | ||||
|     "noVersionDetection": "禁用版本检测", | ||||
| @@ -319,6 +317,7 @@ | ||||
|     "crowdsourcedConfigsShort": "众包应用程序配置", | ||||
|     "allowInsecure": "允许不安全的 HTTP 请求", | ||||
|     "stayOneVersionBehind": "比最新版本晚一个版本", | ||||
|     "refreshBeforeDownload": "下载前刷新应用程序详细信息", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "是否删除应用?", | ||||
|         "other": "是否删除应用?" | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| <p>Obtainium allows you to install and update Apps directly from their releases pages, and receive notifications when new releases are made available.</p> | ||||
| <p>Read the <a href="https://github.com/ImranR98/Obtainium/wiki">Wiki</a></p> | ||||
| <p>Read the <a href="https://wiki.obtainium.imranr.dev/">Wiki</a></p> | ||||
| <p> | ||||
| 	<b>Currently supported App sources:</b> | ||||
| </p> | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| <p>Obtainium позволяет вам устанавливать и обновлять приложения прямо с их объявлений о выпусках и получать уведомления о новых выпусках.</p> | ||||
| <p>Для деталей читайте <a href="https://github.com/ImranR98/Obtainium/wiki">Вики</a></p> | ||||
| <p>Для деталей читайте <a href="https://wiki.obtainium.imranr.dev/">Вики</a></p> | ||||
| <p> | ||||
| 	<b>Поддерживаемые источники приложений:</b> | ||||
| </p> | ||||
|   | ||||
| @@ -5,6 +5,8 @@ import 'package:html/parser.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/apps_provider.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class APKMirror extends AppSource { | ||||
| @@ -31,6 +33,16 @@ class APKMirror extends AppSource { | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, String>?> getRequestHeaders( | ||||
|       Map<String, dynamic> additionalSettings, | ||||
|       {bool forAPKDownload = false}) async { | ||||
|     return { | ||||
|       "User-Agent": | ||||
|           "Obtainium/${(await getInstalledInfo(obtainiumId))?.versionName ?? '1.0.0'}" | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|   | ||||
| @@ -105,11 +105,7 @@ class APKPure extends AppSource { | ||||
|                         .map((e) => e.text.trim()) | ||||
|                         .map((t) => t == 'APKs' ? 'APK' : t) ?? | ||||
|                     []; | ||||
|                 String type = types.isEmpty | ||||
|                     ? 'APK' | ||||
|                     : types.length == 1 | ||||
|                         ? types.first | ||||
|                         : types.last; | ||||
|                 String type = types.isEmpty ? 'APK' : types.first; | ||||
|                 String? dateString = apkInfo | ||||
|                     ?.querySelector('div.info-bottom span.time') | ||||
|                     ?.text | ||||
| @@ -135,12 +131,18 @@ class APKPure extends AppSource { | ||||
|         throw NoAPKError(); | ||||
|       } | ||||
|       String version = Uri.parse(link).pathSegments.last; | ||||
|       String author = html | ||||
|       String? author; | ||||
|       try { | ||||
|         author = html | ||||
|                 .querySelector('span.info-sdk') | ||||
|                 ?.text | ||||
|                 .trim() | ||||
|                 .substring(version.length + 4) ?? | ||||
|             Uri.parse(standardUrl).pathSegments.reversed.last; | ||||
|       } catch (e) { | ||||
|         author = html.querySelector('span.info-sdk')?.text.trim() ?? | ||||
|             Uri.parse(standardUrl).pathSegments.reversed.last; | ||||
|       } | ||||
|       String appName = | ||||
|           html.querySelector('h1.info-title')?.text.trim() ?? appId; | ||||
|       String? changeLog = html | ||||
| @@ -164,9 +166,15 @@ class APKPure extends AppSource { | ||||
|     String host = Uri.parse(standardUrl).host; | ||||
|  | ||||
|     var res0 = await sourceRequest('$standardUrl/versions', additionalSettings); | ||||
|     var decodedStandardUrl = standardUrl; | ||||
|     try { | ||||
|       decodedStandardUrl = Uri.decodeFull(decodedStandardUrl); | ||||
|     } catch (e) { | ||||
|       // | ||||
|     } | ||||
|     var versionLinks = await grabLinksCommon(res0, { | ||||
|       'skipSort': true, | ||||
|       'customLinkFilterRegex': '${Uri.decodeFull(standardUrl)}/download/[^/]+\$' | ||||
|       'customLinkFilterRegex': '$decodedStandardUrl/download/[^/]+\$' | ||||
|     }); | ||||
|  | ||||
|     var supportedArchs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|   | ||||
| @@ -45,7 +45,7 @@ class FDroid extends AppSource { | ||||
|     RegExpMatch? match = standardUrlRegExB.firstMatch(url); | ||||
|     if (match != null) { | ||||
|       url = | ||||
|           'https://${Uri.parse(match.group(0)!).host}/packages/${Uri.parse(url).pathSegments.last}'; | ||||
|           'https://${Uri.parse(match.group(0)!).host}/packages/${Uri.parse(url).pathSegments.where((s) => s.trim().isNotEmpty).last}'; | ||||
|     } | ||||
|     RegExp standardUrlRegExA = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+packages/+[^/]+', | ||||
|   | ||||
| @@ -275,14 +275,15 @@ class GitHub extends AppSource { | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       List<MapEntry<String, String>> getReleaseAssetUrls(dynamic release) => | ||||
|       findReleaseAssetUrls(dynamic release) => | ||||
|           (release['assets'] as List<dynamic>?)?.map((e) { | ||||
|             var url = !e['name'].toString().toLowerCase().endsWith('.apk') | ||||
|                 ? (e['browser_download_url'] ?? e['url']) | ||||
|                 : (e['url'] ?? e['browser_download_url']); | ||||
|             return (e['name'] != null) && (url != null) | ||||
|             e['final_url'] = (e['name'] != null) && (url != null) | ||||
|                 ? MapEntry(e['name'] as String, url as String) | ||||
|                 : const MapEntry('', ''); | ||||
|             return e; | ||||
|           }).toList() ?? | ||||
|           []; | ||||
|  | ||||
| @@ -293,7 +294,9 @@ class GitHub extends AppSource { | ||||
|                   ? DateTime.parse(rel['commit']['created']) | ||||
|                   : null; | ||||
|       DateTime? getNewestAssetDateFromRelease(dynamic rel) { | ||||
|         var t = (rel['assets'] as List<dynamic>?) | ||||
|         var allAssets = rel['assets'] as List<dynamic>?; | ||||
|         var filteredAssets = rel['filteredAssets'] as List<dynamic>?; | ||||
|         var t = (filteredAssets ?? allAssets) | ||||
|             ?.map((e) { | ||||
|               return e?['updated_at'] != null | ||||
|                   ? DateTime.parse(e['updated_at']) | ||||
| @@ -387,18 +390,37 @@ class GitHub extends AppSource { | ||||
|                 .hasMatch(((releases[i]['body'] as String?) ?? '').trim())) { | ||||
|           continue; | ||||
|         } | ||||
|         var allAssetUrls = getReleaseAssetUrls(releases[i]); | ||||
|         List<MapEntry<String, String>> apkUrls = allAssetUrls | ||||
|             .where((element) => element.key.toLowerCase().endsWith('.apk')) | ||||
|         var allAssetsWithUrls = findReleaseAssetUrls(releases[i]); | ||||
|         List<MapEntry<String, String>> allAssetUrls = allAssetsWithUrls | ||||
|             .map((e) => e['final_url'] as MapEntry<String, String>) | ||||
|             .toList(); | ||||
|         var apkAssetsWithUrls = allAssetsWithUrls | ||||
|             .where((element) => | ||||
|                 (element['final_url'] as MapEntry<String, String>) | ||||
|                     .key | ||||
|                     .toLowerCase() | ||||
|                     .endsWith('.apk')) | ||||
|             .toList(); | ||||
|  | ||||
|         apkUrls = filterApks(apkUrls, additionalSettings['apkFilterRegEx'], | ||||
|         var filteredApkUrls = filterApks( | ||||
|             apkAssetsWithUrls | ||||
|                 .map((e) => e['final_url'] as MapEntry<String, String>) | ||||
|                 .toList(), | ||||
|             additionalSettings['apkFilterRegEx'], | ||||
|             additionalSettings['invertAPKFilter']); | ||||
|         if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) { | ||||
|         var filteredApks = apkAssetsWithUrls | ||||
|             .where((e) => filteredApkUrls | ||||
|                 .where((e2) => | ||||
|                     e2.key == (e['final_url'] as MapEntry<String, String>).key) | ||||
|                 .isNotEmpty) | ||||
|             .toList(); | ||||
|  | ||||
|         if (filteredApks.isEmpty && additionalSettings['trackOnly'] != true) { | ||||
|           continue; | ||||
|         } | ||||
|         targetRelease = releases[i]; | ||||
|         targetRelease['apkUrls'] = apkUrls; | ||||
|         targetRelease['apkUrls'] = filteredApkUrls; | ||||
|         targetRelease['filteredAssets'] = filteredApks; | ||||
|         targetRelease['version'] = | ||||
|             additionalSettings['releaseTitleAsVersion'] == true | ||||
|                 ? nameToFilter | ||||
| @@ -420,6 +442,7 @@ class GitHub extends AppSource { | ||||
|         throw NoReleasesError(); | ||||
|       } | ||||
|       String? version = targetRelease['version']; | ||||
|  | ||||
|       DateTime? releaseDate = getReleaseDateFromRelease( | ||||
|           targetRelease, useLatestAssetDateAsReleaseDate); | ||||
|       if (version == null) { | ||||
|   | ||||
| @@ -7,6 +7,9 @@ import 'package:obtainium/providers/apps_provider.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| String ensureAbsoluteUrl(String ambiguousUrl, Uri referenceAbsoluteUrl) { | ||||
|   if (ambiguousUrl.startsWith('//')) { | ||||
|     ambiguousUrl = '${referenceAbsoluteUrl.scheme}:$ambiguousUrl'; | ||||
|   } | ||||
|   try { | ||||
|     Uri.parse(ambiguousUrl).origin; | ||||
|     return ambiguousUrl; | ||||
| @@ -18,7 +21,7 @@ String ensureAbsoluteUrl(String ambiguousUrl, Uri referenceAbsoluteUrl) { | ||||
|       .where((element) => element.trim().isNotEmpty) | ||||
|       .toList(); | ||||
|   String absoluteUrl; | ||||
|   if (ambiguousUrl.startsWith('/') || currPathSegments.isEmpty) { | ||||
|   if (ambiguousUrl.startsWith('/')) { | ||||
|     absoluteUrl = '${referenceAbsoluteUrl.origin}$ambiguousUrl'; | ||||
|   } else if (currPathSegments.isEmpty) { | ||||
|     absoluteUrl = '${referenceAbsoluteUrl.origin}/$ambiguousUrl'; | ||||
| @@ -353,7 +356,12 @@ class HTML extends AppSource { | ||||
|                     forAPKDownload: true), | ||||
|                 allowInsecure: additionalSettings['allowInsecure'] == true)) | ||||
|             .toString(); | ||||
|     return APKDetails(version, [rel].map((e) => MapEntry(e, e)).toList(), | ||||
|     return APKDetails( | ||||
|         version, | ||||
|         [rel] | ||||
|             .map((e) => | ||||
|                 MapEntry('${e.hashCode}-${Uri.parse(e).pathSegments.last}', e)) | ||||
|             .toList(), | ||||
|         AppNames(uri.host, tr('app'))); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,37 +0,0 @@ | ||||
| import 'dart:convert'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class Signal extends AppSource { | ||||
|   Signal() { | ||||
|     hosts = ['signal.org']; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     Response res = await sourceRequest( | ||||
|         'https://updates.${hosts[0]}/android/latest.json', additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       var json = jsonDecode(res.body); | ||||
|       String? apkUrl = json['url']; | ||||
|       List<String> apkUrls = apkUrl == null ? [] : [apkUrl]; | ||||
|       String? version = json['versionName']; | ||||
|       if (version == null) { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|       return APKDetails( | ||||
|           version, getApkUrlsFromUrls(apkUrls), AppNames(name, 'Signal')); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,63 +0,0 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/parser.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 SteamMobile extends AppSource { | ||||
|   SteamMobile() { | ||||
|     hosts = ['store.steampowered.com']; | ||||
|     name = 'Steam'; | ||||
|     additionalSourceAppSpecificSettingFormItems = [ | ||||
|       [ | ||||
|         GeneratedFormDropdown('app', apks.entries.toList(), | ||||
|             label: tr('app'), defaultValue: apks.entries.toList()[0].key) | ||||
|       ] | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   final apks = {'steam': tr('steamMobile'), 'steam-chat-app': tr('steamChat')}; | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     Response res = | ||||
|         await sourceRequest('https://${hosts[0]}/mobile', additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       var apkNamePrefix = additionalSettings['app'] as String?; | ||||
|       if (apkNamePrefix == null) { | ||||
|         throw NoReleasesError(); | ||||
|       } | ||||
|       String apkInURLRegexPattern = | ||||
|           '/$apkNamePrefix-([0-9]+\\.)*[0-9]+\\.apk\$'; | ||||
|       var links = parse(res.body) | ||||
|           .querySelectorAll('a') | ||||
|           .map((e) => e.attributes['href'] ?? '') | ||||
|           .where((e) => RegExp('https://.*$apkInURLRegexPattern').hasMatch(e)) | ||||
|           .toList(); | ||||
|  | ||||
|       if (links.isEmpty) { | ||||
|         throw NoReleasesError(); | ||||
|       } | ||||
|       var versionMatch = RegExp(apkInURLRegexPattern).firstMatch(links[0]); | ||||
|       if (versionMatch == null) { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|       var version = links[0].substring( | ||||
|           versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4); | ||||
|       var apkUrls = [links[0]]; | ||||
|       return APKDetails(version, getApkUrlsFromUrls(apkUrls), | ||||
|           AppNames(name, apks[apkNamePrefix]!)); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -33,7 +33,9 @@ class TelegramApp extends AppSource { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|       String? apkUrl = 'https://telegram.org/dl/android/apk'; | ||||
|       return APKDetails(version, getApkUrlsFromUrls([apkUrl]), | ||||
|       return APKDetails( | ||||
|           version, | ||||
|           [MapEntry<String, String>('telegram-$version.apk', apkUrl)], | ||||
|           AppNames('Telegram', 'Telegram')); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|   | ||||
| @@ -42,19 +42,16 @@ class Uptodown extends AppSource { | ||||
|     String? version = html.querySelector('div.version')?.innerHtml; | ||||
|     String? name = html.querySelector('#detail-app-name')?.innerHtml.trim(); | ||||
|     String? author = html.querySelector('#author-link')?.innerHtml.trim(); | ||||
|     var detailElements = html.querySelectorAll('#technical-information td'); | ||||
|     String? appId = (detailElements.elementAtOrNull(2))?.innerHtml.trim(); | ||||
|     String? dateStr = (detailElements.elementAtOrNull(29))?.innerHtml.trim(); | ||||
|     var detailElements = html | ||||
|         .querySelectorAll('#technical-information td') | ||||
|         .map((e) => e.innerHtml.trim()) | ||||
|         .where((e) => !e.startsWith('<')) | ||||
|         .toList(); | ||||
|     String? appId = detailElements.elementAtOrNull(0); | ||||
|     String? dateStr = detailElements.elementAtOrNull(6); | ||||
|     String? fileId = | ||||
|         html.querySelector('#detail-app-name')?.attributes['data-file-id']; | ||||
|     String? extension = html | ||||
|         .querySelectorAll('td') | ||||
|         .where((e) => e.text.toLowerCase().trim() == 'file type') | ||||
|         .firstOrNull | ||||
|         ?.nextElementSibling | ||||
|         ?.text | ||||
|         .toLowerCase() | ||||
|         .trim(); | ||||
|     String? extension = detailElements.elementAtOrNull(7)?.toLowerCase(); | ||||
|     return Map.fromEntries([ | ||||
|       MapEntry('version', version), | ||||
|       MapEntry('appId', appId), | ||||
|   | ||||
| @@ -1,110 +0,0 @@ | ||||
| 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 VLC extends AppSource { | ||||
|   VLC() { | ||||
|     hosts = ['videolan.org']; | ||||
|   } | ||||
|   get dwUrlBase => 'https://get.${hosts[0]}/vlc-android/'; | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, String>?> getRequestHeaders( | ||||
|       Map<String, dynamic> additionalSettings, | ||||
|       {bool forAPKDownload = false}) async { | ||||
|     return { | ||||
|       "User-Agent": | ||||
|           "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Mobile Safari/537.36" | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   Future<String?> getLatestVersion( | ||||
|       String standardUrl, Map<String, dynamic> additionalSettings) async { | ||||
|     Response res = await sourceRequest(dwUrlBase, additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       var dwLinks = parse(res.body) | ||||
|           .querySelectorAll('a') | ||||
|           .where((element) => element.attributes['href'] != 'last/') | ||||
|           .map((e) => e.attributes['href']?.split('/')[0]) | ||||
|           .toList(); | ||||
|       String? version = dwLinks.isNotEmpty ? dwLinks.last : null; | ||||
|       if (version == null) { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|       return version; | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     Response res = await get( | ||||
|         Uri.parse('https://www.videolan.org/vlc/download-android.html')); | ||||
|     if (res.statusCode == 200) { | ||||
|       var dwUrlBase = 'get.videolan.org/vlc-android'; | ||||
|       var dwLinks = parse(res.body) | ||||
|           .querySelectorAll('a') | ||||
|           .where((element) => | ||||
|               element.attributes['href']?.contains(dwUrlBase) ?? false) | ||||
|           .toList(); | ||||
|       String? version = dwLinks.isNotEmpty | ||||
|           ? dwLinks.first.attributes['href'] | ||||
|               ?.split('/') | ||||
|               .where((s) => s.isNotEmpty) | ||||
|               .last | ||||
|           : null; | ||||
|       if (version == null) { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|       String? targetUrl = 'https://$dwUrlBase/$version/'; | ||||
|       var apkUrls = ['arm64-v8a', 'armeabi-v7a', 'x86', 'x86_64'] | ||||
|           .map((e) => '${targetUrl}VLC-Android-$version-$e.apk') | ||||
|           .toList(); | ||||
|       return APKDetails( | ||||
|           version, getApkUrlsFromUrls(apkUrls), AppNames('VideoLAN', 'VLC')); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String> apkUrlPrefetchModifier(String apkUrl, String standardUrl, | ||||
|       Map<String, dynamic> additionalSettings) async { | ||||
|     Response res = await sourceRequest(apkUrl, additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       String? apkUrl = | ||||
|           parse(res.body).querySelector('#alt_link')?.attributes['href']; | ||||
|       if (apkUrl == null) { | ||||
|         throw NoAPKError(); | ||||
|       } | ||||
|       return apkUrl; | ||||
|     } else if (res.statusCode == 500 && | ||||
|         res.body.toLowerCase().indexOf('mirror') > 0) { | ||||
|       var html = parse(res.body); | ||||
|       var err = ''; | ||||
|       html.body?.nodes.forEach((element) { | ||||
|         if (element.text != null) { | ||||
|           err += '${element.text}\n'; | ||||
|         } | ||||
|       }); | ||||
|       err = err.trim(); | ||||
|       if (err.isEmpty) { | ||||
|         err = tr('err'); | ||||
|       } | ||||
|       throw ObtainiumError(err); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,55 +0,0 @@ | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class WhatsApp extends AppSource { | ||||
|   WhatsApp() { | ||||
|     hosts = ['whatsapp.com']; | ||||
|     versionDetectionDisallowed = true; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String> apkUrlPrefetchModifier(String apkUrl, String standardUrl, | ||||
|       Map<String, dynamic> additionalSettings) async { | ||||
|     Response res = | ||||
|         await sourceRequest('$standardUrl/android', additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       var targetLinks = parse(res.body) | ||||
|           .querySelectorAll('a') | ||||
|           .map((e) => e.attributes['href'] ?? '') | ||||
|           .where((e) => e.isNotEmpty) | ||||
|           .where((e) => e.contains('WhatsApp.apk')) | ||||
|           .toList(); | ||||
|       if (targetLinks.isEmpty) { | ||||
|         throw NoAPKError(); | ||||
|       } | ||||
|       return targetLinks[0]; | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     // This is a CDN link that is consistent per version | ||||
|     // But it has query params that change constantly | ||||
|     Uri apkUri = Uri.parse(await apkUrlPrefetchModifier( | ||||
|         standardUrl, standardUrl, additionalSettings)); | ||||
|     var unusableApkUrl = '${apkUri.origin}/${apkUri.path}'; | ||||
|     // So we use the param-less URL is a pseudo-version to add the app and check for updates | ||||
|     // See #357 for why we can't scrape the version number directly | ||||
|     // But we re-fetch the URL again with its latest query params at the actual download time | ||||
|     String version = unusableApkUrl.hashCode.toString(); | ||||
|     return APKDetails(version, getApkUrlsFromUrls([unusableApkUrl]), | ||||
|         AppNames('Meta', 'WhatsApp')); | ||||
|   } | ||||
| } | ||||
| @@ -538,6 +538,38 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|                       }); | ||||
|                     } | ||||
|                   }), | ||||
|             if (pickedSource != null && pickedSource!.enforceTrackOnly) | ||||
|               GeneratedForm( | ||||
|                   key: Key( | ||||
|                       '${pickedSource.runtimeType.toString()}-${pickedSource?.hostChanged.toString()}-${pickedSource?.hostIdenticalDespiteAnyChange.toString()}-appId'), | ||||
|                   items: [ | ||||
|                     [ | ||||
|                       GeneratedFormTextField('appId', | ||||
|                           label: '${tr('appId')} - ${tr('custom')}', | ||||
|                           required: false, | ||||
|                           additionalValidators: [ | ||||
|                             (value) { | ||||
|                               if (value == null || value.isEmpty) { | ||||
|                                 return null; | ||||
|                               } | ||||
|                               final isValid = RegExp( | ||||
|                                       r'^([A-Za-z]{1}[A-Za-z\d_]*\.)+[A-Za-z][A-Za-z\d_]*$') | ||||
|                                   .hasMatch(value); | ||||
|                               if (!isValid) { | ||||
|                                 return tr('invalidInput'); | ||||
|                               } | ||||
|                               return null; | ||||
|                             } | ||||
|                           ]), | ||||
|                     ] | ||||
|                   ], | ||||
|                   onValueChanges: (values, valid, isBuilding) { | ||||
|                     if (!isBuilding) { | ||||
|                       setState(() { | ||||
|                         additionalSettings['appId'] = values['appId']; | ||||
|                       }); | ||||
|                     } | ||||
|                   }), | ||||
|           ], | ||||
|         ); | ||||
|  | ||||
|   | ||||
| @@ -838,30 +838,6 @@ class AppsPageState extends State<AppsPage> { | ||||
|       Navigator.of(context).pop(); | ||||
|     } | ||||
|  | ||||
|     resetSelectedAppsInstallStatuses() async { | ||||
|       try { | ||||
|         var values = await showDialog( | ||||
|             context: context, | ||||
|             builder: (BuildContext ctx) { | ||||
|               return GeneratedFormModal( | ||||
|                 title: tr('resetInstallStatusForSelectedAppsQuestion'), | ||||
|                 items: const [], | ||||
|                 initValid: true, | ||||
|                 message: tr('installStatusOfXWillBeResetExplanation', | ||||
|                     args: [plural('apps', selectedAppIds.length)]), | ||||
|               ); | ||||
|             }); | ||||
|         if (values != null) { | ||||
|           appsProvider.saveApps(selectedApps.map((e) { | ||||
|             e.installedVersion = null; | ||||
|             return e; | ||||
|           }).toList()); | ||||
|         } | ||||
|       } finally { | ||||
|         Navigator.of(context).pop(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     showMoreOptionsDialog() { | ||||
|       return showDialog( | ||||
|           context: context, | ||||
|   | ||||
| @@ -130,13 +130,18 @@ class _HomePageState extends State<HomePage> { | ||||
|  | ||||
|     // Check initial link if app was in cold state (terminated) | ||||
|     final appLink = await _appLinks.getInitialLink(); | ||||
|     var initLinked = false; | ||||
|     if (appLink != null) { | ||||
|       await interpretLink(appLink); | ||||
|       initLinked = true; | ||||
|     } | ||||
|  | ||||
|     // Handle link when app is in warm state (front or background) | ||||
|     _linkSubscription = _appLinks.uriLinkStream.listen((uri) async { | ||||
|       if (!initLinked) { | ||||
|         await interpretLink(uri); | ||||
|       } else { | ||||
|         initLinked = false; | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -886,7 +886,7 @@ class _SettingsPageState extends State<SettingsPage> { | ||||
|                     ), | ||||
|                     IconButton( | ||||
|                       onPressed: () { | ||||
|                         launchUrlString('${settingsProvider.sourceUrl}/wiki', | ||||
|                         launchUrlString('https://wiki.obtainium.imranr.dev/', | ||||
|                             mode: LaunchMode.externalApplication); | ||||
|                       }, | ||||
|                       icon: const Icon(Icons.help_outline_rounded), | ||||
|   | ||||
| @@ -19,6 +19,8 @@ import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:http/io_client.dart'; | ||||
| import 'package:obtainium/app_sources/directAPKLink.dart'; | ||||
| import 'package:obtainium/app_sources/html.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/components/generated_form_modal.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| @@ -516,11 +518,29 @@ class AppsProvider with ChangeNotifier { | ||||
|             .listSync() | ||||
|             .where((e) => e.path.toLowerCase().endsWith('.apk')) | ||||
|             .toList(); | ||||
|  | ||||
|         FileSystemEntity? temp; | ||||
|         apks.removeWhere((element) { | ||||
|           bool res = element.uri.pathSegments.last.startsWith(app.id); | ||||
|           if (res) { | ||||
|             temp = element; | ||||
|           } | ||||
|           return res; | ||||
|         }); | ||||
|         if (temp != null) { | ||||
|           apks = [ | ||||
|             temp!, | ||||
|             ...apks, | ||||
|           ]; | ||||
|         } | ||||
|  | ||||
|         for (var i = 0; i < apks.length; i++) { | ||||
|           try { | ||||
|             newInfo = await pm.getPackageArchiveInfo( | ||||
|                 archiveFilePath: apks.first.path); | ||||
|             newInfo = | ||||
|                 await pm.getPackageArchiveInfo(archiveFilePath: apks[i].path); | ||||
|             if (newInfo != null) { | ||||
|               break; | ||||
|             } | ||||
|           } catch (e) { | ||||
|             if (i == apks.length - 1) { | ||||
|               rethrow; | ||||
| @@ -644,28 +664,47 @@ class AppsProvider with ChangeNotifier { | ||||
|     var somethingInstalled = false; | ||||
|     try { | ||||
|       MultiAppMultiError errors = MultiAppMultiError(); | ||||
|       List<File> APKFiles = []; | ||||
|       for (var file in dir.extracted | ||||
|           .listSync(recursive: true, followLinks: false) | ||||
|           .whereType<File>()) { | ||||
|         if (file.path.toLowerCase().endsWith('.apk')) { | ||||
|           try { | ||||
|             somethingInstalled = somethingInstalled || | ||||
|                 await installApk( | ||||
|                     DownloadedApk(dir.appId, file), firstTimeWithContext, | ||||
|                     needsBGWorkaround: needsBGWorkaround, | ||||
|                     shizukuPretendToBeGooglePlay: shizukuPretendToBeGooglePlay); | ||||
|           } catch (e) { | ||||
|             logs.add( | ||||
|                 'Could not install APK from XAPK \'${file.path}\': ${e.toString()}'); | ||||
|             errors.add(dir.appId, e, appName: apps[dir.appId]?.name); | ||||
|           } | ||||
|           APKFiles.add(file); | ||||
|         } else if (file.path.toLowerCase().endsWith('.obb')) { | ||||
|           await moveObbFile(file, dir.appId); | ||||
|         } | ||||
|       } | ||||
|       if (somethingInstalled) { | ||||
|  | ||||
|       File? temp; | ||||
|       APKFiles.removeWhere((element) { | ||||
|         bool res = element.uri.pathSegments.last.startsWith(dir.appId); | ||||
|         if (res) { | ||||
|           temp = element; | ||||
|         } | ||||
|         return res; | ||||
|       }); | ||||
|       if (temp != null) { | ||||
|         APKFiles = [ | ||||
|           temp!, | ||||
|           ...APKFiles, | ||||
|         ]; | ||||
|       } | ||||
|  | ||||
|       try { | ||||
|         await installApk( | ||||
|             DownloadedApk(dir.appId, APKFiles[0]), firstTimeWithContext, | ||||
|             needsBGWorkaround: needsBGWorkaround, | ||||
|             shizukuPretendToBeGooglePlay: shizukuPretendToBeGooglePlay, | ||||
|             additionalAPKs: APKFiles.sublist(1) | ||||
|                 .map((a) => DownloadedApk(dir.appId, a)) | ||||
|                 .toList()); | ||||
|         somethingInstalled = true; | ||||
|         dir.file.delete(recursive: true); | ||||
|       } else if (errors.idsByErrorString.isNotEmpty) { | ||||
|       } catch (e) { | ||||
|         logs.add('Could not install APKs from XAPK: ${e.toString()}'); | ||||
|         errors.add(dir.appId, e, appName: apps[dir.appId]?.name); | ||||
|       } | ||||
|       if (errors.idsByErrorString.isNotEmpty) { | ||||
|         throw errors; | ||||
|       } | ||||
|     } finally { | ||||
| @@ -677,7 +716,8 @@ class AppsProvider with ChangeNotifier { | ||||
|   Future<bool> installApk( | ||||
|       DownloadedApk file, BuildContext? firstTimeWithContext, | ||||
|       {bool needsBGWorkaround = false, | ||||
|       bool shizukuPretendToBeGooglePlay = false}) async { | ||||
|       bool shizukuPretendToBeGooglePlay = false, | ||||
|       List<DownloadedApk> additionalAPKs = const []}) async { | ||||
|     if (firstTimeWithContext != null && | ||||
|         settingsProvider.beforeNewInstallsShareToAppVerifier && | ||||
|         (await getInstalledInfo('dev.soupslurpr.appverifier')) != null) { | ||||
| @@ -693,6 +733,7 @@ class AppsProvider with ChangeNotifier { | ||||
|     if (newInfo == null) { | ||||
|       try { | ||||
|         file.file.deleteSync(recursive: true); | ||||
|         additionalAPKs.forEach((a) => a.file.deleteSync(recursive: true)); | ||||
|       } catch (e) { | ||||
|         // | ||||
|       } finally { | ||||
| @@ -700,6 +741,8 @@ class AppsProvider with ChangeNotifier { | ||||
|       } | ||||
|     } | ||||
|     PackageInfo? appInfo = await getInstalledInfo(apps[file.appId]!.app.id); | ||||
|     logs.add( | ||||
|         'Installing "${newInfo.packageName}" version "${newInfo.versionName}" versionCode "${newInfo.versionCode}"${appInfo != null ? ' (from existing version "${appInfo.versionName}" versionCode "${appInfo.versionCode}")' : ''}'); | ||||
|     if (appInfo != null && | ||||
|         newInfo.versionCode! < appInfo.versionCode! && | ||||
|         !(await canDowngradeApps())) { | ||||
| @@ -718,8 +761,10 @@ class AppsProvider with ChangeNotifier { | ||||
|     } | ||||
|     int? code; | ||||
|     if (!settingsProvider.useShizuku) { | ||||
|       code = | ||||
|           await AndroidPackageInstaller.installApk(apkFilePath: file.file.path); | ||||
|       var allAPKs = [file.file.path]; | ||||
|       allAPKs.addAll(additionalAPKs.map((a) => a.file.path)); | ||||
|       code = await AndroidPackageInstaller.installApk( | ||||
|           apkFilePath: allAPKs.join(',')); | ||||
|     } else { | ||||
|       code = await ShizukuApkInstaller.installAPK(file.file.uri.toString(), | ||||
|           shizukuPretendToBeGooglePlay ? "com.android.vending" : ""); | ||||
| @@ -743,6 +788,10 @@ class AppsProvider with ChangeNotifier { | ||||
|     return installed; | ||||
|   } | ||||
|  | ||||
|   Future<String> getStorageRootPath() async { | ||||
|     return '/${(await getExternalStorageDirectory())!.uri.pathSegments.sublist(0, 3).join('/')}'; | ||||
|   } | ||||
|  | ||||
|   Future<void> moveObbFile(File file, String appId) async { | ||||
|     if (!file.path.toLowerCase().endsWith('.obb')) return; | ||||
|  | ||||
| @@ -751,7 +800,7 @@ class AppsProvider with ChangeNotifier { | ||||
|       await Permission.storage.request(); | ||||
|     } | ||||
|  | ||||
|     String obbDirPath = "/storage/emulated/0/Android/obb/$appId"; | ||||
|     String obbDirPath = "${await getStorageRootPath()}/Android/obb/$appId"; | ||||
|     Directory(obbDirPath).createSync(recursive: true); | ||||
|  | ||||
|     String obbFileName = file.path.split("/").last; | ||||
| @@ -841,6 +890,11 @@ class AppsProvider with ChangeNotifier { | ||||
|       } | ||||
|       MapEntry<String, String>? apkUrl; | ||||
|       var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true; | ||||
|       var refreshBeforeDownload = | ||||
|           apps[id]!.app.additionalSettings['refreshBeforeDownload'] == true; | ||||
|       if (refreshBeforeDownload) { | ||||
|         await checkUpdate(apps[id]!.app.id); | ||||
|       } | ||||
|       if (!trackOnly) { | ||||
|         // ignore: use_build_context_synchronously | ||||
|         apkUrl = await confirmAppFileUrl(apps[id]!.app, context, false); | ||||
| @@ -1027,11 +1081,25 @@ class AppsProvider with ChangeNotifier { | ||||
|         throw ObtainiumError(tr('appNotFound')); | ||||
|       } | ||||
|       MapEntry<String, String>? fileUrl; | ||||
|       var refreshBeforeDownload = | ||||
|           apps[id]!.app.additionalSettings['refreshBeforeDownload'] == true; | ||||
|       if (refreshBeforeDownload) { | ||||
|         await checkUpdate(apps[id]!.app.id); | ||||
|       } | ||||
|       if (apps[id]!.app.apkUrls.isNotEmpty || | ||||
|           apps[id]!.app.otherAssetUrls.isNotEmpty) { | ||||
|         // ignore: use_build_context_synchronously | ||||
|         fileUrl = await confirmAppFileUrl(apps[id]!.app, context, true, | ||||
|         MapEntry<String, String>? tempFileUrl = await confirmAppFileUrl( | ||||
|             apps[id]!.app, context, true, | ||||
|             evenIfSingleChoice: true); | ||||
|         if (tempFileUrl != null) { | ||||
|           fileUrl = MapEntry( | ||||
|               tempFileUrl.key, | ||||
|               await (SourceProvider().getSource(apps[id]!.app.url, | ||||
|                       overrideSource: apps[id]!.app.overrideSource)) | ||||
|                   .apkUrlPrefetchModifier(tempFileUrl.value, apps[id]!.app.url, | ||||
|                       apps[id]!.app.additionalSettings)); | ||||
|         } | ||||
|       } | ||||
|       if (fileUrl != null) { | ||||
|         filesToDownload.add(MapEntry(fileUrl, apps[id]!.app)); | ||||
| @@ -1044,17 +1112,7 @@ class AppsProvider with ChangeNotifier { | ||||
|  | ||||
|     Future<void> downloadFn(MapEntry<String, String> fileUrl, App app) async { | ||||
|       try { | ||||
|         var exportDir = await settingsProvider.getExportDir(); | ||||
|         String downloadPath = '/storage/emulated/0/Download'; | ||||
|         bool downloadsAccessible = false; | ||||
|         try { | ||||
|           downloadsAccessible = Directory(downloadPath).existsSync(); | ||||
|         } catch (e) { | ||||
|           // | ||||
|         } | ||||
|         if (!downloadsAccessible && exportDir != null) { | ||||
|           downloadPath = exportDir.path; | ||||
|         } | ||||
|         String downloadPath = '${await getStorageRootPath()}/Download'; | ||||
|         await downloadFile(fileUrl.value, fileUrl.key, true, | ||||
|             (double? progress) { | ||||
|           notificationsProvider | ||||
| @@ -1103,17 +1161,25 @@ class AppsProvider with ChangeNotifier { | ||||
|     if (app?.app == null) { | ||||
|       return false; | ||||
|     } | ||||
|     var source = SourceProvider() | ||||
|         .getSource(app!.app.url, overrideSource: app.app.overrideSource); | ||||
|     var naiveStandardVersionDetection = | ||||
|         app!.app.additionalSettings['naiveStandardVersionDetection'] == true || | ||||
|             SourceProvider() | ||||
|                 .getSource(app.app.url, overrideSource: app.app.overrideSource) | ||||
|                 .naiveStandardVersionDetection; | ||||
|         app.app.additionalSettings['naiveStandardVersionDetection'] == true || | ||||
|             source.naiveStandardVersionDetection; | ||||
|     String? realInstalledVersion = | ||||
|         app.app.additionalSettings['useVersionCodeAsOSVersion'] == true | ||||
|             ? app.installedInfo?.versionCode.toString() | ||||
|             : app.installedInfo?.versionName; | ||||
|     bool isHTMLWithNoVersionDetection = | ||||
|         (source.runtimeType == HTML().runtimeType && | ||||
|             (app.app.additionalSettings['versionExtractionRegEx'] as String?) | ||||
|                     ?.isNotEmpty != | ||||
|                 true); | ||||
|     bool isDirectAPKLink = source.runtimeType == DirectAPKLink().runtimeType; | ||||
|     return app.app.additionalSettings['trackOnly'] != true && | ||||
|         app.app.additionalSettings['releaseDateAsVersion'] != true && | ||||
|         !isHTMLWithNoVersionDetection && | ||||
|         !isDirectAPKLink && | ||||
|         realInstalledVersion != null && | ||||
|         app.app.installedVersion != null && | ||||
|         (reconcileVersionDifferences( | ||||
| @@ -1184,6 +1250,7 @@ class AppsProvider with ChangeNotifier { | ||||
|         !isVersionDetectionPossible( | ||||
|             AppInMemory(app, null, installedInfo, null))) { | ||||
|       app.additionalSettings['versionDetection'] = false; | ||||
|       app.installedVersion = app.latestVersion; | ||||
|       logs.add('Could not reconcile version formats for: ${app.id}'); | ||||
|       modded = true; | ||||
|     } | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter_local_notifications/flutter_local_notifications.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class ObtainiumNotification { | ||||
| @@ -44,23 +45,19 @@ class SilentUpdateNotification extends ObtainiumNotification { | ||||
|   SilentUpdateNotification(List<App> updates, bool succeeded, {int? id}) | ||||
|       : super( | ||||
|             id ?? 3, | ||||
|             succeeded | ||||
|                 ? tr('appsUpdated') | ||||
|                 : tr('appsNotUpdated'), | ||||
|             succeeded ? tr('appsUpdated') : tr('appsNotUpdated'), | ||||
|             '', | ||||
|             'APPS_UPDATED', | ||||
|             tr('appsUpdatedNotifChannel'), | ||||
|             tr('appsUpdatedNotifDescription'), | ||||
|             Importance.defaultImportance) { | ||||
|     message = updates.length == 1 | ||||
|         ? tr(succeeded | ||||
|             ? 'xWasUpdatedToY' | ||||
|             : 'xWasNotUpdatedToY', | ||||
|         ? tr(succeeded ? 'xWasUpdatedToY' : 'xWasNotUpdatedToY', | ||||
|             args: [updates[0].finalName, updates[0].latestVersion]) | ||||
|         : plural(succeeded | ||||
|             ? 'xAndNMoreUpdatesInstalled' | ||||
|             : "xAndNMoreUpdatesFailed", | ||||
|                 updates.length - 1, args: [updates[0].finalName, (updates.length - 1).toString()]); | ||||
|         : plural( | ||||
|             succeeded ? 'xAndNMoreUpdatesInstalled' : "xAndNMoreUpdatesFailed", | ||||
|             updates.length - 1, | ||||
|             args: [updates[0].finalName, (updates.length - 1).toString()]); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -214,7 +211,7 @@ class NotificationsProvider { | ||||
|                 channelDescription: channelDescription, | ||||
|                 importance: importance, | ||||
|                 priority: importanceToPriority[importance]!, | ||||
|                 groupKey: 'dev.imranr.obtainium.$channelCode', | ||||
|                 groupKey: '$obtainiumId.$channelCode', | ||||
|                 progress: progPercent ?? 0, | ||||
|                 maxProgress: 100, | ||||
|                 showProgress: progPercent != null, | ||||
|   | ||||
| @@ -23,18 +23,15 @@ import 'package:obtainium/app_sources/izzyondroid.dart'; | ||||
| import 'package:obtainium/app_sources/html.dart'; | ||||
| import 'package:obtainium/app_sources/jenkins.dart'; | ||||
| import 'package:obtainium/app_sources/neutroncode.dart'; | ||||
| import 'package:obtainium/app_sources/signal.dart'; | ||||
| import 'package:obtainium/app_sources/sourceforge.dart'; | ||||
| import 'package:obtainium/app_sources/sourcehut.dart'; | ||||
| import 'package:obtainium/app_sources/steammobile.dart'; | ||||
| import 'package:obtainium/app_sources/telegramapp.dart'; | ||||
| import 'package:obtainium/app_sources/tencent.dart'; | ||||
| import 'package:obtainium/app_sources/uptodown.dart'; | ||||
| import 'package:obtainium/app_sources/vlc.dart'; | ||||
| import 'package:obtainium/app_sources/whatsapp.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/mass_app_sources/githubstars.dart'; | ||||
| import 'package:obtainium/providers/logs_provider.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
|  | ||||
| class AppNames { | ||||
| @@ -155,10 +152,6 @@ appJSONCompatibilityModifiers(Map<String, dynamic> json) { | ||||
|     additionalSettings['autoApkFilterByArch'] = false; | ||||
|   } | ||||
|   if (source.runtimeType == HTML().runtimeType) { | ||||
|     // HTML 'fixed URL' support should be disabled if it previously did not exist | ||||
|     if (originalAdditionalSettings['supportFixedAPKURL'] == null) { | ||||
|       additionalSettings['supportFixedAPKURL'] = false; | ||||
|     } | ||||
|     // HTML key rename | ||||
|     if (originalAdditionalSettings['sortByFileNamesNotLinks'] != null) { | ||||
|       additionalSettings['sortByLastLinkSegment'] = | ||||
| @@ -183,7 +176,7 @@ appJSONCompatibilityModifiers(Map<String, dynamic> json) { | ||||
|       }).toList(); | ||||
|     } | ||||
|     // Steam source apps should be converted to HTML (#1244) | ||||
|     var legacySteamSourceApps = SteamMobile().apks.keys; | ||||
|     var legacySteamSourceApps = ['steam', 'steam-chat-app']; | ||||
|     if (legacySteamSourceApps.contains(additionalSettings['app'] ?? '')) { | ||||
|       json['url'] = '${json['url']}/mobile'; | ||||
|       var replacementAdditionalSettings = getDefaultValuesFromFormItems( | ||||
| @@ -200,6 +193,72 @@ appJSONCompatibilityModifiers(Map<String, dynamic> json) { | ||||
|       replacementAdditionalSettings['matchGroupToUse'] = '\$1'; | ||||
|       additionalSettings = replacementAdditionalSettings; | ||||
|     } | ||||
|     // Signal apps from before it was removed should be converted to HTML (#1928) | ||||
|     if (json['url'] == 'https://signal.org' && | ||||
|         json['id'] == 'org.thoughtcrime.securesms' && | ||||
|         json['author'] == 'Signal' && | ||||
|         json['name'] == 'Signal' && | ||||
|         json['overrideSource'] == null && | ||||
|         additionalSettings['trackOnly'] == false && | ||||
|         additionalSettings['versionExtractionRegEx'] == '' && | ||||
|         json['lastUpdateCheck'] != null) { | ||||
|       json['url'] = 'https://updates.signal.org/android/latest.json'; | ||||
|       var replacementAdditionalSettings = getDefaultValuesFromFormItems( | ||||
|           HTML().combinedAppSpecificSettingFormItems); | ||||
|       replacementAdditionalSettings['versionExtractionRegEx'] = | ||||
|           '\\d+.\\d+.\\d+'; | ||||
|       additionalSettings = replacementAdditionalSettings; | ||||
|     } | ||||
|     // WhatsApp from before it was removed should be converted to HTML (#1943) | ||||
|     if (json['url'] == 'https://whatsapp.com' && | ||||
|         json['id'] == 'com.whatsapp' && | ||||
|         json['author'] == 'Meta' && | ||||
|         json['name'] == 'WhatsApp' && | ||||
|         json['overrideSource'] == null && | ||||
|         additionalSettings['trackOnly'] == false && | ||||
|         additionalSettings['versionExtractionRegEx'] == '' && | ||||
|         json['lastUpdateCheck'] != null) { | ||||
|       json['url'] = 'https://whatsapp.com/android'; | ||||
|       var replacementAdditionalSettings = getDefaultValuesFromFormItems( | ||||
|           HTML().combinedAppSpecificSettingFormItems); | ||||
|       replacementAdditionalSettings['refreshBeforeDownload'] = true; | ||||
|       additionalSettings = replacementAdditionalSettings; | ||||
|     } | ||||
|     // VLC from before it was removed should be converted to HTML (#1943) | ||||
|     if (json['url'] == 'https://videolan.org' && | ||||
|         json['id'] == 'org.videolan.vlc' && | ||||
|         json['author'] == 'VideoLAN' && | ||||
|         json['name'] == 'VLC' && | ||||
|         json['overrideSource'] == null && | ||||
|         additionalSettings['trackOnly'] == false && | ||||
|         additionalSettings['versionExtractionRegEx'] == '' && | ||||
|         json['lastUpdateCheck'] != null) { | ||||
|       json['url'] = 'https://www.videolan.org/vlc/download-android.html'; | ||||
|       var replacementAdditionalSettings = getDefaultValuesFromFormItems( | ||||
|           HTML().combinedAppSpecificSettingFormItems); | ||||
|       replacementAdditionalSettings['refreshBeforeDownload'] = true; | ||||
|       replacementAdditionalSettings['intermediateLink'] = | ||||
|           <Map<String, dynamic>>[ | ||||
|         { | ||||
|           'customLinkFilterRegex': 'APK', | ||||
|           'filterByLinkText': true, | ||||
|           'skipSort': false, | ||||
|           'reverseSort': false, | ||||
|           'sortByLastLinkSegment': false | ||||
|         }, | ||||
|         { | ||||
|           'customLinkFilterRegex': 'arm64-v8a\\.apk\$', | ||||
|           'filterByLinkText': false, | ||||
|           'skipSort': false, | ||||
|           'reverseSort': false, | ||||
|           'sortByLastLinkSegment': false | ||||
|         } | ||||
|       ]; | ||||
|       replacementAdditionalSettings['versionExtractionRegEx'] = | ||||
|           '/vlc-android/([^/]+)/'; | ||||
|       replacementAdditionalSettings['matchGroupToUse'] = "1"; | ||||
|       additionalSettings = replacementAdditionalSettings; | ||||
|     } | ||||
|   } | ||||
|   json['additionalSettings'] = jsonEncode(additionalSettings); | ||||
|   // F-Droid no longer needs cloudflare exception since override can be used - migrate apps appropriately | ||||
| @@ -226,7 +285,7 @@ class App { | ||||
|   late String name; | ||||
|   String? installedVersion; | ||||
|   late String latestVersion; | ||||
|   List<MapEntry<String, String>> apkUrls = []; | ||||
|   List<MapEntry<String, String>> apkUrls = []; // Key is name, value is URL | ||||
|   List<MapEntry<String, String>> otherAssetUrls = []; | ||||
|   late int preferredApkIndex; | ||||
|   late Map<String, dynamic> additionalSettings; | ||||
| @@ -290,7 +349,14 @@ class App { | ||||
|       otherAssetUrls: otherAssetUrls); | ||||
|  | ||||
|   factory App.fromJson(Map<String, dynamic> json) { | ||||
|     Map<String, dynamic> originalJSON = new Map.from(json); | ||||
|     try { | ||||
|       json = appJSONCompatibilityModifiers(json); | ||||
|     } catch (e) { | ||||
|       json = originalJSON; | ||||
|       LogsProvider().add( | ||||
|           'Error running JSON compat modifiers: ${e.toString()}: ${originalJSON.toString()}'); | ||||
|     } | ||||
|     return App( | ||||
|       json['id'] as String, | ||||
|       json['url'] as String, | ||||
| @@ -571,7 +637,11 @@ abstract class AppSource { | ||||
|       GeneratedFormSwitch('skipUpdateNotifications', | ||||
|           label: tr('skipUpdateNotifications')) | ||||
|     ], | ||||
|     [GeneratedFormTextField('about', label: tr('about'), required: false)] | ||||
|     [GeneratedFormTextField('about', label: tr('about'), required: false)], | ||||
|     [ | ||||
|       GeneratedFormSwitch('refreshBeforeDownload', | ||||
|           label: tr('refreshBeforeDownload')) | ||||
|     ] | ||||
|   ]; | ||||
|  | ||||
|   // Previous 2 variables combined into one at runtime for convenient usage | ||||
| @@ -794,9 +864,6 @@ class SourceProvider { | ||||
|         Tencent(), | ||||
|         Jenkins(), | ||||
|         APKMirror(), | ||||
|         Signal(), | ||||
|         VLC(), | ||||
|         WhatsApp(), | ||||
|         TelegramApp(), | ||||
|         NeutronCode(), | ||||
|         DirectAPKLink(), | ||||
| @@ -922,6 +989,9 @@ class SourceProvider { | ||||
|     name = name.isNotEmpty ? name : apk.names.name; | ||||
|     App finalApp = App( | ||||
|         currentApp?.id ?? | ||||
|             ((additionalSettings['appId'] != null) | ||||
|                 ? additionalSettings['appId'] | ||||
|                 : null) ?? | ||||
|             (!trackOnly && | ||||
|                     (!source.appIdInferIsOptional || | ||||
|                         (source.appIdInferIsOptional && inferAppIdIfOptional)) | ||||
|   | ||||
							
								
								
									
										170
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										170
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -5,16 +5,16 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: android_intent_plus | ||||
|       sha256: "38921ec22ebb3b9a7eb678792cf6fab0b6f458b61b9d327688573449c9b47db3" | ||||
|       sha256: "53136214d506d3128c9f4e5bfce3d026abe7e8038958629811a8d3223b1757c1" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.2.0" | ||||
|     version: "5.2.1" | ||||
|   android_package_installer: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       path: "." | ||||
|       ref: main | ||||
|       resolved-ref: ba2aa7a11edc2649d1d80c25ed9291521262f714 | ||||
|       resolved-ref: bcad19e964d377da8816718032e5dbf6dd16ba3a | ||||
|       url: "https://github.com/ImranR98/android_package_installer" | ||||
|     source: git | ||||
|     version: "0.0.1" | ||||
| @@ -47,10 +47,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: app_links | ||||
|       sha256: ad1a6d598e7e39b46a34f746f9a8b011ee147e4c275d407fa457e7a62f84dd99 | ||||
|       sha256: "433df2e61b10519407475d7f69e470789d23d593f28224c38ba1068597be7950" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.3.2" | ||||
|     version: "6.3.3" | ||||
|   app_links_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -79,10 +79,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: archive | ||||
|       sha256: cb6a278ef2dbb298455e1a713bda08524a175630ec643a242c399c932a0a1f7d | ||||
|       sha256: "6199c74e3db4fbfbd04f66d739e72fe11c8a8957d5f219f1f4482dbde6420b5a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.6.1" | ||||
|     version: "4.0.2" | ||||
|   args: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -111,10 +111,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: battery_plus | ||||
|       sha256: "4b6dc87ffa72f8d1e63ae17c8700ee374a462e521f0152e2c76cfff484610764" | ||||
|       sha256: a0409fe7d21905987eb1348ad57c634f913166f14f0c8936b73d3f5940fac551 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.1.0" | ||||
|     version: "6.2.1" | ||||
|   battery_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -151,10 +151,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: cli_util | ||||
|       sha256: c05b7406fdabc7a49a3929d4af76bcaccbbffcbcdcf185b082e1ae07da323d19 | ||||
|       sha256: ff6785f7e9e3c38ac98b2fb035701789de90154024a75b6cb926445e83197d1c | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.4.1" | ||||
|     version: "0.4.2" | ||||
|   clock: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -175,10 +175,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: connectivity_plus | ||||
|       sha256: "876849631b0c7dc20f8b471a2a03142841b482438e3b707955464f5ffca3e4c3" | ||||
|       sha256: e0817759ec6d2d8e57eb234e6e57d2173931367a865850c7acea40d4b4f9c27d | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.1.0" | ||||
|     version: "6.1.1" | ||||
|   connectivity_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -207,10 +207,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: csslib | ||||
|       sha256: "706b5707578e0c1b4b7550f64078f0a0f19dec3f50a178ffae7006b0a9ca58fb" | ||||
|       sha256: "09bad715f418841f976c77db72d5398dc1253c21fb9c0c7f0b0b985860b2d58e" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.0" | ||||
|     version: "1.0.2" | ||||
|   cupertino_icons: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -231,18 +231,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: device_info_plus | ||||
|       sha256: c4af09051b4f0508f6c1dc0a5c085bf014d5c9a4a0678ce1799c2b4d716387a0 | ||||
|       sha256: "4fa68e53e26ab17b70ca39f072c285562cfc1589df5bb1e9295db90f6645f431" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "11.1.0" | ||||
|     version: "11.2.0" | ||||
|   device_info_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: device_info_plus_platform_interface | ||||
|       sha256: "282d3cf731045a2feb66abfe61bbc40870ae50a3ed10a4d3d217556c35c8c2ba" | ||||
|       sha256: "0b04e02b30791224b31969eb1b50d723498f402971bff3630bca2ba839bd1ed2" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "7.0.1" | ||||
|     version: "7.0.2" | ||||
|   dynamic_color: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -303,10 +303,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: file_picker | ||||
|       sha256: aac85f20436608e01a6ffd1fdd4e746a7f33c93a2c83752e626bdfaea139b877 | ||||
|       sha256: c2376a6aae82358a9f9ccdd7d1f4006d08faa39a2767cce01031d9f593a8bd3b | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "8.1.3" | ||||
|     version: "8.1.6" | ||||
|   fixnum: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -404,10 +404,10 @@ packages: | ||||
|     dependency: "direct dev" | ||||
|     description: | ||||
|       name: flutter_launcher_icons | ||||
|       sha256: "619817c4b65b322b5104b6bb6dfe6cda62d9729bd7ad4303ecc8b4e690a67a77" | ||||
|       sha256: "31cd0885738e87c72d6f055564d37fabcdacee743b396b78c7636c169cac64f5" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.14.1" | ||||
|     version: "0.14.2" | ||||
|   flutter_lints: | ||||
|     dependency: "direct dev" | ||||
|     description: | ||||
| @@ -420,26 +420,26 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_local_notifications | ||||
|       sha256: "49eeef364fddb71515bc78d5a8c51435a68bccd6e4d68e25a942c5e47761ae71" | ||||
|       sha256: ef41ae901e7529e52934feba19ed82827b11baa67336829564aeab3129460610 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "17.2.3" | ||||
|     version: "18.0.1" | ||||
|   flutter_local_notifications_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_local_notifications_linux | ||||
|       sha256: c49bd06165cad9beeb79090b18cd1eb0296f4bf4b23b84426e37dd7c027fc3af | ||||
|       sha256: "8f685642876742c941b29c32030f6f4f6dacd0e4eaecb3efbb187d6a3812ca01" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.1" | ||||
|     version: "5.0.0" | ||||
|   flutter_local_notifications_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_local_notifications_platform_interface | ||||
|       sha256: "85f8d07fe708c1bdcf45037f2c0109753b26ae077e9d9e899d55971711a4ea66" | ||||
|       sha256: "6c5b83c86bf819cdb177a9247a3722067dd8cc6313827ce7c77a4b238a26fd52" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "7.2.0" | ||||
|     version: "8.0.0" | ||||
|   flutter_localizations: | ||||
|     dependency: transitive | ||||
|     description: flutter | ||||
| @@ -449,18 +449,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_markdown | ||||
|       sha256: bd9c475d9aae256369edacafc29d1e74c81f78a10cdcdacbbbc9e3c43d009e4a | ||||
|       sha256: "255b00afa1a7bad19727da6a7780cf3db6c3c12e68d302d85e0ff1fdf173db9e" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.7.4" | ||||
|     version: "0.7.4+3" | ||||
|   flutter_plugin_android_lifecycle: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_plugin_android_lifecycle | ||||
|       sha256: "9b78450b89f059e96c9ebb355fa6b3df1d6b330436e0b885fb49594c41721398" | ||||
|       sha256: "615a505aef59b151b46bbeef55b36ce2b6ed299d160c51d84281946f0aa0ce0e" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.23" | ||||
|     version: "2.0.24" | ||||
|   flutter_test: | ||||
|     dependency: "direct dev" | ||||
|     description: flutter | ||||
| @@ -483,10 +483,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: fluttertoast | ||||
|       sha256: "95f349437aeebe524ef7d6c9bde3e6b4772717cf46a0eb6a3ceaddc740b297cc" | ||||
|       sha256: "24467dc20bbe49fd63e57d8e190798c4d22cbbdac30e54209d153a15273721d1" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "8.2.8" | ||||
|     version: "8.2.10" | ||||
|   fraction: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -515,10 +515,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: html | ||||
|       sha256: "3a7812d5bcd2894edf53dfaf8cd640876cf6cef50a8f238745c8b8120ea74d3a" | ||||
|       sha256: "1fc58edeaec4307368c60d59b7e15b9d658b57d7f3125098b6294153c75337ec" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.15.4" | ||||
|     version: "0.15.5" | ||||
|   http: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -539,10 +539,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: image | ||||
|       sha256: f31d52537dc417fdcde36088fdf11d191026fd5e4fae742491ebd40e5a8bea7d | ||||
|       sha256: "8346ad4b5173924b5ddddab782fc7d8a6300178c8b1dc427775405a01701c4a6" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.3.0" | ||||
|     version: "4.5.2" | ||||
|   intl: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -659,26 +659,26 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: path_provider | ||||
|       sha256: fec0d61223fba3154d87759e3cc27fe2c8dc498f6386c6d6fc80d1afdd1bf378 | ||||
|       sha256: "50c5dd5b6e1aaf6fb3a78b33f6aa3afca52bf903a8a5298f53101fdaee55bbcd" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.4" | ||||
|     version: "2.1.5" | ||||
|   path_provider_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_android | ||||
|       sha256: c464428172cb986b758c6d1724c603097febb8fb855aa265aeecc9280c294d4a | ||||
|       sha256: "4adf4fd5423ec60a29506c76581bc05854c55e3a0b72d35bb28d661c9686edf2" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.12" | ||||
|     version: "2.2.15" | ||||
|   path_provider_foundation: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_foundation | ||||
|       sha256: f234384a3fdd67f989b4d54a5d73ca2a6c422fa55ae694381ae0f4375cd1ea16 | ||||
|       sha256: "4843174df4d288f5e29185bd6e72a6fbdf5a4a4602717eed565497429f179942" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.0" | ||||
|     version: "2.4.1" | ||||
|   path_provider_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -731,10 +731,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: permission_handler_html | ||||
|       sha256: af26edbbb1f2674af65a8f4b56e1a6f526156bc273d0e65dd8075fab51c78851 | ||||
|       sha256: "38f000e83355abb3392140f6bc3030660cfaef189e1f87824facb76300b4ff24" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.1.3+2" | ||||
|     version: "0.1.3+5" | ||||
|   permission_handler_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -807,6 +807,14 @@ packages: | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.10.2+1" | ||||
|   posix: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: posix | ||||
|       sha256: a0117dc2167805aa9125b82eee515cc891819bac2f538c83646d355b16f58b9a | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.0.1" | ||||
|   provider: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -819,42 +827,42 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: share_plus | ||||
|       sha256: "334fcdf0ef9c0df0e3b428faebcac9568f35c747d59831474b2fc56e156d244e" | ||||
|       sha256: "6327c3f233729374d0abaafd61f6846115b2a481b4feddd8534211dc10659400" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "10.1.0" | ||||
|     version: "10.1.3" | ||||
|   share_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: share_plus_platform_interface | ||||
|       sha256: c57c0bbfec7142e3a0f55633be504b796af72e60e3c791b44d5a017b985f7a48 | ||||
|       sha256: cc012a23fc2d479854e6c80150696c4a5f5bb62cb89af4de1c505cf78d0a5d0b | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.0.1" | ||||
|     version: "5.0.2" | ||||
|   shared_preferences: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: shared_preferences | ||||
|       sha256: "746e5369a43170c25816cc472ee016d3a66bc13fcf430c0bc41ad7b4b2922051" | ||||
|       sha256: "3c7e73920c694a436afaf65ab60ce3453d91f84208d761fbd83fc21182134d93" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.2" | ||||
|     version: "2.3.4" | ||||
|   shared_preferences_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_android | ||||
|       sha256: "3b9febd815c9ca29c9e3520d50ec32f49157711e143b7a4ca039eb87e8ade5ab" | ||||
|       sha256: "02a7d8a9ef346c9af715811b01fbd8e27845ad2c41148eefd31321471b41863d" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.3" | ||||
|     version: "2.4.0" | ||||
|   shared_preferences_foundation: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_foundation | ||||
|       sha256: "07e050c7cd39bad516f8d64c455f04508d09df104be326d8c02551590a0d513d" | ||||
|       sha256: "6a52cfcdaeac77cad8c97b539ff688ccfc458c007b4db12be584fbe5c0e49e03" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.5.3" | ||||
|     version: "2.5.4" | ||||
|   shared_preferences_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -929,10 +937,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: sqflite | ||||
|       sha256: "79a297dc3cc137e758c6a4baf83342b039e5a6d2436fcdf3f96a00adaaf2ad62" | ||||
|       sha256: "2d7299468485dca85efeeadf5d38986909c5eb0cd71fd3db2c2f000e6c9454bb" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.0" | ||||
|     version: "2.4.1" | ||||
|   sqflite_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -945,18 +953,18 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_common | ||||
|       sha256: "4468b24876d673418a7b7147e5a08a715b4998a7ae69227acafaab762e0e5490" | ||||
|       sha256: "761b9740ecbd4d3e66b8916d784e581861fd3c3553eda85e167bc49fdb68f709" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.5.4+5" | ||||
|     version: "2.5.4+6" | ||||
|   sqflite_darwin: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_darwin | ||||
|       sha256: "769733dddf94622d5541c73e4ddc6aa7b252d865285914b6fcd54a63c4b4f027" | ||||
|       sha256: "96a698e2bc82bd770a4d6aab00b42396a7c63d9e33513a56945cbccb594c2474" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.1-1" | ||||
|     version: "2.4.1" | ||||
|   sqflite_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1017,10 +1025,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: timezone | ||||
|       sha256: "2236ec079a174ce07434e89fcd3fcda430025eb7692244139a9cf54fdcf1fc7d" | ||||
|       sha256: ffc9d5f4d1193534ef051f9254063fa53d588609418c84299956c3db9383587d | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.9.4" | ||||
|     version: "0.10.0" | ||||
|   typed_data: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1049,34 +1057,34 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_android | ||||
|       sha256: "8fc3bae0b68c02c47c5c86fa8bfa74471d42687b0eded01b78de87872db745e2" | ||||
|       sha256: "6fc2f56536ee873eeb867ad176ae15f304ccccc357848b351f6f0d8d4a40d193" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.3.12" | ||||
|     version: "6.3.14" | ||||
|   url_launcher_ios: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_ios | ||||
|       sha256: e43b677296fadce447e987a2f519dcf5f6d1e527dc35d01ffab4fff5b8a7063e | ||||
|       sha256: "16a513b6c12bb419304e72ea0ae2ab4fed569920d1c7cb850263fe3acc824626" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.3.1" | ||||
|     version: "6.3.2" | ||||
|   url_launcher_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_linux | ||||
|       sha256: e2b9622b4007f97f504cd64c0128309dfb978ae66adbe944125ed9e1750f06af | ||||
|       sha256: "4e9ba368772369e3e08f231d2301b4ef72b9ff87c31192ef471b380ef29a4935" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.2.0" | ||||
|     version: "3.2.1" | ||||
|   url_launcher_macos: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_macos | ||||
|       sha256: "769549c999acdb42b8bcfa7c43d72bf79a382ca7441ab18a808e101149daf672" | ||||
|       sha256: "17ba2000b847f334f16626a574c702b196723af2a289e7a93ffcb79acff855c2" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.2.1" | ||||
|     version: "3.2.2" | ||||
|   url_launcher_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1145,10 +1153,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_android | ||||
|       sha256: "74693a212d990b32e0b7055d27db973a18abf31c53942063948cdfaaef9787ba" | ||||
|       sha256: "3d535126f7244871542b2f0b0fcf94629c9a14883250461f9abe1a6644c1c379" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.0" | ||||
|     version: "4.2.0" | ||||
|   webview_flutter_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1161,18 +1169,18 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_wkwebview | ||||
|       sha256: d4034901d96357beb1b6717ebf7d583c88e40cfc6eb85fe76dd1bf0979a9f251 | ||||
|       sha256: b7e92f129482460951d96ef9a46b49db34bd2e1621685de26e9eaafd9674e7eb | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.16.0" | ||||
|     version: "3.16.3" | ||||
|   win32: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: win32 | ||||
|       sha256: "2735daae5150e8b1dfeb3eb0544b4d3af0061e9e82cef063adcd583bdae4306a" | ||||
|       sha256: "8b338d4486ab3fbc0ba0db9f9b4f5239b6697fcee427939a40e720cbb9ee0a69" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.7.0" | ||||
|     version: "5.9.0" | ||||
|   win32_registry: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1201,10 +1209,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: yaml | ||||
|       sha256: "75769501ea3489fca56601ff33454fe45507ea3bfb014161abc3b43ae25989d5" | ||||
|       sha256: b9da305ac7c39faa3f030eccd175340f968459dae4af175130b3fc47e40d76ce | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.2" | ||||
|     version: "3.1.3" | ||||
| sdks: | ||||
|   dart: ">=3.5.0 <4.0.0" | ||||
|   flutter: ">=3.24.0" | ||||
|   | ||||
| @@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev | ||||
| # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | ||||
| # In Windows, build-name is used as the major, minor, and patch parts | ||||
| # of the product and file versions while build-number is used as the build suffix. | ||||
| version: 1.1.28+2285 | ||||
| version: 1.1.37+2294 | ||||
|  | ||||
| environment: | ||||
|   sdk: '>=3.0.0 <4.0.0' | ||||
| @@ -38,7 +38,7 @@ dependencies: | ||||
|   cupertino_icons: ^1.0.5 | ||||
|   path_provider: ^2.0.11 | ||||
|   flutter_fgbg: ^0.6.0 | ||||
|   flutter_local_notifications: ^17.0.0 | ||||
|   flutter_local_notifications: ^18.0.0 | ||||
|   provider: ^6.0.3 | ||||
|   http: ^1.0.0 | ||||
|   webview_flutter: ^4.0.0 | ||||
|   | ||||
							
								
								
									
										38
									
								
								sign.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										38
									
								
								sign.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # Script to sign unsigned APKs | ||||
| # Assumptions: | ||||
| # 1. Your PGP key is already imported on the locally running agent | ||||
| # 2. An Android SDK is located at $ANDROID_HOME (or ~/Android/Sdk as fallback) | ||||
|  | ||||
| usage() { | ||||
|   echo "sign.sh <PATH_TO_KEYSTORE> <PATH_TO_BUILD_DIR>" | ||||
|   exit 1 | ||||
| } | ||||
|  | ||||
| if [ -z "$1" ] || [ -z "$2" ] || [ ! -f "$1" ] || [ ! -d "$2" ]; then | ||||
|   usage | ||||
| fi | ||||
|  | ||||
| KEYSTORE_LOCATION="$1" | ||||
| BUILD_DIR="$2" | ||||
|  | ||||
| read -s -p "Enter your keystore password: " KEYSTORE_PASSWORD | ||||
|  | ||||
| if [ -z "$ANDROID_HOME" ]; then | ||||
|   ANDROID_HOME=~/Android/Sdk | ||||
| fi | ||||
| if [ ! -d "$ANDROID_HOME" ]; then | ||||
|   echo "Could not find Android SDK!" >&2 | ||||
|   exit 1 | ||||
| fi | ||||
|  | ||||
| for apk in "$BUILD_DIR"/*-release*.apk; do | ||||
|   unsignedApk=${apk/-release/-unsigned} | ||||
|   mv "$apk" "$unsignedApk" | ||||
|   ${ANDROID_HOME}/build-tools/$(ls ${ANDROID_HOME}/build-tools/ | tail -1)/apksigner sign --ks "$KEYSTORE_LOCATION" --ks-pass pass:"${KEYSTORE_PASSWORD}" --out "${apk}" "${unsignedApk}" | ||||
|   sha256sum ${apk} | cut -d " " -f 1 >"$apk".sha256 | ||||
|   gpg --batch --sign --detach-sig "$apk".sha256 | ||||
|   rm "$unsignedApk" | ||||
| done | ||||
		Reference in New Issue
	
	Block a user