Compare commits
	
		
			295 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | e061e99451 | ||
|  | a282080fea | ||
|  | 0b812b508a | ||
|  | e639758b15 | ||
|  | f159c0bd44 | ||
|  | 950bf28289 | ||
|  | ecf4326b47 | ||
|  | 98182d9873 | ||
|  | c7c6731732 | ||
|  | b62b60d9df | ||
|  | 3e41913153 | ||
|  | 6b4943349a | ||
|  | 2f60835801 | ||
|  | 3de2121ed8 | ||
|  | e1c80229ab | ||
|  | e9feaf0d8b | ||
|  | 3175597a2a | ||
|  | 6af1748a78 | ||
|  | c9aed8dfc4 | ||
|  | 9c3bdafa47 | ||
|  | d4e857f7f4 | ||
|  | 74d6ffcfb3 | ||
|  | 10634e8ed2 | ||
|  | 2317c5e0d3 | ||
|  | e9dfb494f7 | ||
|  | 71ca60244b | ||
|  | 94ab83ff75 | ||
|  | 2aa91ed535 | ||
|  | 23a72f1c83 | ||
|  | 12a25d5cbc | ||
|  | eb57d150b4 | ||
|  | 5dc8461341 | ||
|  | 5cf751caa8 | ||
|  | 526ebdfc32 | ||
|  | 48f2164fe0 | ||
|  | 0d4bc44aa4 | ||
|  | e78961f0ab | ||
|  | 6784c6c0c6 | ||
|  | b6ef153bfa | ||
|  | 6a3805723d | ||
|  | ebc6671d62 | ||
|  | c87fc6f242 | ||
|  | b23db43620 | ||
|  | b2b10739e3 | ||
|  | 32cc97a18a | ||
|  | 4815e5180e | ||
|  | 14325eb710 | ||
|  | 7607b747c1 | ||
|  | 774bc07663 | ||
|  | 4e43001276 | ||
|  | 46835d0876 | ||
|  | 48d8eb214d | ||
|  | 988f9a6f9f | ||
|  | 51f0d745c9 | ||
|  | ae72302f4c | ||
|  | e157bda0eb | ||
|  | adf05abfac | ||
|  | 6a8cf0096d | ||
|  | 5e6e3f457c | ||
|  | 044bd7f8a0 | ||
|  | 902f29fdcf | ||
|  | 6c6f256976 | ||
|  | fbf2330c0f | ||
|  | 86ac573edd | ||
|  | 6a07138389 | ||
|  | 23c2664ead | ||
|  | 4200e1d954 | ||
|  | 3eb3cf25bf | ||
|  | 1b039fb5a5 | ||
|  | 6a4a15ab4d | ||
|  | a70c6ef1ed | ||
|  | 85e5dddd34 | ||
|  | 0fd496e660 | ||
|  | b3af899ba1 | ||
|  | 9a58643088 | ||
|  | b223522801 | ||
|  | 375caa4511 | ||
|  | 92ebbd9138 | ||
|  | a3c2761aba | ||
|  | 68e38259bd | ||
|  | 4aed749b44 | ||
|  | e135476d86 | ||
|  | 7490942fad | ||
|  | 794be438b0 | ||
|  | f2f055ad83 | ||
|  | 4c94f278e2 | ||
|  | 22acb6a2dd | ||
|  | d32befb832 | ||
|  | 94d8295992 | ||
|  | dcf9f5732a | ||
|  | c89790d58f | ||
|  | 3633c58bea | ||
|  | 9770501aec | ||
|  | 167d0ccced | ||
|  | 430d1f2690 | ||
|  | eb21ba3f6e | ||
|  | 3ab14e2311 | ||
|  | fff3b22e74 | ||
|  | 46354e648a | ||
|  | 4f67ba3f3b | ||
|  | e5db702a67 | ||
|  | 53451fd883 | ||
|  | 491d3cb723 | ||
|  | 6c7644c9b3 | ||
|  | 8539581fe9 | ||
|  | f301f6cedb | ||
|  | 93c8bca038 | ||
|  | 0961e044a7 | ||
|  | 7da631e429 | ||
|  | 66c08f0bfd | ||
|  | 79a2484d68 | ||
|  | c3c7c844dc | ||
|  | 075d0a0854 | ||
|  | cc0416e344 | ||
|  | af7088b754 | ||
|  | ba21137da7 | ||
|  | a7a5749d4f | ||
|  | fcf3c8b635 | ||
|  | 69f578652c | ||
|  | 7593af25b9 | ||
|  | 1e8fd33469 | ||
|  | 03f50e501e | ||
|  | 21dacb6171 | ||
|  | 16ecc88fcd | ||
|  | 1f829289ed | ||
|  | be739b7639 | ||
|  | 4b66aefb33 | ||
|  | 3f948ae73c | ||
|  | de2e881e82 | ||
|  | 9af3aaa82a | ||
|  | 3a5157ced0 | ||
|  | e16320f995 | ||
|  | aac01885f1 | ||
|  | 7d330cd074 | ||
|  | 8b0e2aaef1 | ||
|  | 5db53d9c71 | ||
|  | ff5bba933b | ||
|  | 2b48ee81b7 | ||
|  | fe6d0040c4 | ||
|  | 17d32cd1c6 | ||
|  | c6a62fe15a | ||
|  | 2b22de585e | ||
|  | 346c670488 | ||
|  | 2137335e56 | ||
|  | cea96275b7 | ||
|  | 9954b68332 | ||
|  | 974a822b29 | ||
|  | 48b89335fa | ||
|  | e834630688 | ||
|  | 3dd1ef076f | ||
|  | e70edae831 | ||
|  | d9b2d18d85 | ||
|  | 96b260e431 | ||
|  | 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 | ||
|  | 2ed2c2c5f9 | ||
|  | 7f35589d49 | ||
|  | 9d5ce75e27 | ||
|  | 09f1f27fa8 | ||
|  | ea239ffa3a | ||
|  | 07ef86ae75 | ||
|  | c670fce652 | ||
|  | ba8ef7e5b0 | ||
|  | 0ec3fb0064 | ||
|  | c1c06b3f9c | ||
|  | d121463bad | ||
|  | 862e85f882 | ||
|  | 5e0333c4c8 | ||
|  | efb75afb9b | ||
|  | b4510e10a7 | ||
|  | 25233f3259 | ||
|  | cff6b86997 | ||
|  | 344a6efbf6 | ||
|  | 59a4b88a88 | ||
|  | cf7c41bf4c | ||
|  | d32543b5d0 | ||
|  | 892a3b3c30 | ||
|  | 56c791d3e5 | ||
|  | e4103c8352 | ||
|  | 672bc159cc | ||
|  | ee5d121bf0 | ||
|  | e5d1fafc13 | ||
|  | 8bb4f46209 | ||
|  | 504e1d551b | ||
|  | 5e7e143bba | ||
|  | 91a82af418 | ||
|  | 7af2145e9a | ||
|  | 74b9de3516 | ||
|  | f5b540dd8b | ||
|  | 01d701d8cd | ||
|  | 4017253470 | ||
|  | b0b6ddb8fd | ||
|  | 192b7fc6ce | ||
|  | b74dbb973c | ||
|  | 46ccf8478b | ||
|  | 9a354ecf0e | ||
|  | 738dd5649f | ||
|  | 9f50d8db2d | ||
|  | eeb57dbe35 | ||
|  | cbcc8c4eaf | ||
|  | f4d27c8494 | ||
|  | 87c66b4d09 | 
							
								
								
									
										2
									
								
								.flutter
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										38
									
								
								.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,8 @@ jobs: | ||||
|        | ||||
|       - uses: actions/checkout@v3 | ||||
|       - uses: subosito/flutter-action@v2 | ||||
|         with: | ||||
|           channel: stable | ||||
|       - uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           distribution: 'temurin' # See 'Supported distributions' for available options | ||||
| @@ -28,13 +27,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 +49,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 +64,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 | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -5,9 +5,11 @@ | ||||
| *.swp | ||||
| .DS_Store | ||||
| .atom/ | ||||
| .build/ | ||||
| .buildlog/ | ||||
| .history | ||||
| .svn/ | ||||
| .swiftpm/ | ||||
| migrate_working_dir/ | ||||
| .vscode/ | ||||
|  | ||||
| @@ -43,6 +45,7 @@ app.*.map.json | ||||
| /android/app/debug | ||||
| /android/app/profile | ||||
| /android/app/release | ||||
| /android/app/.cxx | ||||
|  | ||||
| # Custom | ||||
| TODO.txt | ||||
							
								
								
									
										30
									
								
								.metadata
									
									
									
									
									
								
							
							
						
						| @@ -1,11 +1,11 @@ | ||||
| # This file tracks properties of this Flutter project. | ||||
| # Used by Flutter tool to assess capabilities and perform upgrades etc. | ||||
| # | ||||
| # This file should be version controlled. | ||||
| # This file should be version controlled and should not be manually edited. | ||||
|  | ||||
| version: | ||||
|   revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|   channel: master | ||||
|   revision: "17025dd88227cd9532c33fa78f5250d548d87e9a" | ||||
|   channel: "stable" | ||||
|  | ||||
| project_type: app | ||||
|  | ||||
| @@ -13,14 +13,26 @@ project_type: app | ||||
| migration: | ||||
|   platforms: | ||||
|     - platform: root | ||||
|       create_revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|       base_revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|     - platform: android | ||||
|       create_revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|       base_revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|     - platform: ios | ||||
|       create_revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|       base_revision: 543dac2f35de563fcb6905e9146bdce712c44000 | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|     - platform: linux | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|     - platform: macos | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|     - platform: web | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|     - platform: windows | ||||
|       create_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|       base_revision: 17025dd88227cd9532c33fa78f5250d548d87e9a | ||||
|  | ||||
|   # User provided section | ||||
|  | ||||
|   | ||||
							
								
								
									
										23
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -7,10 +7,13 @@ 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)) | ||||
| - [Obtainium 101](https://www.youtube.com/watch?v=0MF_v2OBncw) - Tutorial video | ||||
| - [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: | ||||
| @@ -27,15 +30,13 @@ Currently supported App sources: | ||||
|   - [Uptodown](https://uptodown.com/) | ||||
|   - [Huawei AppGallery](https://appgallery.huawei.com/) | ||||
|   - [Tencent App Store](https://sj.qq.com/) | ||||
|   - [CoolApk](https://coolapk.com/) | ||||
|   - [RuStore](https://rustore.ru/) | ||||
|   - 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) | ||||
|   - [Telegram App](https://telegram.org/) | ||||
|   - [Neutron Code](https://neutroncode.com/) | ||||
| - Direct APK Link | ||||
| - "HTML" (Fallback): Any other URL that returns an HTML page with links to APK files | ||||
|  | ||||
| @@ -59,7 +60,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. | ||||
|   | ||||
| @@ -13,8 +13,7 @@ linter: | ||||
|   # The lint rules applied to this project can be customized in the | ||||
|   # section below to disable rules from the `package:flutter_lints/flutter.yaml` | ||||
|   # included above or to enable additional rules. A list of all available lints | ||||
|   # and their documentation is published at | ||||
|   # https://dart-lang.github.io/linter/lints/index.html. | ||||
|   # and their documentation is published at https://dart.dev/lints. | ||||
|   # | ||||
|   # Instead of disabling a lint rule for the entire project in the | ||||
|   # section below, it can also be suppressed for a single line of code | ||||
|   | ||||
							
								
								
									
										2
									
								
								android/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -7,7 +7,7 @@ gradle-wrapper.jar | ||||
| GeneratedPluginRegistrant.java | ||||
|  | ||||
| # Remember to never publicly share your keystore. | ||||
| # See https://flutter.dev/docs/deployment/android#reference-the-keystore-from-the-app | ||||
| # See https://flutter.dev/to/reference-keystore | ||||
| key.properties | ||||
| **/*.keystore | ||||
| **/*.jks | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| plugins { | ||||
|     id "com.android.application" | ||||
|     id "kotlin-android" | ||||
|     // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. | ||||
|     id "dev.flutter.flutter-gradle-plugin" | ||||
| } | ||||
|  | ||||
| @@ -29,31 +30,28 @@ if (keystorePropertiesFile.exists()) { | ||||
| } | ||||
|  | ||||
| android { | ||||
|     namespace "dev.imranr.obtainium" | ||||
|     compileSdk flutter.compileSdkVersion | ||||
|     ndkVersion flutter.ndkVersion | ||||
|     namespace = "dev.imranr.obtainium" | ||||
|     compileSdk = flutter.compileSdkVersion | ||||
|     ndkVersion = flutter.ndkVersion | ||||
|  | ||||
|     compileOptions { | ||||
|         sourceCompatibility JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility JavaVersion.VERSION_1_8 | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|     } | ||||
|  | ||||
|     kotlinOptions { | ||||
|         jvmTarget = '1.8' | ||||
|     } | ||||
|  | ||||
|     sourceSets { | ||||
|         main.java.srcDirs += 'src/main/kotlin' | ||||
|         jvmTarget = JavaVersion.VERSION_1_8 | ||||
|     } | ||||
|  | ||||
|     defaultConfig { | ||||
|         applicationId "dev.imranr.obtainium" | ||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). | ||||
|         applicationId = "dev.imranr.obtainium" | ||||
|         // You can update the following values to match your application needs. | ||||
|         // For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-build-configuration. | ||||
|         minSdkVersion 24 | ||||
|         targetSdkVersion flutter.targetSdkVersion | ||||
|         versionCode flutterVersionCode.toInteger() | ||||
|         versionName flutterVersionName | ||||
|         // For more information, see: https://flutter.dev/to/review-gradle-config. | ||||
|         minSdk = 24 | ||||
|         targetSdk = flutter.targetSdkVersion | ||||
|         versionCode = flutter.versionCode | ||||
|         versionName = flutter.versionName | ||||
|     } | ||||
|  | ||||
|     flavorDimensions "flavor" | ||||
| @@ -77,19 +75,20 @@ android { | ||||
|             storePassword keystoreProperties['storePassword'] | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     buildTypes { | ||||
|         release { | ||||
|             signingConfig signingConfigs.release | ||||
|         } | ||||
|         debug { | ||||
|             applicationIdSuffix = ".debug" | ||||
|             versionNameSuffix = "-debug" | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| flutter { | ||||
|     source '../..' | ||||
| } | ||||
|  | ||||
| repositories { | ||||
|     maven { url 'https://jitpack.io' } | ||||
|     source = "../.." | ||||
| } | ||||
|  | ||||
| ext.abiCodes = ["x86_64": 1, "armeabi-v7a": 2, "arm64-v8a": 3] | ||||
|   | ||||
							
								
								
									
										4
									
								
								android/app/src/debug/res/values/string.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|     <string name="label">Obtainium Debug</string> | ||||
| </resources> | ||||
| @@ -2,7 +2,7 @@ | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     package="dev.imranr.obtainium"> | ||||
|     <application | ||||
|         android:label="Obtainium" | ||||
|         android:label="@string/label" | ||||
|         android:name="${applicationName}" | ||||
|         android:icon="@mipmap/ic_launcher" | ||||
|         android:requestLegacyExternalStorage="true" | ||||
| @@ -12,6 +12,7 @@ | ||||
|             android:name=".MainActivity" | ||||
|             android:exported="true" | ||||
|             android:launchMode="singleTop" | ||||
|             android:taskAffinity="" | ||||
|             android:theme="@style/LaunchTheme" | ||||
|             android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode" | ||||
|             android:hardwareAccelerated="true" | ||||
| @@ -25,8 +26,8 @@ | ||||
|               android:resource="@style/NormalTheme" | ||||
|               /> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.MAIN" /> | ||||
|                 <category android:name="android.intent.category.LAUNCHER" /> | ||||
|                 <action android:name="android.intent.action.MAIN"/> | ||||
|                 <category android:name="android.intent.category.LAUNCHER"/> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action | ||||
| @@ -39,7 +40,6 @@ | ||||
|                 <category android:name="android.intent.category.BROWSABLE" /> | ||||
|                 <data android:scheme="obtainium" /> | ||||
|             </intent-filter> | ||||
|  | ||||
|         </activity> | ||||
|         <!-- Don't delete the meta-data below. | ||||
|              This is used by the Flutter tool to generate GeneratedPluginRegistrant.java --> | ||||
| @@ -71,6 +71,17 @@ | ||||
|     <uses-permission android:name="android.permission.REQUEST_DELETE_PACKAGES" /> | ||||
|     <uses-permission | ||||
|         android:name="android.permission.WRITE_EXTERNAL_STORAGE" | ||||
|         android:maxSdkVersion="29" />\ | ||||
|         android:maxSdkVersion="29" /> | ||||
|     <uses-permission android:name="android.permission.QUERY_ALL_PACKAGES" /> | ||||
|     <!-- Required to query activities that can process text, see: | ||||
|          https://developer.android.com/training/package-visibility and | ||||
|          https://developer.android.com/reference/android/content/Intent#ACTION_PROCESS_TEXT. | ||||
|  | ||||
|          In particular, this is used by the Flutter engine in io.flutter.plugin.text.ProcessTextPlugin. --> | ||||
|     <queries> | ||||
|         <intent> | ||||
|             <action android:name="android.intent.action.PROCESS_TEXT"/> | ||||
|             <data android:mimeType="text/plain"/> | ||||
|         </intent> | ||||
|     </queries> | ||||
| </manifest> | ||||
|   | ||||
							
								
								
									
										4
									
								
								android/app/src/main/res/values/string.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|     <string name="label">Obtainium</string> | ||||
| </resources> | ||||
| @@ -9,12 +9,12 @@ allprojects { | ||||
|     } | ||||
| } | ||||
|  | ||||
| rootProject.buildDir = '../build' | ||||
| rootProject.buildDir = "../build" | ||||
| subprojects { | ||||
|     project.buildDir = "${rootProject.buildDir}/${project.name}" | ||||
| } | ||||
| subprojects { | ||||
|     project.evaluationDependsOn(':app') | ||||
|     project.evaluationDependsOn(":app") | ||||
| } | ||||
|  | ||||
| tasks.register("clean", Delete) { | ||||
|   | ||||
| @@ -1,3 +1,3 @@ | ||||
| org.gradle.jvmargs=-Xmx2048M | ||||
| org.gradle.jvmargs=-Xmx4G -XX:MaxMetaspaceSize=2G -XX:+HeapDumpOnOutOfMemoryError | ||||
| android.useAndroidX=true | ||||
| android.enableJetifier=true | ||||
|   | ||||
| @@ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME | ||||
| distributionPath=wrapper/dists | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
| zipStorePath=wrapper/dists | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-all.zip | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-all.zip | ||||
|   | ||||
| @@ -18,8 +18,8 @@ pluginManagement { | ||||
|  | ||||
| plugins { | ||||
|     id "dev.flutter.flutter-plugin-loader" version "1.0.0" | ||||
|     id "com.android.application" version "7.4.2" apply false | ||||
|     id "org.jetbrains.kotlin.android" version "1.8.10" apply false | ||||
|     id "com.android.application" version "8.1.0" apply false | ||||
|     id "org.jetbrains.kotlin.android" version "1.8.22" apply false | ||||
| } | ||||
|  | ||||
| include ":app" | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								assets/fonts/Montserrat-Regular.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								assets/graphics/badge_obtainium_i18n/badge_obtainium_zh-Hans.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 26 KiB | 
| Before Width: | Height: | Size: 234 KiB After Width: | Height: | Size: 346 KiB | 
| Before Width: | Height: | Size: 238 KiB After Width: | Height: | Size: 354 KiB | 
| Before Width: | Height: | Size: 140 KiB After Width: | Height: | Size: 265 KiB | 
| Before Width: | Height: | Size: 139 KiB After Width: | Height: | Size: 227 KiB | 
| Before Width: | Height: | Size: 118 KiB After Width: | Height: | Size: 178 KiB | 
| Before Width: | Height: | Size: 262 KiB After Width: | Height: | Size: 264 KiB | 
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Deinstaliraj s uređaja", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Radi samo za aplikacije s onemogućenom detekcijom verzije.", | ||||
|     "releaseDateAsVersion": "Koristi datum izdanja kao verziju", | ||||
|     "releaseTitleAsVersion": "Use release title as version string", | ||||
|     "releaseDateAsVersionExplanation": "Ova opcija bi se trebala koristiti samo za aplikacije gdje detekcija verzije ne radi ispravno, ali je datum izdavanja dostupan.", | ||||
|     "changes": "Promjene", | ||||
|     "releaseDate": "Datum izdavanja", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Detekcija standardne verzije", | ||||
|     "groupByCategory": "Grupiši po kategoriji", | ||||
|     "autoApkFilterByArch": "Pokušajte filtrirati APK-ove po arhitekturi procesora ako je moguće", | ||||
|     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||
|     "overrideSource": "Premosti izvor", | ||||
|     "dontShowAgain": "Ne prikazuj ovo ponovo", | ||||
|     "dontShowTrackOnlyWarnings": "Ne prikazuj upozorenja „Samo za  praćenje”", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "Intermediate veza nije nađena", | ||||
|     "intermediateLink": "Intermediate veza", | ||||
|     "exemptFromBackgroundUpdates": "Izuzmi iz ažuriranja u pozadini (ako su uključeni)", | ||||
|     "bgUpdatesOnWiFiOnly": "Isključite ažuriranje u pozadini kada niste na WiFi-ju", | ||||
|     "bgUpdatesOnWiFiOnly": "Isključite ažuriranje u pozadini kada niste na Wi-Fi-ju", | ||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||
|     "autoSelectHighestVersionCode": "Automatski izaberite najveću (verziju) versionCode APK-a", | ||||
|     "versionExtractionRegEx": "RegEx ekstrakcija verzije", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Dijeli sa AppVerifier-om, zatim se vratite kada ste spremni.", | ||||
|     "wiki": "Pomoć/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfiguracije aplikacije obezbeđene pomoću velikog broja ljudi (crowdsourcing) (koristite na svoju odgovornost)", | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configurations", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Name", | ||||
|     "smartname": "Name (Smart)", | ||||
|     "sortMethod": "Sort Method", | ||||
|     "welcome": "Welcome", | ||||
|     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Želite li ukloniti aplikaciju?", | ||||
|         "other": "Želite li ukloniti aplikacije?" | ||||
|   | ||||
							
								
								
									
										389
									
								
								assets/translations/ca.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,389 @@ | ||||
| { | ||||
|     "invalidURLForSource": "L'URL de l'aplicació {} no es vàlid", | ||||
|     "noReleaseFound": "No s'ha pogut trobar una versió adequada", | ||||
|     "noVersionFound": "No s'ha pogut determinar la versió", | ||||
|     "urlMatchesNoSource": "L'URL no coincideix amb cap font coneguda", | ||||
|     "cantInstallOlderVersion": "No és possible instal·lar una versió més antiga de l'aplicació", | ||||
|     "appIdMismatch": "L'ID del paquet descarregat no coincideix amb l'ID de l'aplicació instal·lada", | ||||
|     "functionNotImplemented": "Aquesta classe no ha implentat aquesta funció", | ||||
|     "placeholder": "Espai reservat", | ||||
|     "someErrors": "S'han produït alguns errors", | ||||
|     "unexpectedError": "Error inesperat", | ||||
|     "ok": "Accepta", | ||||
|     "and": "i", | ||||
|     "githubPATLabel": "Token d'accés personal a GitHub (augmenta el límit d'accés)", | ||||
|     "includePrereleases": "Inclou les versions preliminars", | ||||
|     "fallbackToOlderReleases": "Torna a les versions anteriors", | ||||
|     "filterReleaseTitlesByRegEx": "Filtra el títol de la versió per una expressió regular", | ||||
|     "invalidRegEx": "Expressió regular invàlida", | ||||
|     "noDescription": "Sense descripció", | ||||
|     "cancel": "Cancel·la", | ||||
|     "continue": "Continua", | ||||
|     "requiredInBrackets": "(requerit)", | ||||
|     "dropdownNoOptsError": "ERROR: EL DESPLEGABLE HA DE TENIR ALMENYS UNA OPCIÓ", | ||||
|     "colour": "Color", | ||||
|     "standard": "Estàndard", | ||||
|     "custom": "Personalitzat", | ||||
|     "useMaterialYou": "Usa 'Material You'", | ||||
|     "githubStarredRepos": "Repositoris favorits de GitHub", | ||||
|     "uname": "Nom d'usuari", | ||||
|     "wrongArgNum": "Nombre d'arguments proveït invàlid", | ||||
|     "xIsTrackOnly": "{} és només per a seguiment", | ||||
|     "source": "Font", | ||||
|     "app": "Aplicació", | ||||
|     "appsFromSourceAreTrackOnly": "Les aplicacions d'aquesta font són només per a seguiment.", | ||||
|     "youPickedTrackOnly": "Has seleccionat l'opció 'només per a seguiment'.", | ||||
|     "trackOnlyAppDescription": "Farem el seguiment de les actualitzacions per a l'aplicació, però Obtainium no podrà descarregar-la ni actualitzar-la.", | ||||
|     "cancelled": "Cancel·lat", | ||||
|     "appAlreadyAdded": "Aplicació ja afegida", | ||||
|     "alreadyUpToDateQuestion": "Aplicació ja actualitzada?", | ||||
|     "addApp": "Afegeix l'aplicació", | ||||
|     "appSourceURL": "URL font de l'aplicació", | ||||
|     "error": "Error", | ||||
|     "add": "Afegeix", | ||||
|     "searchSomeSourcesLabel": "Cerca (només algunes fonts)", | ||||
|     "search": "Cerca", | ||||
|     "additionalOptsFor": "Opcions addicionals per a {}", | ||||
|     "supportedSources": "Fonts suportades", | ||||
|     "trackOnlyInBrackets": "(només per a seguiment)", | ||||
|     "searchableInBrackets": "(permet la cerca)", | ||||
|     "appsString": "Aplicacions", | ||||
|     "noApps": "No hi ha aplicacions", | ||||
|     "noAppsForFilter": "No hi ha aplicacions per filtrar", | ||||
|     "byX": "Per: {}", | ||||
|     "percentProgress": "Progrés: {} %", | ||||
|     "pleaseWait": "Espera...", | ||||
|     "updateAvailable": "Actualització disponible", | ||||
|     "notInstalled": "No instal·lat", | ||||
|     "pseudoVersion": "pseudo-versió", | ||||
|     "selectAll": "Selecciona-ho tot", | ||||
|     "deselectX": "Desselecciona {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} s'eliminarà d'Obtainium però romandrà instal·lada al dispositiu.", | ||||
|     "removeSelectedAppsQuestion": "Elimino les aplicacions seleccionades?", | ||||
|     "removeSelectedApps": "Elimina les aplicacions seleccionades", | ||||
|     "updateX": "Actualitza {}", | ||||
|     "installX": "Instal·la {}", | ||||
|     "markXTrackOnlyAsUpdated": "Marca {}\n(només per a seguiment)\ncom a actualitzada", | ||||
|     "changeX": "Canvia {}", | ||||
|     "installUpdateApps": "Instal·la/actualitza les aplicacions", | ||||
|     "installUpdateSelectedApps": "Instal·la/actualitza les aplicacions seleccionades", | ||||
|     "markXSelectedAppsAsUpdated": "Marco {} les aplicaciones seleccionades com a actualitzades?", | ||||
|     "no": "No", | ||||
|     "yes": "Sí", | ||||
|     "markSelectedAppsUpdated": "Marca les aplicacions seleccionades com a actualitzades", | ||||
|     "pinToTop": "Ancora-la al capdamunt", | ||||
|     "unpinFromTop": "Desancora-la del capdamunt", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Restableixo l'estat d'instal·lació per a les aplicacions seleccionades?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Es restablirà l'estat d'instal·lació de les aplicacions seleccionades.\n\nAçò pot ser útil quan la versió de l'aplicació mostrada per Obtainium és incorrecta a conseqüència d'una actualització no reeixida o d'algun altre problema.", | ||||
|     "customLinkMessage": "Aquests enllaços funcionen en dispositius amb Obtainium instal·lat", | ||||
|     "shareAppConfigLinks": "Comparteix la configuració de l'aplicació com a enllaç HTML", | ||||
|     "shareSelectedAppURLs": "Comparteix els URL de les aplicacions seleccionades", | ||||
|     "resetInstallStatus": "Restableix l'estat de la instal·lació", | ||||
|     "more": "Més", | ||||
|     "removeOutdatedFilter": "Elimina el filtre d'aplicacions desactualitzades", | ||||
|     "showOutdatedOnly": "Mostra només les aplicacions desactualitzades", | ||||
|     "filter": "Filtra", | ||||
|     "filterApps": "Filtra les aplicacions", | ||||
|     "appName": "Nom de l'aplicació", | ||||
|     "author": "Autor", | ||||
|     "upToDateApps": "Aplicacions actualizades", | ||||
|     "nonInstalledApps": "Aplicacions no instal·lades", | ||||
|     "importExport": "Importa/exporta", | ||||
|     "settings": "Paràmetres", | ||||
|     "exportedTo": "Exportat a {}", | ||||
|     "obtainiumExport": "Exporta Obtainium", | ||||
|     "invalidInput": "Entrada no vàlida", | ||||
|     "importedX": "Importat {}", | ||||
|     "obtainiumImport": "Importa Obtainium", | ||||
|     "importFromURLList": "Importa des de la llista d'URL", | ||||
|     "searchQuery": "Terme de cerca", | ||||
|     "appURLList": "Llista d'URL d'aplicacions", | ||||
|     "line": "Línia", | ||||
|     "searchX": "Cerca {}", | ||||
|     "noResults": "No hi ha resultats", | ||||
|     "importX": "Importa des de {}", | ||||
|     "importedAppsIdDisclaimer": "Les aplicacions importades podrien mostrar-se incorrectament com a «No instal·lada».\nPer solventar-ho reinstal·la-les a través d'Obtainium.\nAixò no hauria d'afectar les dades de les aplicacions.\n\nNomés afecta els URL i els mètodes d'importació de tercers.", | ||||
|     "importErrors": "Errors d'importació", | ||||
|     "importedXOfYApps": "{} de {} aplicacions importades.", | ||||
|     "followingURLsHadErrors": "Els següents URLs han tingut problemes:", | ||||
|     "selectURL": "Selecciona URL", | ||||
|     "selectURLs": "Selecciona URLs", | ||||
|     "pick": "Escull", | ||||
|     "theme": "Tema", | ||||
|     "dark": "Fosc", | ||||
|     "light": "Clar", | ||||
|     "followSystem": "Segueix el sistema", | ||||
|     "followSystemThemeExplanation": "Seguir el tema del sistema només és possible si uses aplicacions de tercers", | ||||
|     "useBlackTheme": "Fes servir el negre pur en el tema fosc", | ||||
|     "appSortBy": "Ordena les aplicacions per", | ||||
|     "authorName": "Autor/nom", | ||||
|     "nameAuthor": "Nom/Autor", | ||||
|     "asAdded": "Per l'ordre en què es van afegir", | ||||
|     "appSortOrder": "Per ordre de classificació", | ||||
|     "ascending": "Ascendent", | ||||
|     "descending": "Descendent", | ||||
|     "bgUpdateCheckInterval": "Comprova les actualitzacions en segon pla", | ||||
|     "neverManualOnly": "Mai, només manual", | ||||
|     "appearance": "Aparença", | ||||
|     "showWebInAppView": "Mostra el web d'origen en la vista de l'aplicació", | ||||
|     "pinUpdates": "Ancora les actualitzacions al capdamunt de les aplicacions", | ||||
|     "updates": "Actualitzacions", | ||||
|     "sourceSpecific": "Font específica", | ||||
|     "appSource": "Font de l'aplicació", | ||||
|     "noLogs": "Cap registre", | ||||
|     "appLogs": "Registres de l'aplicació", | ||||
|     "close": "Tanca", | ||||
|     "share": "Comparteix", | ||||
|     "appNotFound": "No s'ha trobat l'aplicació", | ||||
|     "obtainiumExportHyphenatedLowercase": "Exportació d'Obtainium", | ||||
|     "pickAnAPK": "Escull una APK", | ||||
|     "appHasMoreThanOnePackage": "{} té més d'un paquet:", | ||||
|     "deviceSupportsXArch": "Aquest dispositiu admet l'aquitectura de CPU: {}.", | ||||
|     "deviceSupportsFollowingArchs": "Aquest dispositiu admet les següents arquitectures de CPU:", | ||||
|     "warning": "Avís", | ||||
|     "sourceIsXButPackageFromYPrompt": "La font de l'aplicació és «{}» però el paquet de l'actualització ve de «{}». Vols continuar?", | ||||
|     "updatesAvailable": "Actualitzacions disponibles", | ||||
|     "updatesAvailableNotifDescription": "Notifica l'usuari que hi ha actualitzacions per a una o més aplicacions seguides per Obtainium", | ||||
|     "noNewUpdates": "No hi ha noves actualitzacions.", | ||||
|     "xHasAnUpdate": "{} té una actualització.", | ||||
|     "appsUpdated": "Aplicacions actualitzades", | ||||
|     "appsNotUpdated": "Error en actualitzar les aplicacions", | ||||
|     "appsUpdatedNotifDescription": "Notifica l'usuari que una o més aplicacions s'han actualitzat en segon pla", | ||||
|     "xWasUpdatedToY": "{} s'ha actualitzat a {}.", | ||||
|     "xWasNotUpdatedToY": "Error en actualitzar {} a {}.", | ||||
|     "errorCheckingUpdates": "Error en cercar actualitzacions", | ||||
|     "errorCheckingUpdatesNotifDescription": "Una notificació que es mostra quan la comprovació d'actualizacions en segon pla ha fallat", | ||||
|     "appsRemoved": "Aplicacions suprimides", | ||||
|     "appsRemovedNotifDescription": "Notifica l'usuari que una o més aplicacions s'han suprimit per errors en carregar-les", | ||||
|     "xWasRemovedDueToErrorY": "{} s'ha suprimit per aquest error: {}", | ||||
|     "completeAppInstallation": "Instal·lació completa de l'aplicació", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium ha d'estar obert per poder instal·lar aplicacions", | ||||
|     "completeAppInstallationNotifDescription": "Demana l'usuari de tornar a Obtainium per acabar d'instal·lar una aplicació", | ||||
|     "checkingForUpdates": "S'estan cercant actualitzacions...", | ||||
|     "checkingForUpdatesNotifDescription": "Notificació temporal que apareix en cercar actualitzacions", | ||||
|     "pleaseAllowInstallPerm": "Permet que Obtainium instal·li aplicacions", | ||||
|     "trackOnly": "Només per a seguiment", | ||||
|     "errorWithHttpStatusCode": "Error {}", | ||||
|     "versionCorrectionDisabled": "Correcció de versions desactivada (el plugin sembla que no funciona)", | ||||
|     "unknown": "Desconegut", | ||||
|     "none": "Cap", | ||||
|     "never": "Mai", | ||||
|     "latestVersionX": "Última versió: {}", | ||||
|     "installedVersionX": "Versió instal·lada: {}", | ||||
|     "lastUpdateCheckX": "Última comprovació d'actualització: {}", | ||||
|     "remove": "Suprimeix", | ||||
|     "yesMarkUpdated": "Sí, marca com a actualitzada", | ||||
|     "fdroid": "Repositori oficial F-Droid", | ||||
|     "appIdOrName": "ID o nom de l'aplicació", | ||||
|     "appId": "ID de l'aplicació", | ||||
|     "appWithIdOrNameNotFound": "No s'han trobat aplicacions amb aquest ID o nom", | ||||
|     "reposHaveMultipleApps": "Els repositoris poden contenir diverses aplicacions", | ||||
|     "fdroidThirdPartyRepo": "Repositori F-Droid de tercers", | ||||
|     "install": "Instal·la", | ||||
|     "markInstalled": "Marca com a instal·lada", | ||||
|     "update": "Actualitza", | ||||
|     "markUpdated": "Marca com a actualitzada", | ||||
|     "additionalOptions": "Opcions addicionals", | ||||
|     "disableVersionDetection": "Desactiva la detecció de versions", | ||||
|     "noVersionDetectionExplanation": "Només has d'usar aquesta opció en les aplicacions en què la detecció de versions no funcioni correctament.", | ||||
|     "downloadingX": "Descarregant {}", | ||||
|     "downloadX": "Descarrega {}", | ||||
|     "downloadedX": "Descarregada {}", | ||||
|     "releaseAsset": "Recurs actualitzat", | ||||
|     "downloadNotifDescription": "Notifica l'usuari del progrés de la descàrrega d'una aplicació", | ||||
|     "noAPKFound": "No s'ha trobat l'APK", | ||||
|     "noVersionDetection": "No s'han detectat versions", | ||||
|     "categorize": "Categoritza", | ||||
|     "categories": "Categories", | ||||
|     "category": "Categoria", | ||||
|     "noCategory": "No hi ha la categoria", | ||||
|     "noCategories": "No hi ha les categories", | ||||
|     "deleteCategoriesQuestion": "Suprimeixo les categories?", | ||||
|     "categoryDeleteWarning": "Totes les aplicacions de les categories suprimides es marcaran com a no categoritzades.", | ||||
|     "addCategory": "Afegeix una categoria", | ||||
|     "label": "Nom", | ||||
|     "language": "Idioma", | ||||
|     "copiedToClipboard": "Copiat al porta-retalls", | ||||
|     "storagePermissionDenied": "Permís d'emmagatzematge denegat", | ||||
|     "selectedCategorizeWarning": "Açò substituirà els paràmetres de categorització per a les aplicacions selecionades.", | ||||
|     "filterAPKsByRegEx": "Filtra les APKs per l'expressió regular", | ||||
|     "removeFromObtainium": "Suprimeix d'Obtainium", | ||||
|     "uninstallFromDevice": "Desinstal·la del dispositiu", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Només funciona per a aplicacions amb la detecció de versions desactivada.", | ||||
|     "releaseDateAsVersion": "Usa la data de llançament com a cadena de la versió", | ||||
|     "releaseTitleAsVersion": "Usa el títol com a cadena de la versió", | ||||
|     "releaseDateAsVersionExplanation": "Aquest opció només s'hauria d'usar per a aplicacions en què la detecció de la versió no funciona correctament però disposem de la data de publicació.", | ||||
|     "changes": "Canvis", | ||||
|     "releaseDate": "Data de publicació", | ||||
|     "importFromURLsInFile": "Importa els URLs des d'un fitxer (com ara OPML)", | ||||
|     "versionDetectionExplanation": "Concilia la cadena de la versió amb la versió detectada del Sistema Operatiu", | ||||
|     "versionDetection": "Detecció de la versió", | ||||
|     "standardVersionDetection": "Detecció de la versió estàndard", | ||||
|     "groupByCategory": "Agrupa per categories", | ||||
|     "autoApkFilterByArch": "Intenta filtrar les APKs per l'aquitectura de la CPU, si és possible", | ||||
|     "autoLinkFilterByArch": "Intenta filtrar els enllaços per l'aquitectura de la CPU, si és possible", | ||||
|     "overrideSource": "Força la font", | ||||
|     "dontShowAgain": "No ho tornis a mostrar", | ||||
|     "dontShowTrackOnlyWarnings": "No mostris avisos de les aplicacions 'només per a seguiment'", | ||||
|     "dontShowAPKOriginWarnings": "No mostris avisos dels orígens de les APKs", | ||||
|     "moveNonInstalledAppsToBottom": "Desplaça les aplicacions no instal·lades al capdavall de les aplicacions", | ||||
|     "gitlabPATLabel": "Token d'accés personal a GitLab", | ||||
|     "about": "Quant a", | ||||
|     "requiresCredentialsInSettings": "{} requereix credencials addicionals (a Paràmetres)", | ||||
|     "checkOnStart": "Comprova si hi ha actualitzacions en iniciar Obtainium", | ||||
|     "tryInferAppIdFromCode": "Intenta deduir l'ID de l'aplicació des del codi font", | ||||
|     "removeOnExternalUninstall": "Suprimeix de forma automàtica les aplicacions desinstal·lades externament", | ||||
|     "pickHighestVersionCode": "Selecciona de forma automàtica la versió superior de l'APK", | ||||
|     "checkUpdateOnDetailPage": "Comprova les actualitzacions en obrir la pàgina de detalls de l'aplicació", | ||||
|     "disablePageTransitions": "Inhabilita les animacions de transició de pàgina", | ||||
|     "reversePageTransitions": "Inverteix les animacions de transició de pàgina", | ||||
|     "minStarCount": "Nombre mínim d'estrelles", | ||||
|     "addInfoBelow": "Afegeix aquesta informació a sota.", | ||||
|     "addInfoInSettings": "Afegeix aquesta informació a Paràmetres.", | ||||
|     "githubSourceNote": "La limitació de peticions a GitHub es pot evitar amb una clau API.", | ||||
|     "sortByLastLinkSegment": "Ordena per 'només el darrer fragment de l'enllaç'", | ||||
|     "filterReleaseNotesByRegEx": "Filtra les notes de la publicació de la versió per una expressió regular", | ||||
|     "customLinkFilterRegex": "Filtre personalitzat de l'enllaç de l'APK per una expressió regular (Per_defecte '.apk$')", | ||||
|     "appsPossiblyUpdated": "S'ha intentat l'actualització de l'aplicació", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Notifica l'usuari que les actualitzacions per a una o més aplicacions podrien haver-se fet en segon pla", | ||||
|     "xWasPossiblyUpdatedToY": "{} podria haver-se actualitzat a {}.", | ||||
|     "enableBackgroundUpdates": "Habilita les actualizacions en segon pla", | ||||
|     "backgroundUpdateReqsExplanation": "Les actualitzacions en segon pla és possible que no estiguin disponibles per a totes les aplicacions.", | ||||
|     "backgroundUpdateLimitsExplanation": "Les instal·lacions en segon pla reexides només es poden comprovar amb Obtainium obert.", | ||||
|     "verifyLatestTag": "Comprova l'etiqueta 'Latest' (última versió)", | ||||
|     "intermediateLinkRegex": "Filtra per un enllaç 'intermediari' per anar-hi", | ||||
|     "filterByLinkText": "Filtra els enllaços pel text de l'enllaç", | ||||
|     "intermediateLinkNotFound": "No s'ha trobat l'enllaç intermediari", | ||||
|     "intermediateLink": "Enllaç intermediari", | ||||
|     "exemptFromBackgroundUpdates": "Exempta d'actualitzacions en segon pla (si han estat habilitades)", | ||||
|     "bgUpdatesOnWiFiOnly": "Inhabilita les actualitzacions en segon pla sense Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Inhabilita les actualitzacions en segon pla quan no s'estigui carregant el mòbil", | ||||
|     "autoSelectHighestVersionCode": "Selecciona de forma automàtica la versió més recent de l'APK", | ||||
|     "versionExtractionRegEx": "Extracció de la cadena de la versió amb una expressió regular", | ||||
|     "trimVersionString": "Retalla la cadena de la versió amb una expressió regular", | ||||
|     "matchGroupToUseForX": "Grup de coincidència a usar per a \"{}\"", | ||||
|     "matchGroupToUse": "Grup de coincidència a usar per a l'extracció de la cadena de la versió amb una expressió regular", | ||||
|     "highlightTouchTargets": "Ressalta els elements de selecció menys obvis", | ||||
|     "pickExportDir": "Selecciona el directori d'exportació", | ||||
|     "autoExportOnChanges": "Exporta automàticament quan hi hagi canvis", | ||||
|     "includeSettings": "Inclou paràmetres", | ||||
|     "filterVersionsByRegEx": "Filtra les versions per una expressió regular", | ||||
|     "trySelectingSuggestedVersionCode": "Prova a seleccionar la versió de l'APK suggerida", | ||||
|     "dontSortReleasesList": "Mantén l'ordre de publicació de l'API", | ||||
|     "reverseSort": "Ordre invers", | ||||
|     "takeFirstLink": "Usa el primer enllaç", | ||||
|     "skipSort": "Omet l'ordre", | ||||
|     "debugMenu": "Menú de depuració", | ||||
|     "bgTaskStarted": "S'ha iniciat la tasca en segon pla (revisa-ho als registres).", | ||||
|     "runBgCheckNow": "Executa la comprovació d'actualitzacions en segon pla", | ||||
|     "versionExtractWholePage": "Aplica l'extracció de la cadena de la versió amb una expressió regular a tota la pàgina", | ||||
|     "installing": "Instal·lant", | ||||
|     "skipUpdateNotifications": "No notifiquis les actualitzacions", | ||||
|     "updatesAvailableNotifChannel": "Actualitzacions disponibles", | ||||
|     "appsUpdatedNotifChannel": "Aplicacions actualitzades", | ||||
|     "appsPossiblyUpdatedNotifChannel": "S'ha intentat actualitzar l'aplicació", | ||||
|     "errorCheckingUpdatesNotifChannel": "Error en cercar actualitzacions", | ||||
|     "appsRemovedNotifChannel": "Aplicacions suprimides", | ||||
|     "downloadingXNotifChannel": "Descarregant {}", | ||||
|     "completeAppInstallationNotifChannel": "Instal·lació finalitzada", | ||||
|     "checkingForUpdatesNotifChannel": "S'estan cercant actualitzacions", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Comprova les actualitzacions només per a aplicacions instal·lades o en seguiment", | ||||
|     "supportFixedAPKURL": "Suport per als URLs fixos de l'APK", | ||||
|     "selectX": "Selecciona {}", | ||||
|     "parallelDownloads": "Permet les descàrregues paralel·les", | ||||
|     "useShizuku": "Usa Shizuku o Sui per instal·lar", | ||||
|     "shizukuBinderNotFound": "Shizuku no s'està executant", | ||||
|     "shizukuOld": "Versió antiga de Shizuku (<11) - Actualitza-la", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku s'executa en Android < 8.1 amb ADB - Actualitza Android o usa Sui com a alternativa", | ||||
|     "shizukuPretendToBeGooglePlay": "Defineix Google Play com a font d'instal·lació (si uses Shizuku)", | ||||
|     "useSystemFont": "Usa la font del sistema", | ||||
|     "useVersionCodeAsOSVersion": "Usa la versió de l'aplicació com a versió detectada del Sistema Operatiu", | ||||
|     "requestHeader": "Capçalera de sol·licitud", | ||||
|     "useLatestAssetDateAsReleaseDate": "Usa el darrer recurs carregat com a data de llançament", | ||||
|     "defaultPseudoVersioningMethod": "Mètode de pseudo-versionat predeterminat", | ||||
|     "partialAPKHash": "Hash de l'APK parcial", | ||||
|     "APKLinkHash": "Hash de l'enllaç de l'APK", | ||||
|     "directAPKLink": "Enllaç de l'APK directe", | ||||
|     "pseudoVersionInUse": "S'està usant una pseudo-versió", | ||||
|     "installed": "Instal·lada", | ||||
|     "latest": "Versió més recent", | ||||
|     "invertRegEx": "Inverteix l'expressió regular", | ||||
|     "note": "Nota", | ||||
|     "selfHostedNote": "El desplegable «{}» es pot usar per accedir a instàncies autoallotjades/personalitzades de qualsevol font.", | ||||
|     "badDownload": "L'APK no s'ha pogut analitzar (incompatible o descàrrega parcial)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Comparteix les aplicacions noves amb AppVerifier (si està instal·lat)", | ||||
|     "appVerifierInstructionToast": "Comparteix amb AppVerifier i torna aquí quan estigui fet.", | ||||
|     "wiki": "Ajuda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configuració de les aplicacions crowdsourcing (usa-ho sota la teva responsabilitat)", | ||||
|     "crowdsourcedConfigsShort": "Configuració de les aplicacions crowdsourcing", | ||||
|     "allowInsecure": "Permet les sol·licituds HTTP insegures", | ||||
|     "stayOneVersionBehind": "Roman a la versió anterior a l'última", | ||||
|     "refreshBeforeDownload": "Actualitza les dades de l'aplicació abans de descarregar-la", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nom", | ||||
|     "smartname": "Nom (smart)", | ||||
|     "sortMethod": "Mètode d'ordenació", | ||||
|     "welcome": "Benvinguda", | ||||
|     "documentationLinksNote": "La pàgina GitHub d'Obtainium enllaçada a sota conté enllaços a vídeos, articles, debats i altres recursos que t'ajudaran a entendre com usar l'aplicació.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "¿Suprimeixo l'aplicació?", | ||||
|         "other": "¿Suprimeixo les aplicacions?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Massa peticions (límit excedit), torna-hi en {} minut", | ||||
|         "other": "Massa peticions (límit excedit), torna-hi en {} minuts" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "La comprovació d'actualitzacions en segon pla ha trobat un {}, es tornarà a provar en {} minut", | ||||
|         "other": "La comprovació d'actualitzacions en segon pla ha trobat un {}, es tornarà a provar en {} minuts" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "La comprovació d'actualitzacions en segon pla ha trobat {} actualització, t'ho notificarem si cal", | ||||
|         "other": "La comprovació d'actualitzacions en segon pla ha trobat {} actualitzacions, t'ho notificarem si cal" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} Aplicació", | ||||
|         "other": "{} Aplicacions" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URLs" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Minut", | ||||
|         "other": "{} Minuts" | ||||
|     }, | ||||
|     "hour": { | ||||
|         "one": "{} Hora", | ||||
|         "other": "{} Hores" | ||||
|     }, | ||||
|     "day": { | ||||
|         "one": "{} Dia", | ||||
|         "other": "{} Dies" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "Suprimit {n} registre (anterior a = {before}, posterior a = {after})", | ||||
|         "other": "Suprimits {n} registres (anteriors a = {before}, posteriors a = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} i 1 aplicació més tenen actualitzacions.", | ||||
|         "other": "{} i {} aplicacions més tenen actualitzacions." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} i 1 aplicació més s'han actualitzat.", | ||||
|         "other": "{} i {} aplicacions més s'han actualitzat." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Error en actualitzar {} i 1 aplicació més.", | ||||
|         "other": "No s'ha pogut actualizar {} i {} aplicacions més." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} i 1 aplicació més podrien haver estat actualizades.", | ||||
|         "other": "{} i {} aplicacions més podrien haver estat actualitzades." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APKs" | ||||
|     } | ||||
| } | ||||
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Odinstalovat ze zařízení", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Funguje pouze pro aplikace s vypnutou detekcí verze.", | ||||
|     "releaseDateAsVersion": "Použít datum vydání jako verzi", | ||||
|     "releaseTitleAsVersion": "Použít název verze jako řetězec verze", | ||||
|     "releaseDateAsVersionExplanation": "Tato možnost by měla být použita pouze u aplikace, kde detekce verzí nefunguje správně, ale je k dispozici datum vydání.", | ||||
|     "changes": "Změny", | ||||
|     "releaseDate": "Datum vydání", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Standardní detekce verze", | ||||
|     "groupByCategory": "Seskupit podle kategorie", | ||||
|     "autoApkFilterByArch": "Pokud je to možné, pokuste se filtrovat soubory APK podle architektury procesoru", | ||||
|     "autoLinkFilterByArch": "Pokus o filtrování odkazů podle architektury procesoru, pokud je to možné.", | ||||
|     "overrideSource": "Přepsat zdroj", | ||||
|     "dontShowAgain": "Nezobrazovat znovu", | ||||
|     "dontShowTrackOnlyWarnings": "Nezobrazovat varování pro 'Jen sledované'", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "Připojený odkaz", | ||||
|     "exemptFromBackgroundUpdates": "Vyloučit z aktualizací na pozadí (je-li povoleno)", | ||||
|     "bgUpdatesOnWiFiOnly": "Deaktivovat aktualizace na pozadí, pokud není k dispozici Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Zakázat aktualizace na pozadí, když se nenabíjí", | ||||
|     "autoSelectHighestVersionCode": "Automaticky vybrat nejvyšší verzi APK", | ||||
|     "versionExtractionRegEx": "Extrakce verze pomocí RegEx", | ||||
|     "trimVersionString": "Oříznutí řetězce verze pomocí příkazu RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Sdílejte do aplikace AppVerifier a po dokončení se sem vraťte.", | ||||
|     "wiki": "Nápověda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfigurace aplikací s využitím crowdsourcingu (použití na vlastní nebezpečí)", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Název", | ||||
|     "smartname": "Název (Smart)", | ||||
|     "sortMethod": "Metoda třídění", | ||||
|     "welcome": "Vítejte na", | ||||
|     "documentationLinksNote": "Níže odkazovaná stránka Obtainium GitHub obsahuje odkazy na videa, články, diskuse a další zdroje, které vám pomohou pochopit, jak aplikaci používat.", | ||||
|     "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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Afinstaller fra enhed", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Virker kun for apps med versionsregistrering deaktiveret.", | ||||
|     "releaseDateAsVersion": "Brug udgivelsesdato som versionsstreng", | ||||
|     "releaseTitleAsVersion": "Brug udgivelsestitel som versionsstreng", | ||||
|     "releaseDateAsVersionExplanation": "Denne indstilling bør kun bruges til apps, hvor versionsregistrering ikke virker korrekt, men hvor en udgivelsesdato er tilgængelig.", | ||||
|     "changes": "Ændringer", | ||||
|     "releaseDate": "Udgivelsesdato", | ||||
| @@ -222,11 +221,12 @@ | ||||
|     "standardVersionDetection": "Standard versionsregistrering", | ||||
|     "groupByCategory": "Gruppér efter kategori", | ||||
|     "autoApkFilterByArch": "Forsøg at filtrere APK'er efter CPU-arkitektur, hvis muligt", | ||||
|     "autoLinkFilterByArch": "Forsøg at filtrere links efter CPU-arkitektur, hvis det er 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)", | ||||
| @@ -239,43 +239,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", | ||||
|     "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", | ||||
| @@ -286,7 +287,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", | ||||
| @@ -296,8 +297,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", | ||||
| @@ -311,10 +312,20 @@ | ||||
|     "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": "Forbliv én version bagud den seneste", | ||||
|     "refreshBeforeDownload": "Opdater app-detaljer før download", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Navn", | ||||
|     "smartname": "Navn (Smart)", | ||||
|     "sortMethod": "Sorteringsmetode", | ||||
|     "welcome": "Velkommen", | ||||
|     "documentationLinksNote": "Obtainiums GitHub-side, som der linkes til nedenfor, indeholder links til videoer, artikler, diskussioner og andre ressourcer, som kan hjælpe dig med at forstå, hvordan du bruger appen.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Fjern app?", | ||||
|         "other": "Fjern apps?" | ||||
| @@ -353,7 +364,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.", | ||||
|   | ||||
| @@ -32,8 +32,8 @@ | ||||
|     "source": "Quelle", | ||||
|     "app": "App", | ||||
|     "appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind nur zur Nachverfolgung.", | ||||
|     "youPickedTrackOnly": "Sie haben die Option „Nur nachverfolgen“ gewählt.", | ||||
|     "trackOnlyAppDescription": "Die App wird auf Aktualisierungen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", | ||||
|     "youPickedTrackOnly": "Du hast die Option „Nur nachverfolgen“ gewählt.", | ||||
|     "trackOnlyAppDescription": "Die App wird auf neue verfügbare Versionen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", | ||||
|     "cancelled": "Abgebrochen", | ||||
|     "appAlreadyAdded": "App bereits hinzugefügt", | ||||
|     "alreadyUpToDateQuestion": "App bereits auf dem neuesten Stand?", | ||||
| @@ -85,8 +85,8 @@ | ||||
|     "filter": "Filter", | ||||
|     "filterApps": "Apps filtern", | ||||
|     "appName": "App-Name", | ||||
|     "author": "Autor", | ||||
|     "upToDateApps": "Apps mit aktueller Version", | ||||
|     "author": "Herausgebende", | ||||
|     "upToDateApps": "Apps mit aktuellster Version", | ||||
|     "nonInstalledApps": "Nicht installierte Apps", | ||||
|     "importExport": "Import/Export", | ||||
|     "settings": "Einstellungen", | ||||
| @@ -102,22 +102,22 @@ | ||||
|     "searchX": "{} suchen", | ||||
|     "noResults": "Keine Ergebnisse gefunden", | ||||
|     "importX": "{} importieren", | ||||
|     "importedAppsIdDisclaimer": "Importierte Apps werden manchmal fälschlicherweise als „Nicht installiert“ angezeigt. Um dies zu beheben, installieren Sie sie erneut über Obtainium. Dies hat keine Auswirkungen auf App-Daten. Es betrifft nur URL- und Drittanbieter-Importmethoden.", | ||||
|     "importedAppsIdDisclaimer": "Importierte Apps werden manchmal fälschlicherweise als „Nicht installiert“ angezeigt. Um dies zu beheben, installiere sie erneut über Obtainium. Dies hat keine Auswirkungen auf App-Daten. Es betrifft nur URL- und Drittanbieter-Importmethoden.", | ||||
|     "importErrors": "Importfehler", | ||||
|     "importedXOfYApps": "{} von {} Apps importiert.", | ||||
|     "followingURLsHadErrors": "Bei folgenden URLs traten Fehler auf:", | ||||
|     "selectURL": "URL auswählen", | ||||
|     "selectURLs": "URLs auswählen", | ||||
|     "pick": "Auswählen", | ||||
|     "theme": "Theme", | ||||
|     "theme": "Design", | ||||
|     "dark": "Dunkel", | ||||
|     "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", | ||||
|     "followSystem": "Systemstandard", | ||||
|     "followSystemThemeExplanation": "Das Abrufen des Systemdesigns ist unter Android < 10 nur mit Hilfe von Drittanbieterapps möglich", | ||||
|     "useBlackTheme": "Rein schwarzen Hintergrund verwenden", | ||||
|     "appSortBy": "App sortieren nach", | ||||
|     "authorName": "Autor/Name", | ||||
|     "nameAuthor": "Name/Autor", | ||||
|     "authorName": "Hrsg./Name", | ||||
|     "nameAuthor": "Name/Hrsg.", | ||||
|     "asAdded": "Wie hinzugefügt", | ||||
|     "appSortOrder": "App sortieren nach", | ||||
|     "ascending": "Aufsteigend", | ||||
| @@ -138,30 +138,30 @@ | ||||
|     "obtainiumExportHyphenatedLowercase": "Obtainium-Export", | ||||
|     "pickAnAPK": "APK auswählen", | ||||
|     "appHasMoreThanOnePackage": "{} verfügt über mehr als ein Paket:", | ||||
|     "deviceSupportsXArch": "Ihr Gerät unterstützt die CPU-Architektur {}.", | ||||
|     "deviceSupportsFollowingArchs": "Ihr Gerät unterstützt die folgenden CPU-Architekturen:", | ||||
|     "deviceSupportsXArch": "Dein Gerät unterstützt die CPU-Architektur {}.", | ||||
|     "deviceSupportsFollowingArchs": "Dein Gerät unterstützt die folgenden CPU-Architekturen:", | ||||
|     "warning": "Warnung", | ||||
|     "sourceIsXButPackageFromYPrompt": "Die App-Quelle ist '{}', aber das Release-Paket stammt von '{}'. Fortfahren?", | ||||
|     "updatesAvailable": "Aktualisierungen verfügbar", | ||||
|     "updatesAvailableNotifDescription": "Benachrichtigt den Nutzer, dass Aktualisierungen für eine oder mehrere von Obtainium verfolgte Apps verfügbar sind", | ||||
|     "updatesAvailableNotifDescription": "Benachrichtigt, wenn Aktualisierungen für eine oder mehrere von Obtainium verfolgte Apps verfügbar sind", | ||||
|     "noNewUpdates": "Keine neuen Aktualisierungen.", | ||||
|     "xHasAnUpdate": "{} hat eine Aktualisierung.", | ||||
|     "appsUpdated": "App wurde aktualisiert", | ||||
|     "appsNotUpdated": "Aktualisierung der Apps fehlgeschlagen", | ||||
|     "appsUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps im Hintergrund durchgeführt wurden", | ||||
|     "appsUpdatedNotifDescription": "Benachrichtigt, wenn Aktualisierungen für eine oder mehrere Apps im Hintergrund durchgeführt wurden", | ||||
|     "xWasUpdatedToY": "{} wurde auf {} aktualisiert.", | ||||
|     "xWasNotUpdatedToY": "Die Aktualisierung von {} auf {} ist fehlgeschlagen.", | ||||
|     "errorCheckingUpdates": "Fehler beim Prüfen auf Aktualisierungen", | ||||
|     "errorCheckingUpdatesNotifDescription": "Eine Benachrichtigung, die angezeigt wird, wenn die Prüfung der Hintergrundaktualisierung fehlschlägt", | ||||
|     "errorCheckingUpdatesNotifDescription": "Benachrichtigt, wenn die Prüfung der Hintergrundaktualisierung fehlgeschlagen ist", | ||||
|     "appsRemoved": "Apps entfernt", | ||||
|     "appsRemovedNotifDescription": "Benachrichtigt den Benutzer, dass eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden", | ||||
|     "appsRemovedNotifDescription": "Benachrichtigt, wenn eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden", | ||||
|     "xWasRemovedDueToErrorY": "{} wurde aufgrund des folgenden Fehlers entfernt: {}", | ||||
|     "completeAppInstallation": "App-Installation abschließen", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium muss geöffnet sein, um Apps zu installieren", | ||||
|     "completeAppInstallationNotifDescription": "Aufforderung an den Benutzer, zu Obtainium zurückzukehren, um die Installation einer App abzuschließen", | ||||
|     "completeAppInstallationNotifDescription": "Aufforderung zu Obtainium zurückzukehren, um die Installation einer App abzuschließen", | ||||
|     "checkingForUpdates": "Nach Aktualisierungen suchen", | ||||
|     "checkingForUpdatesNotifDescription": "Vorübergehende Benachrichtigung, die bei der Suche nach Aktualisierungen angezeigt wird", | ||||
|     "pleaseAllowInstallPerm": "Bitte erlauben Sie Obtainium die Installation von Apps", | ||||
|     "pleaseAllowInstallPerm": "Bitte erlaube Obtainium die Installation von Apps", | ||||
|     "trackOnly": "Nur nachverfolgen", | ||||
|     "errorWithHttpStatusCode": "Fehler {}", | ||||
|     "versionCorrectionDisabled": "Versionskorrektur deaktiviert (Plugin scheint nicht zu funktionieren)", | ||||
| @@ -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", | ||||
| @@ -192,7 +190,7 @@ | ||||
|     "downloadX": "{} herunterladen", | ||||
|     "downloadedX": "{} heruntergeladen", | ||||
|     "releaseAsset": "Release-Asset", | ||||
|     "downloadNotifDescription": "Benachrichtigt den Nutzer über den Fortschritt beim Herunterladen einer App", | ||||
|     "downloadNotifDescription": "Zeigt den Fortschritt beim Herunterladen einer App", | ||||
|     "noAPKFound": "Keine APK gefunden", | ||||
|     "noVersionDetection": "Keine Versionserkennung", | ||||
|     "categorize": "Kategorisieren", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Vom Gerät deinstallieren", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Funktioniert nur bei Apps mit deaktivierter Versionserkennung.", | ||||
|     "releaseDateAsVersion": "Veröffentlichungsdatum als Version verwenden", | ||||
|     "releaseTitleAsVersion": "Versionstitel als Versionsstring verwenden", | ||||
|     "releaseDateAsVersionExplanation": "Diese Option sollte nur für Apps verwendet werden, bei denen die Versionserkennung nicht korrekt funktioniert, aber ein Veröffentlichungsdatum verfügbar ist.", | ||||
|     "changes": "Änderungen", | ||||
|     "releaseDate": "Veröffentlichungsdatum", | ||||
| @@ -222,9 +221,10 @@ | ||||
|     "standardVersionDetection": "Standardversionserkennung", | ||||
|     "groupByCategory": "Nach Kategorie gruppieren", | ||||
|     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", | ||||
|     "autoLinkFilterByArch": "Versuchen Sie, Links nach CPU-Architektur zu filtern, wenn möglich", | ||||
|     "overrideSource": "Quelle überschreiben", | ||||
|     "dontShowAgain": "Nicht noch einmal anzeigen", | ||||
|     "dontShowTrackOnlyWarnings": "Warnung für 'Nur nachverfolgen' nicht anzeigen", | ||||
|     "dontShowTrackOnlyWarnings": "Warnung für „Nur nachverfolgen“ nicht anzeigen", | ||||
|     "dontShowAPKOriginWarnings": "Warnung für APK-Herkunft nicht anzeigen", | ||||
|     "moveNonInstalledAppsToBottom": "Nicht installierte Apps ans Ende der App-Ansicht verschieben", | ||||
|     "gitlabPATLabel": "Persönlicher Zugangstoken für GitLab", | ||||
| @@ -232,7 +232,7 @@ | ||||
|     "requiresCredentialsInSettings": "{}: Benötigt zusätzliche Anmeldedaten (in den Einstellungen)", | ||||
|     "checkOnStart": "Einmalig beim Start überprüfen", | ||||
|     "tryInferAppIdFromCode": "Versuchen, die App-ID aus dem Quellcode zu ermitteln", | ||||
|     "removeOnExternalUninstall": "Automatisches Entfernen von extern deinstallierten Apps", | ||||
|     "removeOnExternalUninstall": "Extern deinstallierte Apps automatische entfernen", | ||||
|     "pickHighestVersionCode": "Automatische Auswahl des APK mit höchstem Versionscode", | ||||
|     "checkUpdateOnDetailPage": "Nach Aktualisierungen suchen, wenn eine App-Detailseite geöffnet wird", | ||||
|     "disablePageTransitions": "Animationen für Seitenübergänge deaktivieren", | ||||
| @@ -242,25 +242,26 @@ | ||||
|     "addInfoInSettings": "Diese Information in den Einstellungen hinzufügen.", | ||||
|     "githubSourceNote": "Die GitHub-Ratenbegrenzung kann mit einem API-Schlüssel umgangen werden.", | ||||
|     "sortByLastLinkSegment": "Nur nach dem letzten Teil des Links sortieren", | ||||
|     "filterReleaseNotesByRegEx": "Versionshinweise nach regulärem Ausdruck filtern", | ||||
|     "filterReleaseNotesByRegEx": "Versionshinweise nach regulärem Ausdruck\nfiltern", | ||||
|     "customLinkFilterRegex": "Benutzerdefinierter APK-Linkfilter durch regulären Ausdruck (Standard '.apk$')", | ||||
|     "appsPossiblyUpdated": "App-Aktualisierungen wurden versucht", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps möglicherweise im Hintergrund durchgeführt wurden", | ||||
|     "xWasPossiblyUpdatedToY": "{} wurde möglicherweise aktualisiert auf {}.", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Benachrichtigt, dass möglicherweise eine oder mehrere Apps im Hintergrund aktualisiert wurden", | ||||
|     "xWasPossiblyUpdatedToY": "{} wurde eventuell auf Version {} aktualisiert.", | ||||
|     "enableBackgroundUpdates": "Hintergrundaktualisierungen aktivieren", | ||||
|     "backgroundUpdateReqsExplanation": "Die Hintergrundaktualisierung ist möglicherweise nicht für alle Apps möglich.", | ||||
|     "backgroundUpdateReqsExplanation": "Die Hintergrundaktualisierung ist unter Umständen nicht für alle Apps möglich.", | ||||
|     "backgroundUpdateLimitsExplanation": "Der Erfolg einer Hintergrundinstallation kann nur festgestellt werden, wenn Obtainium geöffnet wird.", | ||||
|     "verifyLatestTag": "„Latest“-Tag überprüfen", | ||||
|     "intermediateLinkRegex": "Filter für einen „Zwischen“-Link, der zuerst besucht werden soll", | ||||
|     "filterByLinkText": "Links durch Linktext filtern", | ||||
|     "intermediateLinkNotFound": "„Zwischen“-Link nicht gefunden", | ||||
|     "intermediateLink": "„Zwischen“-Link", | ||||
|     "exemptFromBackgroundUpdates": "Ausschluss von Hintergrundaktualisierungen (falls aktiviert)", | ||||
|     "bgUpdatesOnWiFiOnly": "Hintergrundaktualisierungen deaktivieren, wenn kein WLAN vorhanden ist", | ||||
|     "exemptFromBackgroundUpdates": "Von Hintergrundaktualisierungen (falls aktiviert) ausschließen", | ||||
|     "bgUpdatesOnWiFiOnly": "Hintergrundaktualisierungen nur im WLAN ausführen", | ||||
|     "bgUpdatesWhileChargingOnly": "Hintergrundaktualisierungen nur ausführen, während das Gerät geladen wird", | ||||
|     "autoSelectHighestVersionCode": "Automatisch höchste APK-Version auswählen", | ||||
|     "versionExtractionRegEx": "Versionsextraktion per RegEx", | ||||
|     "trimVersionString": "Versionszeichenfolge mit RegEx kürzen", | ||||
|     "matchGroupToUseForX": "Zu verwendende Abgleichsgruppe für „{}“", | ||||
|     "matchGroupToUseForX": "Zu verwendende Abgleichsgruppe für\n„{}“", | ||||
|     "matchGroupToUse": "Zu verwendende Gruppe abgleichen", | ||||
|     "highlightTouchTargets": "Weniger offensichtliche Touch-Ziele hervorheben", | ||||
|     "pickExportDir": "Export-Verzeichnis wählen", | ||||
| @@ -268,7 +269,7 @@ | ||||
|     "includeSettings": "Einstellungen einbeziehen", | ||||
|     "filterVersionsByRegEx": "Versionen nach regulären Ausdrücken filtern", | ||||
|     "trySelectingSuggestedVersionCode": "Versuchen, den vorgeschlagenen APK-Versionscode auszuwählen", | ||||
|     "dontSortReleasesList": "Freigaberelease von der API ordern", | ||||
|     "dontSortReleasesList": "Releases-Sortierung der API beibehalten", | ||||
|     "reverseSort": "Umgekehrtes Sortieren", | ||||
|     "takeFirstLink": "Ersten Link verwenden", | ||||
|     "skipSort": "Sortieren überspringen", | ||||
| @@ -292,12 +293,12 @@ | ||||
|     "parallelDownloads": "Parallele Downloads erlauben", | ||||
|     "useShizuku": "Shizuku oder Sui zur Installation verwenden", | ||||
|     "shizukuBinderNotFound": "Kompatibler Shizuku-Dienst wurde nicht gefunden", | ||||
|     "shizukuOld": "Alte Shizuku-Version (< 11) - aktualisieren Sie sie", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku läuft auf Android < 8.1 mit ADB - aktualisieren Sie Android oder verwenden Sie stattdessen Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Google Play als Installationsquelle festlegen (wenn Shizuku verwendet wird)", | ||||
|     "shizukuOld": "Veraltete Shizuku-Version (< 11) - bitte aktualisiere sie", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku läuft auf Android < 8.1 mit ADB - aktualisiere die Android-Version oder verwende stattdessen Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "(Mittels Shizuku) Google Play als Installationsquelle registrieren", | ||||
|     "useSystemFont": "Systemschriftart verwenden", | ||||
|     "useVersionCodeAsOSVersion": "App-Version als erkannte Version vom Betriebssystem verwenden", | ||||
|     "requestHeader": "Kopfzeile anfordern", | ||||
|     "useVersionCodeAsOSVersion": "Versionscode (versionCode) als erkannte Version vom Betriebssystem verwenden", | ||||
|     "requestHeader": "Kopfzeile („Header“) anfordern", | ||||
|     "useLatestAssetDateAsReleaseDate": "Letzten Asset-Upload als Veröffentlichungsdatum verwenden", | ||||
|     "defaultPseudoVersioningMethod": "Standardmäßiges Verfahren zur Pseudo-Versionierung", | ||||
|     "partialAPKHash": "Partieller APK-Hash", | ||||
| @@ -309,27 +310,37 @@ | ||||
|     "invertRegEx": "Regulären Ausdruck invertieren", | ||||
|     "note": "Hinweis", | ||||
|     "selfHostedNote": "Das „{}“-Drop-down-Menü kann verwendet werden, um selbst gehostete/angepasste Instanzen einer beliebigen Quelle zu erreichen.", | ||||
|     "badDownload": "Die APK konnte nicht geparst werden (inkompatibler oder teilweiser Download)", | ||||
|     "badDownload": "Die APK konnte nicht gelesen werden (inkompatibler oder unfertiger Download)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Neue Apps mit AppVerifier teilen (falls verfügbar)", | ||||
|     "appVerifierInstructionToast": "Geben Sie die Daten an AppVerifier weiter und kehren Sie dann hierher zurück, wenn Sie fertig sind.", | ||||
|     "appVerifierInstructionToast": "Teile die Daten mit AppVerifier und kehre im Anschluss hierher zurück.", | ||||
|     "wiki": "Hilfe/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App-Konfigurationen (Verwendung auf eigene Gefahr)", | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App-Konfigurationen", | ||||
|     "allowInsecure": "Unsichere HTTP-Anfragen zulassen", | ||||
|     "stayOneVersionBehind": "Eine Version hinter der neuesten Version bleiben", | ||||
|     "refreshBeforeDownload": "App-Details vor dem Download aktualisieren", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Name", | ||||
|     "smartname": "Name (Smart)", | ||||
|     "sortMethod": "Sortierverfahren", | ||||
|     "welcome": "Willkommen", | ||||
|     "documentationLinksNote": "Die unten verlinkte GitHub-Seite von Obtainium enthält Links zu Videos, Artikeln, Diskussionen und anderen Ressourcen, die Ihnen helfen werden, die Verwendung der App zu verstehen.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App entfernen?", | ||||
|         "other": "Apps entfernen?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minute erneut", | ||||
|         "other": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minuten erneut" | ||||
|         "one": "Zu viele Anfragen (Rate begrenzt) – versuche es in {} Minute erneut", | ||||
|         "other": "Zu viele Anfragen (Rate begrenzt) – versuche es in {} Minuten erneut" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant", | ||||
|         "other": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minuten geplant" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierung – benachrichtigt den Benutzer, falls erforderlich", | ||||
|         "other": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierungen – benachrichtigt den Benutzer, falls erforderlich" | ||||
|         "one": "Die Aktualisierungsprüfung fand {} Aktualisierung – benachrichtigt, falls erforderlich", | ||||
|         "other": "Die Aktualisierungsprüfung fand {} Aktualisierungen – benachrichtigt, falls erforderlich" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} App", | ||||
| @@ -356,8 +367,8 @@ | ||||
|         "other": "{n} Logs gelöscht (vorher = {before}, nachher = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} und 1 weitere App haben Aktualisierungen.", | ||||
|         "other": "{} und {} weitere Apps haben Aktualisierungen." | ||||
|         "one": "{} und 1 weitere App können aktualisiert werden.", | ||||
|         "other": "{} und {} weitere Apps können aktualisiert werden." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} und 1 weitere App wurden aktualisiert.", | ||||
|   | ||||
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Malinstali el la disponaĵo", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Funkcias nur por apoj kun malaktiva versiodetekto.", | ||||
|     "releaseDateAsVersion": "Uzi eldondato kiel versioĉeno", | ||||
|     "releaseTitleAsVersion": "Use release title as version string", | ||||
|     "releaseDateAsVersionExplanation": "Tiu opcio devas esti uzata nur por apoj, por kiu la versiodetekto ne funkcias ĝuste, sed eldondato estas havebla.", | ||||
|     "changes": "Modifoj", | ||||
|     "releaseDate": "Eldondato", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Norma versiodetekto", | ||||
|     "groupByCategory": "Grupigi per kategorio", | ||||
|     "autoApkFilterByArch": "Provi filtri APKj per CPU-arkitekturo se ebla", | ||||
|     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||
|     "overrideSource": "Substitui la fonton", | ||||
|     "dontShowAgain": "Ne plu montri", | ||||
|     "dontShowTrackOnlyWarnings": "Ne plu montri 'Nur sekvita' avertojn", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "Pera ligilo", | ||||
|     "exemptFromBackgroundUpdates": "Escepti el la fonaj ĝisdatigoj (se aktiva)", | ||||
|     "bgUpdatesOnWiFiOnly": "Malaktivigi fonajn ĝisdatigojn se sen vifio", | ||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||
|     "autoSelectHighestVersionCode": "Aŭtomate selekti la plej ĵusan version de la APK-kodo", | ||||
|     "versionExtractionRegEx": "Ekstrakti la versioĉenon per regula esprimo", | ||||
|     "trimVersionString": "Mallongigi la versioĉenon per regula esprimo", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Diskonigu kun AppVerifier, poste revenu ĉi tie kiam preta.", | ||||
|     "wiki": "Helpo/Vikio", | ||||
|     "crowdsourcedConfigsLabel": "Komunumaj apo-agordoj (uzu kun singardo)", | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configurations", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Name", | ||||
|     "smartname": "Name (Smart)", | ||||
|     "sortMethod": "Sort Method", | ||||
|     "welcome": "Welcome", | ||||
|     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Forigi la aplikaĵon?", | ||||
|         "other": "Forigi la aplikaĵojn?" | ||||
|   | ||||
| @@ -1,110 +1,110 @@ | ||||
| { | ||||
|     "invalidURLForSource": "Not a valid {} App URL", | ||||
|     "invalidURLForSource": "Not a valid {} app URL", | ||||
|     "noReleaseFound": "Could not find a suitable release", | ||||
|     "noVersionFound": "Could not determine release version", | ||||
|     "urlMatchesNoSource": "URL does not match a known source", | ||||
|     "cantInstallOlderVersion": "Cannot install an older version of an App", | ||||
|     "appIdMismatch": "Downloaded package ID does not match existing App ID", | ||||
|     "cantInstallOlderVersion": "Cannot install an older version of an app", | ||||
|     "appIdMismatch": "Downloaded package ID does not match existing app ID", | ||||
|     "functionNotImplemented": "This class has not implemented this function", | ||||
|     "placeholder": "Placeholder", | ||||
|     "someErrors": "Some Errors Occurred", | ||||
|     "unexpectedError": "Unexpected Error", | ||||
|     "someErrors": "Some errors occurred", | ||||
|     "unexpectedError": "Unexpected error", | ||||
|     "ok": "Okay", | ||||
|     "and": "and", | ||||
|     "githubPATLabel": "GitHub Personal Access Token (Increases Rate Limit)", | ||||
|     "githubPATLabel": "GitHub personal access token (increases rate limit)", | ||||
|     "includePrereleases": "Include prereleases", | ||||
|     "fallbackToOlderReleases": "Fallback to older releases", | ||||
|     "filterReleaseTitlesByRegEx": "Filter Release Titles by Regular Expression", | ||||
|     "filterReleaseTitlesByRegEx": "Filter release titles by regular expression", | ||||
|     "invalidRegEx": "Invalid regular expression", | ||||
|     "noDescription": "No description", | ||||
|     "cancel": "Cancel", | ||||
|     "continue": "Continue", | ||||
|     "requiredInBrackets": "(Required)", | ||||
|     "requiredInBrackets": "(required)", | ||||
|     "dropdownNoOptsError": "ERROR: DROPDOWN MUST HAVE AT LEAST ONE OPT", | ||||
|     "colour": "Colour", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Custom", | ||||
|     "useMaterialYou": "Use Material You", | ||||
|     "githubStarredRepos": "GitHub Starred Repos", | ||||
|     "githubStarredRepos": "GitHub starred repos", | ||||
|     "uname": "Username", | ||||
|     "wrongArgNum": "Wrong number of arguments provided", | ||||
|     "xIsTrackOnly": "{} is Track-Only", | ||||
|     "xIsTrackOnly": "{} is track-only", | ||||
|     "source": "Source", | ||||
|     "app": "App", | ||||
|     "appsFromSourceAreTrackOnly": "Apps from this source are 'Track-Only'.", | ||||
|     "youPickedTrackOnly": "You have selected the 'Track-Only' option.", | ||||
|     "trackOnlyAppDescription": "The App will be tracked for updates, but Obtainium will not be able to download or install it.", | ||||
|     "appsFromSourceAreTrackOnly": "Apps from this source are 'track-only'.", | ||||
|     "youPickedTrackOnly": "You have selected the 'track-only' option.", | ||||
|     "trackOnlyAppDescription": "The app will be tracked for updates, but Obtainium will not be able to download or install it.", | ||||
|     "cancelled": "Cancelled", | ||||
|     "appAlreadyAdded": "App already added", | ||||
|     "alreadyUpToDateQuestion": "App Already up to Date?", | ||||
|     "alreadyUpToDateQuestion": "App already up to date?", | ||||
|     "addApp": "Add App", | ||||
|     "appSourceURL": "App Source URL", | ||||
|     "appSourceURL": "App source URL", | ||||
|     "error": "Error", | ||||
|     "add": "Add", | ||||
|     "searchSomeSourcesLabel": "Search (Some Sources Only)", | ||||
|     "searchSomeSourcesLabel": "Search (some sources only)", | ||||
|     "search": "Search", | ||||
|     "additionalOptsFor": "Additional Options for {}", | ||||
|     "supportedSources": "Supported Sources", | ||||
|     "trackOnlyInBrackets": "(Track-Only)", | ||||
|     "searchableInBrackets": "(Searchable)", | ||||
|     "additionalOptsFor": "Additional options for {}", | ||||
|     "supportedSources": "Supported sources", | ||||
|     "trackOnlyInBrackets": "(track-only)", | ||||
|     "searchableInBrackets": "(searchable)", | ||||
|     "appsString": "Apps", | ||||
|     "noApps": "No Apps", | ||||
|     "noAppsForFilter": "No Apps for Filter", | ||||
|     "noApps": "No apps", | ||||
|     "noAppsForFilter": "No apps for filter", | ||||
|     "byX": "By {}", | ||||
|     "percentProgress": "Progress: {}%", | ||||
|     "pleaseWait": "Please Wait", | ||||
|     "updateAvailable": "Update Available", | ||||
|     "notInstalled": "Not Installed", | ||||
|     "pleaseWait": "Please wait", | ||||
|     "updateAvailable": "Update available", | ||||
|     "notInstalled": "Not installed", | ||||
|     "pseudoVersion": "pseudo-version", | ||||
|     "selectAll": "Select All", | ||||
|     "selectAll": "Select all", | ||||
|     "deselectX": "Deselect {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} will be removed from Obtainium but remain installed on device.", | ||||
|     "removeSelectedAppsQuestion": "Remove Selected Apps?", | ||||
|     "removeSelectedApps": "Remove Selected Apps", | ||||
|     "removeSelectedAppsQuestion": "Remove selected apps?", | ||||
|     "removeSelectedApps": "Remove selected apps", | ||||
|     "updateX": "Update {}", | ||||
|     "installX": "Install {}", | ||||
|     "markXTrackOnlyAsUpdated": "Mark {}\n(Track-Only)\nas Updated", | ||||
|     "markXTrackOnlyAsUpdated": "Mark {}\n(track-only)\nas updated", | ||||
|     "changeX": "Change {}", | ||||
|     "installUpdateApps": "Install/Update Apps", | ||||
|     "installUpdateSelectedApps": "Install/Update Selected Apps", | ||||
|     "markXSelectedAppsAsUpdated": "Mark {} Selected Apps as Updated?", | ||||
|     "installUpdateApps": "Install/update apps", | ||||
|     "installUpdateSelectedApps": "Install/update selected apps", | ||||
|     "markXSelectedAppsAsUpdated": "Mark {} selected apps as updated?", | ||||
|     "no": "No", | ||||
|     "yes": "Yes", | ||||
|     "markSelectedAppsUpdated": "Mark Selected Apps as Updated", | ||||
|     "markSelectedAppsUpdated": "Mark selected apps as updated", | ||||
|     "pinToTop": "Pin to top", | ||||
|     "unpinFromTop": "Unpin from top", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Reset Install Status for Selected Apps?", | ||||
|     "installStatusOfXWillBeResetExplanation": "The install status of any selected Apps will be reset.\n\nThis can help when the App version shown in Obtainium is incorrect due to failed updates or other issues.", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Reset install status for selected apps?", | ||||
|     "installStatusOfXWillBeResetExplanation": "The install status of any selected apps will be reset.\n\nThis can help when the app version shown in Obtainium is incorrect due to failed updates or other issues.", | ||||
|     "customLinkMessage": "These links work on devices with Obtainium installed", | ||||
|     "shareAppConfigLinks": "Share app configuration as HTML link", | ||||
|     "shareSelectedAppURLs": "Share Selected App URLs", | ||||
|     "resetInstallStatus": "Reset Install Status", | ||||
|     "shareSelectedAppURLs": "Share selected app URLs", | ||||
|     "resetInstallStatus": "Reset install status", | ||||
|     "more": "More", | ||||
|     "removeOutdatedFilter": "Remove Out-of-Date App Filter", | ||||
|     "showOutdatedOnly": "Show Out-of-Date Apps Only", | ||||
|     "removeOutdatedFilter": "Remove out-of-date app filter", | ||||
|     "showOutdatedOnly": "Show out-of-date apps only", | ||||
|     "filter": "Filter", | ||||
|     "filterApps": "Filter Apps", | ||||
|     "appName": "App Name", | ||||
|     "filterApps": "Filter apps", | ||||
|     "appName": "App name", | ||||
|     "author": "Author", | ||||
|     "upToDateApps": "Up to Date Apps", | ||||
|     "nonInstalledApps": "Non-Installed Apps", | ||||
|     "upToDateApps": "Up to date apps", | ||||
|     "nonInstalledApps": "Non-installed apps", | ||||
|     "importExport": "Import/Export", | ||||
|     "settings": "Settings", | ||||
|     "exportedTo": "Exported to {}", | ||||
|     "obtainiumExport": "Obtainium Export", | ||||
|     "obtainiumExport": "Obtainium export", | ||||
|     "invalidInput": "Invalid input", | ||||
|     "importedX": "Imported {}", | ||||
|     "obtainiumImport": "Obtainium Import", | ||||
|     "importFromURLList": "Import from URL List", | ||||
|     "searchQuery": "Search Query", | ||||
|     "appURLList": "App URL List", | ||||
|     "obtainiumImport": "Obtainium import", | ||||
|     "importFromURLList": "Import from URL list", | ||||
|     "searchQuery": "Search query", | ||||
|     "appURLList": "App URL list", | ||||
|     "line": "Line", | ||||
|     "searchX": "Search {}", | ||||
|     "noResults": "No results found", | ||||
|     "importX": "Import {}", | ||||
|     "importedAppsIdDisclaimer": "Imported Apps may incorrectly show as \"Not Installed\".\nTo fix this, re-install them through Obtainium.\nThis should not affect App data.\n\nOnly affects URL and third-party import methods.", | ||||
|     "importErrors": "Import Errors", | ||||
|     "importedXOfYApps": "{} of {} Apps imported.", | ||||
|     "importedAppsIdDisclaimer": "Imported apps may incorrectly show as \"not installed\".\nTo fix this, re-install them through Obtainium.\nThis should not affect app data.\n\nOnly affects URL and third-party import methods.", | ||||
|     "importErrors": "Import errors", | ||||
|     "importedXOfYApps": "{} of {} apps imported.", | ||||
|     "followingURLsHadErrors": "The following URLs had errors:", | ||||
|     "selectURL": "Select URL", | ||||
|     "selectURLs": "Select URLs", | ||||
| @@ -112,26 +112,26 @@ | ||||
|     "theme": "Theme", | ||||
|     "dark": "Dark", | ||||
|     "light": "Light", | ||||
|     "followSystem": "Follow System", | ||||
|     "followSystem": "Follow system", | ||||
|     "followSystemThemeExplanation": "Following system theme is possible only by using third-party applications", | ||||
|     "useBlackTheme": "Use pure black dark theme", | ||||
|     "appSortBy": "App Sort By", | ||||
|     "authorName": "Author/Name", | ||||
|     "nameAuthor": "Name/Author", | ||||
|     "asAdded": "As Added", | ||||
|     "appSortOrder": "App Sort Order", | ||||
|     "appSortBy": "App sort by", | ||||
|     "authorName": "Author/name", | ||||
|     "nameAuthor": "Name/author", | ||||
|     "asAdded": "As added", | ||||
|     "appSortOrder": "App sort order", | ||||
|     "ascending": "Ascending", | ||||
|     "descending": "Descending", | ||||
|     "bgUpdateCheckInterval": "Background Update Checking Interval", | ||||
|     "neverManualOnly": "Never - Manual Only", | ||||
|     "bgUpdateCheckInterval": "Background update checking interval", | ||||
|     "neverManualOnly": "Never - manual only", | ||||
|     "appearance": "Appearance", | ||||
|     "showWebInAppView": "Show Source webpage in App view", | ||||
|     "pinUpdates": "Pin updates to top of Apps view", | ||||
|     "showWebInAppView": "Show source webpage in app view", | ||||
|     "pinUpdates": "Pin updates to top of apps view", | ||||
|     "updates": "Updates", | ||||
|     "sourceSpecific": "Source-Specific", | ||||
|     "appSource": "App Source", | ||||
|     "noLogs": "No Logs", | ||||
|     "appLogs": "App Logs", | ||||
|     "sourceSpecific": "Source-specific", | ||||
|     "appSource": "App source", | ||||
|     "noLogs": "No logs", | ||||
|     "appLogs": "App logs", | ||||
|     "close": "Close", | ||||
|     "share": "Share", | ||||
|     "appNotFound": "App not found", | ||||
| @@ -141,28 +141,28 @@ | ||||
|     "deviceSupportsXArch": "Your device supports the {} CPU architecture.", | ||||
|     "deviceSupportsFollowingArchs": "Your device supports the following CPU architectures:", | ||||
|     "warning": "Warning", | ||||
|     "sourceIsXButPackageFromYPrompt": "The App source is '{}' but the release package comes from '{}'. Continue?", | ||||
|     "updatesAvailable": "Updates Available", | ||||
|     "updatesAvailableNotifDescription": "Notifies the user that updates are available for one or more Apps tracked by Obtainium", | ||||
|     "sourceIsXButPackageFromYPrompt": "The app source is '{}' but the release package comes from '{}'. Continue?", | ||||
|     "updatesAvailable": "Updates available", | ||||
|     "updatesAvailableNotifDescription": "Notifies the user that updates are available for one or more apps tracked by Obtainium", | ||||
|     "noNewUpdates": "No new updates.", | ||||
|     "xHasAnUpdate": "{} has an update.", | ||||
|     "appsUpdated": "Apps Updated", | ||||
|     "appsUpdated": "Apps updated", | ||||
|     "appsNotUpdated": "Failed to update applications", | ||||
|     "appsUpdatedNotifDescription": "Notifies the user that updates to one or more Apps were applied in the background", | ||||
|     "appsUpdatedNotifDescription": "Notifies the user that updates to one or more apps were applied in the background", | ||||
|     "xWasUpdatedToY": "{} was updated to {}.", | ||||
|     "xWasNotUpdatedToY": "Failed to update {} to {}.", | ||||
|     "errorCheckingUpdates": "Error Checking for Updates", | ||||
|     "errorCheckingUpdates": "Error checking for updates", | ||||
|     "errorCheckingUpdatesNotifDescription": "A notification that shows when background update checking fails", | ||||
|     "appsRemoved": "Apps Removed", | ||||
|     "appsRemovedNotifDescription": "Notifies the user that one or more Apps were removed due to errors while loading them", | ||||
|     "appsRemoved": "Apps removed", | ||||
|     "appsRemovedNotifDescription": "Notifies the user that one or more apps were removed due to errors while loading them", | ||||
|     "xWasRemovedDueToErrorY": "{} was removed due to this error: {}", | ||||
|     "completeAppInstallation": "Complete App Installation", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium must be open to install Apps", | ||||
|     "completeAppInstallationNotifDescription": "Asks the user to return to Obtainium to finish installing an App", | ||||
|     "checkingForUpdates": "Checking for Updates", | ||||
|     "completeAppInstallation": "Complete app installation", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium must be open to install apps", | ||||
|     "completeAppInstallationNotifDescription": "Asks the user to return to Obtainium to finish installing an app", | ||||
|     "checkingForUpdates": "Checking for updates", | ||||
|     "checkingForUpdatesNotifDescription": "Transient notification that appears when checking for updates", | ||||
|     "pleaseAllowInstallPerm": "Please allow Obtainium to install Apps", | ||||
|     "trackOnly": "Track-Only", | ||||
|     "pleaseAllowInstallPerm": "Please allow Obtainium to install apps", | ||||
|     "trackOnly": "Track-only", | ||||
|     "errorWithHttpStatusCode": "Error {}", | ||||
|     "versionCorrectionDisabled": "Version correction disabled (plugin doesn't seem to work)", | ||||
|     "unknown": "Unknown", | ||||
| @@ -170,123 +170,124 @@ | ||||
|     "never": "Never", | ||||
|     "latestVersionX": "Latest: {}", | ||||
|     "installedVersionX": "Installed: {}", | ||||
|     "lastUpdateCheckX": "Last Update Check: {}", | ||||
|     "lastUpdateCheckX": "Last update check: {}", | ||||
|     "remove": "Remove", | ||||
|     "yesMarkUpdated": "Yes, Mark as Updated", | ||||
|     "fdroid": "F-Droid Official", | ||||
|     "appIdOrName": "App ID or Name", | ||||
|     "yesMarkUpdated": "Yes, mark as updated", | ||||
|     "fdroid": "F-Droid official", | ||||
|     "appIdOrName": "App ID or name", | ||||
|     "appId": "App ID", | ||||
|     "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", | ||||
|     "appWithIdOrNameNotFound": "No app was found with that ID or name", | ||||
|     "reposHaveMultipleApps": "Repos may contain multiple apps", | ||||
|     "fdroidThirdPartyRepo": "F-Droid third-party repo", | ||||
|     "install": "Install", | ||||
|     "markInstalled": "Mark Installed", | ||||
|     "markInstalled": "Mark installed", | ||||
|     "update": "Update", | ||||
|     "markUpdated": "Mark Updated", | ||||
|     "additionalOptions": "Additional Options", | ||||
|     "disableVersionDetection": "Disable Version Detection", | ||||
|     "noVersionDetectionExplanation": "This option should only be used for Apps where version detection does not work correctly.", | ||||
|     "markUpdated": "Mark updated", | ||||
|     "additionalOptions": "Additional options", | ||||
|     "disableVersionDetection": "Disable version detection", | ||||
|     "noVersionDetectionExplanation": "This option should only be used for apps where version detection does not work correctly.", | ||||
|     "downloadingX": "Downloading {}", | ||||
|     "downloadX": "Download {}", | ||||
|     "downloadedX": "Downloaded {}", | ||||
|     "releaseAsset": "Release Asset", | ||||
|     "downloadNotifDescription": "Notifies the user of the progress in downloading an App", | ||||
|     "releaseAsset": "Release asset", | ||||
|     "downloadNotifDescription": "Notifies the user of the progress in downloading an app", | ||||
|     "noAPKFound": "No APK found", | ||||
|     "noVersionDetection": "No version detection", | ||||
|     "categorize": "Categorize", | ||||
|     "categories": "Categories", | ||||
|     "category": "Category", | ||||
|     "noCategory": "No Category", | ||||
|     "noCategories": "No Categories", | ||||
|     "deleteCategoriesQuestion": "Delete Categories?", | ||||
|     "categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.", | ||||
|     "addCategory": "Add Category", | ||||
|     "noCategory": "No category", | ||||
|     "noCategories": "No categories", | ||||
|     "deleteCategoriesQuestion": "Delete categories?", | ||||
|     "categoryDeleteWarning": "All apps in deleted categories will be set to uncategorized.", | ||||
|     "addCategory": "Add category", | ||||
|     "label": "Label", | ||||
|     "language": "Language", | ||||
|     "copiedToClipboard": "Copied to Clipboard", | ||||
|     "copiedToClipboard": "Copied to clipboard", | ||||
|     "storagePermissionDenied": "Storage permission denied", | ||||
|     "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.", | ||||
|     "filterAPKsByRegEx": "Filter APKs by Regular Expression", | ||||
|     "selectedCategorizeWarning": "This will replace any existing category settings for the selected apps.", | ||||
|     "filterAPKsByRegEx": "Filter APKs by regular expression", | ||||
|     "removeFromObtainium": "Remove from Obtainium", | ||||
|     "uninstallFromDevice": "Uninstall from Device", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.", | ||||
|     "uninstallFromDevice": "Uninstall from device", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Only works for apps with version detection disabled.", | ||||
|     "releaseDateAsVersion": "Use release date as version string", | ||||
|     "releaseDateAsVersionExplanation": "This option should only be used for Apps where version detection does not work correctly, but a release date is available.", | ||||
|     "releaseTitleAsVersion": "Use release title as version string", | ||||
|     "releaseDateAsVersionExplanation": "This option should only be used for apps where version detection does not work correctly, but a release date is available.", | ||||
|     "changes": "Changes", | ||||
|     "releaseDate": "Release Date", | ||||
|     "importFromURLsInFile": "Import from URLs in File (like OPML)", | ||||
|     "releaseDate": "Release date", | ||||
|     "importFromURLsInFile": "Import from URLs in file (like OPML)", | ||||
|     "versionDetectionExplanation": "Reconcile version string with version detected from OS", | ||||
|     "versionDetection": "Version Detection", | ||||
|     "versionDetection": "Version detection", | ||||
|     "standardVersionDetection": "Standard version detection", | ||||
|     "groupByCategory": "Group by Category", | ||||
|     "groupByCategory": "Group by category", | ||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||
|     "overrideSource": "Override Source", | ||||
|     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||
|     "overrideSource": "Override source", | ||||
|     "dontShowAgain": "Don't show this again", | ||||
|     "dontShowTrackOnlyWarnings": "Don't show 'Track-Only' warnings", | ||||
|     "dontShowTrackOnlyWarnings": "Don't show 'track-only' warnings", | ||||
|     "dontShowAPKOriginWarnings": "Don't show APK origin warnings", | ||||
|     "moveNonInstalledAppsToBottom": "Move non-installed Apps to bottom of Apps view", | ||||
|     "gitlabPATLabel": "GitLab Personal Access Token", | ||||
|     "moveNonInstalledAppsToBottom": "Move non-installed apps to bottom of apps view", | ||||
|     "gitlabPATLabel": "GitLab personal access token", | ||||
|     "about": "About", | ||||
|     "requiresCredentialsInSettings": "{} needs additional credentials (in Settings)", | ||||
|     "checkOnStart": "Check for updates on startup", | ||||
|     "tryInferAppIdFromCode": "Try inferring App ID from source code", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "tryInferAppIdFromCode": "Try inferring app ID from source code", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an app detail page", | ||||
|     "disablePageTransitions": "Disable page transition animations", | ||||
|     "reversePageTransitions": "Reverse page transition animations", | ||||
|     "minStarCount": "Minimum Star Count", | ||||
|     "minStarCount": "Minimum star count", | ||||
|     "addInfoBelow": "Add this info below.", | ||||
|     "addInfoInSettings": "Add this info in the Settings.", | ||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|     "sortByLastLinkSegment": "Sort by only the last segment of the link", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "customLinkFilterRegex": "Custom APK Link Filter by Regular Expression (Default '.apk$')", | ||||
|     "appsPossiblyUpdated": "App Updates Attempted", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Notifies the user that updates to one or more Apps were potentially applied in the background", | ||||
|     "filterReleaseNotesByRegEx": "Filter release notes by regular expression", | ||||
|     "customLinkFilterRegex": "Custom APK link filter by regular expression (default '.apk$')", | ||||
|     "appsPossiblyUpdated": "App updates attempted", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Notifies the user that updates to one or more apps were potentially applied in the background", | ||||
|     "xWasPossiblyUpdatedToY": "{} may have been updated to {}.", | ||||
|     "enableBackgroundUpdates": "Enable background updates", | ||||
|     "backgroundUpdateReqsExplanation": "Background updates may not be possible for all apps.", | ||||
|     "backgroundUpdateLimitsExplanation": "The success of a background install can only be determined when Obtainium is opened.", | ||||
|     "verifyLatestTag": "Verify the 'latest' tag", | ||||
|     "intermediateLinkRegex": "Filter for an 'Intermediate' Link to Visit", | ||||
|     "intermediateLinkRegex": "Filter for an 'intermediate' link to visit", | ||||
|     "filterByLinkText": "Filter links by link text", | ||||
|     "intermediateLinkNotFound": "Intermediate link not found", | ||||
|     "intermediateLink": "Intermediate link", | ||||
|     "exemptFromBackgroundUpdates": "Exempt from background updates (if enabled)", | ||||
|     "bgUpdatesOnWiFiOnly": "Disable background updates when not on WiFi", | ||||
|     "bgUpdatesOnWiFiOnly": "Disable background updates when not on Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||
|     "autoSelectHighestVersionCode": "Auto-select highest versionCode APK", | ||||
|     "versionExtractionRegEx": "Version String Extraction RegEx", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
|     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||
|     "matchGroupToUse": "Match Group to Use for Version String Extraction RegEx", | ||||
|     "versionExtractionRegEx": "Version string extraction RegEx", | ||||
|     "trimVersionString": "Trim version string with RegEx", | ||||
|     "matchGroupToUseForX": "Match group to use for \"{}\"", | ||||
|     "matchGroupToUse": "Match group to use for version string extraction RegEx", | ||||
|     "highlightTouchTargets": "Highlight less obvious touch targets", | ||||
|     "pickExportDir": "Pick Export Directory", | ||||
|     "autoExportOnChanges": "Auto-export on changes", | ||||
|     "pickExportDir": "Pick export directory", | ||||
|     "autoExportOnChanges": "Automatically export on changes", | ||||
|     "includeSettings": "Include settings", | ||||
|     "filterVersionsByRegEx": "Filter Versions by Regular Expression", | ||||
|     "filterVersionsByRegEx": "Filter versions by regular expression", | ||||
|     "trySelectingSuggestedVersionCode": "Try selecting suggested versionCode APK", | ||||
|     "dontSortReleasesList": "Retain release order from API", | ||||
|     "reverseSort": "Reverse sorting", | ||||
|     "takeFirstLink": "Take first link", | ||||
|     "skipSort": "Skip sorting", | ||||
|     "debugMenu": "Debug Menu", | ||||
|     "debugMenu": "Debug menu", | ||||
|     "bgTaskStarted": "Background task started - check logs.", | ||||
|     "runBgCheckNow": "Run Background Update Check Now", | ||||
|     "runBgCheckNow": "Run background update check now", | ||||
|     "versionExtractWholePage": "Apply version string extraction Regex to entire page", | ||||
|     "installing": "Installing", | ||||
|     "skipUpdateNotifications": "Skip update notifications", | ||||
|     "updatesAvailableNotifChannel": "Updates Available", | ||||
|     "appsUpdatedNotifChannel": "Apps Updated", | ||||
|     "appsPossiblyUpdatedNotifChannel": "App Updates Attempted", | ||||
|     "errorCheckingUpdatesNotifChannel": "Error Checking for Updates", | ||||
|     "appsRemovedNotifChannel": "Apps Removed", | ||||
|     "updatesAvailableNotifChannel": "Updates available", | ||||
|     "appsUpdatedNotifChannel": "Apps updated", | ||||
|     "appsPossiblyUpdatedNotifChannel": "App updates attempted", | ||||
|     "errorCheckingUpdatesNotifChannel": "Error checking for updates", | ||||
|     "appsRemovedNotifChannel": "Apps removed", | ||||
|     "downloadingXNotifChannel": "Downloading {}", | ||||
|     "completeAppInstallationNotifChannel": "Complete App Installation", | ||||
|     "checkingForUpdatesNotifChannel": "Checking for Updates", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Only check installed and Track-Only apps for updates", | ||||
|     "completeAppInstallationNotifChannel": "Complete app installation", | ||||
|     "checkingForUpdatesNotifChannel": "Checking for updates", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Only check installed and track-only apps for updates", | ||||
|     "supportFixedAPKURL": "Support fixed APK URLs", | ||||
|     "selectX": "Select {}", | ||||
|     "parallelDownloads": "Allow parallel downloads", | ||||
| @@ -299,22 +300,32 @@ | ||||
|     "useVersionCodeAsOSVersion": "Use app versionCode as OS-detected version", | ||||
|     "requestHeader": "Request header", | ||||
|     "useLatestAssetDateAsReleaseDate": "Use latest asset upload as release date", | ||||
|     "defaultPseudoVersioningMethod": "Default Pseudo-Versioning Method", | ||||
|     "partialAPKHash": "Partial APK Hash", | ||||
|     "APKLinkHash": "APK Link Hash", | ||||
|     "directAPKLink": "Direct APK Link", | ||||
|     "pseudoVersionInUse": "A Pseudo-Version is in Use", | ||||
|     "defaultPseudoVersioningMethod": "Default pseudo-versioning method", | ||||
|     "partialAPKHash": "Partial APK hash", | ||||
|     "APKLinkHash": "APK link hash", | ||||
|     "directAPKLink": "Direct APK link", | ||||
|     "pseudoVersionInUse": "A pseudo-version is in use", | ||||
|     "installed": "Installed", | ||||
|     "latest": "Latest", | ||||
|     "invertRegEx": "Invert regular expression", | ||||
|     "note": "Note", | ||||
|     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", | ||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Share new apps with AppVerifier (if available)", | ||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced app configurations (use at your own risk)", | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced app configurations", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Name", | ||||
|     "smartname": "Name (smart)", | ||||
|     "sortMethod": "Sort method", | ||||
|     "welcome": "Welcome", | ||||
|     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||
|     "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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Desinstalar del dispositivo", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Solo funciona para aplicaciones con la detección de versiones desactivada.", | ||||
|     "releaseDateAsVersion": "Por fecha de publicación", | ||||
|     "releaseTitleAsVersion": "Utilizar el título de la versión como cadena de versión", | ||||
|     "releaseDateAsVersionExplanation": "Esta opción solo se debería usar con aplicaciones en las que la detección de versiones no funciona pero hay disponible una fecha de publicación.", | ||||
|     "changes": "Cambios", | ||||
|     "releaseDate": "Fecha de publicación", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Por versión", | ||||
|     "groupByCategory": "Agrupar por categorías", | ||||
|     "autoApkFilterByArch": "Filtrar APK por arquitectura del procesador (si es posible)", | ||||
|     "autoLinkFilterByArch": "Intente filtrar los enlaces por arquitectura de CPU si es posible", | ||||
|     "overrideSource": "Forzar desde la fuente", | ||||
|     "dontShowAgain": "No mostrar de nuevo", | ||||
|     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps 'solo para seguimiento'", | ||||
| @@ -256,12 +256,13 @@ | ||||
|     "intermediateLinkNotFound": "Enlace intermedio no encontrado", | ||||
|     "intermediateLink": "Enlace intermedio", | ||||
|     "exemptFromBackgroundUpdates": "Exenta de actualizciones en segundo plano (si están habilitadas)", | ||||
|     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin WiFi", | ||||
|     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin Wi-Fi", | ||||
|     "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", | ||||
| @@ -275,7 +276,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", | ||||
| @@ -310,11 +311,21 @@ | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nombre", | ||||
|     "smartname": "Nombre (Smart)", | ||||
|     "sortMethod": "Método de clasificación", | ||||
|     "welcome": "Bienvenido", | ||||
|     "documentationLinksNote": "La página GitHub de Obtainium enlazada a continuación contiene enlaces a vídeos, artículos, debates y otros recursos que te ayudarán a entender cómo utilizar la aplicación.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "¿Eliminar aplicación?", | ||||
|         "other": "¿Eliminar aplicaciones?" | ||||
|   | ||||
| @@ -179,8 +179,6 @@ | ||||
|     "appWithIdOrNameNotFound": "هیچ برنامه ای با آن شناسه یا نام یافت نشد", | ||||
|     "reposHaveMultipleApps": "مخازن ممکن است شامل چندین برنامه باشد", | ||||
|     "fdroidThirdPartyRepo": "مخازن شخص ثالث F-Droid", | ||||
|     "steamMobile": "استیم موبایل", | ||||
|     "steamChat": "چت استیم", | ||||
|     "install": "نصب", | ||||
|     "markInstalled": "علامت گذاری به عنوان نصب شده", | ||||
|     "update": "به روز رسانی", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "حذف نصب از دستگاه", | ||||
|     "onlyWorksWithNonVersionDetectApps": "فقط برای برنامههایی کار میکند که تشخیص نسخه غیرفعال است.", | ||||
|     "releaseDateAsVersion": "از تاریخ انتشار به عنوان نسخه استفاده کنید", | ||||
|     "releaseTitleAsVersion": "از عنوان انتشار به عنوان رشته نسخه استفاده کنید", | ||||
|     "releaseDateAsVersionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند، اما تاریخ انتشار در دسترس است.", | ||||
|     "changes": "تغییرات", | ||||
|     "releaseDate": "تاریخ انتشار", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||
|     "groupByCategory": "گروه بر اساس دسته", | ||||
|     "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید", | ||||
|     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||
|     "overrideSource": "نادیده گرفتن منبع", | ||||
|     "dontShowAgain": "دوباره این را نشان نده", | ||||
|     "dontShowTrackOnlyWarnings": "هشدار 'فقط ردیابی' را نشان ندهید", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "لینک میانی پیدا نشد", | ||||
|     "intermediateLink": "پیوند میانی", | ||||
|     "exemptFromBackgroundUpdates": "معاف از بهروزرسانیهای پسزمینه (در صورت فعال بودن)", | ||||
|     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به WiFi غیرفعال کنید", | ||||
|     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به Wi-Fi غیرفعال کنید", | ||||
|     "bgUpdatesWhileChargingOnly": "بهروزرسانیهای پسزمینه را هنگام شارژ نشدن غیرفعال کنید", | ||||
|     "autoSelectHighestVersionCode": "انتخاب خودکار بالاترین نسخه کد APK", | ||||
|     "versionExtractionRegEx": "نسخه استخراج RegEx", | ||||
|     "trimVersionString": "برش رشته نسخه با RegEx", | ||||
| @@ -313,8 +314,18 @@ | ||||
|     "beforeNewInstallsShareToAppVerifier": "اشتراکگذاری برنامههای جدید با AppVerifier (در صورت وجود)", | ||||
|     "appVerifierInstructionToast": "در AppVerifier به اشتراک بگذارید، سپس پس از آماده شدن به اینجا برگردید.", | ||||
|     "wiki": "راهنما/ویکی", | ||||
|     "crowdsourcedConfigsLabel": "تنظیمات برنامه Crowdsourced (با مسئولیت خود استفاده کنید)", | ||||
|     "crowdsourcedConfigsLabel": "تنظیمات برنامه های مشارکت جمعی (با مسئولیت خود استفاده کنید)", | ||||
|     "crowdsourcedConfigsShort": "تنظیمات برنامه های مشارکت جمعی", | ||||
|     "allowInsecure": "درخواست های HTTP ناامن را مجاز کنید", | ||||
|     "stayOneVersionBehind": "یک نسخه از آخرین نسخه پشت سر بگذارید", | ||||
|     "refreshBeforeDownload": "قبل از دانلود، جزئیات برنامه را بازخوانی کنید", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Name", | ||||
|     "smartname": "Name (Smart)", | ||||
|     "sortMethod": "Sort Method", | ||||
|     "welcome": "Welcome", | ||||
|     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "برنامه حذف شود؟", | ||||
|         "other": "برنامه ها حذف شوند؟" | ||||
|   | ||||
| @@ -9,11 +9,11 @@ | ||||
|     "placeholder": "Espace réservé", | ||||
|     "someErrors": "Des erreurs sont survenues", | ||||
|     "unexpectedError": "Erreur inattendue", | ||||
|     "ok": "Ok", | ||||
|     "ok": "OK", | ||||
|     "and": "et", | ||||
|     "githubPATLabel": "Jeton d'accès personnel GitHub (augmente la limite de débit)", | ||||
|     "includePrereleases": "Inclure les versions préliminaires", | ||||
|     "fallbackToOlderReleases": "Retour aux anciennes versions", | ||||
|     "fallbackToOlderReleases": "Revenir aux anciennes versions", | ||||
|     "filterReleaseTitlesByRegEx": "Filtrer les titres de version par expression régulière", | ||||
|     "invalidRegEx": "Expression régulière invalide", | ||||
|     "noDescription": "Aucune description", | ||||
| @@ -25,19 +25,19 @@ | ||||
|     "standard": "Standard", | ||||
|     "custom": "Personnalisé", | ||||
|     "useMaterialYou": "Utiliser Material You", | ||||
|     "githubStarredRepos": "Dépôts étoilés GitHub", | ||||
|     "githubStarredRepos": "dépôts étoilés GitHub", | ||||
|     "uname": "Nom d'utilisateur", | ||||
|     "wrongArgNum": "Nombre incorrect des arguments fournis", | ||||
|     "xIsTrackOnly": "{} en Suivi uniquement", | ||||
|     "source": "Source", | ||||
|     "app": "Application", | ||||
|     "source": "source", | ||||
|     "app": "Appli", | ||||
|     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", | ||||
|     "youPickedTrackOnly": "Vous avez sélectionné l'option 'Suivi uniquement'.", | ||||
|     "trackOnlyAppDescription": "L'application sera suivie pour les mises à jour, mais Obtainium ne pourra pas la télécharger ou l'installer.", | ||||
|     "cancelled": "Annulé", | ||||
|     "appAlreadyAdded": "Application déjà ajoutée", | ||||
|     "alreadyUpToDateQuestion": "L'application est à jour?", | ||||
|     "addApp": "Ajouter Appli", | ||||
|     "alreadyUpToDateQuestion": "L'application est déjà à jour?", | ||||
|     "addApp": "Ajouter appli", | ||||
|     "appSourceURL": "URL source de l'application", | ||||
|     "error": "Erreur", | ||||
|     "add": "Ajouter", | ||||
| @@ -47,30 +47,30 @@ | ||||
|     "supportedSources": "Sources prises en charge", | ||||
|     "trackOnlyInBrackets": "(Suivi uniquement)", | ||||
|     "searchableInBrackets": "(Interrogeable)", | ||||
|     "appsString": "Applications", | ||||
|     "appsString": "Applis", | ||||
|     "noApps": "Aucune application", | ||||
|     "noAppsForFilter": "Aucune application à filtrer", | ||||
|     "noAppsForFilter": "Aucune application correspondant au filtre", | ||||
|     "byX": "Par {}", | ||||
|     "percentProgress": "Progression : {}%", | ||||
|     "pleaseWait": "Veuillez patienter", | ||||
|     "updateAvailable": "Mise à jour disponible", | ||||
|     "notInstalled": "Non installé", | ||||
|     "notInstalled": "Non installée", | ||||
|     "pseudoVersion": "Version fictive", | ||||
|     "selectAll": "Tout sélectionner", | ||||
|     "deselectX": "Déselectionner {}", | ||||
|     "deselectX": "Désélectionner {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} sera supprimée d'Obtainium mais restera installée sur l'appareil.", | ||||
|     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", | ||||
|     "removeSelectedApps": "Les applications sélectionnées ont été supprimées", | ||||
|     "removeSelectedApps": "Supprimer les applications sélectionnées", | ||||
|     "updateX": "Mettre à jour {}", | ||||
|     "installX": "Installer {}", | ||||
|     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\ncomme étant à jour", | ||||
|     "changeX": "Changer {}", | ||||
|     "changeX": "Modifier {}", | ||||
|     "installUpdateApps": "Installer/Mettre à jour les applications", | ||||
|     "installUpdateSelectedApps": "Installer/Mettre à jour les applications sélectionnées", | ||||
|     "markXSelectedAppsAsUpdated": "Marquer les {} applications sélectionnées comme étant à jour ?", | ||||
|     "no": "Non", | ||||
|     "yes": "Oui", | ||||
|     "markSelectedAppsUpdated": "Marquer les application sélectionnées comme étant à jour", | ||||
|     "markSelectedAppsUpdated": "Marquer les applications sélectionnées comme étant à jour", | ||||
|     "pinToTop": "Épingler en haut", | ||||
|     "unpinFromTop": "Désépingler du haut", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Réinitialiser l'état d'installation des applications sélectionnées ?", | ||||
| @@ -78,7 +78,7 @@ | ||||
|     "customLinkMessage": "Ces liens fonctionnent sur les appareils sur lesquels Obtainium est installé", | ||||
|     "shareAppConfigLinks": "Partager la configuration de l'application sous forme de lien HTML", | ||||
|     "shareSelectedAppURLs": "Partager les URL des applications sélectionnées", | ||||
|     "resetInstallStatus": "L'état d'installation des applications a été réinitialisé", | ||||
|     "resetInstallStatus": "Réinitialiser l'état d'installation des applications", | ||||
|     "more": "Plus", | ||||
|     "removeOutdatedFilter": "Supprimer le filtre des applications obsolètes", | ||||
|     "showOutdatedOnly": "Afficher uniquement les applications obsolètes", | ||||
| @@ -88,33 +88,33 @@ | ||||
|     "author": "Auteur", | ||||
|     "upToDateApps": "Applications à jour", | ||||
|     "nonInstalledApps": "Applications non installées", | ||||
|     "importExport": "Importer/Exporter", | ||||
|     "importExport": "Import/Export", | ||||
|     "settings": "Paramètres", | ||||
|     "exportedTo": "Exporté vers {}", | ||||
|     "obtainiumExport": "Exporter Obtainium", | ||||
|     "obtainiumExport": "Exporter la configuration d'Obtainium", | ||||
|     "invalidInput": "Entrée invalide", | ||||
|     "importedX": "Importé {}", | ||||
|     "obtainiumImport": "Importer sur Obtainium", | ||||
|     "obtainiumImport": "Importer la configuration sur Obtainium", | ||||
|     "importFromURLList": "Importer depuis une liste d'URL", | ||||
|     "searchQuery": "Requête de recherche", | ||||
|     "appURLList": "Liste d'URL de l'application", | ||||
|     "appURLList": "Liste d'URL des applications", | ||||
|     "line": "Ligne", | ||||
|     "searchX": "Rechercher {}", | ||||
|     "noResults": "Aucun résultat", | ||||
|     "importX": "Importation de {}", | ||||
|     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher de manière incorrecte comme étant \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela n'affectera pas les données des applications.\n\nN'affecte que les méthodes d'importation d'URL et par des tiers.", | ||||
|     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher de manière incorrecte comme étant \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela n'affectera pas les données des applications.\n\nCela n'affecte que les méthodes d'importation d'URL et par des tiers.", | ||||
|     "importErrors": "Erreurs lors de l'importation", | ||||
|     "importedXOfYApps": "{} applications sur {} ont été importés.", | ||||
|     "followingURLsHadErrors": "Les URL suivants comportent des erreurs :", | ||||
|     "importedXOfYApps": "{} applications sur {} ont été importées.", | ||||
|     "followingURLsHadErrors": "Les URL suivantes comportent des erreurs :", | ||||
|     "selectURL": "Sélectionner l'URL", | ||||
|     "selectURLs": "Sélectionner les URL", | ||||
|     "pick": "Choisir", | ||||
|     "theme": "Thème", | ||||
|     "dark": "Sombre", | ||||
|     "light": "Clair", | ||||
|     "followSystem": "Correspondre au système", | ||||
|     "followSystemThemeExplanation": "Correspondre au thème du système est possible en utilisant des applications tierces.", | ||||
|     "useBlackTheme": "Utiliser un thème Noir", | ||||
|     "followSystem": "Suivre le système", | ||||
|     "followSystemThemeExplanation": "Suivre le thème du système est possible en utilisant des applications tierces.", | ||||
|     "useBlackTheme": "Utiliser un thème noir", | ||||
|     "appSortBy": "Trier les applications par", | ||||
|     "authorName": "Auteur/Nom", | ||||
|     "nameAuthor": "Nom/Auteur", | ||||
| @@ -136,10 +136,10 @@ | ||||
|     "share": "Partager", | ||||
|     "appNotFound": "Application introuvable", | ||||
|     "obtainiumExportHyphenatedLowercase": "export-obtainium", | ||||
|     "pickAnAPK": "Selectionner un APK", | ||||
|     "appHasMoreThanOnePackage": "{} a plus d'un paquet:", | ||||
|     "pickAnAPK": "Sélectionner un APK", | ||||
|     "appHasMoreThanOnePackage": "{} a plus d'un paquet :", | ||||
|     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture CPU {}.", | ||||
|     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivants: ", | ||||
|     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivantes : ", | ||||
|     "warning": "Avertissement", | ||||
|     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de mise à jour provient de '{}'. Continuer ?", | ||||
|     "updatesAvailable": "Mises à jour disponibles", | ||||
| @@ -149,7 +149,7 @@ | ||||
|     "appsUpdated": "Applications mises à jour", | ||||
|     "appsNotUpdated": "Échec de la mise à jour des applications", | ||||
|     "appsUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont été installées en arrière-plan.", | ||||
|     "xWasUpdatedToY": "{} a été mis à jour en {}.", | ||||
|     "xWasUpdatedToY": "{} a été mise à jour vers {}.", | ||||
|     "xWasNotUpdatedToY": "Échec de la mise à jour de {} vers {}.", | ||||
|     "errorCheckingUpdates": "Erreur lors de la recherche de mises à jour", | ||||
|     "errorCheckingUpdatesNotifDescription": "Notifie l'utilisateur lorsque la recherche de mises à jour en arrière-plan échoue.", | ||||
| @@ -179,29 +179,27 @@ | ||||
|     "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é", | ||||
|     "markInstalled": "Marquer comme installée", | ||||
|     "update": "Mettre à jour", | ||||
|     "markUpdated": "Marquer comme étant à jour", | ||||
|     "markUpdated": "Marquer comme à jour", | ||||
|     "additionalOptions": "Options supplémentaires", | ||||
|     "disableVersionDetection": "Désactiver la détection de la version", | ||||
|     "noVersionDetectionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de la version ne fonctionne pas correctement.", | ||||
|     "downloadingX": "Téléchargement {}", | ||||
|     "downloadingX": "Téléchargement de {}", | ||||
|     "downloadX": "Télécharger {}", | ||||
|     "downloadedX": "Téléchargé {}", | ||||
|     "releaseAsset": "Version active", | ||||
|     "releaseAsset": "Élément de version", | ||||
|     "downloadNotifDescription": "Notifie l'utilisateur sur l'avancement du téléchargement d'une application", | ||||
|     "noAPKFound": "Aucun APK trouvé", | ||||
|     "noVersionDetection": "Aucune version trouvée", | ||||
|     "noVersionDetection": "Aucune détection de version", | ||||
|     "categorize": "Catégoriser", | ||||
|     "categories": "Catégories", | ||||
|     "category": "Catégorie", | ||||
|     "noCategory": "Aucune catégorie", | ||||
|     "noCategories": "Aucune catégories", | ||||
|     "deleteCategoriesQuestion": "Supprimer les catégories?", | ||||
|     "categoryDeleteWarning": "Toutes les applications des catégories supprimées seront définies comme non catégorisées .", | ||||
|     "noCategories": "Aucune catégorie", | ||||
|     "deleteCategoriesQuestion": "Supprimer les catégories ?", | ||||
|     "categoryDeleteWarning": "Toutes les applications des catégories supprimées seront définies comme non catégorisées.", | ||||
|     "addCategory": "Ajouter une catégorie", | ||||
|     "label": "Nom", | ||||
|     "language": "Langue", | ||||
| @@ -211,25 +209,27 @@ | ||||
|     "filterAPKsByRegEx": "Filtrer les APK par expression régulière", | ||||
|     "removeFromObtainium": "Supprimer d'Obtainium", | ||||
|     "uninstallFromDevice": "Désinstaller de l'appareil", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Ne fonctionne que pour les applications dont la détection de la version est désactivée.", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Ne fonctionne qu'avec les applications dont la détection de la version est désactivée.", | ||||
|     "releaseDateAsVersion": "Utiliser la date de sortie comme version", | ||||
|     "releaseTitleAsVersion": "Utiliser le titre de la version comme chaîne de version", | ||||
|     "releaseDateAsVersionExplanation": "Cette option ne doit être utilisée que pour les applications pour lesquelles la détection de la version ne fonctionne pas correctement, mais dont une date de sortie est disponible.", | ||||
|     "changes": "Modifications", | ||||
|     "releaseDate": "Date de sortie", | ||||
|     "importFromURLsInFile": "Importer à partir des URLs d'un fichier (comme OPML)", | ||||
|     "versionDetectionExplanation": "Reporter la chaîne de version selon la version détectée par le système d'exploitation", | ||||
|     "versionDetectionExplanation": "Réconcilier la chaîne de version avec la version détectée par le système d'exploitation", | ||||
|     "versionDetection": "Détection de la version", | ||||
|     "standardVersionDetection": "Détection de la version standard", | ||||
|     "groupByCategory": "Grouper par catégorie", | ||||
|     "autoApkFilterByArch": "Essayer de filtrer les APKs par architecture CPU si possible", | ||||
|     "autoLinkFilterByArch": "Essayer de filtrer les liens en fonction de l'architecture de l'unité centrale si possible", | ||||
|     "overrideSource": "Remplacer la source", | ||||
|     "dontShowAgain": "Ne plus afficher", | ||||
|     "dontShowTrackOnlyWarnings": "Ne plus afficher les erreurs 'Suivi uniquement'", | ||||
|     "dontShowAPKOriginWarnings": "Ne plus afficher les erreurs sur l'origine de l'APK", | ||||
|     "dontShowTrackOnlyWarnings": "Ne plus afficher les avertissements 'Suivi uniquement'", | ||||
|     "dontShowAPKOriginWarnings": "Ne plus afficher les avertissements sur l'origine de l'APK", | ||||
|     "moveNonInstalledAppsToBottom": "Déplacer les applications non installées vers le bas de la vue Applications", | ||||
|     "gitlabPATLabel": "Jeton d'accès personnel GitLab", | ||||
|     "about": "À propos", | ||||
|     "requiresCredentialsInSettings": "{} a besoin d'un complément d'information (dans les Paramètres)", | ||||
|     "requiresCredentialsInSettings": "{} a besoin d'informations d'identification supplémentaires (dans les Paramètres)", | ||||
|     "checkOnStart": "Rechercher les mises à jour au démarrage", | ||||
|     "tryInferAppIdFromCode": "Essayer de déduire l'identifiant de l'application à partir du code source", | ||||
|     "removeOnExternalUninstall": "Supprimer automatiquement les applications désinstallées en externe", | ||||
| @@ -244,66 +244,67 @@ | ||||
|     "sortByLastLinkSegment": "Trier par le dernier segment du lien", | ||||
|     "filterReleaseNotesByRegEx": "Filtrer les notes de version par expression régulière", | ||||
|     "customLinkFilterRegex": "Filtre de lien APK personnalisé par expression régulière (par défaut '.apk$')", | ||||
|     "appsPossiblyUpdated": "Tentative de mise à jour des applications", | ||||
|     "appsPossiblyUpdated": "Tentatives de mise à jour d'applications", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont potentiellement été appliquées en arrière-plan", | ||||
|     "xWasPossiblyUpdatedToY": "{} peut être mis à jour en {}.", | ||||
|     "xWasPossiblyUpdatedToY": "{} a peut-être été mise à jour vers {}.", | ||||
|     "enableBackgroundUpdates": "Activer les mises à jour en arrière-plan", | ||||
|     "backgroundUpdateReqsExplanation": "Les mises à jour en arrière-plan peuvent ne pas être possibles pour toutes les applications.", | ||||
|     "backgroundUpdateLimitsExplanation": "Le résultat d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", | ||||
|     "backgroundUpdateLimitsExplanation": "Le succès d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", | ||||
|     "verifyLatestTag": "Vérifier la balise 'latest'", | ||||
|     "intermediateLinkRegex": "Filtrer un lien 'intermédiaire' à visiter", | ||||
|     "filterByLinkText": "Filtrer les liens par texte du lien", | ||||
|     "intermediateLinkNotFound": "Lien intermédiaire introuvable", | ||||
|     "intermediateLink": "Lien intermédiaire", | ||||
|     "exemptFromBackgroundUpdates": "Exclure de la mise à jour en arrière-plan (si activé)", | ||||
|     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas en WiFi", | ||||
|     "exemptFromBackgroundUpdates": "Exclure des mises à jour en arrière-plan (si activées)", | ||||
|     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas en Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Désactiver les mises à jour en arrière-plan lorsque l'appareil n'est pas en charge", | ||||
|     "autoSelectHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", | ||||
|     "versionExtractionRegEx": "Extraire la version par Expression régulière", | ||||
|     "trimVersionString": "Découper la version par Expression régulière", | ||||
|     "versionExtractionRegEx": "Expression régulière d'extraction de version", | ||||
|     "trimVersionString": "Découper la chaîne de version avec une expression régulière", | ||||
|     "matchGroupToUseForX": "Groupe de correspondance à utiliser pour \"{}\"", | ||||
|     "matchGroupToUse": "Groupe de correspondance à utiliser pour l'extraction de la version par Expression régulière", | ||||
|     "highlightTouchTargets": "Mettre en évidence les touches moins évidentes", | ||||
|     "pickExportDir": "Selectionner le dossier d'exportation", | ||||
|     "matchGroupToUse": "Groupe de correspondance à utiliser pour l'extraction de la version par expression régulière", | ||||
|     "highlightTouchTargets": "Mettre en évidence les zones tactiles moins évidentes", | ||||
|     "pickExportDir": "Sélectionner le dossier d'exportation", | ||||
|     "autoExportOnChanges": "Exporter automatiquement lors de modifications", | ||||
|     "includeSettings": "Inclure les paramètres", | ||||
|     "filterVersionsByRegEx": "Filtrer les versions par expression régulière", | ||||
|     "trySelectingSuggestedVersionCode": "Essayer de sélectionner la version suggérée du code APK", | ||||
|     "dontSortReleasesList": "Conserver l'ordre de la version de l'API", | ||||
|     "trySelectingSuggestedVersionCode": "Essayer de sélectionner le code de version APK suggéré", | ||||
|     "dontSortReleasesList": "Conserver l'ordre de version de l'API", | ||||
|     "reverseSort": "Tri inversé", | ||||
|     "takeFirstLink": "Utiliser le premier lien", | ||||
|     "skipSort": "Ignorer le tri", | ||||
|     "debugMenu": "Menu de déboggage", | ||||
|     "debugMenu": "Menu de débogage", | ||||
|     "bgTaskStarted": "Tâche en arrière-plan démarrée - vérifier les journaux.", | ||||
|     "runBgCheckNow": "Exécuter la recherche de mise à jour en arrière-plan maintenant", | ||||
|     "versionExtractWholePage": "Appliquer l'extraction de la version par expression régulière à l'ensemble de la page", | ||||
|     "runBgCheckNow": "Exécuter la recherche de mises à jour en arrière-plan maintenant", | ||||
|     "versionExtractWholePage": "Appliquer l'expression régulière d'extraction de version à l'ensemble de la page", | ||||
|     "installing": "Installation", | ||||
|     "skipUpdateNotifications": "Ignorer les notifications de mise à jour", | ||||
|     "skipUpdateNotifications": "Désactiver les notifications de mise à jour", | ||||
|     "updatesAvailableNotifChannel": "Mises à jour disponibles", | ||||
|     "appsUpdatedNotifChannel": "Applications mises à jour", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Essayer de mettre à jour les applications", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Tentatives de mise à jour d'applications", | ||||
|     "errorCheckingUpdatesNotifChannel": "Erreur lors de la recherche de mises à jour", | ||||
|     "appsRemovedNotifChannel": "Applications supprimées", | ||||
|     "downloadingXNotifChannel": "Téléchargement {}", | ||||
|     "completeAppInstallationNotifChannel": "Installation complète de l'application", | ||||
|     "downloadingXNotifChannel": "Téléchargement de {}", | ||||
|     "completeAppInstallationNotifChannel": "Terminer l'installation de l'application", | ||||
|     "checkingForUpdatesNotifChannel": "Recherche de mises à jour", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Rechercher uniquement les mises à jour des applications installées et des applications 'Suivi uniquement'", | ||||
|     "supportFixedAPKURL": "Prise en charge des URL APK fixes", | ||||
|     "selectX": "Sélectionner {}", | ||||
|     "parallelDownloads": "Autoriser les téléchargements simultanés", | ||||
|     "useShizuku": "Utiliser Shizuku ou Sui pour l'installation", | ||||
|     "shizukuBinderNotFound": "Le service Shizuku est introuvable", | ||||
|     "shizukuBinderNotFound": "Le service Shizuku n'est pas en cours d'exécution", | ||||
|     "shizukuOld": "Ancienne version de Shizuku (<11) - veuillez le mettre à jour", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku fonctionne sur Android < 8.1 avec ADB - veuillez mettre à jour Android ou utiliser Sui à la place", | ||||
|     "shizukuPretendToBeGooglePlay": "Définir Google Play comme source d'installation (si Shizuku est utilisé)", | ||||
|     "useSystemFont": "Utiliser la police du système", | ||||
|     "useVersionCodeAsOSVersion": "Utiliser le code de version de l'application détectée par le système d'exploitation", | ||||
|     "requestHeader": "Intitulé de la demande", | ||||
|     "useLatestAssetDateAsReleaseDate": "Utiliser le dernier élément mis en ligne comme date de sortie", | ||||
|     "defaultPseudoVersioningMethod": "Méthode de version fictive par défaut", | ||||
|     "requestHeader": "En-tête de requête", | ||||
|     "useLatestAssetDateAsReleaseDate": "Utiliser la date du dernier élément mis en ligne comme date de sortie", | ||||
|     "defaultPseudoVersioningMethod": "Méthode de versionnage fictif par défaut", | ||||
|     "partialAPKHash": "Hash partiel de l'APK", | ||||
|     "APKLinkHash": "Hash du lien APK", | ||||
|     "directAPKLink": "Lien direct de l'APK", | ||||
|     "pseudoVersionInUse": "Version fictive utilisé", | ||||
|     "pseudoVersionInUse": "Une version fictive est utilisée", | ||||
|     "installed": "Installée", | ||||
|     "latest": "Dernière version", | ||||
|     "invertRegEx": "Inverser l'expression régulière", | ||||
| @@ -313,23 +314,33 @@ | ||||
|     "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)", | ||||
|     "crowdsourcedConfigsLabel": "Configurations d'applications communautaires (à utiliser à vos risques et périls)", | ||||
|     "crowdsourcedConfigsShort": "Applications communautaires", | ||||
|     "allowInsecure": "Autoriser les requêtes HTTP non sécurisées", | ||||
|     "stayOneVersionBehind": "Rester une version en arrière de la dernière", | ||||
|     "refreshBeforeDownload": "Actualiser les détails de l'application avant de la télécharger", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nom", | ||||
|     "smartname": "Nom (Smart)", | ||||
|     "sortMethod": "Méthode de tri", | ||||
|     "welcome": "Bienvenue", | ||||
|     "documentationLinksNote": "La page GitHub d'Obtainium, dont le lien figure ci-dessous, contient des liens vers des vidéos, des articles, des discussions et d'autres ressources qui vous aideront à comprendre comment utiliser l'application.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Supprimer l'application ?", | ||||
|         "other": "Supprimer les applications ?" | ||||
|         "one": "Supprimer l'application ?", | ||||
|         "other": "Supprimer les applications ?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Trop de requêtes (taux limité) - réessayez dans {} minute", | ||||
|         "other": "Trop de requêtes (taux limité) - réessayez dans {} minutes" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minute", | ||||
|         "other": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minutes" | ||||
|         "one": "La recherche de mise à jour en arrière-plan a rencontré une erreur {}, une nouvelle tentative programmée dans {} minute", | ||||
|         "other": "La recherche de mise à jour en arrière-plan a rencontré une erreur {}, une nouvelle tentative programmée dans {} minutes" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "La recherche de mises à jour en arrière-plan à trouvée {} mise à jour - l'utilisateur sera notifié si nécessaire", | ||||
|         "other": "La recherche de mises à jour en arrière-plan à trouvée {} mises à jour - l'utilisateur sera notifié si nécessaire" | ||||
|         "one": "La recherche de mises à jour en arrière-plan a trouvé {} mise à jour - l'utilisateur sera notifié si nécessaire", | ||||
|         "other": "La recherche de mises à jour en arrière-plan a trouvé {} mises à jour - l'utilisateur sera notifié si nécessaire" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} Application", | ||||
| @@ -337,7 +348,7 @@ | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URL" | ||||
|         "other": "{} URLs" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Minute", | ||||
| @@ -361,7 +372,7 @@ | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} et 1 autre application ont été mises à jour.", | ||||
|         "other": "{} et {} autres applications ont étés mis à jour." | ||||
|         "other": "{} et {} autres applications ont été mises à jour." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Échec de la mise à jour de {} et 1 autre application.", | ||||
| @@ -369,7 +380,7 @@ | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} et 1 autre application ont peut-être été mises à jour.", | ||||
|         "other": "{} et {} autres applications ont peut-être étés mis à jour." | ||||
|         "other": "{} et {} autres applications ont peut-être été mises à jour." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
|     "placeholder": "Helyőrző", | ||||
|     "someErrors": "Néhány hiba történt", | ||||
|     "unexpectedError": "Váratlan hiba", | ||||
|     "ok": "Rendben", | ||||
|     "ok": "OK", | ||||
|     "and": "és", | ||||
|     "githubPATLabel": "GitHub személyes hozzáférési token (megnöveli a lekérdezés-korlátozást)", | ||||
|     "includePrereleases": "Tartalmazza az előzetes kiadásokat", | ||||
| @@ -28,11 +28,11 @@ | ||||
|     "githubStarredRepos": "Csillagozott GitHub tárolók", | ||||
|     "uname": "Felhasználónév", | ||||
|     "wrongArgNum": "A megadott argumentumok száma nem megfelelő", | ||||
|     "xIsTrackOnly": "A(z) {} csak nyomonkövethető", | ||||
|     "xIsTrackOnly": "A(z) {} csak nyomon-követhető", | ||||
|     "source": "Forrás", | ||||
|     "app": "Alkalmazás", | ||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások „csak nyomonkövethetők”.", | ||||
|     "youPickedTrackOnly": "„Csak nyomonkövetés” opciót választotta.", | ||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások „csak nyomon-követhetők”.", | ||||
|     "youPickedTrackOnly": "„Csak nyomon-követés” opciót választotta.", | ||||
|     "trackOnlyAppDescription": "Az alkalmazás frissítéseit nyomon követi, de az Obtainium nem tudja letölteni vagy telepíteni.", | ||||
|     "cancelled": "Visszavonva", | ||||
|     "appAlreadyAdded": "Az alkalmazás már hozzá van adva", | ||||
| @@ -45,7 +45,7 @@ | ||||
|     "search": "Keresés", | ||||
|     "additionalOptsFor": "További lehetőségek a következőhöz: {}", | ||||
|     "supportedSources": "Támogatott források", | ||||
|     "trackOnlyInBrackets": "(Csak nyomonkövetés)", | ||||
|     "trackOnlyInBrackets": "(Csak nyomon-követés)", | ||||
|     "searchableInBrackets": "(Kereshető)", | ||||
|     "appsString": "Alkalmazások", | ||||
|     "noApps": "Nincsenek alkalmazások", | ||||
| @@ -61,10 +61,10 @@ | ||||
|     "xWillBeRemovedButRemainInstalled": "A(z) {} el lesz távolítva az Obtainiumból, de továbbra is telepítve marad az eszközön.", | ||||
|     "removeSelectedAppsQuestion": "A kiválasztott alkalmazások eltávolítása?", | ||||
|     "removeSelectedApps": "A kiválasztott alkalmazások eltávolítása", | ||||
|     "updateX": "A(z) {} frissítése", | ||||
|     "installX": "A(z) {} telepítése", | ||||
|     "markXTrackOnlyAsUpdated": "Megjelölés: {}\n(Csak nyomonkövetés)\nFrissítettként", | ||||
|     "changeX": "{} változtatás", | ||||
|     "updateX": "{} frissítése", | ||||
|     "installX": "{} telepítése", | ||||
|     "markXTrackOnlyAsUpdated": "Megjelölés: {}\n(Csak nyomon-követés)\nFrissítettként", | ||||
|     "changeX": "{}-változás", | ||||
|     "installUpdateApps": "Alkalmazások telepítése/frissítése", | ||||
|     "installUpdateSelectedApps": "A kiválasztott alkalmazások telepítése/frissítése", | ||||
|     "markXSelectedAppsAsUpdated": "A(z) {} kiválasztott alkalmazás megjelölése frissítettként?", | ||||
| @@ -80,7 +80,7 @@ | ||||
|     "shareSelectedAppURLs": "A kiválasztott alkalmazás hivatkozásának megosztása", | ||||
|     "resetInstallStatus": "Telepítési állapot visszaállítása", | ||||
|     "more": "További", | ||||
|     "removeOutdatedFilter": "Elavult-alkalmazás szűrő eltávolítása", | ||||
|     "removeOutdatedFilter": "Elavult alkalmazás-szűrő eltávolítása", | ||||
|     "showOutdatedOnly": "Csak az elavult alkalmazások megjelenítése", | ||||
|     "filter": "Szűrő", | ||||
|     "filterApps": "Alkalmazások szűrése", | ||||
| @@ -88,13 +88,13 @@ | ||||
|     "author": "Szerző", | ||||
|     "upToDateApps": "Naprakész alkalmazások", | ||||
|     "nonInstalledApps": "Nem telepített alkalmazások", | ||||
|     "importExport": "Import/Export", | ||||
|     "importExport": "Adatmozgatás", | ||||
|     "settings": "Beállítások", | ||||
|     "exportedTo": "Exportálva ide: {}", | ||||
|     "obtainiumExport": "Obtainium adatok exportálása", | ||||
|     "obtainiumExport": "Obtainium-adatok exportálása", | ||||
|     "invalidInput": "Hibás bemenet", | ||||
|     "importedX": "Importálva innen: {}", | ||||
|     "obtainiumImport": "Obtainium adatok importálása", | ||||
|     "obtainiumImport": "Obtainium-adatok importálása", | ||||
|     "importFromURLList": "Importálás webcímlistából", | ||||
|     "searchQuery": "Keresési lekérdezés", | ||||
|     "appURLList": "Alkalmazás-webcímlista", | ||||
| @@ -115,11 +115,11 @@ | ||||
|     "followSystem": "Rendszerbeállítás használata", | ||||
|     "followSystemThemeExplanation": "A következő rendszer téma csak harmadik féltől származó alkalmazások használatával lehetséges", | ||||
|     "useBlackTheme": "Használjon teljesen fekete sötét témát", | ||||
|     "appSortBy": "Elrendezés", | ||||
|     "appSortBy": "Rendezési szempont", | ||||
|     "authorName": "Szerző/Név", | ||||
|     "nameAuthor": "Név/Szerző", | ||||
|     "asAdded": "Hozzáadás dátuma", | ||||
|     "appSortOrder": "Elrendezés sorrendje", | ||||
|     "appSortOrder": "Rendezési sorrend", | ||||
|     "ascending": "Növekvő", | ||||
|     "descending": "Csökkenő", | ||||
|     "bgUpdateCheckInterval": "Időtartam a frissítések háttérellenőrzése között", | ||||
| @@ -143,7 +143,7 @@ | ||||
|     "warning": "Figyelem", | ||||
|     "sourceIsXButPackageFromYPrompt": "Az alkalmazás forrása a(z) „{}” tároló, de a kiadási csomag innen származik: „{}”. Folytatja?", | ||||
|     "updatesAvailable": "Frissítések érhetők el", | ||||
|     "updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy egy vagy több, az Obtainium által nyomonkövetett alkalmazáshoz frissítések állnak rendelkezésre", | ||||
|     "updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy egy vagy több, az Obtainium által nyomon-követett alkalmazáshoz frissítések állnak rendelkezésre", | ||||
|     "noNewUpdates": "Nincsenek új frissítések.", | ||||
|     "xHasAnUpdate": "A(z) {} frissítést kapott.", | ||||
|     "appsUpdated": "Alkalmazások frissítve", | ||||
| @@ -151,20 +151,20 @@ | ||||
|     "appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás frissítése a háttérben történt.", | ||||
|     "xWasUpdatedToY": "A(z) {} frissítve lett a következőre: {}.", | ||||
|     "xWasNotUpdatedToY": "Nem sikerült frissíteni a következőt: {}, erre: {}.", | ||||
|     "errorCheckingUpdates": "Hiba a frissítések keresésekor", | ||||
|     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérfrissítés ellenőrzése nem sikerül", | ||||
|     "errorCheckingUpdates": "Hiba a frissítések ellenőrzésekor", | ||||
|     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, amikor a frissítések ellenőrzése a háttérben nem sikerül", | ||||
|     "appsRemoved": "Alkalmazások eltávolítva", | ||||
|     "appsRemovedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás betöltés közbeni hiba miatt eltávolításra került", | ||||
|     "xWasRemovedDueToErrorY": "A(z) {} eltávolításra került a következő hiba miatt: {}", | ||||
|     "appsRemovedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás egy betöltés közbeni hiba miatt el lesz(nek) távolítva", | ||||
|     "xWasRemovedDueToErrorY": "A(z) {} el lett távolítva a következő hiba miatt: {}", | ||||
|     "completeAppInstallation": "Teljes alkalmazástelepítés", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Az alkalmazások telepítéséhez az Obtainiumnak megnyitva kell lennie", | ||||
|     "completeAppInstallationNotifDescription": "Megkéri a felhasználót, hogy térjen vissza az Obtainiumhoz, hogy befejezze az alkalmazás telepítését", | ||||
|     "checkingForUpdates": "Frissítések keresése", | ||||
|     "checkingForUpdatesNotifDescription": "Átmeneti értesítés, amely a frissítések keresésekor jelenik meg", | ||||
|     "checkingForUpdates": "Frissítések ellenőrzése", | ||||
|     "checkingForUpdatesNotifDescription": "Átmeneti értesítés, amely a frissítések ellenőrzésekor jelenik meg", | ||||
|     "pleaseAllowInstallPerm": "Engedélyezze az Obtainiumnak az alkalmazások telepítését", | ||||
|     "trackOnly": "Csak nyomonkövetés", | ||||
|     "trackOnly": "Csak nyomon-követés", | ||||
|     "errorWithHttpStatusCode": "Hiba {}", | ||||
|     "versionCorrectionDisabled": "Verziókorrekció letiltva (úgy tűnik, hogy a bővítmény nem működik)", | ||||
|     "versionCorrectionDisabled": "Verzió-korrekció letiltva (úgy tűnik, hogy a bővítmény nem működik)", | ||||
|     "unknown": "Ismeretlen", | ||||
|     "none": "Semmi", | ||||
|     "never": "Soha", | ||||
| @@ -179,22 +179,20 @@ | ||||
|     "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", | ||||
|     "markUpdated": "Frissítettnek jelölés", | ||||
|     "additionalOptions": "További beállítások", | ||||
|     "disableVersionDetection": "Verzióérzékelés letiltása", | ||||
|     "noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően.", | ||||
|     "disableVersionDetection": "Verzió-érzékelés letiltása", | ||||
|     "noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzió-érzékelés nem működik megfelelően.", | ||||
|     "downloadingX": "{} letöltése", | ||||
|     "downloadX": "{} letöltése", | ||||
|     "downloadedX": "{} letöltve", | ||||
|     "releaseAsset": "Kiadási csomag", | ||||
|     "downloadNotifDescription": "Értesíti a felhasználót az alkalmazás letöltésének előrehaladásáról", | ||||
|     "noAPKFound": "Nem található APK", | ||||
|     "noVersionDetection": "Nincs verzióérzékelés", | ||||
|     "noVersionDetection": "Nincs verzió-érzékelés", | ||||
|     "categorize": "Kategorizálás", | ||||
|     "categories": "Kategóriák", | ||||
|     "category": "Kategória", | ||||
| @@ -211,30 +209,32 @@ | ||||
|     "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel", | ||||
|     "removeFromObtainium": "Eltávolítás az Obtainiumból", | ||||
|     "uninstallFromDevice": "Eltávolítás az eszközről", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzióérzékelés le van tiltva.", | ||||
|     "releaseDateAsVersion": "Használja a kiadás dátumát verzióké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", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzió-érzékelés le van tiltva.", | ||||
|     "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á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 észlelt verzióval", | ||||
|     "versionDetection": "Verzióérzékelés", | ||||
|     "standardVersionDetection": "Alapértelmezett verzióérzékelés", | ||||
|     "versionDetectionExplanation": "A verzió-karakterlánc egyeztetése az rendszer által érzékelt verzióval", | ||||
|     "versionDetection": "Verzió-érzékelés", | ||||
|     "standardVersionDetection": "Alapértelmezett verzió-érzékelés", | ||||
|     "groupByCategory": "Csoportosítás kategória alapján", | ||||
|     "autoApkFilterByArch": "Ha lehetséges, próbálja CPU architektúra szerint szűrni az APK-kat", | ||||
|     "autoApkFilterByArch": "Ha lehetséges, próbálja meg az APK-kat CPU-architektúra szerint szűrni", | ||||
|     "autoLinkFilterByArch": "Ha lehetséges, próbálja meg a hivatkozásokat CPU-architektúra szerint szűrni", | ||||
|     "overrideSource": "Forrás felülírása", | ||||
|     "dontShowAgain": "Ne jelenítse meg ezt többé", | ||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítse meg a „Csak nyomonkövetés” figyelmeztetést", | ||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítse meg a „Csak nyomon-követés” figyelmeztetést", | ||||
|     "dontShowAPKOriginWarnings": "Ne jelenítse meg az APK eredetére vonatkozó figyelmeztetéseket", | ||||
|     "moveNonInstalledAppsToBottom": "Helyezze át a nem telepített alkalmazásokat az alkalmazásnézet aljára", | ||||
|     "gitlabPATLabel": "GitLab személyes hozzáférési token", | ||||
|     "about": "Névjegy", | ||||
|     "requiresCredentialsInSettings": "A(z) {} alkalmazásnak további hitelesítő adatokra van szüksége (a beállításokban)", | ||||
|     "checkOnStart": "Frissítések keresése indításkor", | ||||
|     "checkOnStart": "Frissítések ellenőrzése az alkalmazás megnyitásakor", | ||||
|     "tryInferAppIdFromCode": "Próbálja meg kikövetkeztetni az alkalmazás azonosítóját a forráskódból", | ||||
|     "removeOnExternalUninstall": "A külsőleg eltávolított alkalmazások automatikus eltávolítása", | ||||
|     "pickHighestVersionCode": "A legmagasabb verziószámú APK automatikus kiválasztása", | ||||
|     "checkUpdateOnDetailPage": "Frissítések keresése az alkalmazás részleteit tartalmazó oldal megnyitásakor", | ||||
|     "checkUpdateOnDetailPage": "Frissítések ellenőrzése az alkalmazás részleteit tartalmazó oldal megnyitásakor", | ||||
|     "disablePageTransitions": "Lap áttűnési animációk letiltása", | ||||
|     "reversePageTransitions": "Fordított lap áttűnési animációk", | ||||
|     "minStarCount": "Minimális csillagozási szám", | ||||
| @@ -256,37 +256,38 @@ | ||||
|     "intermediateLinkNotFound": "Köztes hivatkozás nem található", | ||||
|     "intermediateLink": "Köztes hivatkozás", | ||||
|     "exemptFromBackgroundUpdates": "Mentes a háttérben történő frissítések alól (ha engedélyezett)", | ||||
|     "bgUpdatesOnWiFiOnly": "A háttérben futó frissítések letiltása, ha nincs Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "A legmagasabb verziószámú APK automatikus kiválasztása", | ||||
|     "versionExtractionRegEx": "Verziókarakterlánc-kivonatolása reguláris kifejezéssel", | ||||
|     "trimVersionString": "Verziókarakterlánc levágása reguláris kifejezéssel", | ||||
|     "bgUpdatesOnWiFiOnly": "Háttérfrissítések letiltása, amikor az eszköz nem csatlakozik a Wi-Fi-hez", | ||||
|     "bgUpdatesWhileChargingOnly": "Háttérfrissítések letiltása, amikor az eszköz nincs a töltőn", | ||||
|     "autoSelectHighestVersionCode": "A legmagasabb verziókódú APK automatikus kiválasztása", | ||||
|     "versionExtractionRegEx": "Verzió-karakterlánc kivonatolása reguláris kifejezéssel", | ||||
|     "trimVersionString": "Verzió-karakterlánc levágása reguláris kifejezéssel", | ||||
|     "matchGroupToUseForX": "A(z) „{}” esetén használandó csoport egyeztetése", | ||||
|     "matchGroupToUse": "Verziókarakterlánc-kivonatoláshoz használandó csoport reguláris kifejezéssel való egyeztetése", | ||||
|     "matchGroupToUse": "A verzió-karakterlánc kivonatolásához használandó csoport reguláris kifejezéssel való egyeztetése", | ||||
|     "highlightTouchTargets": "A kevésbé nyilvánvaló érintési pontok kiemelése", | ||||
|     "pickExportDir": "Válassza ki a könyvtárat, ahová exportálni szeretne", | ||||
|     "autoExportOnChanges": "Automatikus exportálás a változások után", | ||||
|     "autoExportOnChanges": "Automatikus exportálás a változtatások után", | ||||
|     "includeSettings": "Tartalmazza a beállításokat", | ||||
|     "filterVersionsByRegEx": "Verziók szűrése reguláris kifejezéssel", | ||||
|     "trySelectingSuggestedVersionCode": "Próbálja ki a javasolt verziókódú APK-t", | ||||
|     "trySelectingSuggestedVersionCode": "Próbálja kiválasztani a „Javasolt” verziókódú APK-t", | ||||
|     "dontSortReleasesList": "Az API-ból származó kiadási sorrend megőrzése", | ||||
|     "reverseSort": "Fordított elrendezés", | ||||
|     "reverseSort": "Fordított rendezés", | ||||
|     "takeFirstLink": "Vegye az első hivatkozást", | ||||
|     "skipSort": "Rendezés kihagyása", | ||||
|     "debugMenu": "Hibakereső menü", | ||||
|     "bgTaskStarted": "A háttérfeladat elindult – ellenőrizze a naplókat.", | ||||
|     "runBgCheckNow": "Frissítések keresése a háttérben", | ||||
|     "versionExtractWholePage": "„Verziókarakterlánc-kivonatolása reguláris kifejezéssel” alkalmazása az egész oldalra", | ||||
|     "runBgCheckNow": "Frissítések ellenőrzése a háttérben most", | ||||
|     "versionExtractWholePage": "„Verzió-karakterlánc kivonatolása reguláris kifejezéssel” alkalmazása az egész oldalra", | ||||
|     "installing": "Telepítés", | ||||
|     "skipUpdateNotifications": "A frissítési értesítések kihagyása", | ||||
|     "updatesAvailableNotifChannel": "Frissítések érhetők el", | ||||
|     "appsUpdatedNotifChannel": "Alkalmazások frissítve", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Megkísérelt alkalmazás-frissítések", | ||||
|     "errorCheckingUpdatesNotifChannel": "Hiba a frissítések keresésekor", | ||||
|     "errorCheckingUpdatesNotifChannel": "Hiba a frissítések ellenőrzésekor", | ||||
|     "appsRemovedNotifChannel": "Eltávolított alkalmazások", | ||||
|     "downloadingXNotifChannel": "A(z) {} letöltése", | ||||
|     "completeAppInstallationNotifChannel": "Teljes alkalmazás telepítés", | ||||
|     "checkingForUpdatesNotifChannel": "Frissítések keresése", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak nyomonkövethető alkalmazások frissítéseinek ellenőrzése", | ||||
|     "checkingForUpdatesNotifChannel": "Frissítések ellenőrzése", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak nyomon-követhető alkalmazások frissítéseinek ellenőrzése", | ||||
|     "supportFixedAPKURL": "Támogatja a rögzített APK webcímeket", | ||||
|     "selectX": "{} kiválasztása", | ||||
|     "parallelDownloads": "Párhuzamos letöltések engedélyezése", | ||||
| @@ -294,9 +295,9 @@ | ||||
|     "shizukuBinderNotFound": "A Shizuku szolgáltatás nem fut", | ||||
|     "shizukuOld": "Régi Shizuku verzió (<11) - frissítse", | ||||
|     "shizukuOldAndroidWithADB": "A Shizuku csak Android < 8.1 ADB-vel fut - frissítse az Androidot vagy használja a Sui-t helyette", | ||||
|     "shizukuPretendToBeGooglePlay": "Állítsa be a Google Playt telepítési forrásként (ha Shizukut használ)", | ||||
|     "shizukuPretendToBeGooglePlay": "A Google Play beállítása telepítési forrásként (ha Shizukut használ)", | ||||
|     "useSystemFont": "A rendszer betűtípusának használata", | ||||
|     "useVersionCodeAsOSVersion": "Az alkalmazás verziókódjának használata a rendszer által észlelt verzióként", | ||||
|     "useVersionCodeAsOSVersion": "Az alkalmazás verziókódjának használata a rendszer által érzékelt verzióként", | ||||
|     "requestHeader": "Kérelemfejléc", | ||||
|     "useLatestAssetDateAsReleaseDate": "A kiadás dátumaként használja a legutóbbi csomagfeltöltést", | ||||
|     "defaultPseudoVersioningMethod": "Alapértelmezett pszeudoverziós módszer", | ||||
| @@ -313,8 +314,18 @@ | ||||
|     "beforeNewInstallsShareToAppVerifier": "Új alkalmazások megosztása az AppVerifierrel (ha elérhető)", | ||||
|     "appVerifierInstructionToast": "Ossza meg az AppVerifierrel, majd térjen vissza ide, ha kész.", | ||||
|     "wiki": "Súgó/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsource-ből származó alkalmazások beállítása (saját felelősségére használja)", | ||||
|     "crowdsourcedConfigsLabel": "Közreműködők által összeállított alkalmazásbeállítások (saját felelősségére használja)", | ||||
|     "crowdsourcedConfigsShort": "Alkalmazáslista", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent Appstore", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Név", | ||||
|     "smartname": "Név (Okos)", | ||||
|     "sortMethod": "Rendezési eljárás", | ||||
|     "welcome": "Üdvözöljük!", | ||||
|     "documentationLinksNote": "Az alábbi hivatkozás az Obtainium GitHub oldalára vezet, amely további videók, cikkek, beszélgetések és egyéb források hivatkozásait tartalmazza, amelyek segítenek megérteni az alkalmazás használatát.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Eltávolítja az alkalmazást?", | ||||
|         "other": "Eltávolítja az alkalmazásokat?" | ||||
| @@ -352,8 +363,8 @@ | ||||
|         "other": "{} nap" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "{n} napló törölve lett ({after} után és {before} előtt)", | ||||
|         "other": "{n} napló törölve lett ({after} után és {before} előtt)" | ||||
|         "one": "{n} napló törölve lett ({after} után, és {before} előtt)", | ||||
|         "other": "{n} napló törölve lett ({after} után, és {before} előtt)" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "A(z) {} és 1 további alkalmazás frissítést kapott.", | ||||
|   | ||||
| @@ -154,7 +154,7 @@ | ||||
|     "errorCheckingUpdates": "Kesalahan dalam memeriksa pembaruan", | ||||
|     "errorCheckingUpdatesNotifDescription": "Notifikasi yang ditampilkan saat pemeriksaan pembaruan latar belakang gagal", | ||||
|     "appsRemoved": "Aplikasi dihapus", | ||||
|     "appsRemovedNotifDescription": "Memberi tahu pengguna bahwa aplikasi aplikasi dihapus karena kesalahan saat memuatnya", | ||||
|     "appsRemovedNotifDescription": "Memberi tahu pengguna bahwa aplikasi dihapus karena kesalahan saat memuatnya", | ||||
|     "xWasRemovedDueToErrorY": "{} dihapus karena kesalahan berikut: {}", | ||||
|     "completeAppInstallation": "Selesaikan pemasangan aplikasi", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium harus terbuka untuk memasang aplikasi", | ||||
| @@ -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", | ||||
| @@ -204,7 +202,7 @@ | ||||
|     "categoryDeleteWarning": "Semua aplikasi dalam kategori yang dihapus akan diatur sebagai tidak terkategori.", | ||||
|     "addCategory": "Tambah kategori", | ||||
|     "label": "Label", | ||||
|     "language": "Language", | ||||
|     "language": "Bahasa", | ||||
|     "copiedToClipboard": "Disalin ke papan klip", | ||||
|     "storagePermissionDenied": "Izin penyimpanan ditolak", | ||||
|     "selectedCategorizeWarning": "Ini akan mengganti pengaturan kategori yang ada untuk aplikasi terpilih.", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Copot pemasangan dari perangkat", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Hanya bekerja untuk aplikasi dengan pendeteksi versi yang nonaktif.", | ||||
|     "releaseDateAsVersion": "Gunakan tanggal rilis sebagai versi string", | ||||
|     "releaseTitleAsVersion": "Gunakan judul rilis sebagai string versi", | ||||
|     "releaseDateAsVersionExplanation": "Opsi ini hanya boleh digunakan untuk aplikasi dengan pendeteksi versi yang tidak berfungsi dengan benar, tetapi tanggal rilisnya tersedia.", | ||||
|     "changes": "Perubahan", | ||||
|     "releaseDate": "Tanggal rilis", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Pendeteksi versi standar", | ||||
|     "groupByCategory": "Kelompokkan berdasarkan kategori", | ||||
|     "autoApkFilterByArch": "Coba filter APK berdasarkan arsitektur CPU jika memungkinkan", | ||||
|     "autoLinkFilterByArch": "Mencoba memfilter tautan berdasarkan arsitektur CPU jika memungkinkan", | ||||
|     "overrideSource": "Ganti sumber", | ||||
|     "dontShowAgain": "Jangan tampilkan lagi", | ||||
|     "dontShowTrackOnlyWarnings": "Jangan tampilkan peringatan 'Pelacakan Saja'", | ||||
| @@ -231,8 +231,8 @@ | ||||
|     "about": "Tentang", | ||||
|     "requiresCredentialsInSettings": "{} memerlukan kredensial tambahan (dalam pengaturan)", | ||||
|     "checkOnStart": "Periksa pembaruan saat proses memulai", | ||||
|     "tryInferAppIdFromCode": "Coba simpulkan ID aplikasi dari kode sumber", | ||||
|     "removeOnExternalUninstall": "Secara otomatis menghapus aplikasi yang dicopot pemasangannya dari luar", | ||||
|     "tryInferAppIdFromCode": "Coba untuk menebak ID aplikasi dari kode sumber", | ||||
|     "removeOnExternalUninstall": "Hapus otomatis aplikasi yang dicopot pemasangannya dari luar", | ||||
|     "pickHighestVersionCode": "Pilih otomatis APK dengan versi kode tertinggi", | ||||
|     "checkUpdateOnDetailPage": "Periksa pembaruan saat membuka halaman detail aplikasi", | ||||
|     "disablePageTransitions": "Nonaktifkan animasi transisi halaman", | ||||
| @@ -256,8 +256,9 @@ | ||||
|     "intermediateLinkNotFound": "Tautan perantara tidak ditemukan", | ||||
|     "intermediateLink": "Tautan perantara", | ||||
|     "exemptFromBackgroundUpdates": "Dikecualikan dari pembaruan latar belakang (jika diaktifkan)", | ||||
|     "bgUpdatesOnWiFiOnly": "Nonaktifkan pembaruan latar belakang saat tidak menggunakan WiFi", | ||||
|     "autoSelectHighestVersionCode": "Secara otomatis pilih APK dengan versi kode tertinggi", | ||||
|     "bgUpdatesOnWiFiOnly": "Nonaktifkan pembaruan latar belakang saat tidak menggunakan Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Menonaktifkan pembaruan latar belakang saat tidak mengisi daya", | ||||
|     "autoSelectHighestVersionCode": "Pilih otomatis APK dengan versi kode tertinggi", | ||||
|     "versionExtractionRegEx": "Reguler ekspresi terkait ekstraksi versi string", | ||||
|     "trimVersionString": "Potong versi string dengan reguler ekspresi", | ||||
|     "matchGroupToUseForX": "Cocokkan grup yang akan digunakan untuk \"{}\"", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Bagikan ke AppVerifier, lalu kembali ke sini jika sudah siap.", | ||||
|     "wiki": "Bantuan/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfigurasi aplikasi Crowdsourced (risiko penggunaan ditanggung sendiri)", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nama", | ||||
|     "smartname": "Nama (Cerdas)", | ||||
|     "sortMethod": "Metode Penyortiran", | ||||
|     "welcome": "Selamat datang.", | ||||
|     "documentationLinksNote": "Halaman GitHub Obtainium yang ditautkan di bawah ini berisi tautan ke video, artikel, diskusi, dan sumber daya lain yang akan membantu Anda memahami cara menggunakan aplikasi.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Hapus aplikasi?", | ||||
|         "other": "Hapus aplikasi?" | ||||
| @@ -356,8 +367,8 @@ | ||||
|         "other": "Menghapus {n} catatan (sebelum = {before}, setelah = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} dan 1 aplikasi lainnya mendapat pembaruan.", | ||||
|         "other": "{} dan {} aplikasi lainnya mendapat pembaruan." | ||||
|         "one": "{} dan 1 aplikasi lainnya memiliki pembaruan.", | ||||
|         "other": "{} dan {} aplikasi lainnya memiliki pembaruan." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} dan 1 aplikasi lainnya telah diperbarui.", | ||||
|   | ||||
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Disinstalla dal dispositivo", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Funziona solo per le app con il rilevamento della versione disattivato.", | ||||
|     "releaseDateAsVersion": "Usa data di rilascio come versione", | ||||
|     "releaseTitleAsVersion": "Usa il titolo della versione come stringa di versione", | ||||
|     "releaseDateAsVersionExplanation": "Questa opzione dovrebbe essere usata solo per le app in cui il rilevamento della versione non funziona correttamente, ma è disponibile una data di rilascio.", | ||||
|     "changes": "Novità", | ||||
|     "releaseDate": "Data di rilascio", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Rilevamento di versione standard", | ||||
|     "groupByCategory": "Raggruppa per categoria", | ||||
|     "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile", | ||||
|     "autoLinkFilterByArch": "Tentare di filtrare i collegamenti in base all'architettura della CPU, se possibile.", | ||||
|     "overrideSource": "Sovrascrivi fonte", | ||||
|     "dontShowAgain": "Non mostrarlo più", | ||||
|     "dontShowTrackOnlyWarnings": "Non mostrare gli avvisi 'Solo-Monitoraggio'", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "Link intermedio non trovato", | ||||
|     "intermediateLink": "Collegamento intermedio", | ||||
|     "exemptFromBackgroundUpdates": "Esente da aggiornamenti in secondo piano (se attivo)", | ||||
|     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il WiFi", | ||||
|     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Disabilita gli aggiornamenti in background quando non è in carica", | ||||
|     "autoSelectHighestVersionCode": "Auto-seleziona APK con versionCode più alto", | ||||
|     "versionExtractionRegEx": "RegEx di estrazione versione", | ||||
|     "trimVersionString": "Tagliare la stringa della versione con RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Condividete con AppVerifier, quindi tornate qui quando siete pronti.", | ||||
|     "wiki": "Aiuto/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurazioni di app in crowdsourcing (uso a proprio rischio)", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nome", | ||||
|     "smartname": "Nome (intelligente)", | ||||
|     "sortMethod": "Metodo di ordinamento", | ||||
|     "welcome": "Benvenuti", | ||||
|     "documentationLinksNote": "La pagina GitHub di Obtainium collegata qui sotto contiene collegamenti a video, articoli, discussioni e altre risorse che vi aiuteranno a capire come utilizzare l'applicazione.", | ||||
|     "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,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "デバイスからアンインストールする", | ||||
|     "onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。", | ||||
|     "releaseDateAsVersion": "リリース日をバージョンとして使用する", | ||||
|     "releaseTitleAsVersion": "リリースタイトルをバージョンとして使用する", | ||||
|     "releaseDateAsVersionExplanation": "このオプションは、バージョン検出が正しく機能しないアプリで、リリース日が利用可能な場合にのみ使用する必要があります。", | ||||
|     "changes": "変更点", | ||||
|     "releaseDate": "リリース日", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "標準のバージョン検出", | ||||
|     "groupByCategory": "カテゴリ別にグループ化する", | ||||
|     "autoApkFilterByArch": "可能であれば、CPUアーキテクチャによるAPKのフィルタリングを試みる", | ||||
|     "autoLinkFilterByArch": "可能であれば、CPUアーキテクチャによるリンクのフィルタリングを試みる。", | ||||
|     "overrideSource": "ソースの上書き", | ||||
|     "dontShowAgain": "二度と表示しない", | ||||
|     "dontShowTrackOnlyWarnings": "「追跡のみ」の警告を表示しない", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "中間リンク", | ||||
|     "exemptFromBackgroundUpdates": "バックグラウンドアップデートを行わない (有効な場合)", | ||||
|     "bgUpdatesOnWiFiOnly": "WiFiを使用していない場合、バックグラウンドアップデートを無効にする", | ||||
|     "bgUpdatesWhileChargingOnly": "非充電時にバックグラウンドアップデートを無効にする", | ||||
|     "autoSelectHighestVersionCode": "最も高いバージョンコードのAPKを自動で選択する", | ||||
|     "versionExtractionRegEx": "バージョン抽出の正規表現", | ||||
|     "trimVersionString": "正規表現でバージョン文字列をトリムする", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "AppVerifierに共有し、準備ができたらここに戻ってください。", | ||||
|     "wiki": "ヘルプ/ウィキ", | ||||
|     "crowdsourcedConfigsLabel": "クラウドソーシングによるアプリの設定(利用は自己責任で)", | ||||
|     "crowdsourcedConfigsShort": "クラウドソーシングによるアプリの設定", | ||||
|     "allowInsecure": "安全でないHTTPリクエストを許可する", | ||||
|     "stayOneVersionBehind": "最新のバージョンから1つ前のものを使用する", | ||||
|     "refreshBeforeDownload": "ダウンロード前にアプリの詳細を更新する", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "名称", | ||||
|     "smartname": "名前(スマート)", | ||||
|     "sortMethod": "ソート方法", | ||||
|     "welcome": "ようこそ", | ||||
|     "documentationLinksNote": "以下のリンクにあるObtainium GitHubページには、ビデオ、記事、ディスカッション、その他のリソースへのリンクがあり、アプリの使い方を理解するのに役立ちます。", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "アプリを削除しますか?", | ||||
|         "other": "アプリを削除しますか?" | ||||
| @@ -356,23 +367,23 @@ | ||||
|         "other": "{n} 個のログをクリアしました (前 = {before}, 後 = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} とさらに {} 個のアプリのアップデートが利用可能です。", | ||||
|         "one": "{} とさらに 1 個のアプリのアップデートが利用可能です。", | ||||
|         "other": "{} とさらに {} 個のアプリのアップデートが利用可能です。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} とさらに {} 個のアプリがアップデートされました。", | ||||
|         "one": "{} とさらに 1 個のアプリがアップデートされました。", | ||||
|         "other": "{} とさらに {} 個のアプリがアップデートされました。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "更新に失敗しました。", | ||||
|         "other": "アプリのアップデートに失敗しました。" | ||||
|         "one": "{} とさらに 1 個のアプリのアップデートに失敗しました。", | ||||
|         "other": "{} とさらに {} 個のアプリのアップデートに失敗しました。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} とさらに 1 個のアプリがアップデートされた可能性があります。", | ||||
|         "other": "{} とさらに {} 個のアプリがアップデートされた可能性があります。" | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{}APK", | ||||
|         "other": "{}APK" | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK" | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										389
									
								
								assets/translations/ko.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,389 @@ | ||||
| { | ||||
|     "invalidURLForSource": "유효한 {} 앱 URL이 아닙니다", | ||||
|     "noReleaseFound": "적절한 릴리스를 찾을 수 없습니다", | ||||
|     "noVersionFound": "릴리스 버전을 결정할 수 없습니다", | ||||
|     "urlMatchesNoSource": "URL이 알려진 소스와 일치하지 않습니다", | ||||
|     "cantInstallOlderVersion": "앱의 이전 버전을 설치할 수 없습니다", | ||||
|     "appIdMismatch": "다운로드된 패키지 ID가 기존 앱 ID와 일치하지 않습니다", | ||||
|     "functionNotImplemented": "이 클래스는 이 기능을 구현하지 않았습니다", | ||||
|     "placeholder": "플레이스홀더", | ||||
|     "someErrors": "일부 오류가 발생했습니다", | ||||
|     "unexpectedError": "예기치 않은 오류", | ||||
|     "ok": "확인", | ||||
|     "and": "그리고", | ||||
|     "githubPATLabel": "GitHub 개인 액세스 토큰 (속도 제한 증가)", | ||||
|     "includePrereleases": "사전 릴리스 포함", | ||||
|     "fallbackToOlderReleases": "이전 릴리스로 대체", | ||||
|     "filterReleaseTitlesByRegEx": "정규 표현식으로 릴리스 제목 필터링", | ||||
|     "invalidRegEx": "잘못된 정규 표현식", | ||||
|     "noDescription": "설명 없음", | ||||
|     "cancel": "취소", | ||||
|     "continue": "계속", | ||||
|     "requiredInBrackets": "(필수)", | ||||
|     "dropdownNoOptsError": "오류: 드롭다운에는 최소 하나의 옵션이 있어야 합니다", | ||||
|     "colour": "색상", | ||||
|     "standard": "표준", | ||||
|     "custom": "사용자 정의", | ||||
|     "useMaterialYou": "Material You 사용", | ||||
|     "githubStarredRepos": "GitHub 즐겨찾기 저장소", | ||||
|     "uname": "사용자 이름", | ||||
|     "wrongArgNum": "잘못된 인수 수 제공", | ||||
|     "xIsTrackOnly": "{}는 추적 전용입니다", | ||||
|     "source": "소스", | ||||
|     "app": "앱", | ||||
|     "appsFromSourceAreTrackOnly": "이 소스의 앱은 '추적 전용'입니다.", | ||||
|     "youPickedTrackOnly": "당신은 '추적 전용' 옵션을 선택했습니다.", | ||||
|     "trackOnlyAppDescription": "앱은 업데이트를 위해 추적되지만 Obtainium은 다운로드하거나 설치할 수 없습니다.", | ||||
|     "cancelled": "취소됨", | ||||
|     "appAlreadyAdded": "앱이 이미 추가되었습니다", | ||||
|     "alreadyUpToDateQuestion": "앱이 이미 최신 상태입니까?", | ||||
|     "addApp": "앱 추가", | ||||
|     "appSourceURL": "앱 소스 URL", | ||||
|     "error": "오류", | ||||
|     "add": "추가", | ||||
|     "searchSomeSourcesLabel": "검색 (일부 소스만)", | ||||
|     "search": "검색", | ||||
|     "additionalOptsFor": "{}에 대한 추가 옵션", | ||||
|     "supportedSources": "지원되는 소스", | ||||
|     "trackOnlyInBrackets": "(추적 전용)", | ||||
|     "searchableInBrackets": "(검색 가능)", | ||||
|     "appsString": "앱", | ||||
|     "noApps": "앱 없음", | ||||
|     "noAppsForFilter": "필터에 대한 앱 없음", | ||||
|     "byX": "{}에 의해", | ||||
|     "percentProgress": "진행률: {}%", | ||||
|     "pleaseWait": "기다려 주세요", | ||||
|     "updateAvailable": "업데이트 가능", | ||||
|     "notInstalled": "설치되지 않음", | ||||
|     "pseudoVersion": "의사 버전", | ||||
|     "selectAll": "모두 선택", | ||||
|     "deselectX": "{} 선택 해제", | ||||
|     "xWillBeRemovedButRemainInstalled": "{}는 Obtainium에서 제거되지만 장치에 설치된 상태로 남아 있습니다.", | ||||
|     "removeSelectedAppsQuestion": "선택한 앱을 제거하시겠습니까?", | ||||
|     "removeSelectedApps": "선택한 앱 제거", | ||||
|     "updateX": "{} 업데이트", | ||||
|     "installX": "{} 설치", | ||||
|     "markXTrackOnlyAsUpdated": "{}\n(추적 전용)\n업데이트됨으로 표시", | ||||
|     "changeX": "{} 변경", | ||||
|     "installUpdateApps": "앱 설치/업데이트", | ||||
|     "installUpdateSelectedApps": "선택한 앱 설치/업데이트", | ||||
|     "markXSelectedAppsAsUpdated": "{} 선택한 앱을 업데이트됨으로 표시하시겠습니까?", | ||||
|     "no": "아니요", | ||||
|     "yes": "예", | ||||
|     "markSelectedAppsUpdated": "선택한 앱을 업데이트됨으로 표시", | ||||
|     "pinToTop": "상단에 고정", | ||||
|     "unpinFromTop": "상단에서 고정 해제", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "선택한 앱의 설치 상태를 재설정하시겠습니까?", | ||||
|     "installStatusOfXWillBeResetExplanation": "선택한 앱의 설치 상태가 재설정됩니다.\n\n이것은 실패한 업데이트나 기타 문제로 인해 Obtainium에 표시된 앱 버전이 잘못된 경우에 도움이 될 수 있습니다.", | ||||
|     "customLinkMessage": "이 링크는 Obtainium이 설치된 장치에서 작동합니다", | ||||
|     "shareAppConfigLinks": "앱 구성 HTML 링크로 공유", | ||||
|     "shareSelectedAppURLs": "선택한 앱 URL 공유", | ||||
|     "resetInstallStatus": "설치 상태 재설정", | ||||
|     "more": "더보기", | ||||
|     "removeOutdatedFilter": "구식 앱 필터 제거", | ||||
|     "showOutdatedOnly": "구식 앱만 표시", | ||||
|     "filter": "필터", | ||||
|     "filterApps": "앱 필터", | ||||
|     "appName": "앱 이름", | ||||
|     "author": "저자", | ||||
|     "upToDateApps": "최신 상태의 앱", | ||||
|     "nonInstalledApps": "설치되지 않은 앱", | ||||
|     "importExport": "가져오기/내보내기", | ||||
|     "settings": "설정", | ||||
|     "exportedTo": "{}로 내보내기 완료", | ||||
|     "obtainiumExport": "Obtainium 내보내기", | ||||
|     "invalidInput": "잘못된 입력", | ||||
|     "importedX": "{} 가져오기 완료", | ||||
|     "obtainiumImport": "Obtainium 가져오기", | ||||
|     "importFromURLList": "URL 목록에서 가져오기", | ||||
|     "searchQuery": "검색 쿼리", | ||||
|     "appURLList": "앱 URL 목록", | ||||
|     "line": "줄", | ||||
|     "searchX": "{} 검색", | ||||
|     "noResults": "결과가 없습니다", | ||||
|     "importX": "{} 가져오기", | ||||
|     "importedAppsIdDisclaimer": "가져온 앱은 \"설치되지 않음\"으로 잘못 표시될 수 있습니다.\n이를 수정하려면 Obtainium을 통해 다시 설치하십시오.\n앱 데이터에는 영향을 미치지 않습니다.\n\nURL 및 타사 가져오기 방법에만 영향을 미칩니다.", | ||||
|     "importErrors": "가져오기 오류", | ||||
|     "importedXOfYApps": "{}개의 앱 중 {}개 가져오기 완료.", | ||||
|     "followingURLsHadErrors": "다음 URL에 오류가 있었습니다:", | ||||
|     "selectURL": "URL 선택", | ||||
|     "selectURLs": "URL 선택", | ||||
|     "pick": "선택", | ||||
|     "theme": "테마", | ||||
|     "dark": "다크", | ||||
|     "light": "라이트", | ||||
|     "followSystem": "시스템 따르기", | ||||
|     "followSystemThemeExplanation": "시스템 테마를 따르려면 타사 애플리케이션을 사용해야 합니다", | ||||
|     "useBlackTheme": "순수한 검은색 다크 테마 사용", | ||||
|     "appSortBy": "앱 정렬 기준", | ||||
|     "authorName": "저자/이름", | ||||
|     "nameAuthor": "이름/저자", | ||||
|     "asAdded": "추가된 순서대로", | ||||
|     "appSortOrder": "앱 정렬 순서", | ||||
|     "ascending": "오름차순", | ||||
|     "descending": "내림차순", | ||||
|     "bgUpdateCheckInterval": "백그라운드 업데이트 확인 간격", | ||||
|     "neverManualOnly": "절대 - 수동만", | ||||
|     "appearance": "외관", | ||||
|     "showWebInAppView": "앱 보기에서 소스 웹페이지 표시", | ||||
|     "pinUpdates": "앱 보기 상단에 업데이트 고정", | ||||
|     "updates": "업데이트", | ||||
|     "sourceSpecific": "소스별", | ||||
|     "appSource": "앱 소스", | ||||
|     "noLogs": "로그 없음", | ||||
|     "appLogs": "앱 로그", | ||||
|     "close": "닫기", | ||||
|     "share": "공유", | ||||
|     "appNotFound": "앱을 찾을 수 없습니다", | ||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||
|     "pickAnAPK": "APK 선택", | ||||
|     "appHasMoreThanOnePackage": "{}에는 둘 이상의 패키지가 있습니다:", | ||||
|     "deviceSupportsXArch": "장치는 {} CPU 아키텍처를 지원합니다.", | ||||
|     "deviceSupportsFollowingArchs": "장치는 다음 CPU 아키텍처를 지원합니다:", | ||||
|     "warning": "경고", | ||||
|     "sourceIsXButPackageFromYPrompt": "앱 소스는 '{}'이지만 릴리스 패키지는 '{}'에서 제공됩니다. 계속하시겠습니까?", | ||||
|     "updatesAvailable": "업데이트 가능", | ||||
|     "updatesAvailableNotifDescription": "Obtainium이 추적하는 하나 이상의 앱에 대한 업데이트가 있음을 사용자에게 알립니다", | ||||
|     "noNewUpdates": "새로운 업데이트가 없습니다.", | ||||
|     "xHasAnUpdate": "{}에 업데이트가 있습니다.", | ||||
|     "appsUpdated": "앱 업데이트됨", | ||||
|     "appsNotUpdated": "앱 업데이트 실패", | ||||
|     "appsUpdatedNotifDescription": "백그라운드에서 하나 이상의 앱에 대한 업데이트가 적용되었음을 사용자에게 알립니다", | ||||
|     "xWasUpdatedToY": "{}가 {}로 업데이트되었습니다.", | ||||
|     "xWasNotUpdatedToY": "{}를 {}로 업데이트하지 못했습니다.", | ||||
|     "errorCheckingUpdates": "업데이트 확인 오류", | ||||
|     "errorCheckingUpdatesNotifDescription": "백그라운드 업데이트 확인이 실패할 때 표시되는 알림", | ||||
|     "appsRemoved": "앱 제거됨", | ||||
|     "appsRemovedNotifDescription": "로드 중 오류로 인해 하나 이상의 앱이 제거되었음을 사용자에게 알립니다", | ||||
|     "xWasRemovedDueToErrorY": "{}가 다음 오류로 인해 제거되었습니다: {}", | ||||
|     "completeAppInstallation": "앱 설치 완료", | ||||
|     "obtainiumMustBeOpenToInstallApps": "앱을 설치하려면 Obtainium이 열려 있어야 합니다", | ||||
|     "completeAppInstallationNotifDescription": "앱 설치를 완료하려면 Obtainium으로 돌아가도록 사용자에게 요청합니다", | ||||
|     "checkingForUpdates": "업데이트 확인 중", | ||||
|     "checkingForUpdatesNotifDescription": "업데이트 확인 시 나타나는 일시적인 알림", | ||||
|     "pleaseAllowInstallPerm": "Obtainium이 앱을 설치할 수 있도록 허용해 주세요", | ||||
|     "trackOnly": "추적 전용", | ||||
|     "errorWithHttpStatusCode": "오류 {}", | ||||
|     "versionCorrectionDisabled": "버전 수정 비활성화됨 (플러그인이 작동하지 않는 것 같습니다)", | ||||
|     "unknown": "알 수 없음", | ||||
|     "none": "없음", | ||||
|     "never": "절대", | ||||
|     "latestVersionX": "최신: {}", | ||||
|     "installedVersionX": "설치됨: {}", | ||||
|     "lastUpdateCheckX": "마지막 업데이트 확인: {}", | ||||
|     "remove": "제거", | ||||
|     "yesMarkUpdated": "예, 업데이트됨으로 표시", | ||||
|     "fdroid": "F-Droid 공식", | ||||
|     "appIdOrName": "앱 ID 또는 이름", | ||||
|     "appId": "앱 ID", | ||||
|     "appWithIdOrNameNotFound": "해당 ID 또는 이름의 앱을 찾을 수 없습니다", | ||||
|     "reposHaveMultipleApps": "저장소에는 여러 앱이 포함될 수 있습니다", | ||||
|     "fdroidThirdPartyRepo": "F-Droid 타사 저장소", | ||||
|     "install": "설치", | ||||
|     "markInstalled": "설치됨으로 표시", | ||||
|     "update": "업데이트", | ||||
|     "markUpdated": "업데이트됨으로 표시", | ||||
|     "additionalOptions": "추가 옵션", | ||||
|     "disableVersionDetection": "버전 감지 비활성화", | ||||
|     "noVersionDetectionExplanation": "이 옵션은 버전 감지가 올바르게 작동하지 않는 앱에만 사용해야 합니다.", | ||||
|     "downloadingX": "{} 다운로드 중", | ||||
|     "downloadX": "{} 다운로드", | ||||
|     "downloadedX": "{} 다운로드 완료", | ||||
|     "releaseAsset": "릴리스 자산", | ||||
|     "downloadNotifDescription": "앱 다운로드 진행 상황을 사용자에게 알립니다", | ||||
|     "noAPKFound": "APK를 찾을 수 없습니다", | ||||
|     "noVersionDetection": "버전 감지 없음", | ||||
|     "categorize": "분류", | ||||
|     "categories": "카테고리", | ||||
|     "category": "카테고리", | ||||
|     "noCategory": "카테고리 없음", | ||||
|     "noCategories": "카테고리 없음", | ||||
|     "deleteCategoriesQuestion": "카테고리를 삭제하시겠습니까?", | ||||
|     "categoryDeleteWarning": "삭제된 카테고리의 모든 앱은 미분류로 설정됩니다.", | ||||
|     "addCategory": "카테고리 추가", | ||||
|     "label": "레이블", | ||||
|     "language": "언어", | ||||
|     "copiedToClipboard": "클립보드에 복사됨", | ||||
|     "storagePermissionDenied": "저장소 권한 거부됨", | ||||
|     "selectedCategorizeWarning": "이 작업은 선택한 앱의 기존 카테고리 설정을 대체합니다.", | ||||
|     "filterAPKsByRegEx": "정규 표현식으로 APK 필터링", | ||||
|     "removeFromObtainium": "Obtainium에서 제거", | ||||
|     "uninstallFromDevice": "장치에서 제거", | ||||
|     "onlyWorksWithNonVersionDetectApps": "버전 감지가 비활성화된 앱에만 작동합니다.", | ||||
|     "releaseDateAsVersion": "릴리스 날짜를 버전 문자열로 사용", | ||||
|     "releaseTitleAsVersion": "릴리스 제목을 버전 문자열로 사용", | ||||
|     "releaseDateAsVersionExplanation": "이 옵션은 버전 감지가 올바르게 작동하지 않지만 릴리스 날짜가 있는 앱에만 사용해야 합니다.", | ||||
|     "changes": "변경 사항", | ||||
|     "releaseDate": "릴리스 날짜", | ||||
|     "importFromURLsInFile": "파일의 URL에서 가져오기 (OPML과 같은)", | ||||
|     "versionDetectionExplanation": "OS에서 감지된 버전과 버전 문자열 조정", | ||||
|     "versionDetection": "버전 감지", | ||||
|     "standardVersionDetection": "표준 버전 감지", | ||||
|     "groupByCategory": "카테고리별 그룹화", | ||||
|     "autoApkFilterByArch": "가능한 경우 CPU 아키텍처별로 APK 필터링 시도", | ||||
|     "autoLinkFilterByArch": "가능하면 CPU 아키텍처별로 링크를 필터링해 보세요.", | ||||
|     "overrideSource": "소스 재정의", | ||||
|     "dontShowAgain": "다시 표시하지 않기", | ||||
|     "dontShowTrackOnlyWarnings": "'추적 전용' 경고 표시 안 함", | ||||
|     "dontShowAPKOriginWarnings": "APK 출처 경고 표시 안 함", | ||||
|     "moveNonInstalledAppsToBottom": "설치되지 않은 앱을 앱 보기 하단으로 이동", | ||||
|     "gitlabPATLabel": "GitLab 개인 액세스 토큰", | ||||
|     "about": "정보", | ||||
|     "requiresCredentialsInSettings": "{}는 추가 자격 증명이 필요합니다 (설정에서)", | ||||
|     "checkOnStart": "시작 시 업데이트 확인", | ||||
|     "tryInferAppIdFromCode": "소스 코드에서 앱 ID 추론 시도", | ||||
|     "removeOnExternalUninstall": "외부에서 제거된 앱 자동 제거", | ||||
|     "pickHighestVersionCode": "가장 높은 버전 코드 APK 자동 선택", | ||||
|     "checkUpdateOnDetailPage": "앱 세부 정보 페이지 열 때 업데이트 확인", | ||||
|     "disablePageTransitions": "페이지 전환 애니메이션 비활성화", | ||||
|     "reversePageTransitions": "페이지 전환 애니메이션 반전", | ||||
|     "minStarCount": "최소 별 개수", | ||||
|     "addInfoBelow": "아래에 이 정보를 추가하십시오.", | ||||
|     "addInfoInSettings": "설정에 이 정보를 추가하십시오.", | ||||
|     "githubSourceNote": "GitHub 속도 제한은 API 키를 사용하여 피할 수 있습니다.", | ||||
|     "sortByLastLinkSegment": "링크의 마지막 세그먼트로만 정렬", | ||||
|     "filterReleaseNotesByRegEx": "정규 표현식으로 릴리스 노트 필터링", | ||||
|     "customLinkFilterRegex": "정규 표현식으로 사용자 정의 APK 링크 필터링 (기본값 '.apk$')", | ||||
|     "appsPossiblyUpdated": "앱 업데이트 시도됨", | ||||
|     "appsPossiblyUpdatedNotifDescription": "백그라운드에서 하나 이상의 앱에 대한 업데이트가 잠재적으로 적용되었음을 사용자에게 알립니다", | ||||
|     "xWasPossiblyUpdatedToY": "{}가 {}로 업데이트되었을 수 있습니다.", | ||||
|     "enableBackgroundUpdates": "백그라운드 업데이트 활성화", | ||||
|     "backgroundUpdateReqsExplanation": "모든 앱에 대해 백그라운드 업데이트가 가능하지 않을 수 있습니다.", | ||||
|     "backgroundUpdateLimitsExplanation": "백그라운드 설치의 성공 여부는 Obtainium이 열릴 때만 확인할 수 있습니다.", | ||||
|     "verifyLatestTag": "'최신' 태그 확인", | ||||
|     "intermediateLinkRegex": "'중간' 링크 방문 필터", | ||||
|     "filterByLinkText": "링크 텍스트로 링크 필터링", | ||||
|     "intermediateLinkNotFound": "중간 링크를 찾을 수 없습니다", | ||||
|     "intermediateLink": "중간 링크", | ||||
|     "exemptFromBackgroundUpdates": "백그라운드 업데이트에서 제외 (활성화된 경우)", | ||||
|     "bgUpdatesOnWiFiOnly": "WiFi가 아닐 때 백그라운드 업데이트 비활성화", | ||||
|     "bgUpdatesWhileChargingOnly": "충전 중이 아닐 때 백그라운드 업데이트 비활성화", | ||||
|     "autoSelectHighestVersionCode": "가장 높은 versionCode APK 자동 선택", | ||||
|     "versionExtractionRegEx": "버전 문자열 추출 정규 표현식", | ||||
|     "trimVersionString": "정규 표현식으로 버전 문자열 자르기", | ||||
|     "matchGroupToUseForX": "\"{}\"에 사용할 일치 그룹", | ||||
|     "matchGroupToUse": "버전 문자열 추출 정규 표현식에 사용할 일치 그룹", | ||||
|     "highlightTouchTargets": "덜 명확한 터치 대상 강조", | ||||
|     "pickExportDir": "내보내기 디렉토리 선택", | ||||
|     "autoExportOnChanges": "변경 시 자동 내보내기", | ||||
|     "includeSettings": "설정 포함", | ||||
|     "filterVersionsByRegEx": "정규 표현식으로 버전 필터링", | ||||
|     "trySelectingSuggestedVersionCode": "제안된 versionCode APK 선택 시도", | ||||
|     "dontSortReleasesList": "API에서 릴리스 순서 유지", | ||||
|     "reverseSort": "정렬 반전", | ||||
|     "takeFirstLink": "첫 번째 링크 선택", | ||||
|     "skipSort": "정렬 건너뛰기", | ||||
|     "debugMenu": "디버그 메뉴", | ||||
|     "bgTaskStarted": "백그라운드 작업 시작됨 - 로그를 확인하세요.", | ||||
|     "runBgCheckNow": "지금 백그라운드 업데이트 확인 실행", | ||||
|     "versionExtractWholePage": "전체 페이지에 버전 문자열 추출 정규 표현식 적용", | ||||
|     "installing": "설치 중", | ||||
|     "skipUpdateNotifications": "업데이트 알림 건너뛰기", | ||||
|     "updatesAvailableNotifChannel": "업데이트 가능", | ||||
|     "appsUpdatedNotifChannel": "앱 업데이트됨", | ||||
|     "appsPossiblyUpdatedNotifChannel": "앱 업데이트 시도됨", | ||||
|     "errorCheckingUpdatesNotifChannel": "업데이트 확인 오류", | ||||
|     "appsRemovedNotifChannel": "앱 제거됨", | ||||
|     "downloadingXNotifChannel": "{} 다운로드 중", | ||||
|     "completeAppInstallationNotifChannel": "앱 설치 완료", | ||||
|     "checkingForUpdatesNotifChannel": "업데이트 확인 중", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "설치된 앱과 추적 전용 앱만 업데이트 확인", | ||||
|     "supportFixedAPKURL": "고정 APK URL 지원", | ||||
|     "selectX": "{} 선택", | ||||
|     "parallelDownloads": "병렬 다운로드 허용", | ||||
|     "useShizuku": "Shizuku 또는 Sui를 사용하여 설치", | ||||
|     "shizukuBinderNotFound": "Shizuku 서비스가 실행 중이 아닙니다", | ||||
|     "shizukuOld": "오래된 Shizuku 버전 (<11) - 업데이트 필요", | ||||
|     "shizukuOldAndroidWithADB": "ADB로 Android < 8.1에서 실행 중인 Shizuku - Android를 업데이트하거나 대신 Sui를 사용하세요", | ||||
|     "shizukuPretendToBeGooglePlay": "설치 소스로 Google Play 설정 (Shizuku 사용 시)", | ||||
|     "useSystemFont": "시스템 글꼴 사용", | ||||
|     "useVersionCodeAsOSVersion": "앱 versionCode를 OS에서 감지된 버전으로 사용", | ||||
|     "requestHeader": "요청 헤더", | ||||
|     "useLatestAssetDateAsReleaseDate": "최신 자산 업로드를 릴리스 날짜로 사용", | ||||
|     "defaultPseudoVersioningMethod": "기본 의사 버전 관리 방법", | ||||
|     "partialAPKHash": "부분 APK 해시", | ||||
|     "APKLinkHash": "APK 링크 해시", | ||||
|     "directAPKLink": "직접 APK 링크", | ||||
|     "pseudoVersionInUse": "의사 버전 사용 중", | ||||
|     "installed": "설치됨", | ||||
|     "latest": "최신", | ||||
|     "invertRegEx": "정규 표현식 반전", | ||||
|     "note": "노트", | ||||
|     "selfHostedNote": "\"{}\" 드롭다운을 사용하여 소스의 자체 호스팅/사용자 정의 인스턴스에 도달할 수 있습니다.", | ||||
|     "badDownload": "APK를 구문 분석할 수 없습니다 (호환되지 않거나 부분 다운로드)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "새 앱을 AppVerifier와 공유 (가능한 경우)", | ||||
|     "appVerifierInstructionToast": "AppVerifier에 공유한 후 준비가 되면 여기로 돌아오세요.", | ||||
|     "wiki": "도움말/위키", | ||||
|     "crowdsourcedConfigsLabel": "크라우드소싱 앱 구성 (자신의 책임 하에 사용)", | ||||
|     "crowdsourcedConfigsShort": "크라우드소싱 앱 구성", | ||||
|     "allowInsecure": "안전하지 않은 HTTP 요청 허용", | ||||
|     "stayOneVersionBehind": "최신 버전보다 한 버전 뒤에 머무르기", | ||||
|     "refreshBeforeDownload": "다운로드 전에 앱 세부 정보 새로 고침", | ||||
|     "tencentAppStore": "텐센트 앱 스토어", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "이름", | ||||
|     "smartname": "이름(스마트)", | ||||
|     "sortMethod": "정렬 방법", | ||||
|     "welcome": "환영", | ||||
|     "documentationLinksNote": "아래에 링크된 Obtainium 깃허브 페이지에는 앱 사용 방법을 이해하는 데 도움이 되는 동영상, 기사, 토론 및 기타 리소스에 대한 링크가 포함되어 있습니다.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "앱을 제거하시겠습니까?", | ||||
|         "other": "앱을 제거하시겠습니까?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "요청이 너무 많습니다 (속도 제한) - {}분 후에 다시 시도하세요", | ||||
|         "other": "요청이 너무 많습니다 (속도 제한) - {}분 후에 다시 시도하세요" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "BG 업데이트 확인 중 {} 오류가 발생했습니다. {}분 후에 다시 확인을 예약합니다", | ||||
|         "other": "BG 업데이트 확인 중 {} 오류가 발생했습니다. {}분 후에 다시 확인을 예약합니다" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "BG 업데이트 확인에서 {}개의 업데이트를 발견했습니다 - 필요 시 사용자에게 알립니다", | ||||
|         "other": "BG 업데이트 확인에서 {}개의 업데이트를 발견했습니다 - 필요 시 사용자에게 알립니다" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} 앱", | ||||
|         "other": "{} 앱" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URL" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} 분", | ||||
|         "other": "{} 분" | ||||
|     }, | ||||
|     "hour": { | ||||
|         "one": "{} 시간", | ||||
|         "other": "{} 시간" | ||||
|     }, | ||||
|     "day": { | ||||
|         "one": "{} 일", | ||||
|         "other": "{} 일" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "{n}개의 로그가 지워졌습니다 (이전 = {before}, 이후 = {after})", | ||||
|         "other": "{n}개의 로그가 지워졌습니다 (이전 = {before}, 이후 = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} 및 1개의 앱에 업데이트가 있습니다.", | ||||
|         "other": "{} 및 {}개의 앱에 업데이트가 있습니다." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} 및 1개의 앱이 업데이트되었습니다.", | ||||
|         "other": "{} 및 {}개의 앱이 업데이트되었습니다." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "{} 및 1개의 앱 업데이트에 실패했습니다.", | ||||
|         "other": "{} 및 {}개의 앱 업데이트에 실패했습니다." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} 및 1개의 앱이 업데이트되었을 수 있습니다.", | ||||
|         "other": "{} 및 {}개의 앱이 업데이트되었을 수 있습니다." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK" | ||||
|     } | ||||
| } | ||||
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Van apparaat verwijderen", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Werkt alleen voor apps waarbij versieherkenning is uitgeschakeld.", | ||||
|     "releaseDateAsVersion": "Releasedatum als versie gebruiken", | ||||
|     "releaseTitleAsVersion": "Gebruik releasetitel als versiestring", | ||||
|     "releaseDateAsVersionExplanation": "Deze optie moet alleen worden gebruikt voor apps waar versieherkenning niet correct werkt, maar waar wel een releasedatum beschikbaar is.", | ||||
|     "changes": "Aanpassingen", | ||||
|     "releaseDate": "Releasedatum", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Standaard versieherkenning", | ||||
|     "groupByCategory": "Groeperen op categorie", | ||||
|     "autoApkFilterByArch": "Probeer APK's te filteren op CPU-architectuur, indien mogelijk", | ||||
|     "autoLinkFilterByArch": "Probeer links zo mogelijk te filteren op CPU-architectuur", | ||||
|     "overrideSource": "Bron overschrijven", | ||||
|     "dontShowAgain": "Laat dit niet meer zien", | ||||
|     "dontShowTrackOnlyWarnings": "Geen waarschuwingen weergeven voor 'Alleen volgen'", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "Intermediaire link niet gevonden", | ||||
|     "intermediateLink": "Intermediaire link", | ||||
|     "exemptFromBackgroundUpdates": "Vrijgesteld van achtergrond-updates (indien ingeschakeld)", | ||||
|     "bgUpdatesOnWiFiOnly": "Achtergrond-updates uitschakelen wanneer niet verbonden met WiFi", | ||||
|     "bgUpdatesOnWiFiOnly": "Achtergrond-updates uitschakelen wanneer niet verbonden met Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Achtergrondupdates uitschakelen als er niet wordt opgeladen", | ||||
|     "autoSelectHighestVersionCode": "De APK met de hoogste versiecode automatisch selecteren", | ||||
|     "versionExtractionRegEx": "Reguliere expressie voor versie-extractie", | ||||
|     "trimVersionString": "Versie string trimmen met RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Deel het met AppVerifier en keer daarna hier terug.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App-configuraties (gebruik op eigen risico)", | ||||
|     "crowdsourcedConfigsShort": "App-configuraties door menigte", | ||||
|     "allowInsecure": "Onveilige HTTP-verzoeken toestaan", | ||||
|     "stayOneVersionBehind": "Blijf een versie achter op de nieuwste", | ||||
|     "refreshBeforeDownload": "Vernieuw app details voor download", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Naam", | ||||
|     "smartname": "Naam (Slim)", | ||||
|     "sortMethod": "Sorteermethode", | ||||
|     "welcome": "Welkom", | ||||
|     "documentationLinksNote": "De GitHub pagina van Obtainium waarnaar hieronder wordt gelinkt bevat links naar video's, artikelen, discussies en andere bronnen die je zullen helpen begrijpen hoe je de app kunt gebruiken.", | ||||
|     "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,15 +211,17 @@ | ||||
|     "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ż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", | ||||
|     "autoApkFilterByArch": "Spróbuj filtrować pliki APK według architektury procesora, jeśli to możliwe", | ||||
|     "autoLinkFilterByArch": "Spróbuj filtrować linki według architektury procesora, jeśli to możliwe.", | ||||
|     "overrideSource": "Nadpisz źródło", | ||||
|     "dontShowAgain": "Nie pokazuj tego ponownie", | ||||
|     "dontShowTrackOnlyWarnings": "Nie pokazuj ostrzeżeń \"Tylko obserwowana\"", | ||||
| @@ -251,15 +251,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", | ||||
| @@ -270,7 +271,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.", | ||||
| @@ -283,38 +284,48 @@ | ||||
|     "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)", | ||||
|     "crowdsourcedConfigsLabel": "Baza konfiguracji", | ||||
|     "crowdsourcedConfigsShort": "Baza konfiguracji", | ||||
|     "allowInsecure": "Zezwalaj na niezabezpieczone żądania HTTP", | ||||
|     "stayOneVersionBehind": "Pozostań jedną wersję w tyle za najnowszą", | ||||
|     "refreshBeforeDownload": "Odśwież szczegóły aplikacji przed pobraniem", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nazwa", | ||||
|     "smartname": "Nazwa (Smart)", | ||||
|     "sortMethod": "Metoda sortowania", | ||||
|     "welcome": "Witamy", | ||||
|     "documentationLinksNote": "Strona Obtainium GitHub, do której link znajduje się poniżej, zawiera linki do filmów, artykułów, dyskusji i innych zasobów, które pomogą ci zrozumieć, jak korzystać z aplikacji.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Usunąć aplikację?", | ||||
|         "few": "Usunąć aplikacje?", | ||||
| @@ -388,8 +399,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.", | ||||
| @@ -398,7 +411,9 @@ | ||||
|         "other": "{} i {} inne apki mogły zostać zaktualizowane." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK" | ||||
|         "one": "{} apk", | ||||
|         "few": "{} apki", | ||||
|         "many": "{} apek", | ||||
|         "other": "{} apki" | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										389
									
								
								assets/translations/pt-BR.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,389 @@ | ||||
| { | ||||
|     "invalidURLForSource": "Não é uma URL de app válida de {}", | ||||
|     "noReleaseFound": "Não foi possível encontrar um lançamento adequado", | ||||
|     "noVersionFound": "Não foi possível determinar a versão do lançamento", | ||||
|     "urlMatchesNoSource": "A URL não corresponde com nenhuma fonte conhecida", | ||||
|     "cantInstallOlderVersion": "Não é possível instalar uma versão mais antiga de um app", | ||||
|     "appIdMismatch": "O ID do pacote baixado não corresponde ao existente", | ||||
|     "functionNotImplemented": "Essa classe não implementou esse recurso ainda", | ||||
|     "placeholder": "Espaço reservado", | ||||
|     "someErrors": "Ocorreram alguns erros", | ||||
|     "unexpectedError": "Erro inesperado", | ||||
|     "ok": "Ok", | ||||
|     "and": "e", | ||||
|     "githubPATLabel": "Token de acesso pessoal do GitHub (aumenta o limite de taxa)", | ||||
|     "includePrereleases": "Incluir pré-lançamentos", | ||||
|     "fallbackToOlderReleases": "Recorrer à lançamentos mais antigos", | ||||
|     "filterReleaseTitlesByRegEx": "Filtrar títulos de lançamentos por expressão regular", | ||||
|     "invalidRegEx": "Expressão regular inválida", | ||||
|     "noDescription": "Sem descrição", | ||||
|     "cancel": "Cancelar", | ||||
|     "continue": "Continuar", | ||||
|     "requiredInBrackets": "(obrigatório)", | ||||
|     "dropdownNoOptsError": "ERRO: O MENU DEVE TER PELO MENOS UMA OPÇÃO", | ||||
|     "colour": "Cor", | ||||
|     "standard": "Padrão", | ||||
|     "custom": "Personalizado", | ||||
|     "useMaterialYou": "Usar Material You", | ||||
|     "githubStarredRepos": "Repositórios com estrela do GitHub", | ||||
|     "uname": "Nome de usuário", | ||||
|     "wrongArgNum": "Número errado de argumentos fornecidos", | ||||
|     "xIsTrackOnly": "{} é somente de rastreio", | ||||
|     "source": "Fonte", | ||||
|     "app": "App", | ||||
|     "appsFromSourceAreTrackOnly": "Apps desta fonte são somente para rastreamento.", | ||||
|     "youPickedTrackOnly": "Você selecionou a opção de somente rastreamento.", | ||||
|     "trackOnlyAppDescription": "As atualizações do app serão rastreadas, mas o Obtainium não baixará ou instalará elas.", | ||||
|     "cancelled": "Cancelado", | ||||
|     "appAlreadyAdded": "O app já foi adicionado", | ||||
|     "alreadyUpToDateQuestion": "O app já está atualizado?", | ||||
|     "addApp": "Adicionar app", | ||||
|     "appSourceURL": "URL da fonte do app", | ||||
|     "error": "Erro", | ||||
|     "add": "Adicionar", | ||||
|     "searchSomeSourcesLabel": "Pesquisar (somente algumas fontes)", | ||||
|     "search": "Pesquisar", | ||||
|     "additionalOptsFor": "Opções adicionais de {}", | ||||
|     "supportedSources": "Fontes suportadas", | ||||
|     "trackOnlyInBrackets": "(somente rastreamento)", | ||||
|     "searchableInBrackets": "(pesquisável)", | ||||
|     "appsString": "Apps", | ||||
|     "noApps": "Nenhum app", | ||||
|     "noAppsForFilter": "Nenhum app pro filtro", | ||||
|     "byX": "Por {}", | ||||
|     "percentProgress": "Progresso: {}%", | ||||
|     "pleaseWait": "Por favor aguarde", | ||||
|     "updateAvailable": "Atualização disponível", | ||||
|     "notInstalled": "Não instalado", | ||||
|     "pseudoVersion": "pseudo-versão", | ||||
|     "selectAll": "Selecionar tudo", | ||||
|     "deselectX": "Desselecionar {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} será removido do Obtainium mas continuará instalado no dispositivo.", | ||||
|     "removeSelectedAppsQuestion": "Remover os apps selecionados?", | ||||
|     "removeSelectedApps": "Remover apps selecionados", | ||||
|     "updateX": "Atualizar {}", | ||||
|     "installX": "Instalar {}", | ||||
|     "markXTrackOnlyAsUpdated": "Marcar {}\n(somente rastreamento)\ncomo atualizado", | ||||
|     "changeX": "Alterar {}", | ||||
|     "installUpdateApps": "Instalar/atualizar apps", | ||||
|     "installUpdateSelectedApps": "Instalar/atualizar apps selecionados", | ||||
|     "markXSelectedAppsAsUpdated": "Marcar os {} apps selecionados como atualizados?", | ||||
|     "no": "Não", | ||||
|     "yes": "Sim", | ||||
|     "markSelectedAppsUpdated": "Marcar apps selecionados como atualizados", | ||||
|     "pinToTop": "Fixar ao topo", | ||||
|     "unpinFromTop": "Desfixar do topo", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Redefinir o estado de instalação dos apps selecionados?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Os estados de instalação dos apps selecionados serão redefinidos.\n\nIsso pode ajudar quando a versão exibida no Obtainium está incorreta devido a atualizações malsucedidas ou outros problemas.", | ||||
|     "customLinkMessage": "Esses links funcionarão em dispositivos com o Obtainium instalado", | ||||
|     "shareAppConfigLinks": "Compartilhar configuração do app como um link HTML", | ||||
|     "shareSelectedAppURLs": "Compartilhar as URLs dos apps selecionados", | ||||
|     "resetInstallStatus": "Redefinir estado de instalação", | ||||
|     "more": "Mais", | ||||
|     "removeOutdatedFilter": "Remover filtro de apps desatualizados", | ||||
|     "showOutdatedOnly": "Mostrar somente apps desatualizados", | ||||
|     "filter": "Filtro", | ||||
|     "filterApps": "Filtrar apps", | ||||
|     "appName": "Nome do app", | ||||
|     "author": "Autor", | ||||
|     "upToDateApps": "Apps atualizados", | ||||
|     "nonInstalledApps": "Apps não instalados", | ||||
|     "importExport": "Importar/Exportar", | ||||
|     "settings": "Configurações", | ||||
|     "exportedTo": "Exportado para {}", | ||||
|     "obtainiumExport": "Exportação do Obtainium", | ||||
|     "invalidInput": "Entrada inválida", | ||||
|     "importedX": "{} importado(s)", | ||||
|     "obtainiumImport": "Importação do Obtainium", | ||||
|     "importFromURLList": "Importar da lista de URLs", | ||||
|     "searchQuery": "Consulta de pesquisa", | ||||
|     "appURLList": "Lista de URLs dos apps", | ||||
|     "line": "Linha", | ||||
|     "searchX": "Pesquisar {}", | ||||
|     "noResults": "Nenhum resultado encontrado", | ||||
|     "importX": "Importar {}", | ||||
|     "importedAppsIdDisclaimer": "Os apps importados podem ser exibidos incorretamente como se não estivessem instalados.\nPara resolver isso, reinstale eles pelo Obtainium.\nIsso não afetará os dados dos apps.\n\nIsso somente afeta a URL e os métodos de importação de terceiros.", | ||||
|     "importErrors": "Erros de importação", | ||||
|     "importedXOfYApps": "{} de {} foram importados.", | ||||
|     "followingURLsHadErrors": "As seguintes URLs tiveram erros:", | ||||
|     "selectURL": "Selecionar URL", | ||||
|     "selectURLs": "Selecionar URLs", | ||||
|     "pick": "Escolher", | ||||
|     "theme": "Tema", | ||||
|     "dark": "Escuro", | ||||
|     "light": "Claro", | ||||
|     "followSystem": "Seguir o sistema", | ||||
|     "followSystemThemeExplanation": "Só é possível seguir o tema do sistema ao usar aplicativos de terceiros", | ||||
|     "useBlackTheme": "Usar o tema escuro de preto profundo", | ||||
|     "appSortBy": "Ordenar apps por", | ||||
|     "authorName": "Autor/nome", | ||||
|     "nameAuthor": "Nome/autor", | ||||
|     "asAdded": "Como adicionados", | ||||
|     "appSortOrder": "Ordem dos apps", | ||||
|     "ascending": "Crescente", | ||||
|     "descending": "Decrescente", | ||||
|     "bgUpdateCheckInterval": "Intervalo de busca por atualizações em segundo plano", | ||||
|     "neverManualOnly": "Nunca - somente manualmente", | ||||
|     "appearance": "Aparência", | ||||
|     "showWebInAppView": "Mostrar a fonte da pagina web na tela de apps", | ||||
|     "pinUpdates": "Fixar atualizações no topo da tela de apps", | ||||
|     "updates": "Atualizações", | ||||
|     "sourceSpecific": "Específico à fonte", | ||||
|     "appSource": "Fonte do app", | ||||
|     "noLogs": "Nenhum registro", | ||||
|     "appLogs": "Registros do app", | ||||
|     "close": "Fechar", | ||||
|     "share": "Compartilhar", | ||||
|     "appNotFound": "O app não foi encontrado", | ||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||
|     "pickAnAPK": "Selecione um APK", | ||||
|     "appHasMoreThanOnePackage": "{} tem mais de um pacote:", | ||||
|     "deviceSupportsXArch": "Seu dispositivo suporta a arquitetura de CPU {}.", | ||||
|     "deviceSupportsFollowingArchs": "Seu dispositivo suporta as seguintes arquiteturas de CPU:", | ||||
|     "warning": "Alerta", | ||||
|     "sourceIsXButPackageFromYPrompt": "A fonte do app é '{}' mas o pacote de lançamento vem de '{}'. Continuar mesmo assim?", | ||||
|     "updatesAvailable": "Atualizações disponíveis", | ||||
|     "updatesAvailableNotifDescription": "Notifica o usuário que atualizações estão disponíveis para um ou mais apps rastreados pelo Obtainium", | ||||
|     "noNewUpdates": "Nenhuma atualização disponível.", | ||||
|     "xHasAnUpdate": "{} tem uma atualização.", | ||||
|     "appsUpdated": "Apps atualizados", | ||||
|     "appsNotUpdated": "Falhou ao atualizar os aplicativos", | ||||
|     "appsUpdatedNotifDescription": "Notifica o usuário que atualizações de um ou mais apps foram aplicadas em segundo plano", | ||||
|     "xWasUpdatedToY": "{} foi atualizado para a versão {}.", | ||||
|     "xWasNotUpdatedToY": "Falha ao atualizar {} para a versão {}.", | ||||
|     "errorCheckingUpdates": "Ocorreu um erro ao buscar atualizações", | ||||
|     "errorCheckingUpdatesNotifDescription": "Uma notificação que mostra quando a busca de atualizações em segundo plano falha", | ||||
|     "appsRemoved": "Apps removidos", | ||||
|     "appsRemovedNotifDescription": "Notifica o usuário que um ou mais apps foram removidos devido a erros ao carregá-los", | ||||
|     "xWasRemovedDueToErrorY": "{} for removido devido ao erro: {}", | ||||
|     "completeAppInstallation": "Concluir instalação do app", | ||||
|     "obtainiumMustBeOpenToInstallApps": "O Obtainium precisa estar aberto para instalar apps", | ||||
|     "completeAppInstallationNotifDescription": "Pede pro usuário voltar ao Obtainium para concluir a instalação de um app", | ||||
|     "checkingForUpdates": "Buscando atualizações", | ||||
|     "checkingForUpdatesNotifDescription": "Notificação transitória que aparece ao buscar atualizações", | ||||
|     "pleaseAllowInstallPerm": "Permita que o Obtainium instale apps", | ||||
|     "trackOnly": "Somente rastreamento", | ||||
|     "errorWithHttpStatusCode": "Erro {}", | ||||
|     "versionCorrectionDisabled": "Correção de versão desativada (o plugin parece não funcionar)", | ||||
|     "unknown": "Desconhecido", | ||||
|     "none": "Nenhum", | ||||
|     "never": "Nunca", | ||||
|     "latestVersionX": "Mais recente: {}", | ||||
|     "installedVersionX": "Instalado: {}", | ||||
|     "lastUpdateCheckX": "Última busca por atualizações: {}", | ||||
|     "remove": "Remover", | ||||
|     "yesMarkUpdated": "Sim, marcar como atualizado", | ||||
|     "fdroid": "Oficial do F-Droid", | ||||
|     "appIdOrName": "ID do app ou nome", | ||||
|     "appId": "ID do app", | ||||
|     "appWithIdOrNameNotFound": "Nenhum app foi encontrado com aquele ID ou nome", | ||||
|     "reposHaveMultipleApps": "Repositórios podem conter vários apps", | ||||
|     "fdroidThirdPartyRepo": "Repositório de terceiros do F-Droid", | ||||
|     "install": "Instalar", | ||||
|     "markInstalled": "Marcar como instalado", | ||||
|     "update": "Atualizar", | ||||
|     "markUpdated": "Marcar como atualizado", | ||||
|     "additionalOptions": "Opções adicionais", | ||||
|     "disableVersionDetection": "Desativar detecção de versão", | ||||
|     "noVersionDetectionExplanation": "Essa opção só seve ser usada para apps aonde a detecção de versão não funciona corretamente.", | ||||
|     "downloadingX": "Baixando {}", | ||||
|     "downloadX": "Baixar {}", | ||||
|     "downloadedX": "{} foi baixado", | ||||
|     "releaseAsset": "Item de lançamento", | ||||
|     "downloadNotifDescription": "Notifica o usuário do progresso ao baixar um app", | ||||
|     "noAPKFound": "Nenhum APK encontrado", | ||||
|     "noVersionDetection": "Sem detecção de versão", | ||||
|     "categorize": "Categorizar", | ||||
|     "categories": "Categorias", | ||||
|     "category": "Categoria", | ||||
|     "noCategory": "Nenhuma categoria", | ||||
|     "noCategories": "Nenhuma categoria", | ||||
|     "deleteCategoriesQuestion": "Excluir categorias?", | ||||
|     "categoryDeleteWarning": "Todos os apps em categorias excluídas ficarão sem categoria.", | ||||
|     "addCategory": "Adicionar categoria", | ||||
|     "label": "Rótulo", | ||||
|     "language": "Idioma", | ||||
|     "copiedToClipboard": "Copiado para a área de transferência", | ||||
|     "storagePermissionDenied": "Permissão de armazenamento negada", | ||||
|     "selectedCategorizeWarning": "Isso substituirá a configuração de categoria existente dos apps selecionados.", | ||||
|     "filterAPKsByRegEx": "Filtrar APKs por expressão regular", | ||||
|     "removeFromObtainium": "Remover do Obtainium", | ||||
|     "uninstallFromDevice": "Desinstalar do dispositivo", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Funciona somente em apps com a detecção de versão desativada.", | ||||
|     "releaseDateAsVersion": "Usar data de lançamento como número da versão", | ||||
|     "releaseTitleAsVersion": "Usar título do lançamento como número da versão", | ||||
|     "releaseDateAsVersionExplanation": "Essa opção só deve ser usada para apps quais a detecção de versão não funciona corretamente, mas uma data de lançamento está disponível.", | ||||
|     "changes": "Alterações", | ||||
|     "releaseDate": "Data de lançamento", | ||||
|     "importFromURLsInFile": "Importar das URLs em arquivo (como OPML)", | ||||
|     "versionDetectionExplanation": "Combinar o número da versão com a versão detectada pelo sistema", | ||||
|     "versionDetection": "Detecção de versão", | ||||
|     "standardVersionDetection": "Detecção de versão padrão", | ||||
|     "groupByCategory": "Agrupar por categoria", | ||||
|     "autoApkFilterByArch": "Tentar filtrar APKs pela arquitetura da CPU quando possível", | ||||
|     "autoLinkFilterByArch": "Tentar filtrar links pela arquitetura da CPU quando possível", | ||||
|     "overrideSource": "Sobrescrever fonte", | ||||
|     "dontShowAgain": "Não mostrar isso novamente", | ||||
|     "dontShowTrackOnlyWarnings": "Não mostrar alertas de \"somente rastreamento\"", | ||||
|     "dontShowAPKOriginWarnings": "Não mostrar alertas de origem dos APKs", | ||||
|     "moveNonInstalledAppsToBottom": "Mover apps não instalados ao final da tela de apps", | ||||
|     "gitlabPATLabel": "Token de acesso pessoal do GitLab", | ||||
|     "about": "Sobre", | ||||
|     "requiresCredentialsInSettings": "{} precisa de credenciais adicionais (nas Configurações)", | ||||
|     "checkOnStart": "Buscar atualizações ao abrir o app", | ||||
|     "tryInferAppIdFromCode": "Tentar inferir o ID do app pelo código fonte", | ||||
|     "removeOnExternalUninstall": "Remover automaticamente apps desinstalados externamente", | ||||
|     "pickHighestVersionCode": "Selecionar APK de versão mais alta automaticamente", | ||||
|     "checkUpdateOnDetailPage": "Buscar atualizações ao abrir a tela de detalhes de um app", | ||||
|     "disablePageTransitions": "Desativar animações de transição de tela", | ||||
|     "reversePageTransitions": "Inverter animações de transição de tela", | ||||
|     "minStarCount": "Número de estrelas mínimo", | ||||
|     "addInfoBelow": "Adicione essa informação abaixo.", | ||||
|     "addInfoInSettings": "Adicione essa informação nas Configurações.", | ||||
|     "githubSourceNote": "O limite de taxa do GitHub pode ser evitado ao usar uma chave de API.", | ||||
|     "sortByLastLinkSegment": "Ordenar somente pelo ultimo segmento do link", | ||||
|     "filterReleaseNotesByRegEx": "Filtrar notas de lançamento por expressão regular", | ||||
|     "customLinkFilterRegex": "Filtro de link de APK personalizado por expressão regular (padrão '.apk$')", | ||||
|     "appsPossiblyUpdated": "Tentativas de atualização de apps", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Notifica o usuário que atualizações de um ou mais apps podem ter sido aplicadas em segundo plano", | ||||
|     "xWasPossiblyUpdatedToY": "{} pode ter sido atualizado para a versão {}.", | ||||
|     "enableBackgroundUpdates": "Ativar atualizações em segundo plano", | ||||
|     "backgroundUpdateReqsExplanation": "Atualizações em segundo plano podem não funcionar com todos os apps.", | ||||
|     "backgroundUpdateLimitsExplanation": "O sucesso de uma instalação em segundo plano só pode ser determinada ao abrir o Obtainium.", | ||||
|     "verifyLatestTag": "Verificar a tag 'mais recente'", | ||||
|     "intermediateLinkRegex": "Filtrar por um link 'intermediário' para visitar", | ||||
|     "filterByLinkText": "Filtrar links por texto do link", | ||||
|     "intermediateLinkNotFound": "Link intermediário não encontrado", | ||||
|     "intermediateLink": "Link intermediário", | ||||
|     "exemptFromBackgroundUpdates": "Isento de atualizações em segundo plano (caso ativadas)", | ||||
|     "bgUpdatesOnWiFiOnly": "Desativar atualizações em segundo plano fora do Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Desativar atualizações em segundo plano fora do carregador", | ||||
|     "autoSelectHighestVersionCode": "Selecionar automaticamente APK com o código de versão mais alto", | ||||
|     "versionExtractionRegEx": "ExReg de extração do número da versão", | ||||
|     "trimVersionString": "Cortar número da versal com ExReg", | ||||
|     "matchGroupToUseForX": "Corresponder grupo para o uso em \"{}\"", | ||||
|     "matchGroupToUse": "Corresponder grupo para o uso para a extração do número da versão por ExReg", | ||||
|     "highlightTouchTargets": "Acentuar alvos de toque menos óbvios", | ||||
|     "pickExportDir": "Selecionar pasta de exportação", | ||||
|     "autoExportOnChanges": "Exportar automaticamente ao ocorrer alterações", | ||||
|     "includeSettings": "Incluir configurações", | ||||
|     "filterVersionsByRegEx": "Filtrar versões por expressão regular", | ||||
|     "trySelectingSuggestedVersionCode": "Tente selecionar o APK com o código de versão sugerido", | ||||
|     "dontSortReleasesList": "Manter ordem de lançamento da API", | ||||
|     "reverseSort": "Ordem inversa", | ||||
|     "takeFirstLink": "Usar o primeiro link", | ||||
|     "skipSort": "Pular ordenação", | ||||
|     "debugMenu": "Menu de depuração", | ||||
|     "bgTaskStarted": "Tarefa em segundo plano iniada - verifique os registros.", | ||||
|     "runBgCheckNow": "Executar busca por atualizações em segundo plano agora", | ||||
|     "versionExtractWholePage": "Aplicar ExReg de extração de número de versão à página inteira", | ||||
|     "installing": "Instalando", | ||||
|     "skipUpdateNotifications": "Pular notificações de atualização", | ||||
|     "updatesAvailableNotifChannel": "Atualizações disponíveis", | ||||
|     "appsUpdatedNotifChannel": "Apps atualizados", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Tentativas de atualização de apps", | ||||
|     "errorCheckingUpdatesNotifChannel": "Erro ao buscar atualizações", | ||||
|     "appsRemovedNotifChannel": "Apps removidos", | ||||
|     "downloadingXNotifChannel": "Baixando {}", | ||||
|     "completeAppInstallationNotifChannel": "Concluir instalação do app", | ||||
|     "checkingForUpdatesNotifChannel": "Buscando atualizações", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Buscar atualizações somente para apps instalados e de somente rastreamento", | ||||
|     "supportFixedAPKURL": "Suportar URLs de APK fixas", | ||||
|     "selectX": "Selecionar {}", | ||||
|     "parallelDownloads": "Permitir downloads em paralelo", | ||||
|     "useShizuku": "Usar Shizuku ou Sui para instalação", | ||||
|     "shizukuBinderNotFound": "Serviço Shizuku não está em execução", | ||||
|     "shizukuOld": "Versão do Shizuku antiga (<11) - atualize", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku sendo executado no Android < 8.1 com ADB - atualize o Android ou use o Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Definir Google Play como a fonte de instalação (se o Shizuku é usado)", | ||||
|     "useSystemFont": "Usar a fonte do sistema", | ||||
|     "useVersionCodeAsOSVersion": "Usar código de versão do app como a versão detectada pelo sistema", | ||||
|     "requestHeader": "Cabeçalho da solicitação", | ||||
|     "useLatestAssetDateAsReleaseDate": "Usar o envio de item mais recente como a data de lançamento", | ||||
|     "defaultPseudoVersioningMethod": "Método de pseudo-versão padrão", | ||||
|     "partialAPKHash": "Hash do APK parcial", | ||||
|     "APKLinkHash": "Hash do link do APK", | ||||
|     "directAPKLink": "Link direto ao APK", | ||||
|     "pseudoVersionInUse": "Uma pseudo-versão está em uso", | ||||
|     "installed": "Instalado", | ||||
|     "latest": "Mais recente", | ||||
|     "invertRegEx": "Inverter expressão regular", | ||||
|     "note": "Observação", | ||||
|     "selfHostedNote": "O menu de opções \"{}\" pode ser usado para alcançar instâncias hospedadas-por-você/personalizadas de qualquer fonte.", | ||||
|     "badDownload": "O APK não pode ser interpretado (incompatível ou baixado parcialmente)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Compartilhar apps novos com o AppVerifier (se disponível)", | ||||
|     "appVerifierInstructionToast": "Compartilhe com o AppVerifier, e volte aqui ao estar pronto.", | ||||
|     "wiki": "Ajuda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurações de app pela comunidade (use ao seu próprio risco)", | ||||
|     "crowdsourcedConfigsShort": "Configurações de app da comunidade", | ||||
|     "allowInsecure": "Permitir solicitações de HTTP inseguras", | ||||
|     "stayOneVersionBehind": "Ficar uma versão antes da mais recente", | ||||
|     "refreshBeforeDownload": "Atualizar detalhes do app antes de baixar", | ||||
|     "tencentAppStore": "Loja de Apps da Tencent", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nome", | ||||
|     "smartname": "Nome (inteligente)", | ||||
|     "sortMethod": "Método de ordenação", | ||||
|     "welcome": "Boas vindas", | ||||
|     "documentationLinksNote": "A página do Obtainium no GitHub visível abaixo contém links de vídeos, artigos, discussões, e outros recursos que podem te ajudar ao usar o app.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Remover app?", | ||||
|         "other": "Remover apps?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Muitas solicitações (limitado) - tente novamente em {} minuto", | ||||
|         "other": "Muitas solicitações (limitado) - tente novamente em {} minutos" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "A busca de atualizações em segundo plano encontrou um {}, será agendado uma nova tentativa em {} minuto", | ||||
|         "other": "A busca de atualizações em segundo plano encontrou um {}, será agendado uma nova tentativa em {} minutos" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "BG update checking found {} update - will notify user if needed", | ||||
|         "other": "BG update checking found {} updates - will notify user if needed" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} app", | ||||
|         "other": "{} apps" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URLs" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} minuto", | ||||
|         "other": "{} minutos" | ||||
|     }, | ||||
|     "hour": { | ||||
|         "one": "{} hora", | ||||
|         "other": "{} horas" | ||||
|     }, | ||||
|     "day": { | ||||
|         "one": "{} dia", | ||||
|         "other": "{} dias" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "Cleared {n} log (before = {before}, after = {after})", | ||||
|         "other": "Cleared {n} logs (before = {before}, after = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} e mais 1 app têm atualizações.", | ||||
|         "other": "{} e mais {} apps têm atualizações." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} e mais 1 app foram atualizados.", | ||||
|         "other": "{} e mais {} apps foram atualizados." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Falha ao atualizar {} e mais 1 app.", | ||||
|         "other": "Falha ao atualizar {} e mais {} apps." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} e mais 1 app podem ter sido atualizados.", | ||||
|         "other": "{} e mais {} apps podem ter sido atualizados." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APKs" | ||||
|     } | ||||
| } | ||||
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Desinstalar do dispositivo", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Apenas funciona para aplicativos com detecção de versão desativada.", | ||||
|     "releaseDateAsVersion": "Usar data de lançamento como versão", | ||||
|     "releaseTitleAsVersion": "Utilizar o título da versão como cadeia de versões", | ||||
|     "releaseDateAsVersionExplanation": "Esta opção só deve ser usada para aplicativos onde a detecção de versão não funciona corretamente, mas há uma data de lançamento disponível.", | ||||
|     "changes": "Alterações", | ||||
|     "releaseDate": "Data de lançamento", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Detecção de versão padrão", | ||||
|     "groupByCategory": "Agroupar por categoria", | ||||
|     "autoApkFilterByArch": "Tente filtrar APKs por arquitetura de CPU, se possível", | ||||
|     "autoLinkFilterByArch": "Tentativa de filtrar as ligações por arquitetura de CPU, se possível", | ||||
|     "overrideSource": "Substituir fonte", | ||||
|     "dontShowAgain": "Não mostrar isso novamente", | ||||
|     "dontShowTrackOnlyWarnings": "Não mostrar avisos 'Apenas monitorar'", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "Link intermediário", | ||||
|     "exemptFromBackgroundUpdates": "Isento de atualizações em segundo-plano (se ativadas)", | ||||
|     "bgUpdatesOnWiFiOnly": "Desative as atualizações em segundo-plano quando não estiver conectado no Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Desativar actualizações em segundo plano quando não estiver a carregar", | ||||
|     "autoSelectHighestVersionCode": "Auto-selecionar a versão mais recente", | ||||
|     "versionExtractionRegEx": "Regex de extração de versão", | ||||
|     "trimVersionString": "Cortar a cadeia de caracteres da versão com RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Partilhe com o AppVerifier e, em seguida, regresse aqui quando estiver pronto.", | ||||
|     "wiki": "Ajuda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurações de aplicações de crowdsourcing (utilização por sua conta e risco)", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Nome", | ||||
|     "smartname": "Nome (Smart)", | ||||
|     "sortMethod": "Método de ordenação", | ||||
|     "welcome": "Bem-vindo", | ||||
|     "documentationLinksNote": "A página do Obtainium no GitHub com a ligação abaixo contém ligações para vídeos, artigos, discussões e outros recursos que o ajudarão a compreender como utilizar a aplicação.", | ||||
|     "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": "Обновить", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Удалить с устройства", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Работает только для приложений с отключенным определением версии", | ||||
|     "releaseDateAsVersion": "Дата выпуска вместо версии", | ||||
|     "releaseTitleAsVersion": "Используйте название релиза в качестве строки версии", | ||||
|     "releaseDateAsVersionExplanation": "Этот параметр следует использовать только для приложений, в которых определение версии не работает правильно, но имеется дата выпуска", | ||||
|     "changes": "Изменения", | ||||
|     "releaseDate": "Дата выпуска", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Стандартное", | ||||
|     "groupByCategory": "Группировать по категориям", | ||||
|     "autoApkFilterByArch": "Попытаться отфильтровать APK-файлы по архитектуре процессора", | ||||
|     "autoLinkFilterByArch": "Попытайтесь отфильтровать ссылки по архитектуре процессора, если это возможно", | ||||
|     "overrideSource": "Переопределить источник", | ||||
|     "dontShowAgain": "Не показывать снова", | ||||
|     "dontShowTrackOnlyWarnings": "Не показывать предупреждения о только отслеживаемых приложениях", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "Промежуточная ссылка", | ||||
|     "exemptFromBackgroundUpdates": "Исключить из фоновых обновлений (если включено)", | ||||
|     "bgUpdatesOnWiFiOnly": "Отключить фоновые обновления, если нет соединения с Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Отключение фоновых обновлений при отсутствии зарядки", | ||||
|     "autoSelectHighestVersionCode": "Автоматически выбирать APK с актуальной версией кода", | ||||
|     "versionExtractionRegEx": "Регулярное выражение для извлечения версии", | ||||
|     "trimVersionString": "Обрезка строки версии с помощью RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Поделитесь с AppVerifier, а затем вернитесь сюда, когда будете готовы.", | ||||
|     "wiki": "Помощь/Вики", | ||||
|     "crowdsourcedConfigsLabel": "Конфигурации приложений на основе краудсорсинга (используйте на свой страх и риск)", | ||||
|     "crowdsourcedConfigsShort": "Конфиги приложений с помощью краудсорсинга", | ||||
|     "allowInsecure": "Разрешить небезопасные HTTP-запросы", | ||||
|     "stayOneVersionBehind": "Не отставайте от последней версии", | ||||
|     "refreshBeforeDownload": "Обновляйте информацию о приложении перед загрузкой", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Имя", | ||||
|     "smartname": "Имя (умное)", | ||||
|     "sortMethod": "Метод сортировки", | ||||
|     "welcome": "Добро пожаловать", | ||||
|     "documentationLinksNote": "На странице Obtainium GitHub, ссылка на которую приведена ниже, содержатся ссылки на видео, статьи, обсуждения и другие ресурсы, которые помогут вам понять, как пользоваться приложением.", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Удалить приложение?", | ||||
|         "other": "Удалить приложения?" | ||||
|   | ||||
| @@ -12,7 +12,11 @@ const neverAutoTranslate = { | ||||
|     root: ['*'], | ||||
|     obtainiumExportHyphenatedLowercase: ['*'], | ||||
|     theme: ['de'], | ||||
|     appId: ['de'] | ||||
|     appId: ['de'], | ||||
|     placeholder: ['pl'], | ||||
|     importExport: ['fr'], | ||||
|     url: ['fr'], | ||||
|     tencentAppStore: ['*'] | ||||
| } | ||||
|  | ||||
| const translateText = async (text, targetLang, authKey) => { | ||||
| @@ -75,12 +79,18 @@ const main = async () => { | ||||
|         const translationKeys = Object.keys(templateTranslation) | ||||
|         for (let j in translationKeys) { | ||||
|             const k = translationKeys[j] | ||||
|             try { | ||||
|                 if (JSON.stringify(thisTranslation[k]) == JSON.stringify(templateTranslation[k])) { | ||||
|                     const lang = file.split('/').pop().split('.')[0] | ||||
|                     if (!neverAutoTranslate[k] || (neverAutoTranslate[k].indexOf('*') < 0 && neverAutoTranslate[k].indexOf(lang) < 0)) { | ||||
|                         const reportLine = `${file} :::: ${k} :::: ${JSON.stringify(thisTranslation[k])}` | ||||
|                         if (deeplAPIKey) { | ||||
|                             const translateFunc = async (str) => { | ||||
|                                 await new Promise((resolve, reject) => { | ||||
|                                     setTimeout(() => { | ||||
|                                         resolve() | ||||
|                                     }, Math.random() * 10000); // Try to avoid rate limit | ||||
|                                 }) | ||||
|                                 const response = await translateText(str, lang, deeplAPIKey) | ||||
|                                 if (response.translations) { | ||||
|                                     return response.translations[0].text | ||||
| @@ -110,6 +120,9 @@ const main = async () => { | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } catch (err) { | ||||
|                 console.error(err) | ||||
|             } | ||||
|         } | ||||
|         fs.writeFileSync(file, `${JSON.stringify(thisTranslation, null, '    ')}\n`) | ||||
|     } | ||||
|   | ||||
| @@ -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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Avinstallera från Enheten", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Fungerar bara för Appar med versionsdetektering inaktiverat..", | ||||
|     "releaseDateAsVersion": "Använd releasedatum som version", | ||||
|     "releaseTitleAsVersion": "Använd release-titel som versionssträng", | ||||
|     "releaseDateAsVersionExplanation": "Det här alternativet bör endast användas för appar där versionsidentifiering inte fungerar korrekt, men ett releasedatum är tillgängligt.", | ||||
|     "changes": "Ändringar", | ||||
|     "releaseDate": "Releasedatum", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Standardversionsdetektering", | ||||
|     "groupByCategory": "Gruppera via Kategori", | ||||
|     "autoApkFilterByArch": "Försök att filtrera APK-filer efter CPU-arkitektur om möjligt", | ||||
|     "autoLinkFilterByArch": "Försök att filtrera länkar efter CPU-arkitektur om möjligt", | ||||
|     "overrideSource": "Överskrid Källa", | ||||
|     "dontShowAgain": "Visa inte detta igen", | ||||
|     "dontShowTrackOnlyWarnings": "Visa inte 'Följ-Endast' varningar", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "Mellanlänk hittades inte", | ||||
|     "intermediateLink": "Mellanlänk", | ||||
|     "exemptFromBackgroundUpdates": "Undta från bakgrundsuppdateringar (om aktiverad)", | ||||
|     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan WiFi", | ||||
|     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Inaktivera bakgrundsuppdateringar när du inte laddar", | ||||
|     "autoSelectHighestVersionCode": "Välj automatiskt högsta versionskod APK", | ||||
|     "versionExtractionRegEx": "Version Extraction RegEx", | ||||
|     "trimVersionString": "Trimma versionssträng med RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Dela till AppVerifier och återvänd sedan hit när du är klar.", | ||||
|     "wiki": "Hjälp/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourcade appkonfigurationer (använd på egen risk)", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Namn", | ||||
|     "smartname": "Namn (Smart)", | ||||
|     "sortMethod": "Sorteringsmetod", | ||||
|     "welcome": "Välkommen", | ||||
|     "documentationLinksNote": "Obtainium GitHub-sidan som länkas nedan innehåller länkar till videor, artiklar, diskussioner och andra resurser som hjälper dig att förstå hur du använder appen.", | ||||
|     "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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Cihazdan Kaldır", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Yalnızca Sürüm Algılaması Devre Dışı Uygulamalar İçin Çalışır.", | ||||
|     "releaseDateAsVersion": "Sürüm Olarak Yayın Tarihi Kullan", | ||||
|     "releaseTitleAsVersion": "Sürüm dizesi olarak sürüm başlığını kullan", | ||||
|     "releaseDateAsVersionExplanation": "Bu seçenek, sürüm algılamanın doğru çalışmadığı ancak bir sürüm tarihinin mevcut olduğu uygulamalar için kullanılmalıdır.", | ||||
|     "changes": "Değişiklikler", | ||||
|     "releaseDate": "Yayın Tarihi", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Standart sürüm tespiti", | ||||
|     "groupByCategory": "Kategoriye Göre Grupla", | ||||
|     "autoApkFilterByArch": "Mümkünse APK'leri CPU mimarisi ile filtreleme girişimi", | ||||
|     "autoLinkFilterByArch": "Mümkünse bağlantıları CPU mimarisine göre filtrelemeye çalışın", | ||||
|     "overrideSource": "Öncelenecek Kaynak", | ||||
|     "dontShowAgain": "Bunu tekrar gösterme", | ||||
|     "dontShowTrackOnlyWarnings": "'Yalnızca Takip Edilen' uyarılarını gösterme", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "Ara bağlantı", | ||||
|     "exemptFromBackgroundUpdates": "Arka plan güncellemelerinden muaf tut (etkinse)", | ||||
|     "bgUpdatesOnWiFiOnly": "WiFi olmadığında arka plan güncellemelerini devre dışı bırak", | ||||
|     "bgUpdatesWhileChargingOnly": "Şarj olmadığında arka plan güncellemelerini devre dışı bırakma", | ||||
|     "autoSelectHighestVersionCode": "Otomatik olarak en yüksek sürüm kodunu seç", | ||||
|     "versionExtractionRegEx": "Sürüm Çıkarma Düzenli İfade", | ||||
|     "trimVersionString": "RegEx ile Sürüm Dizesini Kırpma", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "AppVerifier ile paylaşın, hazır olduğunuzda buraya dönün.", | ||||
|     "wiki": "Yardım/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Kitle Kaynaklı Uygulama Yapılandırmaları (riski size ait olmak üzere kullanın)", | ||||
|     "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", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "İsim", | ||||
|     "smartname": "İsim (Akıllı)", | ||||
|     "sortMethod": "Sıralama Yöntemi", | ||||
|     "welcome": "Hoş geldiniz", | ||||
|     "documentationLinksNote": "Aşağıda bağlantısı verilen Obtainium GitHub sayfası, uygulamayı nasıl kullanacağınızı anlamanıza yardımcı olacak videolara, makalelere, tartışmalara ve diğer kaynaklara bağlantılar içerir.", | ||||
|     "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": "Оновити", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Видалити з пристрою", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Працює лише з застосунками з вимкненим визначенням версії.", | ||||
|     "releaseDateAsVersion": "Використовувати дату випуску як рядок версії", | ||||
|     "releaseTitleAsVersion": "Використовувати назву випуску як рядок версії", | ||||
|     "releaseDateAsVersionExplanation": "Цю опцію слід використовувати лише для застосунків, де визначення версії працює неправильно, але є дата випуску.", | ||||
|     "changes": "Зміни", | ||||
|     "releaseDate": "Дата випуску", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Стандартне визначення версії", | ||||
|     "groupByCategory": "Групувати за категоріями", | ||||
|     "autoApkFilterByArch": "Спробувати фільтрувати APK за архітектурою ЦП, якщо можливо", | ||||
|     "autoLinkFilterByArch": "Спробуйте відфільтрувати посилання за архітектурою процесора, якщо це можливо", | ||||
|     "overrideSource": "Перевизначити джерело", | ||||
|     "dontShowAgain": "Не показувати це знову", | ||||
|     "dontShowTrackOnlyWarnings": "Не показувати попередження про 'Тільки відстеження'", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "Проміжне посилання", | ||||
|     "exemptFromBackgroundUpdates": "Виключено з фонових оновлень (якщо ввімкнено)", | ||||
|     "bgUpdatesOnWiFiOnly": "Вимкнути фонові оновлення поза Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Вимкнути фонові оновлення, коли не заряджається", | ||||
|     "autoSelectHighestVersionCode": "Автоматичний вибір APK з найвищим кодом версії", | ||||
|     "versionExtractionRegEx": "Регулярний вираз для вилучення рядка версії", | ||||
|     "trimVersionString": "Обрізати рядок версії за допомогою RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Надішліть на AppVerifier, а потім поверніться сюди, коли будете готові.", | ||||
|     "wiki": "Довідка/Вікі", | ||||
|     "crowdsourcedConfigsLabel": "Краудсорсингові конфігурації додатків (використовуйте на свій страх і ризик)", | ||||
|     "crowdsourcedConfigsShort": "Налаштування краудсорсингових додатків", | ||||
|     "allowInsecure": "Дозволити незахищені HTTP-запити", | ||||
|     "stayOneVersionBehind": "Залишайтеся на одну версію актуальнішою", | ||||
|     "refreshBeforeDownload": "Оновіть інформацію про програму перед завантаженням", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Ім'я", | ||||
|     "smartname": "Ім'я (Smart)", | ||||
|     "sortMethod": "Метод сортування", | ||||
|     "welcome": "Ласкаво просимо.", | ||||
|     "documentationLinksNote": "Сторінка Obtainium на GitHub, посилання на яку наведено нижче, містить посилання на відео, статті, дискусії та інші ресурси, які допоможуть вам зрозуміти, як користуватися додатком.", | ||||
|     "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", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "Gỡ cài đặt khỏi thiết bị", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Chỉ hoạt động với Ứng dụng đã tắt tính năng phát hiện phiên bản.", | ||||
|     "releaseDateAsVersion": "Sử dụng ngày phát hành làm phiên bản", | ||||
|     "releaseTitleAsVersion": "Use release title as version string", | ||||
|     "releaseDateAsVersionExplanation": "Chỉ nên sử dụng tùy chọn này cho Ứng dụng trong đó tính năng phát hiện phiên bản không hoạt động chính xác nhưng đã có ngày phát hành.", | ||||
|     "changes": "Thay đổi", | ||||
|     "releaseDate": "Ngày phát hành", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "Phát hiện phiên bản tiêu chuẩn", | ||||
|     "groupByCategory": "Nhóm theo danh mục", | ||||
|     "autoApkFilterByArch": "Cố gắng lọc APK theo kiến trúc CPU nếu có thể", | ||||
|     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||
|     "overrideSource": "Ghi đè nguồn", | ||||
|     "dontShowAgain": "Đừng hiển thị thông tin này nữa", | ||||
|     "dontShowTrackOnlyWarnings": "Không hiển thị cảnh báo 'Chỉ theo dõi'", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "Không tìm thấy liên kết trung gian", | ||||
|     "intermediateLink": "Liên kết trung gian", | ||||
|     "exemptFromBackgroundUpdates": "Miễn cập nhật nền (nếu được bật)", | ||||
|     "bgUpdatesOnWiFiOnly": "Tắt cập nhật nền khi không có WiFi", | ||||
|     "bgUpdatesOnWiFiOnly": "Tắt cập nhật nền khi không có Wi-Fi", | ||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||
|     "autoSelectHighestVersionCode": "Tự động chọn APK mã phiên bản cao nhất", | ||||
|     "versionExtractionRegEx": "Trích xuất phiên bản RegEx", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "Chia sẻ lên AppVerifier, sau đó quay lại đây khi sẵn sàng.", | ||||
|     "wiki": "Trợ giúp/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configurations", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "stayOneVersionBehind": "Stay one version behind latest", | ||||
|     "refreshBeforeDownload": "Refresh app details before download", | ||||
|     "tencentAppStore": "Tencent App Store", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "Name", | ||||
|     "smartname": "Name (Smart)", | ||||
|     "sortMethod": "Sort Method", | ||||
|     "welcome": "Welcome", | ||||
|     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||
|     "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": "更新", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "從裝置解除安裝", | ||||
|     "onlyWorksWithNonVersionDetectApps": "僅適用於停用版本偵測的應用程式。", | ||||
|     "releaseDateAsVersion": "使用發佈日期作為版本字串", | ||||
|     "releaseTitleAsVersion": "使用發佈標題作為版本字串", | ||||
|     "releaseDateAsVersionExplanation": "此選項僅應用於版本偵測無法正確工作但有發佈日期的應用程式。", | ||||
|     "changes": "變更", | ||||
|     "releaseDate": "發佈日期", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "標準版本偵測", | ||||
|     "groupByCategory": "按類別分組", | ||||
|     "autoApkFilterByArch": "如果可能,嘗試按 CPU 架構過濾 APK", | ||||
|     "autoLinkFilterByArch": "若可能,自動根據 CPU 架構篩選連結", | ||||
|     "overrideSource": "覆蓋來源", | ||||
|     "dontShowAgain": "不要再顯示", | ||||
|     "dontShowTrackOnlyWarnings": "不要顯示「僅追蹤」警告", | ||||
| @@ -256,7 +256,8 @@ | ||||
|     "intermediateLinkNotFound": "沒有找到中間連結", | ||||
|     "intermediateLink": "中間連結", | ||||
|     "exemptFromBackgroundUpdates": "免除背景更新(若已啟用)", | ||||
|     "bgUpdatesOnWiFiOnly": "停用非 WiFi 的背景更新", | ||||
|     "bgUpdatesOnWiFiOnly": "停用非 Wi-Fi 的背景更新", | ||||
|     "bgUpdatesWhileChargingOnly": "未充電時停用背景更新", | ||||
|     "autoSelectHighestVersionCode": "自動選擇最高 versionCode 的 APK", | ||||
|     "versionExtractionRegEx": "版本字串提取正則表達式", | ||||
|     "trimVersionString": "用正則表達式修剪版本字串", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "分享至 AppVerifier,然後準備好時回到此處。", | ||||
|     "wiki": "幫助/維基", | ||||
|     "crowdsourcedConfigsLabel": "群眾外包的應用程式設定(使用風險自負)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "crowdsourcedConfigsShort": "群眾外包的應用程式設定", | ||||
|     "allowInsecure": "允許不安全的 HTTP 請求", | ||||
|     "stayOneVersionBehind": "保持比最新版本落後一個版本", | ||||
|     "refreshBeforeDownload": "下載前刷新應用程式詳細資訊", | ||||
|     "tencentAppStore": "騰訊應用寶", | ||||
|     "coolApk": "CoolApk", | ||||
|     "name": "名稱", | ||||
|     "smartname": "名稱(智慧)", | ||||
|     "sortMethod": "排序方式", | ||||
|     "welcome": "歡迎", | ||||
|     "documentationLinksNote": "下方連結的 Obtainium GitHub 頁面包含影片、文章、討論及其他資源,能幫助你瞭解如何使用這款應用程式。", | ||||
|     "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": "禁用版本检测", | ||||
| @@ -213,6 +211,7 @@ | ||||
|     "uninstallFromDevice": "从设备中卸载", | ||||
|     "onlyWorksWithNonVersionDetectApps": "仅适用于禁用版本检测的应用。", | ||||
|     "releaseDateAsVersion": "将发行日期作为版本号", | ||||
|     "releaseTitleAsVersion": "使用版本标题作为版本字符串", | ||||
|     "releaseDateAsVersionExplanation": "此选项应该仅用于无法进行版本检测但能够获取发行日期的应用。", | ||||
|     "changes": "更新日志", | ||||
|     "releaseDate": "发行日期", | ||||
| @@ -222,6 +221,7 @@ | ||||
|     "standardVersionDetection": "常规版本检测", | ||||
|     "groupByCategory": "按类别分组显示", | ||||
|     "autoApkFilterByArch": "如果可能,尝试按设备支持的 CPU 架构筛选 APK 文件", | ||||
|     "autoLinkFilterByArch": "尽可能按 CPU 架构过滤链接", | ||||
|     "overrideSource": "覆盖来源", | ||||
|     "dontShowAgain": "不再显示", | ||||
|     "dontShowTrackOnlyWarnings": "忽略“仅追踪”模式警告", | ||||
| @@ -257,6 +257,7 @@ | ||||
|     "intermediateLink": "中转链接", | ||||
|     "exemptFromBackgroundUpdates": "禁用后台更新(仅此应用生效,即使已启用全局后台更新)", | ||||
|     "bgUpdatesOnWiFiOnly": "未连接 Wi-Fi 时禁用后台更新", | ||||
|     "bgUpdatesWhileChargingOnly": "不充电时禁用后台更新", | ||||
|     "autoSelectHighestVersionCode": "自动选择内部版本号最高的 APK 文件", | ||||
|     "versionExtractionRegEx": "提取版本号的正则表达式", | ||||
|     "trimVersionString": "使用 RegEx 修剪版本字符串", | ||||
| @@ -314,7 +315,17 @@ | ||||
|     "appVerifierInstructionToast": "分享至 AppVerifier,完成后返回此处。", | ||||
|     "wiki": "帮助/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "众包应用程序配置(使用风险自负)", | ||||
|     "crowdsourcedConfigsShort": "众包应用程序配置", | ||||
|     "allowInsecure": "允许不安全的 HTTP 请求", | ||||
|     "stayOneVersionBehind": "比最新版本晚一个版本", | ||||
|     "refreshBeforeDownload": "下载前刷新应用程序详细信息", | ||||
|     "tencentAppStore": "腾讯应用宝", | ||||
|     "coolApk": "酷安", | ||||
|     "name": "名称", | ||||
|     "smartname": "姓名(智能)", | ||||
|     "sortMethod": "排序方法", | ||||
|     "welcome": "欢迎光临", | ||||
|     "documentationLinksNote": "下面链接的 Obtainium GitHub 页面包含视频、文章、讨论和其他资源的链接,可帮助您了解如何使用该应用程序。", | ||||
|     "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> | ||||
| @@ -13,7 +13,6 @@ | ||||
| 			<li>F-Droid</li> | ||||
| 			<li>Third Party F-Droid Repos</li> | ||||
| 			<li>IzzyOnDroid</li> | ||||
| 			<li>SourceForge</li> | ||||
| 			<li>SourceHut</li> | ||||
| 		</ul> | ||||
| 	</li> | ||||
| @@ -22,24 +21,18 @@ | ||||
| 		<ul> | ||||
| 			<li>APKPure</li> | ||||
| 			<li>Aptoide</li> | ||||
| 			<li>Uptodowng</li> | ||||
| 			<li>Uptodown</li> | ||||
| 			<li>APKMirror (Track-Only)</li> | ||||
| 			<li>Huawei AppGallery</li> | ||||
| 			<li>Tencent App Store</li> | ||||
| 			<li>CoolApk</li> | ||||
| 			<li>Jenkins Jobs</li> | ||||
| 		</ul> | ||||
| 	</li> | ||||
| 	<li> | ||||
| 		<p>Open Source - App-Specific:</p> | ||||
| 		<ul> | ||||
| 			<li>Mullvad</li> | ||||
| 			<li>Signal</li> | ||||
| 			<li>VLC</li> | ||||
| 			<li>RuStore</li> | ||||
| 		</ul> | ||||
| 	</li> | ||||
| 	<li> | ||||
| 		<p>Other - App-Specific:</p> | ||||
| 		<ul> | ||||
| 			<li>WhatsApp</li> | ||||
| 			<li>Telegram App</li> | ||||
| 			<li>Neutron Code</li> | ||||
| 		</ul> | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| Get android app updates directly from the source | ||||
| Get Android app updates directly from the source | ||||
|   | ||||
| @@ -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> | ||||
| @@ -13,7 +13,6 @@ | ||||
| 			<li>F-Droid</li> | ||||
| 			<li>Third Party F-Droid Repos</li> | ||||
| 			<li>IzzyOnDroid</li> | ||||
| 			<li>SourceForge</li> | ||||
| 			<li>SourceHut</li> | ||||
| 		</ul> | ||||
| 	</li> | ||||
| @@ -22,24 +21,18 @@ | ||||
| 		<ul> | ||||
| 			<li>APKPure</li> | ||||
| 			<li>Aptoide</li> | ||||
| 			<li>Uptodowng</li> | ||||
| 			<li>Uptodown</li> | ||||
| 			<li>APKMirror (Track-Only)</li> | ||||
| 			<li>Huawei AppGallery</li> | ||||
| 			<li>Tencent App Store</li> | ||||
| 			<li>CoolApk</li> | ||||
| 			<li>Jenkins Jobs</li> | ||||
| 		</ul> | ||||
| 	</li> | ||||
| 	<li> | ||||
| 		<p>Свободное ПО - Для отдельных приложений:</p> | ||||
| 		<ul> | ||||
| 			<li>Mullvad</li> | ||||
| 			<li>Signal</li> | ||||
| 			<li>VLC</li> | ||||
| 			<li>RuStore</li> | ||||
| 		</ul> | ||||
| 	</li> | ||||
| 	<li> | ||||
| 		<p>Другие - Для отдельных приложений:</p> | ||||
| 		<ul> | ||||
| 			<li>WhatsApp</li> | ||||
| 			<li>Telegram App</li> | ||||
| 			<li>Neutron Code</li> | ||||
| 		</ul> | ||||
|   | ||||
| @@ -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( | ||||
| @@ -60,7 +72,8 @@ class APKMirror extends AppSource { | ||||
|                 true | ||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||
|             : null; | ||||
|     Response res = await sourceRequest('$standardUrl/feed', additionalSettings); | ||||
|     Response res = | ||||
|         await sourceRequest('$standardUrl/feed/', additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       var items = parse(res.body).querySelectorAll('item'); | ||||
|       dynamic targetRelease; | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import 'package:device_info_plus/device_info_plus.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:obtainium/app_sources/html.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| @@ -26,6 +27,16 @@ class APKPure extends AppSource { | ||||
|     allowSubDomains = true; | ||||
|     naiveStandardVersionDetection = true; | ||||
|     showReleaseDateAsVersionToggle = true; | ||||
|     additionalSourceAppSpecificSettingFormItems = [ | ||||
|       [ | ||||
|         GeneratedFormSwitch('fallbackToOlderReleases', | ||||
|             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||
|       ], | ||||
|       [ | ||||
|         GeneratedFormSwitch('stayOneVersionBehind', | ||||
|             label: tr('stayOneVersionBehind'), defaultValue: false) | ||||
|       ] | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -54,33 +65,14 @@ class APKPure extends AppSource { | ||||
|     return Uri.parse(standardUrl).pathSegments.last; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|   getDetailsForVersionLink( | ||||
|       String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     String appId = (await tryInferringAppId(standardUrl))!; | ||||
|     String host = Uri.parse(standardUrl).host; | ||||
|  | ||||
|     var res0 = await sourceRequest('$standardUrl/versions', additionalSettings); | ||||
|     var versionLinks = await grabLinksCommon(res0, { | ||||
|       'skipSort': true, | ||||
|       'customLinkFilterRegex': '$standardUrl/download/[^/]+\$' | ||||
|     }); | ||||
|  | ||||
|     var supportedArchs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|  | ||||
|     if (additionalSettings['autoApkFilterByArch'] != true) { | ||||
|       // No need to request multiple versions when we're not going to filter them (always pick the top one) | ||||
|       versionLinks = versionLinks.sublist(0, 1); | ||||
|     } | ||||
|     if (versionLinks.isEmpty) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|  | ||||
|     for (var i = 0; i < versionLinks.length; i++) { | ||||
|       var link = versionLinks[i]; | ||||
|       var res = await sourceRequest(link.key, additionalSettings); | ||||
|       String appId, | ||||
|       String host, | ||||
|       List<String> supportedArchs, | ||||
|       String link, | ||||
|       Map<String, dynamic> additionalSettings) async { | ||||
|     var res = await sourceRequest(link, additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|       var html = parse(res.body); | ||||
|       var apksDiv = | ||||
| @@ -103,9 +95,8 @@ class APKPure extends AppSource { | ||||
|                 // Unclear why there can even be multiple APKs for the same version and arch | ||||
|                 var apkInfo = e.nextElementSibling?.querySelector('div.info'); | ||||
|                 String? versionCode = RegExp('[0-9]+') | ||||
|                       .firstMatch(apkInfo | ||||
|                               ?.querySelector('div.info-top span.code') | ||||
|                               ?.text ?? | ||||
|                     .firstMatch( | ||||
|                         apkInfo?.querySelector('div.info-top .code')?.text ?? | ||||
|                             '') | ||||
|                     ?.group(0) | ||||
|                     ?.trim(); | ||||
| @@ -114,17 +105,12 @@ 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 | ||||
|                     .trim(); | ||||
|                   DateTime? releaseDate = | ||||
|                       parseDateTimeMMMddCommayyyy(dateString); | ||||
|                 DateTime? releaseDate = parseDateTimeMMMddCommayyyy(dateString); | ||||
|                 if (additionalSettings['autoApkFilterByArch'] == true && | ||||
|                     architectures.isNotEmpty && | ||||
|                     architectures | ||||
| @@ -142,15 +128,21 @@ class APKPure extends AppSource { | ||||
|               .toList() ?? | ||||
|           []; | ||||
|       if (apkUrls.isEmpty) { | ||||
|           continue; | ||||
|         throw NoAPKError(); | ||||
|       } | ||||
|         String version = Uri.parse(link.key).pathSegments.last; | ||||
|         String author = html | ||||
|       String version = Uri.parse(link).pathSegments.last; | ||||
|       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,6 +156,52 @@ class APKPure extends AppSource { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     String appId = (await tryInferringAppId(standardUrl))!; | ||||
|     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': '$decodedStandardUrl/download/[^/]+\$' | ||||
|     }); | ||||
|  | ||||
|     var supportedArchs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|  | ||||
|     if (additionalSettings['autoApkFilterByArch'] != true) { | ||||
|       // No need to request multiple versions when we're not going to filter them (always pick the top one) | ||||
|       versionLinks = versionLinks.sublist(0, 1); | ||||
|     } | ||||
|     if (versionLinks.isEmpty) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|  | ||||
|     for (var i = 0; i < versionLinks.length; i++) { | ||||
|       var link = versionLinks[i]; | ||||
|       try { | ||||
|         if (i == 0 && additionalSettings['stayOneVersionBehind'] == true) { | ||||
|           throw NoReleasesError(); | ||||
|         } | ||||
|         return await getDetailsForVersionLink(standardUrl, appId, host, | ||||
|             supportedArchs, link.key, additionalSettings); | ||||
|       } catch (e) { | ||||
|         if (additionalSettings['fallbackToOlderReleases'] != true || | ||||
|             i == versionLinks.length - 1) { | ||||
|           rethrow; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     throw NoAPKError(); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										173
									
								
								lib/app_sources/coolapk.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,173 @@ | ||||
| import 'dart:convert'; | ||||
| import 'package:bcrypt/bcrypt.dart'; | ||||
| import 'package:crypto/crypto.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
| import 'dart:math'; | ||||
|  | ||||
| // kanged from https://github.com/DUpdateSystem/UpgradeAll/blob/b2f92c9/core-websdk/src/main/java/net/xzos/upgradeall/core/websdk/api/client_proxy/hubs/CoolApk.kt | ||||
| class CoolApk extends AppSource { | ||||
|   CoolApk() { | ||||
|     name = tr('coolApk'); | ||||
|     hosts = ['www.coolapk.com', 'api2.coolapk.com']; | ||||
|     allowSubDomains = true; | ||||
|     naiveStandardVersionDetection = true; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         r'^https?://(www\.)?coolapk\.com/apk/[^/]+', | ||||
|         caseSensitive: false); | ||||
|     var match = standardUrlRegEx.firstMatch(url); | ||||
|     if (match == null) { | ||||
|       throw InvalidURLError(name); | ||||
|     } | ||||
|     String standardizedUrl = match.group(0)!; | ||||
|     return standardizedUrl; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String?> tryInferringAppId(String standardUrl, | ||||
|       {Map<String, dynamic> additionalSettings = const {}}) async { | ||||
|     String appId = Uri.parse(standardUrl).pathSegments.last; | ||||
|     return appId; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|       String standardUrl, | ||||
|       Map<String, dynamic> additionalSettings, | ||||
|       ) async { | ||||
|     String appId = (await tryInferringAppId(standardUrl))!; | ||||
|     String apiUrl = 'https://api2.coolapk.com'; | ||||
|  | ||||
|     // get latest | ||||
|     var detailUrl = '$apiUrl/v6/apk/detail?id=$appId'; | ||||
|     var headers = await getRequestHeaders(additionalSettings); | ||||
|     var res = await sourceRequest(detailUrl, additionalSettings); | ||||
|  | ||||
|     if (res.statusCode != 200) { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|  | ||||
|     var json = jsonDecode(res.body); | ||||
|     if (json['status'] == -2 || json['data'] == null) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|  | ||||
|     var detail = json['data']; | ||||
|     String version = detail['apkversionname'].toString(); | ||||
|     String appName = detail['title'].toString(); | ||||
|     String author = detail['developername']?.toString() ?? 'CoolApk'; | ||||
|     String changelog = detail['changelog']?.toString() ?? ''; | ||||
|     int? releaseDate = detail['lastupdate'] != null | ||||
|         ? (detail['lastupdate'] is int | ||||
|         ? detail['lastupdate'] * 1000 | ||||
|         : int.parse(detail['lastupdate'].toString()) * 1000) | ||||
|         : null; | ||||
|     String aid = detail['id'].toString(); | ||||
|  | ||||
|     // get apk url | ||||
|     String apkUrl = await _getLatestApkUrl(apiUrl, appId, aid, version, headers); | ||||
|     if (apkUrl.isEmpty) { | ||||
|       throw NoAPKError(); | ||||
|     } | ||||
|  | ||||
|     String apkName = '${appId}_$version.apk'; | ||||
|  | ||||
|     return APKDetails( | ||||
|       version, | ||||
|       [MapEntry(apkName, apkUrl)], | ||||
|       AppNames(author, appName), | ||||
|       releaseDate: releaseDate != null | ||||
|           ? DateTime.fromMillisecondsSinceEpoch(releaseDate) | ||||
|           : null, | ||||
|       changeLog: changelog, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   Future<String> _getLatestApkUrl(String apiUrl, String appId, String aid, | ||||
|       String version, Map<String, String>? headers) async { | ||||
|     String url = '$apiUrl/v6/apk/download?pn=$appId&aid=$aid'; | ||||
|     var res = await sourceRequest(url, {}, followRedirects: false); | ||||
|     if (res.statusCode >= 300 && res.statusCode < 400) { | ||||
|       String location = res.headers['location'] ?? ''; | ||||
|       return location; | ||||
|     } | ||||
|     return ''; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, String>?> getRequestHeaders( | ||||
|       Map<String, dynamic> additionalSettings, | ||||
|       {bool forAPKDownload = false}) async { | ||||
|     var tokenPair = _getToken(); | ||||
|     // CoolAPK header | ||||
|     return { | ||||
|       'User-Agent': | ||||
|       'Dalvik/2.1.0 (Linux; U; Android 9; MI 8 SE MIUI/9.5.9) (#Build; Xiaomi; MI 8 SE; PKQ1.181121.001; 9) +CoolMarket/12.4.2-2208241-universal', | ||||
|       'X-App-Id': 'com.coolapk.market', | ||||
|       'X-Requested-With': 'XMLHttpRequest', | ||||
|       'X-Sdk-Int': '30', | ||||
|       'X-App-Mode': 'universal', | ||||
|       'X-App-Channel': 'coolapk', | ||||
|       'X-Sdk-Locale': 'zh-CN', | ||||
|       'X-App-Version': '12.4.2', | ||||
|       'X-Api-Supported': '2208241', | ||||
|       'X-App-Code': '2208241', | ||||
|       'X-Api-Version': '12', | ||||
|       'X-App-Device': tokenPair['deviceCode']!, | ||||
|       'X-Dark-Mode': '0', | ||||
|       'X-App-Token': tokenPair['token']!, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   Map<String, String> _getToken() { | ||||
|     final rand = Random(); | ||||
|  | ||||
|     String randHexString(int n) => | ||||
|         List.generate(n, (_) => rand.nextInt(256).toRadixString(16).padLeft(2, '0')) | ||||
|             .join() | ||||
|             .toUpperCase(); | ||||
|  | ||||
|     String randMacAddress() => | ||||
|         List.generate(6, (_) => rand.nextInt(256).toRadixString(16).padLeft(2, '0')) | ||||
|             .join(':'); | ||||
|  | ||||
|     // 加密算法来自 https://github.com/XiaoMengXinX/FuckCoolapkTokenV2、https://github.com/Coolapk-UWP/Coolapk-UWP | ||||
|     // device | ||||
|     String aid = randHexString(16); | ||||
|     String mac = randMacAddress(); | ||||
|     const manufactor = 'Google'; | ||||
|     const brand = 'Google'; | ||||
|     const model = 'Pixel 5a'; | ||||
|     const buildNumber = 'SQ1D.220105.007'; | ||||
|  | ||||
|     // generate deviceCode | ||||
|     String deviceCode = | ||||
|     base64.encode('$aid; ; ; $mac; $manufactor; $brand; $model; $buildNumber'.codeUnits); | ||||
|  | ||||
|     // generate timestamp | ||||
|     String timeStamp = (DateTime.now().millisecondsSinceEpoch ~/ 1000).toString(); | ||||
|     String base64TimeStamp = base64.encode(timeStamp.codeUnits); | ||||
|     String md5TimeStamp = md5.convert(timeStamp.codeUnits).toString(); | ||||
|     String md5DeviceCode = md5.convert(deviceCode.codeUnits).toString(); | ||||
|  | ||||
|     // generate token | ||||
|     String token = | ||||
|         'token://com.coolapk.market/dcf01e569c1e3db93a3d0fcf191a622c?$md5TimeStamp\$$md5DeviceCode&com.coolapk.market'; | ||||
|     String base64Token = base64.encode(token.codeUnits); | ||||
|     String md5Base64Token = md5.convert(base64Token.codeUnits).toString(); | ||||
|     String md5Token = md5.convert(token.codeUnits).toString(); | ||||
|  | ||||
|     // generate salt and hash | ||||
|     String bcryptSalt = '\$2a\$10\$${base64TimeStamp.substring(0, 14)}/${md5Token.substring(0, 6)}u'; | ||||
|     String bcryptResult = BCrypt.hashpw(md5Base64Token, bcryptSalt); | ||||
|     String reBcryptResult = bcryptResult.replaceRange(0, 3, '\$2y'); | ||||
|     String finalToken = 'v2${base64.encode(reBcryptResult.codeUnits)}'; | ||||
|  | ||||
|     return {'deviceCode': deviceCode, 'token': finalToken}; | ||||
|   } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:obtainium/app_sources/html.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| @@ -8,12 +9,23 @@ class DirectAPKLink extends AppSource { | ||||
|  | ||||
|   DirectAPKLink() { | ||||
|     name = tr('directAPKLink'); | ||||
|     additionalSourceAppSpecificSettingFormItems = html | ||||
|         .additionalSourceAppSpecificSettingFormItems | ||||
|     additionalSourceAppSpecificSettingFormItems = [ | ||||
|       ...html.additionalSourceAppSpecificSettingFormItems | ||||
|           .where((element) => element | ||||
|               .where((element) => element.key == 'requestHeader') | ||||
|               .isNotEmpty) | ||||
|         .toList(); | ||||
|           .toList(), | ||||
|       [ | ||||
|         GeneratedFormDropdown( | ||||
|             'defaultPseudoVersioningMethod', | ||||
|             [ | ||||
|               MapEntry('partialAPKHash', tr('partialAPKHash')), | ||||
|               MapEntry('ETag', 'ETag') | ||||
|             ], | ||||
|             label: tr('defaultPseudoVersioningMethod'), | ||||
|             defaultValue: 'partialAPKHash') | ||||
|       ] | ||||
|     ]; | ||||
|     excludeCommonSettingKeys = [ | ||||
|       'versionExtractionRegEx', | ||||
|       'matchGroupToUse', | ||||
| @@ -57,9 +69,8 @@ class DirectAPKLink extends AppSource { | ||||
|         additionalSettingsNew[s] = additionalSettings[s]; | ||||
|       } | ||||
|     } | ||||
|     additionalSettingsNew['defaultPseudoVersioningMethod'] = 'partialAPKHash'; | ||||
|     additionalSettingsNew['directAPKLink'] = true; | ||||
|     additionalSettings['versionDetection'] = false; | ||||
|     additionalSettingsNew['versionDetection'] = false; | ||||
|     return html.getLatestAPKDetails(standardUrl, additionalSettingsNew); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -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/+[^/]+', | ||||
|   | ||||
| @@ -75,12 +75,26 @@ class GitHub extends AppSource { | ||||
|       ], | ||||
|       [GeneratedFormSwitch('verifyLatestTag', label: tr('verifyLatestTag'))], | ||||
|       [ | ||||
|         GeneratedFormSwitch('dontSortReleasesList', | ||||
|             label: tr('dontSortReleasesList')) | ||||
|         GeneratedFormDropdown( | ||||
|             'sortMethodChoice', | ||||
|             [ | ||||
|               MapEntry('date', tr('releaseDate')), | ||||
|               MapEntry('smartname', tr('smartname')), | ||||
|               MapEntry('none', tr('none')), | ||||
|               MapEntry('smartname-datefallback', | ||||
|                   '${tr('smartname')} x ${tr('releaseDate')}'), | ||||
|               MapEntry('name', tr('name')), | ||||
|             ], | ||||
|             label: tr('sortMethod'), | ||||
|             defaultValue: 'date') | ||||
|       ], | ||||
|       [ | ||||
|         GeneratedFormSwitch('useLatestAssetDateAsReleaseDate', | ||||
|             label: tr('useLatestAssetDateAsReleaseDate'), defaultValue: false) | ||||
|       ], | ||||
|       [ | ||||
|         GeneratedFormSwitch('releaseTitleAsVersion', | ||||
|             label: tr('releaseTitleAsVersion'), defaultValue: false) | ||||
|       ] | ||||
|     ]; | ||||
|  | ||||
| @@ -240,10 +254,10 @@ class GitHub extends AppSource { | ||||
|             ? additionalSettings['filterReleaseNotesByRegEx'] | ||||
|             : null; | ||||
|     bool verifyLatestTag = additionalSettings['verifyLatestTag'] == true; | ||||
|     bool dontSortReleasesList = | ||||
|         additionalSettings['dontSortReleasesList'] == true; | ||||
|     bool useLatestAssetDateAsReleaseDate = | ||||
|         additionalSettings['useLatestAssetDateAsReleaseDate'] == true; | ||||
|     String sortMethod = | ||||
|         additionalSettings['sortMethodChoice'] ?? 'smartname-datefallback'; | ||||
|     dynamic latestRelease; | ||||
|     if (verifyLatestTag) { | ||||
|       var temp = requestUrl.split('?'); | ||||
| @@ -271,14 +285,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() ?? | ||||
|           []; | ||||
|  | ||||
| @@ -289,7 +304,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']) | ||||
| @@ -309,7 +326,7 @@ class GitHub extends AppSource { | ||||
|               ? getPublishDateFromRelease(rel) | ||||
|               : getNewestAssetDateFromRelease(rel); | ||||
|  | ||||
|       if (dontSortReleasesList) { | ||||
|       if (sortMethod == 'none') { | ||||
|         releases = releases.reversed.toList(); | ||||
|       } else { | ||||
|         releases.sort((a, b) { | ||||
| @@ -323,22 +340,30 @@ class GitHub extends AppSource { | ||||
|           } else { | ||||
|             var nameA = a['tag_name'] ?? a['name']; | ||||
|             var nameB = b['tag_name'] ?? b['name']; | ||||
|             var stdFormats = findStandardFormatsForVersion(nameA, true) | ||||
|                 .intersection(findStandardFormatsForVersion(nameB, true)); | ||||
|             if (stdFormats.isNotEmpty) { | ||||
|               var reg = RegExp(stdFormats.first); | ||||
|               var matchA = reg.firstMatch(nameA); | ||||
|               var matchB = reg.firstMatch(nameB); | ||||
|               return compareAlphaNumeric( | ||||
|                   (nameA as String).substring(matchA!.start, matchA.end), | ||||
|                   (nameB as String).substring(matchB!.start, matchB.end)); | ||||
|             } else { | ||||
|             var stdFormats = findStandardFormatsForVersion(nameA, false) | ||||
|                 .intersection(findStandardFormatsForVersion(nameB, false)); | ||||
|             if (sortMethod == 'date' || | ||||
|                 (sortMethod == 'smartname-datefallback' && | ||||
|                     stdFormats.isEmpty)) { | ||||
|               return (getReleaseDateFromRelease( | ||||
|                           a, useLatestAssetDateAsReleaseDate) ?? | ||||
|                       DateTime(1)) | ||||
|                   .compareTo(getReleaseDateFromRelease( | ||||
|                           b, useLatestAssetDateAsReleaseDate) ?? | ||||
|                       DateTime(0)); | ||||
|             } else { | ||||
|               if (sortMethod != 'name' && stdFormats.isNotEmpty) { | ||||
|                 var reg = RegExp(stdFormats.last); | ||||
|                 var matchA = reg.firstMatch(nameA); | ||||
|                 var matchB = reg.firstMatch(nameB); | ||||
|                 return compareAlphaNumeric( | ||||
|                     (nameA as String).substring(matchA!.start, matchA.end), | ||||
|                     (nameB as String).substring(matchB!.start, matchB.end)); | ||||
|               } else { | ||||
|                 // 'name' | ||||
|                 return compareAlphaNumeric( | ||||
|                     (nameA as String), (nameB as String)); | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         }); | ||||
| @@ -383,20 +408,41 @@ 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'] = | ||||
|             targetRelease['tag_name'] ?? targetRelease['name']; | ||||
|             additionalSettings['releaseTitleAsVersion'] == true | ||||
|                 ? nameToFilter | ||||
|                 : targetRelease['tag_name'] ?? targetRelease['name']; | ||||
|         if (targetRelease['tarball_url'] != null) { | ||||
|           allAssetUrls.add(MapEntry( | ||||
|               (targetRelease['version'] ?? 'source') + '.tar.gz', | ||||
| @@ -414,6 +460,7 @@ class GitHub extends AppSource { | ||||
|         throw NoReleasesError(); | ||||
|       } | ||||
|       String? version = targetRelease['version']; | ||||
|  | ||||
|       DateTime? releaseDate = getReleaseDateFromRelease( | ||||
|           targetRelease, useLatestAssetDateAsReleaseDate); | ||||
|       if (version == null) { | ||||
| @@ -472,7 +519,7 @@ class GitHub extends AppSource { | ||||
|   AppNames getAppNames(String standardUrl) { | ||||
|     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); | ||||
|     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); | ||||
|     return AppNames(names[0], names[1]); | ||||
|     return AppNames(names[0], names.sublist(1).join('/')); | ||||
|   } | ||||
|  | ||||
|   Future<Map<String, List<String>>> searchCommon( | ||||
|   | ||||
| @@ -53,8 +53,12 @@ class GitLab extends AppSource { | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     var urlSegments = url.split('/'); | ||||
|     var cutOffIndex = urlSegments.indexWhere((s) => s == '-'); | ||||
|     url = | ||||
|         urlSegments.sublist(0, cutOffIndex <= 0 ? null : cutOffIndex).join('/'); | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+(/[^((\b/\b)|(\b/-/\b))]+){1,20}', | ||||
|         caseSensitive: false); | ||||
|     RegExpMatch? match = standardUrlRegEx.firstMatch(url); | ||||
|     if (match == null) { | ||||
| @@ -116,7 +120,7 @@ class GitLab extends AppSource { | ||||
|       Map<String, dynamic> additionalSettings) async { | ||||
|     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); | ||||
|     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; | ||||
|     return '$apkUrl?$optionalAuth'; | ||||
|     return '$apkUrl${(Uri.parse(apkUrl).query.isEmpty ? '?' : '&')}$optionalAuth'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -126,6 +130,8 @@ class GitLab extends AppSource { | ||||
|   ) async { | ||||
|     // Prepare request params | ||||
|     var names = GitHub().getAppNames(standardUrl); | ||||
|     String projectUriComponent = | ||||
|         '${Uri.encodeComponent(names.author)}%2F${Uri.encodeComponent(names.name)}'; | ||||
|     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); | ||||
|     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; | ||||
|  | ||||
| @@ -133,7 +139,7 @@ class GitLab extends AppSource { | ||||
|  | ||||
|     // Get project ID | ||||
|     Response res0 = await sourceRequest( | ||||
|         'https://${hosts[0]}/api/v4/projects/${names.author}%2F${names.name}?$optionalAuth', | ||||
|         'https://${hosts[0]}/api/v4/projects/$projectUriComponent?$optionalAuth', | ||||
|         additionalSettings); | ||||
|     if (res0.statusCode != 200) { | ||||
|       throw getObtainiumHttpError(res0); | ||||
| @@ -145,7 +151,7 @@ class GitLab extends AppSource { | ||||
|  | ||||
|     // Request data from REST API | ||||
|     Response res = await sourceRequest( | ||||
|         'https://${hosts[0]}/api/v4/projects/${names.author}%2F${names.name}/${trackOnly ? 'repository/tags' : 'releases'}?$optionalAuth', | ||||
|         'https://${hosts[0]}/api/v4/projects/$projectUriComponent/${trackOnly ? 'repository/tags' : 'releases'}?$optionalAuth', | ||||
|         additionalSettings); | ||||
|     if (res.statusCode != 200) { | ||||
|       throw getObtainiumHttpError(res); | ||||
| @@ -157,12 +163,18 @@ class GitLab extends AppSource { | ||||
|     apkDetailsList = json.map((e) { | ||||
|       var apkUrlsFromAssets = (e['assets']?['links'] as List<dynamic>? ?? []) | ||||
|           .map((e) { | ||||
|             return (e['direct_asset_url'] ?? e['url'] ?? '') as String; | ||||
|             var url = (e['direct_asset_url'] ?? e['url'] ?? '') as String; | ||||
|             var parsedUrl = url.isNotEmpty ? Uri.parse(url) : null; | ||||
|             return MapEntry( | ||||
|                 (e['name'] ?? | ||||
|                     (parsedUrl != null && parsedUrl.pathSegments.isNotEmpty | ||||
|                         ? parsedUrl.pathSegments.last | ||||
|                         : 'unknown')) as String, | ||||
|                 (e['direct_asset_url'] ?? e['url'] ?? '') as String); | ||||
|           }) | ||||
|           .where((s) => s.isNotEmpty) | ||||
|           .where((s) => s.key.isNotEmpty) | ||||
|           .toList(); | ||||
|       List<String> uploadedAPKsFromDescription = | ||||
|           ((e['description'] ?? '') as String) | ||||
|       var uploadedAPKsFromDescription = ((e['description'] ?? '') as String) | ||||
|           .split('](') | ||||
|           .join('\n') | ||||
|           .split('.apk)') | ||||
| @@ -170,17 +182,21 @@ class GitLab extends AppSource { | ||||
|           .split('\n') | ||||
|           .where((s) => s.startsWith('/uploads/') && s.endsWith('apk')) | ||||
|           .map((s) => 'https://${hosts[0]}/-/project/$projectId$s') | ||||
|           .map((l) => MapEntry(Uri.parse(l).pathSegments.last, l)) | ||||
|           .toList(); | ||||
|       var apkUrlsSet = apkUrlsFromAssets.toSet(); | ||||
|       apkUrlsSet.addAll(uploadedAPKsFromDescription); | ||||
|       Map<String, String> apkUrls = {}; | ||||
|       for (var entry in apkUrlsFromAssets) { | ||||
|         apkUrls[entry.key] = entry.value; | ||||
|       } | ||||
|       for (var entry in uploadedAPKsFromDescription) { | ||||
|         apkUrls[entry.key] = entry.value; | ||||
|       } | ||||
|       var releaseDateString = | ||||
|           e['released_at'] ?? e['created_at'] ?? e['commit']?['created_at']; | ||||
|       DateTime? releaseDate = | ||||
|           releaseDateString != null ? DateTime.parse(releaseDateString) : null; | ||||
|       return APKDetails( | ||||
|           e['tag_name'] ?? e['name'], | ||||
|           getApkUrlsFromUrls(apkUrlsSet.toList()), | ||||
|           GitHub().getAppNames(standardUrl), | ||||
|       return APKDetails(e['tag_name'] ?? e['name'], apkUrls.entries.toList(), | ||||
|           AppNames(names.author, names.name.split('/').last), | ||||
|           releaseDate: releaseDate); | ||||
|     }); | ||||
|     if (apkDetailsList.isEmpty) { | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| import 'dart:convert'; | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| @@ -7,6 +9,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 +23,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'; | ||||
| @@ -64,6 +69,27 @@ int compareAlphaNumeric(String a, String b) { | ||||
|   return aParts.length.compareTo(bParts.length); | ||||
| } | ||||
|  | ||||
| List<String> collectAllStringsFromJSONObject(dynamic obj) { | ||||
|   List<String> extractor(dynamic obj) { | ||||
|     final results = <String>[]; | ||||
|     if (obj is String) { | ||||
|       results.add(obj); | ||||
|     } else if (obj is List) { | ||||
|       for (final item in obj) { | ||||
|         results.addAll(extractor(item)); | ||||
|       } | ||||
|     } else if (obj is Map<String, dynamic>) { | ||||
|       for (final value in obj.values) { | ||||
|         results.addAll(extractor(value)); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return results; | ||||
|   } | ||||
|  | ||||
|   return extractor(obj); | ||||
| } | ||||
|  | ||||
| List<String> _splitAlphaNumeric(String s) { | ||||
|   List<String> parts = []; | ||||
|   StringBuffer sb = StringBuffer(); | ||||
| @@ -92,6 +118,13 @@ bool _isNumeric(String s) { | ||||
|   return s.codeUnitAt(0) >= 48 && s.codeUnitAt(0) <= 57; | ||||
| } | ||||
|  | ||||
| List<MapEntry<String, String>> getLinksInLines(String lines) => RegExp( | ||||
|         r'(http|ftp|https)://([\w_-]+(?:(?:\.[\w_-]+)+))([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?') | ||||
|     .allMatches(lines) | ||||
|     .map((match) => | ||||
|         MapEntry(match.group(0)!, match.group(0)?.split('/').last ?? '')) | ||||
|     .toList(); | ||||
|  | ||||
| // Given an HTTP response, grab some links according to the common additional settings | ||||
| // (those that apply to intermediate and final steps) | ||||
| Future<List<MapEntry<String, String>>> grabLinksCommon( | ||||
| @@ -111,12 +144,21 @@ Future<List<MapEntry<String, String>>> grabLinksCommon( | ||||
|       .map((e) => MapEntry(ensureAbsoluteUrl(e.key, res.request!.url), e.value)) | ||||
|       .toList(); | ||||
|   if (allLinks.isEmpty) { | ||||
|     allLinks = RegExp( | ||||
|             r'(http|ftp|https)://([\w_-]+(?:(?:\.[\w_-]+)+))([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?') | ||||
|         .allMatches(res.body) | ||||
|         .map((match) => | ||||
|             MapEntry(match.group(0)!, match.group(0)?.split('/').last ?? '')) | ||||
|         .toList(); | ||||
|     allLinks = getLinksInLines(res.body); | ||||
|   } | ||||
|   if (allLinks.isEmpty) { | ||||
|     // Getting desperate | ||||
|     try { | ||||
|       var jsonStrings = collectAllStringsFromJSONObject(jsonDecode(res.body)); | ||||
|       allLinks = getLinksInLines(jsonStrings.join('\n')); | ||||
|       if (allLinks.isEmpty) { | ||||
|         allLinks = getLinksInLines(jsonStrings.map((l) { | ||||
|           return ensureAbsoluteUrl(l, res.request!.url); | ||||
|         }).join('\n')); | ||||
|       } | ||||
|     } catch (e) { | ||||
|       // | ||||
|     } | ||||
|   } | ||||
|   List<MapEntry<String, String>> links = []; | ||||
|   bool skipSort = additionalSettings['skipSort'] == true; | ||||
| @@ -209,6 +251,10 @@ class HTML extends AppSource { | ||||
|           required: true, | ||||
|           additionalValidators: [(value) => regExValidator(value)]) | ||||
|     ], | ||||
|     [ | ||||
|       GeneratedFormSwitch('autoLinkFilterByArch', | ||||
|           label: tr('autoLinkFilterByArch'), defaultValue: false) | ||||
|     ], | ||||
|   ]; | ||||
|   HTML() { | ||||
|     additionalSourceAppSpecificSettingFormItems = [ | ||||
| @@ -256,7 +302,8 @@ class HTML extends AppSource { | ||||
|             'defaultPseudoVersioningMethod', | ||||
|             [ | ||||
|               MapEntry('partialAPKHash', tr('partialAPKHash')), | ||||
|               MapEntry('APKLinkHash', tr('APKLinkHash')) | ||||
|               MapEntry('APKLinkHash', tr('APKLinkHash')), | ||||
|               MapEntry('ETag', 'ETag') | ||||
|             ], | ||||
|             label: tr('defaultPseudoVersioningMethod'), | ||||
|             defaultValue: 'partialAPKHash') | ||||
| @@ -310,8 +357,12 @@ class HTML extends AppSource { | ||||
|           await sourceRequest(currentUrl, additionalSettings), | ||||
|           additionalSettings['intermediateLink'][i]); | ||||
|       if (intLinks.isEmpty) { | ||||
|         throw NoReleasesError(); | ||||
|         throw NoReleasesError(note: currentUrl); | ||||
|       } else { | ||||
|         if (additionalSettings['intermediateLink'][i]['autoLinkFilterByArch'] == | ||||
|             true) { | ||||
|           intLinks = await filterApksByArch(intLinks); | ||||
|         } | ||||
|         currentUrl = intLinks.last.key; | ||||
|       } | ||||
|     } | ||||
| @@ -326,7 +377,7 @@ class HTML extends AppSource { | ||||
|       links = filterApks(links, additionalSettings['apkFilterRegEx'], | ||||
|           additionalSettings['invertAPKFilter']); | ||||
|       if (links.isEmpty) { | ||||
|         throw NoReleasesError(); | ||||
|         throw NoReleasesError(note: currentUrl); | ||||
|       } | ||||
|     } else { | ||||
|       links = [MapEntry(currentUrl, currentUrl)]; | ||||
| @@ -345,15 +396,32 @@ class HTML extends AppSource { | ||||
|         additionalSettings['versionExtractWholePage'] == true | ||||
|             ? versionExtractionWholePageString | ||||
|             : relDecoded); | ||||
|     version ??= additionalSettings['defaultPseudoVersioningMethod'] == | ||||
|             'APKLinkHash' | ||||
|     var apkReqHeaders = | ||||
|         await getRequestHeaders(additionalSettings, forAPKDownload: true); | ||||
|     if (version == null && | ||||
|         additionalSettings['defaultPseudoVersioningMethod'] == 'ETag') { | ||||
|       version = await checkETagHeader(rel, | ||||
|           headers: apkReqHeaders, | ||||
|           allowInsecure: additionalSettings['allowInsecure'] == true); | ||||
|       if (version == null) { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|     } | ||||
|     version ??= | ||||
|         additionalSettings['defaultPseudoVersioningMethod'] == 'APKLinkHash' | ||||
|             ? rel.hashCode.toString() | ||||
|             : (await checkPartialDownloadHashDynamic(rel, | ||||
|                 headers: await getRequestHeaders(additionalSettings, | ||||
|                     forAPKDownload: true), | ||||
|                     headers: apkReqHeaders, | ||||
|                     allowInsecure: additionalSettings['allowInsecure'] == true)) | ||||
|                 .toString(); | ||||
|     return APKDetails(version, [rel].map((e) => MapEntry(e, e)).toList(), | ||||
|     return APKDetails( | ||||
|         version, | ||||
|         [rel].map((e) { | ||||
|           var uri = Uri.parse(e); | ||||
|           var fileName = | ||||
|               uri.pathSegments.isNotEmpty ? uri.pathSegments.last : uri.origin; | ||||
|           return MapEntry('${e.hashCode}-$fileName', e); | ||||
|         }).toList(), | ||||
|         AppNames(uri.host, tr('app'))); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										98
									
								
								lib/app_sources/rustore.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,98 @@ | ||||
| import 'dart:convert'; | ||||
| import 'dart:typed_data'; | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter_charset_detector/flutter_charset_detector.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class RuStore extends AppSource { | ||||
|   RuStore() { | ||||
|     hosts = ['rustore.ru']; | ||||
|     name = 'RuStore'; | ||||
|     naiveStandardVersionDetection = true; | ||||
|     showReleaseDateAsVersionToggle = true; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/catalog/app/+[^/]+', | ||||
|         caseSensitive: false); | ||||
|     RegExpMatch? match = standardUrlRegEx.firstMatch(url); | ||||
|     if (match == null) { | ||||
|       throw InvalidURLError(name); | ||||
|     } | ||||
|     return match.group(0)!; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String?> tryInferringAppId(String standardUrl, | ||||
|       {Map<String, dynamic> additionalSettings = const {}}) async { | ||||
|     return Uri.parse(standardUrl).pathSegments.last; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     String? appId = await tryInferringAppId(standardUrl); | ||||
|     Response res0 = await sourceRequest( | ||||
|         'https://backapi.rustore.ru/applicationData/overallInfo/$appId', | ||||
|         additionalSettings); | ||||
|     if (res0.statusCode != 200) { | ||||
|       throw getObtainiumHttpError(res0); | ||||
|     } | ||||
|     var appDetails = jsonDecode(res0.body)['body']; | ||||
|     if (appDetails['appId'] == null) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|  | ||||
|     String appName = appDetails['appName'] ?? tr('app'); | ||||
|     String author = appDetails['companyName'] ?? name; | ||||
|     String? dateStr = appDetails['updatedAt']; | ||||
|     String? version = appDetails['versionName']; | ||||
|     String? changeLog = appDetails['whatsNew']; | ||||
|     if (version == null) { | ||||
|       throw NoVersionError(); | ||||
|     } | ||||
|     DateTime? relDate; | ||||
|     if (dateStr != null) { | ||||
|       relDate = DateTime.parse(dateStr); | ||||
|     } | ||||
|  | ||||
|     Response res1 = await sourceRequest( | ||||
|         'https://backapi.rustore.ru/applicationData/download-link', | ||||
|         additionalSettings, | ||||
|         followRedirects: false, | ||||
|         postBody: {"appId": appDetails['appId'], "firstInstall": true}); | ||||
|     var downloadDetails = jsonDecode(res1.body)['body']; | ||||
|     if (res1.statusCode != 200 || downloadDetails['apkUrl'] == null) { | ||||
|       throw NoAPKError(); | ||||
|     } | ||||
|  | ||||
|     appName = (await CharsetDetector.autoDecode( | ||||
|             Uint8List.fromList(appName.codeUnits))) | ||||
|         .string; | ||||
|     author = | ||||
|         (await CharsetDetector.autoDecode(Uint8List.fromList(author.codeUnits))) | ||||
|             .string; | ||||
|     changeLog = changeLog != null | ||||
|         ? (await CharsetDetector.autoDecode( | ||||
|                 Uint8List.fromList(changeLog.codeUnits))) | ||||
|             .string | ||||
|         : null; | ||||
|  | ||||
|     return APKDetails( | ||||
|         version, | ||||
|         getApkUrlsFromUrls([ | ||||
|           (downloadDetails['apkUrl'] as String) | ||||
|               .replaceAll(RegExp('\\.zip\$'), '.apk') | ||||
|         ]), | ||||
|         AppNames(author, appName), | ||||
|         releaseDate: relDate, | ||||
|         changeLog: changeLog); | ||||
|   } | ||||
| } | ||||
| @@ -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); | ||||
|   | ||||
| @@ -1,11 +1,12 @@ | ||||
| import 'dart:convert'; | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class Tencent extends AppSource { | ||||
|   Tencent() { | ||||
|     name = 'Tencent App Store'; | ||||
|     name = tr('tencentAppStore'); | ||||
|     hosts = ['sj.qq.com']; | ||||
|     naiveStandardVersionDetection = true; | ||||
|     showReleaseDateAsVersionToggle = true; | ||||
| @@ -63,11 +64,11 @@ class Tencent extends AppSource { | ||||
|       var author = json['app_detail_records'][appId]['app_info']['author']; | ||||
|       var releaseDate = | ||||
|           json['app_detail_records'][appId]['app_info']['update_time']; | ||||
|       var apkName = Uri.parse(apkUrl).queryParameters['fsname'] ?? | ||||
|           '${appId}_$version.apk'; | ||||
|  | ||||
|       return APKDetails( | ||||
|           version, | ||||
|           [MapEntry(Uri.parse(apkUrl).queryParameters['fsname']!, apkUrl)], | ||||
|           AppNames(author, appName), | ||||
|           version, [MapEntry(apkName, apkUrl)], AppNames(author, appName), | ||||
|           releaseDate: releaseDate != null | ||||
|               ? DateTime.fromMillisecondsSinceEpoch(releaseDate * 1000) | ||||
|               : null); | ||||
|   | ||||
| @@ -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')); | ||||
|   } | ||||
| } | ||||
| @@ -154,7 +154,7 @@ String list2FriendlyString(List<String> list) { | ||||
|               (e.key == list.length - 1 | ||||
|                   ? '' | ||||
|                   : e.key == list.length - 2 | ||||
|                       ? ', and ' | ||||
|                       ? ' and ' | ||||
|                       : ', ')) | ||||
|           .join(''); | ||||
| } | ||||
|   | ||||
| @@ -23,7 +23,7 @@ import 'package:easy_localization/src/localization.dart'; | ||||
| List<MapEntry<Locale, String>> supportedLocales = const [ | ||||
|   MapEntry(Locale('en'), 'English'), | ||||
|   MapEntry(Locale('zh'), '简体中文'), | ||||
|   MapEntry(Locale('zh_Hant_TW'), '臺灣話'), | ||||
|   MapEntry(Locale('zh', 'Hant_TW'), '臺灣話'), | ||||
|   MapEntry(Locale('it'), 'Italiano'), | ||||
|   MapEntry(Locale('ja'), '日本語'), | ||||
|   MapEntry(Locale('hu'), 'Magyar'), | ||||
| @@ -34,7 +34,8 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | ||||
|   MapEntry(Locale('pl'), 'Polski'), | ||||
|   MapEntry(Locale('ru'), 'Русский'), | ||||
|   MapEntry(Locale('bs'), 'Bosanski'), | ||||
|   MapEntry(Locale('pt'), 'Brasileiro'), | ||||
|   MapEntry(Locale('pt'), 'Português'), | ||||
|   MapEntry(Locale('pt', 'BR'), 'Brasileiro'), | ||||
|   MapEntry(Locale('cs'), 'Česky'), | ||||
|   MapEntry(Locale('sv'), 'Svenska'), | ||||
|   MapEntry(Locale('nl'), 'Nederlands'), | ||||
| @@ -44,7 +45,9 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | ||||
|   MapEntry(Locale('da'), 'Dansk'), | ||||
|   MapEntry(Locale('en', 'EO'), | ||||
|       'Esperanto'), // https://github.com/aissat/easy_localization/issues/220#issuecomment-846035493 | ||||
|   MapEntry(Locale('in'), 'Bahasa Indonesia') | ||||
|   MapEntry(Locale('in'), 'Bahasa Indonesia'), | ||||
|   MapEntry(Locale('ko'), '한국어'), | ||||
|   MapEntry(Locale('ca'), 'Català'), | ||||
| ]; | ||||
| const fallbackLocale = Locale('en'); | ||||
| const localeDir = 'assets/translations'; | ||||
| @@ -60,11 +63,11 @@ Future<void> loadTranslations() async { | ||||
|   var forceLocale = s.forcedLocale; | ||||
|   final controller = EasyLocalizationController( | ||||
|     saveLocale: true, | ||||
|     forceLocale: forceLocale != null ? Locale(forceLocale) : null, | ||||
|     forceLocale: forceLocale, | ||||
|     fallbackLocale: fallbackLocale, | ||||
|     supportedLocales: supportedLocales.map((e) => e.key).toList(), | ||||
|     assetLoader: const RootBundleAssetLoader(), | ||||
|     useOnlyLangCode: true, | ||||
|     useOnlyLangCode: false, | ||||
|     useFallbackTranslations: true, | ||||
|     path: localeDir, | ||||
|     onLoadError: (FlutterError e) { | ||||
| @@ -107,18 +110,20 @@ void main() async { | ||||
|     ); | ||||
|     SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge); | ||||
|   } | ||||
|   final np = NotificationsProvider(); | ||||
|   await np.initialize(); | ||||
|   runApp(MultiProvider( | ||||
|     providers: [ | ||||
|       ChangeNotifierProvider(create: (context) => AppsProvider()), | ||||
|       ChangeNotifierProvider(create: (context) => SettingsProvider()), | ||||
|       Provider(create: (context) => NotificationsProvider()), | ||||
|       Provider(create: (context) => np), | ||||
|       Provider(create: (context) => LogsProvider()) | ||||
|     ], | ||||
|     child: EasyLocalization( | ||||
|         supportedLocales: supportedLocales.map((e) => e.key).toList(), | ||||
|         path: localeDir, | ||||
|         fallbackLocale: fallbackLocale, | ||||
|         useOnlyLangCode: true, | ||||
|         useOnlyLangCode: false, | ||||
|         child: const Obtainium()), | ||||
|   )); | ||||
|   BackgroundFetch.registerHeadlessTask(backgroundFetchHeadlessTask); | ||||
| @@ -166,6 +171,7 @@ class _ObtainiumState extends State<Obtainium> { | ||||
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); | ||||
|     AppsProvider appsProvider = context.read<AppsProvider>(); | ||||
|     LogsProvider logs = context.read<LogsProvider>(); | ||||
|     NotificationsProvider notifs = context.read<NotificationsProvider>(); | ||||
|  | ||||
|     if (settingsProvider.prefs == null) { | ||||
|       settingsProvider.initializeSettings(); | ||||
| @@ -202,16 +208,17 @@ class _ObtainiumState extends State<Obtainium> { | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|       if (!supportedLocales | ||||
|               .map((e) => e.key.languageCode) | ||||
|               .contains(context.locale.languageCode) || | ||||
|       if (!supportedLocales.map((e) => e.key).contains(context.locale) || | ||||
|           (settingsProvider.forcedLocale == null && | ||||
|               context.deviceLocale.languageCode != | ||||
|                   context.locale.languageCode)) { | ||||
|               context.deviceLocale != context.locale)) { | ||||
|         settingsProvider.resetLocaleSafe(context); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     WidgetsBinding.instance.addPostFrameCallback((_) { | ||||
|       notifs.checkLaunchByNotif(); | ||||
|     }); | ||||
|  | ||||
|     return DynamicColorBuilder( | ||||
|         builder: (ColorScheme? lightDynamic, ColorScheme? darkDynamic) { | ||||
|       // Decide on a colour/brightness scheme based on OS and user settings | ||||
| @@ -244,22 +251,21 @@ class _ObtainiumState extends State<Obtainium> { | ||||
|           supportedLocales: context.supportedLocales, | ||||
|           locale: context.locale, | ||||
|           navigatorKey: globalNavigatorKey, | ||||
|           debugShowCheckedModeBanner: false, | ||||
|           theme: ThemeData( | ||||
|               useMaterial3: true, | ||||
|               colorScheme: settingsProvider.theme == ThemeSettings.dark | ||||
|                   ? darkColorScheme | ||||
|                   : lightColorScheme, | ||||
|               fontFamily: settingsProvider.useSystemFont | ||||
|                   ? 'SystemFont' | ||||
|                   : 'Wix-Madefor-Display'), | ||||
|               fontFamily: | ||||
|                   settingsProvider.useSystemFont ? 'SystemFont' : 'Montserrat'), | ||||
|           darkTheme: ThemeData( | ||||
|               useMaterial3: true, | ||||
|               colorScheme: settingsProvider.theme == ThemeSettings.light | ||||
|                   ? lightColorScheme | ||||
|                   : darkColorScheme, | ||||
|               fontFamily: settingsProvider.useSystemFont | ||||
|                   ? 'SystemFont' | ||||
|                   : 'Wix-Madefor-Display'), | ||||
|               fontFamily: | ||||
|                   settingsProvider.useSystemFont ? 'SystemFont' : 'Montserrat'), | ||||
|           home: Shortcuts(shortcuts: <LogicalKeySet, Intent>{ | ||||
|             LogicalKeySet(LogicalKeyboardKey.select): const ActivateIntent(), | ||||
|           }, child: const HomePage())); | ||||
|   | ||||
| @@ -30,6 +30,7 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|   String userInput = ''; | ||||
|   String searchQuery = ''; | ||||
|   String? pickedSourceOverride; | ||||
|   String? previousPickedSourceOverride; | ||||
|   AppSource? pickedSource; | ||||
|   Map<String, dynamic> additionalSettings = {}; | ||||
|   bool additionalSettingsValid = true; | ||||
| @@ -58,6 +59,9 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|         if (overrideSource != null) { | ||||
|           pickedSourceOverride = overrideSource; | ||||
|         } | ||||
|         bool overrideChanged = | ||||
|             pickedSourceOverride != previousPickedSourceOverride; | ||||
|         previousPickedSourceOverride = pickedSourceOverride; | ||||
|         if (updateUrlInput) { | ||||
|           urlInputKey++; | ||||
|         } | ||||
| @@ -69,6 +73,7 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|                 overrideSource: pickedSourceOverride) | ||||
|             : null; | ||||
|         if (pickedSource.runtimeType != source.runtimeType || | ||||
|             overrideChanged || | ||||
|             (prevHost != null && prevHost != source?.hosts[0])) { | ||||
|           pickedSource = source; | ||||
|           pickedSource?.runOnAddAppInputChange(userInput); | ||||
| @@ -487,7 +492,8 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|               height: 16, | ||||
|             ), | ||||
|             GeneratedForm( | ||||
|                 key: Key(pickedSource.runtimeType.toString()), | ||||
|                 key: Key( | ||||
|                     '${pickedSource.runtimeType.toString()}-${pickedSource?.hostChanged.toString()}-${pickedSource?.hostIdenticalDespiteAnyChange.toString()}'), | ||||
|                 items: [ | ||||
|                   ...pickedSource!.combinedAppSpecificSettingFormItems, | ||||
|                   ...(pickedSourceOverride != null | ||||
| @@ -532,12 +538,47 @@ 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']; | ||||
|                       }); | ||||
|                     } | ||||
|                   }), | ||||
|           ], | ||||
|         ); | ||||
|  | ||||
|     Widget getSourcesListWidget() => Padding( | ||||
|           padding: const EdgeInsets.all(16), | ||||
|           child: Row( | ||||
|           child: Wrap( | ||||
|             direction: Axis.horizontal, | ||||
|             alignment: WrapAlignment.spaceBetween, | ||||
|             spacing: 12, | ||||
|             children: [ | ||||
|               GestureDetector( | ||||
|                   onTap: () { | ||||
| @@ -594,7 +635,20 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|                         fontWeight: FontWeight.bold, | ||||
|                         decoration: TextDecoration.underline, | ||||
|                         fontStyle: FontStyle.italic), | ||||
|                   )) | ||||
|                   )), | ||||
|               GestureDetector( | ||||
|                 onTap: () { | ||||
|                   launchUrlString('https://apps.obtainium.imranr.dev/', | ||||
|                       mode: LaunchMode.externalApplication); | ||||
|                 }, | ||||
|                 child: Text( | ||||
|                   tr('crowdsourcedConfigsShort'), | ||||
|                   style: const TextStyle( | ||||
|                       fontWeight: FontWeight.bold, | ||||
|                       decoration: TextDecoration.underline, | ||||
|                       fontStyle: FontStyle.italic), | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|         ); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:flutter_markdown/flutter_markdown.dart'; | ||||
| import 'package:obtainium/components/generated_form_modal.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/main.dart'; | ||||
| @@ -12,6 +13,7 @@ import 'package:obtainium/providers/source_provider.dart'; | ||||
| import 'package:url_launcher/url_launcher_string.dart'; | ||||
| import 'package:webview_flutter/webview_flutter.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:markdown/markdown.dart' as md; | ||||
|  | ||||
| class AppPage extends StatefulWidget { | ||||
|   const AppPage({super.key, required this.appId}); | ||||
| @@ -23,9 +25,32 @@ class AppPage extends StatefulWidget { | ||||
| } | ||||
|  | ||||
| class _AppPageState extends State<AppPage> { | ||||
|   late final WebViewController _webViewController; | ||||
|   bool _wasWebViewOpened = false; | ||||
|   AppInMemory? prevApp; | ||||
|   bool updating = false; | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _webViewController = WebViewController() | ||||
|       ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||
|       ..setNavigationDelegate( | ||||
|         NavigationDelegate( | ||||
|           onWebResourceError: (WebResourceError error) { | ||||
|             if (error.isForMainFrame == true) { | ||||
|               showError( | ||||
|                   ObtainiumError(error.description, unexpected: true), context); | ||||
|             } | ||||
|           }, | ||||
|           onNavigationRequest: (NavigationRequest request) => | ||||
|               request.url.startsWith("rustore://") | ||||
|                   ? NavigationDecision.prevent | ||||
|                   : NavigationDecision.navigate, | ||||
|         ), | ||||
|       ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     var appsProvider = context.watch<AppsProvider>(); | ||||
| @@ -79,6 +104,11 @@ class _AppPageState extends State<AppPage> { | ||||
|         (app?.app.installedVersion != null && | ||||
|             app?.app.additionalSettings['versionDetection'] != true); | ||||
|  | ||||
|     if (app != null && !_wasWebViewOpened) { | ||||
|       _wasWebViewOpened = true; | ||||
|       _webViewController.loadRequest(Uri.parse(app.app.url)); | ||||
|     } | ||||
|  | ||||
|     getInfoColumn() { | ||||
|       String versionLines = ''; | ||||
|       bool installed = app?.app.installedVersion != null; | ||||
| @@ -182,7 +212,10 @@ class _AppPageState extends State<AppPage> { | ||||
|                                             Brightness.light | ||||
|                                         ? Theme.of(context).primaryColor | ||||
|                                         : Theme.of(context).primaryColorLight) | ||||
|                                     .withAlpha(20) | ||||
|                                     .withAlpha(Theme.of(context).brightness == | ||||
|                                             Brightness.light | ||||
|                                         ? 20 | ||||
|                                         : 40) | ||||
|                                 : null), | ||||
|                         padding: settingsProvider.highlightTouchTargets | ||||
|                             ? const EdgeInsetsDirectional.fromSTEB(12, 6, 12, 6) | ||||
| @@ -218,6 +251,7 @@ class _AppPageState extends State<AppPage> { | ||||
|           if (app?.app.additionalSettings['about'] is String && | ||||
|               app?.app.additionalSettings['about'].isNotEmpty) | ||||
|             Column( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               children: [ | ||||
|                 const SizedBox( | ||||
|                   height: 48, | ||||
| @@ -230,23 +264,39 @@ class _AppPageState extends State<AppPage> { | ||||
|                         content: Text(tr('copiedToClipboard')), | ||||
|                       )); | ||||
|                     }, | ||||
|                   child: Text( | ||||
|                     app?.app.additionalSettings['about'], | ||||
|                     textAlign: TextAlign.center, | ||||
|                     style: const TextStyle(fontStyle: FontStyle.italic), | ||||
|                     child: Markdown( | ||||
|                       physics: NeverScrollableScrollPhysics(), | ||||
|                       shrinkWrap: true, | ||||
|                       styleSheet: MarkdownStyleSheet( | ||||
|                           blockquoteDecoration: | ||||
|                               BoxDecoration(color: Theme.of(context).cardColor), | ||||
|                           textAlign: WrapAlignment.center), | ||||
|                       data: app?.app.additionalSettings['about'], | ||||
|                       onTapLink: (text, href, title) { | ||||
|                         if (href != null) { | ||||
|                           launchUrlString(href, | ||||
|                               mode: LaunchMode.externalApplication); | ||||
|                         } | ||||
|                       }, | ||||
|                       extensionSet: md.ExtensionSet( | ||||
|                         md.ExtensionSet.gitHubFlavored.blockSyntaxes, | ||||
|                         [ | ||||
|                           md.EmojiSyntax(), | ||||
|                           ...md.ExtensionSet.gitHubFlavored.inlineSyntaxes | ||||
|                         ], | ||||
|                       ), | ||||
|                 ) | ||||
|                     )) | ||||
|               ], | ||||
|             ), | ||||
|         ], | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     getFullInfoColumn() => Column( | ||||
|     getFullInfoColumn({bool small = false}) => Column( | ||||
|           mainAxisAlignment: MainAxisAlignment.center, | ||||
|           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|           children: [ | ||||
|             const SizedBox(height: 20), | ||||
|             SizedBox(height: small ? 5 : 20), | ||||
|             FutureBuilder( | ||||
|                 future: | ||||
|                     appsProvider.updateAppIcon(app?.app.id, ignoreCache: true), | ||||
| @@ -261,24 +311,28 @@ class _AppPageState extends State<AppPage> { | ||||
|                                     : () => pm.openApp(app.app.id), | ||||
|                                 child: Image.memory( | ||||
|                                   app!.icon!, | ||||
|                                   height: 150, | ||||
|                                   height: small ? 70 : 150, | ||||
|                                   gaplessPlayback: true, | ||||
|                                 ), | ||||
|                               ) | ||||
|                             ]) | ||||
|                       : Container(); | ||||
|                 }), | ||||
|             const SizedBox( | ||||
|               height: 25, | ||||
|             SizedBox( | ||||
|               height: small ? 10 : 25, | ||||
|             ), | ||||
|             Text( | ||||
|               app?.name ?? tr('app'), | ||||
|               textAlign: TextAlign.center, | ||||
|               style: Theme.of(context).textTheme.displayLarge, | ||||
|               style: small | ||||
|                   ? Theme.of(context).textTheme.displaySmall | ||||
|                   : Theme.of(context).textTheme.displayLarge, | ||||
|             ), | ||||
|             Text(tr('byX', args: [app?.app.author ?? tr('unknown')]), | ||||
|             Text(tr('byX', args: [app?.author ?? tr('unknown')]), | ||||
|                 textAlign: TextAlign.center, | ||||
|                 style: Theme.of(context).textTheme.headlineMedium), | ||||
|                 style: small | ||||
|                     ? Theme.of(context).textTheme.headlineSmall | ||||
|                     : Theme.of(context).textTheme.headlineMedium), | ||||
|             const SizedBox( | ||||
|               height: 24, | ||||
|             ), | ||||
| @@ -314,22 +368,9 @@ class _AppPageState extends State<AppPage> { | ||||
|  | ||||
|     getAppWebView() => app != null | ||||
|         ? WebViewWidget( | ||||
|             controller: WebViewController() | ||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||
|               ..setBackgroundColor(Theme.of(context).colorScheme.surface) | ||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||
|               ..setNavigationDelegate( | ||||
|                 NavigationDelegate( | ||||
|                   onWebResourceError: (WebResourceError error) { | ||||
|                     if (error.isForMainFrame == true) { | ||||
|                       showError( | ||||
|                           ObtainiumError(error.description, unexpected: true), | ||||
|                           context); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|               ) | ||||
|               ..loadRequest(Uri.parse(app.app.url))) | ||||
|             key: ObjectKey(_webViewController), | ||||
|             controller: _webViewController | ||||
|               ..setBackgroundColor(Theme.of(context).colorScheme.surface)) | ||||
|         : Container(); | ||||
|  | ||||
|     showMarkUpdatedDialog() { | ||||
| @@ -496,11 +537,8 @@ class _AppPageState extends State<AppPage> { | ||||
|                                   builder: (BuildContext ctx) { | ||||
|                                     return AlertDialog( | ||||
|                                       scrollable: true, | ||||
|                                       content: getInfoColumn(), | ||||
|                                       title: Text( | ||||
|                                           '${app.name} ${tr('byX', args: [ | ||||
|                                             app.app.author | ||||
|                                           ])}'), | ||||
|                                       content: getFullInfoColumn(small: true), | ||||
|                                       title: Text(app.name), | ||||
|                                       actions: [ | ||||
|                                         TextButton( | ||||
|                                             onPressed: () { | ||||
|   | ||||
| @@ -60,6 +60,9 @@ showChangeLogDialog(BuildContext context, App app, String? changesUrl, | ||||
|                     width: MediaQuery.of(context).size.width, | ||||
|                     height: MediaQuery.of(context).size.height - 350, | ||||
|                     child: Markdown( | ||||
|                       styleSheet: MarkdownStyleSheet( | ||||
|                           blockquoteDecoration: BoxDecoration( | ||||
|                               color: Theme.of(context).cardColor)), | ||||
|                       data: changeLog, | ||||
|                       onTapLink: (text, href, title) { | ||||
|                         if (href != null) { | ||||
| @@ -213,7 +216,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|           } | ||||
|         } | ||||
|         for (var t in authorTokens) { | ||||
|           if (!app.app.author.toLowerCase().contains(t.toLowerCase())) { | ||||
|           if (!app.author.toLowerCase().contains(t.toLowerCase())) { | ||||
|             return false; | ||||
|           } | ||||
|         } | ||||
| @@ -244,11 +247,11 @@ class AppsPageState extends State<AppsPage> { | ||||
|     listedApps.sort((a, b) { | ||||
|       int result = 0; | ||||
|       if (settingsProvider.sortColumn == SortColumnSettings.authorName) { | ||||
|         result = ((a.app.author + a.name).toLowerCase()) | ||||
|             .compareTo((b.app.author + b.name).toLowerCase()); | ||||
|         result = ((a.author + a.name).toLowerCase()) | ||||
|             .compareTo((b.author + b.name).toLowerCase()); | ||||
|       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { | ||||
|         result = ((a.name + a.app.author).toLowerCase()) | ||||
|             .compareTo((b.name + b.app.author).toLowerCase()); | ||||
|         result = ((a.name + a.author).toLowerCase()) | ||||
|             .compareTo((b.name + b.author).toLowerCase()); | ||||
|       } else if (settingsProvider.sortColumn == | ||||
|           SortColumnSettings.releaseDate) { | ||||
|         result = (a.app.releaseDate)?.compareTo( | ||||
| @@ -481,7 +484,10 @@ class AppsPageState extends State<AppsPage> { | ||||
|                           ? (Theme.of(context).brightness == Brightness.light | ||||
|                                   ? Theme.of(context).primaryColor | ||||
|                                   : Theme.of(context).primaryColorLight) | ||||
|                               .withAlpha(20) | ||||
|                               .withAlpha(Theme.of(context).brightness == | ||||
|                                       Brightness.light | ||||
|                                   ? 20 | ||||
|                                   : 40) | ||||
|                           : null), | ||||
|                   padding: settingsProvider.highlightTouchTargets | ||||
|                       ? const EdgeInsetsDirectional.fromSTEB(12, 0, 12, 0) | ||||
| @@ -521,12 +527,12 @@ class AppsPageState extends State<AppsPage> { | ||||
|       var transparent = | ||||
|           Theme.of(context).colorScheme.surface.withAlpha(0).value; | ||||
|       List<double> stops = [ | ||||
|         ...listedApps[index].app.categories.asMap().entries.map( | ||||
|             (e) => ((e.key / (listedApps[index].app.categories.length - 1)))), | ||||
|         ...listedApps[index].app.categories.asMap().entries.map((e) => | ||||
|             ((e.key / (listedApps[index].app.categories.length - 1)) - 0.0001)), | ||||
|         1 | ||||
|       ]; | ||||
|       if (stops.length == 2) { | ||||
|         stops[0] = 1; | ||||
|         stops[0] = 0.9999; | ||||
|       } | ||||
|       return Container( | ||||
|           decoration: BoxDecoration( | ||||
| @@ -564,7 +570,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|                     : FontWeight.normal, | ||||
|               ), | ||||
|             ), | ||||
|             subtitle: Text(tr('byX', args: [listedApps[index].app.author]), | ||||
|             subtitle: Text(tr('byX', args: [listedApps[index].author]), | ||||
|                 maxLines: 1, | ||||
|                 style: TextStyle( | ||||
|                     overflow: TextOverflow.ellipsis, | ||||
| @@ -838,30 +844,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, | ||||
| @@ -929,13 +911,15 @@ class AppsPageState extends State<AppsPage> { | ||||
|                                               .map((e) => e.id) | ||||
|                                               .toList(), | ||||
|                                           overrideExportSettings: false)); | ||||
|                                   String fn = | ||||
|                                       '${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().toIso8601String().replaceAll(':', '-')}-count-${selectedApps.length}'; | ||||
|                                   XFile f = XFile.fromData( | ||||
|                                       Uint8List.fromList( | ||||
|                                           utf8.encode(exportJSON)), | ||||
|                                       mimeType: 'application/json', | ||||
|                                       name: | ||||
|                                           '${tr('obtainiumExportHyphenatedLowercase')}-${selectedApps.length}-${DateTime.now().millisecondsSinceEpoch}'); | ||||
|                                   Share.shareXFiles([f]); | ||||
|                                       name: fn); | ||||
|                                   Share.shareXFiles([f], | ||||
|                                       fileNameOverrides: ['$fn.json']); | ||||
|                                 }, | ||||
|                           child: Text( | ||||
|                               '${tr('share')} - ${tr('obtainiumExport')}')), | ||||
|   | ||||
| @@ -15,6 +15,7 @@ import 'package:obtainium/providers/apps_provider.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:url_launcher/url_launcher_string.dart'; | ||||
|  | ||||
| class HomePage extends StatefulWidget { | ||||
|   const HomePage({super.key}); | ||||
| @@ -54,6 +55,45 @@ class _HomePageState extends State<HomePage> { | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     initDeepLinks(); | ||||
|     WidgetsBinding.instance.addPostFrameCallback((_) async { | ||||
|       var sp = context.read<SettingsProvider>(); | ||||
|       if (!sp.welcomeShown) { | ||||
|         await showDialog( | ||||
|             context: context, | ||||
|             builder: (BuildContext ctx) { | ||||
|               return AlertDialog( | ||||
|                 title: Text(tr('welcome')), | ||||
|                 content: Column( | ||||
|                   mainAxisSize: MainAxisSize.min, | ||||
|                   spacing: 20, | ||||
|                   children: [ | ||||
|                     Text(tr('documentationLinksNote')), | ||||
|                     GestureDetector( | ||||
|                         onTap: () { | ||||
|                           launchUrlString( | ||||
|                               'https://github.com/ImranR98/Obtainium/blob/main/README.md', | ||||
|                               mode: LaunchMode.externalApplication); | ||||
|                         }, | ||||
|                         child: Text( | ||||
|                           'https://github.com/ImranR98/Obtainium/blob/main/README.md', | ||||
|                           style: const TextStyle( | ||||
|                               decoration: TextDecoration.underline, | ||||
|                               fontWeight: FontWeight.bold), | ||||
|                         )), | ||||
|                   ], | ||||
|                 ), | ||||
|                 actions: [ | ||||
|                   TextButton( | ||||
|                       onPressed: () { | ||||
|                         sp.welcomeShown = true; | ||||
|                         Navigator.of(context).pop(null); | ||||
|                       }, | ||||
|                       child: Text(tr('ok'))), | ||||
|                 ], | ||||
|               ); | ||||
|             }); | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<void> initDeepLinks() async { | ||||
| @@ -130,13 +170,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; | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -490,8 +490,9 @@ class _ImportExportPageState extends State<ImportExportPage> { | ||||
|                                                       searchSource[0]); | ||||
|                                                 } | ||||
|                                               }, | ||||
|                                         child: Text(tr('searchX', | ||||
|                                             args: [tr('source')])))), | ||||
|                                         child: Text(tr('searchX', args: [ | ||||
|                                           tr('source').toLowerCase() | ||||
|                                         ])))), | ||||
|                               ], | ||||
|                             ), | ||||
|                             const SizedBox(height: 8), | ||||
|   | ||||
| @@ -262,14 +262,14 @@ class _SettingsPageState extends State<SettingsPage> { | ||||
|             child: Text(tr('followSystem')), | ||||
|           ), | ||||
|           ...supportedLocales.map((e) => DropdownMenuItem( | ||||
|                 value: e.key.toLanguageTag(), | ||||
|                 value: e.key, | ||||
|                 child: Text(e.value), | ||||
|               )) | ||||
|         ], | ||||
|         onChanged: (value) { | ||||
|           settingsProvider.forcedLocale = value; | ||||
|           if (value != null) { | ||||
|             context.setLocale(Locale(value)); | ||||
|             context.setLocale(value); | ||||
|           } else { | ||||
|             settingsProvider.resetLocaleSafe(context); | ||||
|           } | ||||
| @@ -419,6 +419,25 @@ class _SettingsPageState extends State<SettingsPage> { | ||||
|                                                           }) | ||||
|                                                     ], | ||||
|                                                   ), | ||||
|                                                   height16, | ||||
|                                                   Row( | ||||
|                                                     mainAxisAlignment: | ||||
|                                                         MainAxisAlignment | ||||
|                                                             .spaceBetween, | ||||
|                                                     children: [ | ||||
|                                                       Flexible( | ||||
|                                                           child: Text(tr( | ||||
|                                                               'bgUpdatesWhileChargingOnly'))), | ||||
|                                                       Switch( | ||||
|                                                           value: settingsProvider | ||||
|                                                               .bgUpdatesWhileChargingOnly, | ||||
|                                                           onChanged: (value) { | ||||
|                                                             settingsProvider | ||||
|                                                                     .bgUpdatesWhileChargingOnly = | ||||
|                                                                 value; | ||||
|                                                           }) | ||||
|                                                     ], | ||||
|                                                   ), | ||||
|                                                 ], | ||||
|                                               ), | ||||
|                                           ], | ||||
| @@ -578,6 +597,22 @@ class _SettingsPageState extends State<SettingsPage> { | ||||
|                                     }) | ||||
|                               ], | ||||
|                             ), | ||||
|                             height16, | ||||
|                             Row( | ||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                               children: [ | ||||
|                                 Flexible( | ||||
|                                     child: Text( | ||||
|                                         tr('shizukuPretendToBeGooglePlay'))), | ||||
|                                 Switch( | ||||
|                                     value: settingsProvider | ||||
|                                         .shizukuPretendToBeGooglePlay, | ||||
|                                     onChanged: (value) { | ||||
|                                       settingsProvider | ||||
|                                           .shizukuPretendToBeGooglePlay = value; | ||||
|                                     }) | ||||
|                               ], | ||||
|                             ), | ||||
|                             height32, | ||||
|                             Text( | ||||
|                               tr('sourceSpecific'), | ||||
| @@ -851,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), | ||||
|   | ||||
| @@ -5,9 +5,10 @@ import 'dart:async'; | ||||
| import 'dart:convert'; | ||||
| import 'dart:io'; | ||||
| import 'dart:math'; | ||||
| import 'package:battery_plus/battery_plus.dart'; | ||||
| import 'package:fluttertoast/fluttertoast.dart'; | ||||
| import 'package:http/http.dart' as http; | ||||
| import 'package:crypto/crypto.dart'; | ||||
| import 'dart:typed_data'; | ||||
|  | ||||
| import 'package:android_intent_plus/flag.dart'; | ||||
| import 'package:android_package_installer/android_package_installer.dart'; | ||||
| @@ -18,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'; | ||||
| @@ -50,6 +53,7 @@ class AppInMemory { | ||||
|       AppInMemory(app.deepCopy(), downloadProgress, installedInfo, icon); | ||||
|  | ||||
|   String get name => app.overrideName ?? app.finalName; | ||||
|   String get author => app.overrideAuthor ?? app.finalAuthor; | ||||
| } | ||||
|  | ||||
| class DownloadedApk { | ||||
| @@ -148,13 +152,15 @@ Future<File> downloadFileWithRetry(String url, String fileName, | ||||
|     {bool useExisting = true, | ||||
|     Map<String, String>? headers, | ||||
|     int retries = 3, | ||||
|     bool allowInsecure = false}) async { | ||||
|     bool allowInsecure = false, | ||||
|     LogsProvider? logs}) async { | ||||
|   try { | ||||
|     return await downloadFile( | ||||
|         url, fileName, fileNameHasExt, onProgress, destDir, | ||||
|         useExisting: useExisting, | ||||
|         headers: headers, | ||||
|         allowInsecure: allowInsecure); | ||||
|         allowInsecure: allowInsecure, | ||||
|         logs: logs); | ||||
|   } catch (e) { | ||||
|     if (retries > 0 && e is ClientException) { | ||||
|       await Future.delayed(const Duration(seconds: 5)); | ||||
| @@ -163,7 +169,8 @@ Future<File> downloadFileWithRetry(String url, String fileName, | ||||
|           useExisting: useExisting, | ||||
|           headers: headers, | ||||
|           retries: (retries - 1), | ||||
|           allowInsecure: allowInsecure); | ||||
|           allowInsecure: allowInsecure, | ||||
|           logs: logs); | ||||
|     } else { | ||||
|       rethrow; | ||||
|     } | ||||
| @@ -212,11 +219,8 @@ Future<String> checkPartialDownloadHash(String url, int bytesToGrab, | ||||
|   return hashListOfLists(bytes); | ||||
| } | ||||
|  | ||||
| Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|     Function? onProgress, String destDir, | ||||
|     {bool useExisting = true, | ||||
|     Map<String, String>? headers, | ||||
|     bool allowInsecure = false}) async { | ||||
| Future<String?> checkETagHeader(String url, | ||||
|     {Map<String, String>? headers, bool allowInsecure = false}) async { | ||||
|   // Send the initial request but cancel it as soon as you have the headers | ||||
|   var reqHeaders = headers ?? {}; | ||||
|   var req = Request('GET', Uri.parse(url)); | ||||
| @@ -224,6 +228,26 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|   var client = IOClient(createHttpClient(allowInsecure)); | ||||
|   StreamedResponse response = await client.send(req); | ||||
|   var resHeaders = response.headers; | ||||
|   client.close(); | ||||
|   return resHeaders[HttpHeaders.etagHeader] | ||||
|       ?.replaceAll('"', '') | ||||
|       .hashCode | ||||
|       .toString(); | ||||
| } | ||||
|  | ||||
| Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|     Function? onProgress, String destDir, | ||||
|     {bool useExisting = true, | ||||
|     Map<String, String>? headers, | ||||
|     bool allowInsecure = false, | ||||
|     LogsProvider? logs}) async { | ||||
|   // Send the initial request but cancel it as soon as you have the headers | ||||
|   var reqHeaders = headers ?? {}; | ||||
|   var req = Request('GET', Uri.parse(url)); | ||||
|   req.headers.addAll(reqHeaders); | ||||
|   var headersClient = IOClient(createHttpClient(allowInsecure)); | ||||
|   StreamedResponse headersResponse = await headersClient.send(req); | ||||
|   var resHeaders = headersResponse.headers; | ||||
|  | ||||
|   // Use the headers to decide what the file extension is, and | ||||
|   // whether it supports partial downloads (range request), and | ||||
| @@ -251,21 +275,20 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|     rangeFeatureEnabled = | ||||
|         resHeaders['accept-ranges']?.trim().toLowerCase() == 'bytes'; | ||||
|   } | ||||
|   headersClient.close(); | ||||
|  | ||||
|   // If you have an existing file that is usable, | ||||
|   // decide whether you can use it (either return full or resume partial) | ||||
|   var fullContentLength = response.contentLength; | ||||
|   var fullContentLength = headersResponse.contentLength; | ||||
|   if (useExisting && downloadedFile.existsSync()) { | ||||
|     var length = downloadedFile.lengthSync(); | ||||
|     if (fullContentLength == null || !rangeFeatureEnabled) { | ||||
|       // If there is no content length reported, assume it the existing file is fully downloaded | ||||
|       // Also if the range feature is not supported, don't trust the content length if any (#1542) | ||||
|       client.close(); | ||||
|       return downloadedFile; | ||||
|     } else { | ||||
|       // Check if resume needed/possible | ||||
|       if (length == fullContentLength) { | ||||
|         client.close(); | ||||
|         return downloadedFile; | ||||
|       } | ||||
|       if (length > fullContentLength) { | ||||
| @@ -277,6 +300,41 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|   // Download to a '.temp' file (to distinguish btn. complete/incomplete files) | ||||
|   File tempDownloadedFile = File('${downloadedFile.path}.part'); | ||||
|  | ||||
|   // If there is already a temp file, a download may already be in progress - account for this (see #2073) | ||||
|   bool tempFileExists = tempDownloadedFile.existsSync(); | ||||
|   if (tempFileExists && useExisting) { | ||||
|     logs?.add( | ||||
|         'Partial download exists - will wait: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||
|     bool isDownloading = true; | ||||
|     int currentTempFileSize = await tempDownloadedFile.length(); | ||||
|     bool shouldReturn = false; | ||||
|     while (isDownloading) { | ||||
|       await Future.delayed(Duration(seconds: 7)); | ||||
|       if (tempDownloadedFile.existsSync()) { | ||||
|         int newTempFileSize = await tempDownloadedFile.length(); | ||||
|         if (newTempFileSize > currentTempFileSize) { | ||||
|           currentTempFileSize = newTempFileSize; | ||||
|           logs?.add( | ||||
|               'Existing partial download still in progress: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||
|         } else { | ||||
|           logs?.add( | ||||
|               'Ignoring existing partial download: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||
|           break; | ||||
|         } | ||||
|       } else { | ||||
|         shouldReturn = downloadedFile.existsSync(); | ||||
|       } | ||||
|     } | ||||
|     if (shouldReturn) { | ||||
|       logs?.add( | ||||
|           'Existing partial download completed - not repeating: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||
|       return downloadedFile; | ||||
|     } else { | ||||
|       logs?.add( | ||||
|           'Existing partial download not in progress: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // If the range feature is not available (or you need to start a ranged req from 0), | ||||
|   // complete the already-started request, else cancel it and start a ranged request, | ||||
|   // and open the file for writing in the appropriate mode | ||||
| @@ -285,34 +343,62 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|       : null; | ||||
|   int rangeStart = targetFileLength ?? 0; | ||||
|   IOSink? sink; | ||||
|   if (rangeFeatureEnabled && fullContentLength != null && rangeStart > 0) { | ||||
|     client.close(); | ||||
|     client = IOClient(createHttpClient(allowInsecure)); | ||||
|   req = Request('GET', Uri.parse(url)); | ||||
|   req.headers.addAll(reqHeaders); | ||||
|     req.headers.addAll({'range': 'bytes=$rangeStart-${fullContentLength - 1}'}); | ||||
|     response = await client.send(req); | ||||
|   if (rangeFeatureEnabled && fullContentLength != null && rangeStart > 0) { | ||||
|     reqHeaders.addAll({'range': 'bytes=$rangeStart-${fullContentLength - 1}'}); | ||||
|     sink = tempDownloadedFile.openWrite(mode: FileMode.writeOnlyAppend); | ||||
|   } else if (tempDownloadedFile.existsSync()) { | ||||
|     tempDownloadedFile.deleteSync(recursive: true); | ||||
|   } | ||||
|   var responseWithClient = | ||||
|       await sourceRequestStreamResponse('GET', url, reqHeaders, {}); | ||||
|   HttpClient responseClient = responseWithClient.key; | ||||
|   HttpClientResponse response = responseWithClient.value; | ||||
|   sink ??= tempDownloadedFile.openWrite(mode: FileMode.writeOnly); | ||||
|  | ||||
|   // Perform the download | ||||
|   var received = 0; | ||||
|   double? progress; | ||||
|   DateTime? lastProgressUpdate; // Track last progress update time | ||||
|   if (rangeStart > 0 && fullContentLength != null) { | ||||
|     received = rangeStart; | ||||
|   } | ||||
|   await response.stream.map((s) { | ||||
|     received += s.length; | ||||
|     progress = | ||||
|         (fullContentLength != null ? (received / fullContentLength) * 100 : 30); | ||||
|     if (onProgress != null) { | ||||
|   const downloadUIUpdateInterval = Duration(milliseconds: 500); | ||||
|   const downloadBufferSize = 32 * 1024; // 32KB | ||||
|   final downloadBuffer = BytesBuilder(); | ||||
|   await response | ||||
|       .asBroadcastStream() | ||||
|       .map((chunk) { | ||||
|         received += chunk.length; | ||||
|         final now = DateTime.now(); | ||||
|         if (onProgress != null && | ||||
|             (lastProgressUpdate == null || | ||||
|                 now.difference(lastProgressUpdate!) >= | ||||
|                     downloadUIUpdateInterval)) { | ||||
|           progress = fullContentLength != null | ||||
|               ? (received / fullContentLength) * 100 | ||||
|               : 30; | ||||
|           onProgress(progress); | ||||
|           lastProgressUpdate = now; | ||||
|         } | ||||
|     return s; | ||||
|   }).pipe(sink); | ||||
|         return chunk; | ||||
|       }) | ||||
|       .transform(StreamTransformer<List<int>, List<int>>.fromHandlers( | ||||
|         handleData: (List<int> data, EventSink<List<int>> s) { | ||||
|           downloadBuffer.add(data); | ||||
|           if (downloadBuffer.length >= downloadBufferSize) { | ||||
|             s.add(downloadBuffer.takeBytes()); | ||||
|           } | ||||
|         }, | ||||
|         handleDone: (EventSink<List<int>> s) { | ||||
|           if (downloadBuffer.isNotEmpty) { | ||||
|             s.add(downloadBuffer.takeBytes()); | ||||
|           } | ||||
|           s.close(); | ||||
|         }, | ||||
|       )) | ||||
|       .pipe(sink); | ||||
|   await sink.close(); | ||||
|   progress = null; | ||||
|   if (onProgress != null) { | ||||
| @@ -320,31 +406,15 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|   } | ||||
|   if (response.statusCode < 200 || response.statusCode > 299) { | ||||
|     tempDownloadedFile.deleteSync(recursive: true); | ||||
|     throw response.reasonPhrase ?? tr('unexpectedError'); | ||||
|     throw response.reasonPhrase; | ||||
|   } | ||||
|   if (tempDownloadedFile.existsSync()) { | ||||
|     tempDownloadedFile.renameSync(downloadedFile.path); | ||||
|   } | ||||
|   client.close(); | ||||
|   responseClient.close(); | ||||
|   return downloadedFile; | ||||
| } | ||||
|  | ||||
| Future<Map<String, String>> getHeaders(String url, | ||||
|     {Map<String, String>? headers, bool allowInsecure = false}) async { | ||||
|   var req = http.Request('GET', Uri.parse(url)); | ||||
|   if (headers != null) { | ||||
|     req.headers.addAll(headers); | ||||
|   } | ||||
|   var client = IOClient(createHttpClient(allowInsecure)); | ||||
|   var response = await client.send(req); | ||||
|   if (response.statusCode < 200 || response.statusCode > 299) { | ||||
|     throw ObtainiumError(response.reasonPhrase ?? tr('unexpectedError')); | ||||
|   } | ||||
|   var returnHeaders = response.headers; | ||||
|   client.close(); | ||||
|   return returnHeaders; | ||||
| } | ||||
|  | ||||
| Future<List<PackageInfo>> getAllInstalledInfo() async { | ||||
|   return await pm.getInstalledPackages() ?? []; | ||||
| } | ||||
| @@ -416,9 +486,7 @@ class AppsProvider with ChangeNotifier { | ||||
|         // Delete any partial APKs (if safe to do so) | ||||
|         var cutoff = DateTime.now().subtract(const Duration(days: 7)); | ||||
|         APKDir.listSync() | ||||
|             .where((element) => | ||||
|                 element.path.endsWith('.part') || | ||||
|                 element.statSync().modified.isBefore(cutoff)) | ||||
|             .where((element) => element.statSync().modified.isBefore(cutoff)) | ||||
|             .forEach((partialApk) { | ||||
|           if (!areDownloadsRunning()) { | ||||
|             partialApk.delete(recursive: true); | ||||
| @@ -492,7 +560,8 @@ class AppsProvider with ChangeNotifier { | ||||
|         prevProg = prog; | ||||
|       }, APKDir.path, | ||||
|           useExisting: useExisting, | ||||
|           allowInsecure: app.additionalSettings['allowInsecure'] == true); | ||||
|           allowInsecure: app.additionalSettings['allowInsecure'] == true, | ||||
|           logs: logs); | ||||
|       // Set to 90 for remaining steps, will make null in 'finally' | ||||
|       if (apps[app.id] != null) { | ||||
|         apps[app.id]!.downloadProgress = -1; | ||||
| @@ -515,11 +584,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; | ||||
| @@ -565,9 +652,11 @@ class AppsProvider with ChangeNotifier { | ||||
|       return false; | ||||
|     } | ||||
|     if (app.additionalSettings['exemptFromBackgroundUpdates'] == true) { | ||||
|       logs.add('Exempted from BG updates: ${app.id}'); | ||||
|       return false; | ||||
|     } | ||||
|     if (app.apkUrls.length > 1) { | ||||
|       logs.add('Multiple APK URLs: ${app.id}'); | ||||
|       return false; // Manual API selection means silent install is not possible | ||||
|     } | ||||
|  | ||||
| @@ -579,6 +668,8 @@ class AppsProvider with ChangeNotifier { | ||||
|               ?.installingPackageName | ||||
|           : (await pm.getInstallerPackageName(packageName: app.id)); | ||||
|     } catch (e) { | ||||
|       logs.add( | ||||
|           'Failed to get installed package details: ${app.id} (${e.toString()})'); | ||||
|       return false; // App probably not installed | ||||
|     } | ||||
|  | ||||
| @@ -587,6 +678,7 @@ class AppsProvider with ChangeNotifier { | ||||
|     // The APK should target a new enough API | ||||
|     // https://developer.android.com/reference/android/content/pm/PackageInstaller.SessionParams#setRequireUserAction(int) | ||||
|     if (!(targetSDK != null && targetSDK >= (osInfo.version.sdkInt - 3))) { | ||||
|       logs.add('Multiple APK URLs: ${app.id}'); | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
| @@ -601,8 +693,12 @@ class AppsProvider with ChangeNotifier { | ||||
|       // If we did not install the app, silent install is not possible | ||||
|       return false; | ||||
|     } | ||||
|     if (osInfo.version.sdkInt < 31) { | ||||
|       // The OS must also be new enough | ||||
|     return osInfo.version.sdkInt >= 31; | ||||
|       logs.add('Android SDK too old: ${osInfo.version.sdkInt}'); | ||||
|       return false; | ||||
|     } | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   Future<void> waitForUserToReturnToForeground(BuildContext context) async { | ||||
| @@ -634,28 +730,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 { | ||||
| @@ -667,7 +782,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) { | ||||
| @@ -683,6 +799,9 @@ class AppsProvider with ChangeNotifier { | ||||
|     if (newInfo == null) { | ||||
|       try { | ||||
|         file.file.deleteSync(recursive: true); | ||||
|         for (var a in additionalAPKs) { | ||||
|           a.file.deleteSync(recursive: true); | ||||
|         } | ||||
|       } catch (e) { | ||||
|         // | ||||
|       } finally { | ||||
| @@ -690,6 +809,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())) { | ||||
| @@ -708,8 +829,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" : ""); | ||||
| @@ -733,6 +856,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; | ||||
|  | ||||
| @@ -741,7 +868,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; | ||||
| @@ -831,6 +958,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); | ||||
| @@ -879,22 +1011,20 @@ class AppsProvider with ChangeNotifier { | ||||
|             apps[id]?.installedInfo == null ? context : null; | ||||
|         bool needBGWorkaround = | ||||
|             willBeSilent && context == null && !settingsProvider.useShizuku; | ||||
|         bool shizukuPretendToBeGooglePlay = settingsProvider | ||||
|                 .shizukuPretendToBeGooglePlay || | ||||
|             apps[id]!.app.additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                 true; | ||||
|         if (downloadedFile != null) { | ||||
|           if (needBGWorkaround) { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             installApk(downloadedFile, contextIfNewInstall, | ||||
|                 needsBGWorkaround: true, | ||||
|                 shizukuPretendToBeGooglePlay: apps[id]! | ||||
|                         .app | ||||
|                         .additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                     true); | ||||
|                 shizukuPretendToBeGooglePlay: shizukuPretendToBeGooglePlay); | ||||
|           } else { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             sayInstalled = await installApk(downloadedFile, contextIfNewInstall, | ||||
|                 shizukuPretendToBeGooglePlay: apps[id]! | ||||
|                         .app | ||||
|                         .additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                     true); | ||||
|                 shizukuPretendToBeGooglePlay: shizukuPretendToBeGooglePlay); | ||||
|           } | ||||
|         } else { | ||||
|           if (needBGWorkaround) { | ||||
| @@ -905,10 +1035,7 @@ class AppsProvider with ChangeNotifier { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             sayInstalled = await installXApkDir( | ||||
|                 downloadedDir!, contextIfNewInstall, | ||||
|                 shizukuPretendToBeGooglePlay: apps[id]! | ||||
|                         .app | ||||
|                         .additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                     true); | ||||
|                 shizukuPretendToBeGooglePlay: shizukuPretendToBeGooglePlay); | ||||
|           } | ||||
|         } | ||||
|         if (willBeSilent && context == null) { | ||||
| @@ -1022,11 +1149,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)); | ||||
| @@ -1039,17 +1180,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 | ||||
| @@ -1061,7 +1192,8 @@ class AppsProvider with ChangeNotifier { | ||||
|                     forAPKDownload: | ||||
|                         fileUrl.key.endsWith('.apk') ? true : false), | ||||
|             useExisting: false, | ||||
|             allowInsecure: app.additionalSettings['allowInsecure'] == true); | ||||
|             allowInsecure: app.additionalSettings['allowInsecure'] == true, | ||||
|             logs: logs); | ||||
|         notificationsProvider | ||||
|             .notify(DownloadedNotification(fileUrl.key, fileUrl.value)); | ||||
|       } catch (e) { | ||||
| @@ -1098,17 +1230,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( | ||||
| @@ -1179,6 +1319,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; | ||||
|     } | ||||
| @@ -1342,8 +1483,10 @@ class AppsProvider with ChangeNotifier { | ||||
|         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; | ||||
|       } | ||||
|       if (!onlyIfExists || this.apps.containsKey(app.id)) { | ||||
|         File('${(await getAppsDir()).path}/${app.id}.json') | ||||
|             .writeAsStringSync(jsonEncode(app.toJson())); | ||||
|         String filePath = '${(await getAppsDir()).path}/${app.id}.json'; | ||||
|         File('$filePath.tmp') | ||||
|             .writeAsStringSync(jsonEncode(app.toJson())); // #2089 | ||||
|         File('$filePath.tmp').renameSync(filePath); | ||||
|       } | ||||
|       try { | ||||
|         this.apps.update(app.id, | ||||
| @@ -1881,10 +2024,20 @@ Future<void> bgUpdateCheck(String taskId, Map<String, dynamic>? params) async { | ||||
|         (<List<MapEntry<String, int>>>[])) | ||||
|   ]; | ||||
|  | ||||
|   var networkRestricted = false; | ||||
|   if (appsProvider.settingsProvider.bgUpdatesOnWiFiOnly) { | ||||
|     networkRestricted = !netResult.contains(ConnectivityResult.wifi) && | ||||
|   var networkRestricted = appsProvider.settingsProvider.bgUpdatesOnWiFiOnly && | ||||
|       !netResult.contains(ConnectivityResult.wifi) && | ||||
|       !netResult.contains(ConnectivityResult.ethernet); | ||||
|  | ||||
|   var chargingRestricted = | ||||
|       appsProvider.settingsProvider.bgUpdatesWhileChargingOnly && | ||||
|           (await Battery().batteryState) != BatteryState.charging; | ||||
|  | ||||
|   if (networkRestricted) { | ||||
|     logs.add('BG update task: Network restriction in effect.'); | ||||
|   } | ||||
|  | ||||
|   if (chargingRestricted) { | ||||
|     logs.add('BG update task: Charging restriction in effect.'); | ||||
|   } | ||||
|  | ||||
|   if (toCheck.isNotEmpty) { | ||||
| @@ -1924,14 +2077,6 @@ Future<void> bgUpdateCheck(String taskId, Map<String, dynamic>? params) async { | ||||
|     CheckingUpdatesNotification notif = CheckingUpdatesNotification( | ||||
|         plural('apps', toCheck.length)); // The notif. to show while checking | ||||
|  | ||||
|     // Set a bool for when we're no on wifi/wired and the user doesn't want to download apps in that state | ||||
|     var networkRestricted = false; | ||||
|     if (appsProvider.settingsProvider.bgUpdatesOnWiFiOnly) { | ||||
|       var netResult = await (Connectivity().checkConnectivity()); | ||||
|       networkRestricted = !netResult.contains(ConnectivityResult.wifi) && | ||||
|           !netResult.contains(ConnectivityResult.ethernet); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       // Check for updates | ||||
|       notificationsProvider.notify(notif, cancelExisting: true); | ||||
| @@ -1979,6 +2124,7 @@ Future<void> bgUpdateCheck(String taskId, Map<String, dynamic>? params) async { | ||||
|     // Filter out updates that will be installed silently (the rest go into toNotify) | ||||
|     for (var i = 0; i < updates.length; i++) { | ||||
|       if (networkRestricted || | ||||
|           chargingRestricted || | ||||
|           !(await appsProvider.canInstallSilently(updates[i]))) { | ||||
|         if (updates[i].additionalSettings['skipUpdateNotifications'] != true) { | ||||
|           toNotify.add(updates[i]); | ||||
| @@ -2025,7 +2171,7 @@ Future<void> bgUpdateCheck(String taskId, Map<String, dynamic>? params) async { | ||||
|   } else { | ||||
|     // In install mode... | ||||
|     // If you haven't explicitly been given updates to install, grab all available silent updates | ||||
|     if (toInstall.isEmpty && !networkRestricted) { | ||||
|     if (toInstall.isEmpty && !networkRestricted && !chargingRestricted) { | ||||
|       var temp = appsProvider.findExistingUpdates(installedOnly: true); | ||||
|       for (var i = 0; i < temp.length; i++) { | ||||
|         if (await appsProvider | ||||
|   | ||||
| @@ -2,7 +2,10 @@ | ||||
| // Contains a set of pre-defined ObtainiumNotification objects that should be used throughout the app | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_local_notifications/flutter_local_notifications.dart'; | ||||
| import 'package:obtainium/main.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class ObtainiumNotification { | ||||
| @@ -15,10 +18,11 @@ class ObtainiumNotification { | ||||
|   Importance importance; | ||||
|   int? progPercent; | ||||
|   bool onlyAlertOnce; | ||||
|   String? payload; | ||||
|  | ||||
|   ObtainiumNotification(this.id, this.title, this.message, this.channelCode, | ||||
|       this.channelName, this.channelDescription, this.importance, | ||||
|       {this.onlyAlertOnce = false, this.progPercent}); | ||||
|       {this.onlyAlertOnce = false, this.progPercent, this.payload}); | ||||
| } | ||||
|  | ||||
| class UpdateNotification extends ObtainiumNotification { | ||||
| @@ -44,23 +48,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()]); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -91,7 +91,8 @@ class ErrorCheckingUpdatesNotification extends ObtainiumNotification { | ||||
|             'BG_UPDATE_CHECK_ERROR', | ||||
|             tr('errorCheckingUpdatesNotifChannel'), | ||||
|             tr('errorCheckingUpdatesNotifDescription'), | ||||
|             Importance.high); | ||||
|             Importance.high, | ||||
|             payload: "${tr('errorCheckingUpdates')}\n$error"); | ||||
| } | ||||
|  | ||||
| class AppsRemovedNotification extends ObtainiumNotification { | ||||
| @@ -176,11 +177,50 @@ class NotificationsProvider { | ||||
|   }; | ||||
|  | ||||
|   Future<void> initialize() async { | ||||
|     isInitialized = await notifications.initialize(const InitializationSettings( | ||||
|             android: AndroidInitializationSettings('ic_notification'))) ?? | ||||
|     isInitialized = await notifications.initialize( | ||||
|           const InitializationSettings( | ||||
|               android: AndroidInitializationSettings('ic_notification')), | ||||
|           onDidReceiveNotificationResponse: (NotificationResponse response) { | ||||
|             _showNotificationPayload(response.payload); | ||||
|           }, | ||||
|         ) ?? | ||||
|         false; | ||||
|   } | ||||
|  | ||||
|   checkLaunchByNotif() async { | ||||
|     final NotificationAppLaunchDetails? launchDetails = | ||||
|         await notifications.getNotificationAppLaunchDetails(); | ||||
|     if (launchDetails?.didNotificationLaunchApp ?? false) { | ||||
|       _showNotificationPayload(launchDetails!.notificationResponse?.payload, | ||||
|           doublePop: true); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   _showNotificationPayload(String? payload, {bool doublePop = false}) { | ||||
|     if (payload?.isNotEmpty == true) { | ||||
|       var title = (payload ?? '\n\n').split('\n').first; | ||||
|       var content = (payload ?? '\n\n').split('\n').sublist(1).join('\n'); | ||||
|       globalNavigatorKey.currentState?.push( | ||||
|         PageRouteBuilder( | ||||
|           pageBuilder: (context, _, __) => AlertDialog( | ||||
|             title: Text(title), | ||||
|             content: Text(content), | ||||
|             actions: [ | ||||
|               TextButton( | ||||
|                   onPressed: () { | ||||
|                     Navigator.of(context).pop(null); | ||||
|                     if (doublePop) { | ||||
|                       Navigator.of(context).pop(null); | ||||
|                     } | ||||
|                   }, | ||||
|                   child: Text(tr('ok'))), | ||||
|             ], | ||||
|           ), | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> cancel(int id) async { | ||||
|     if (!isInitialized) { | ||||
|       await initialize(); | ||||
| @@ -198,7 +238,8 @@ class NotificationsProvider { | ||||
|       Importance importance, | ||||
|       {bool cancelExisting = false, | ||||
|       int? progPercent, | ||||
|       bool onlyAlertOnce = false}) async { | ||||
|       bool onlyAlertOnce = false, | ||||
|       String? payload}) async { | ||||
|     if (cancelExisting) { | ||||
|       await cancel(id); | ||||
|     } | ||||
| @@ -214,12 +255,13 @@ 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, | ||||
|                 onlyAlertOnce: onlyAlertOnce, | ||||
|                 indeterminate: progPercent != null && progPercent < 0))); | ||||
|                 indeterminate: progPercent != null && progPercent < 0)), | ||||
|         payload: payload); | ||||
|   } | ||||
|  | ||||
|   Future<void> notify(ObtainiumNotification notif, | ||||
| @@ -228,5 +270,6 @@ class NotificationsProvider { | ||||
|           notif.channelName, notif.channelDescription, notif.importance, | ||||
|           cancelExisting: cancelExisting, | ||||
|           onlyAlertOnce: notif.onlyAlertOnce, | ||||
|           progPercent: notif.progPercent); | ||||
|           progPercent: notif.progPercent, | ||||
|           payload: notif.payload); | ||||
| } | ||||
|   | ||||
| @@ -48,7 +48,7 @@ class SettingsProvider with ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   bool get useShizuku{ | ||||
|   bool get useShizuku { | ||||
|     return prefs?.getBool('useShizuku') ?? false; | ||||
|   } | ||||
|  | ||||
| @@ -69,8 +69,7 @@ class SettingsProvider with ChangeNotifier { | ||||
|  | ||||
|   Color get themeColor { | ||||
|     int? colorCode = prefs?.getInt('themeColor'); | ||||
|     return (colorCode != null) ? | ||||
|         Color(colorCode) : obtainiumThemeColor; | ||||
|     return (colorCode != null) ? Color(colorCode) : obtainiumThemeColor; | ||||
|   } | ||||
|  | ||||
|   set themeColor(Color themeColor) { | ||||
| @@ -151,6 +150,15 @@ class SettingsProvider with ChangeNotifier { | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   bool get welcomeShown { | ||||
|     return prefs?.getBool('welcomeShown') ?? false; | ||||
|   } | ||||
|  | ||||
|   set welcomeShown(bool welcomeShown) { | ||||
|     prefs?.setBool('welcomeShown', welcomeShown); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   bool checkJustStarted() { | ||||
|     if (justStarted) { | ||||
|       justStarted = false; | ||||
| @@ -262,22 +270,24 @@ class SettingsProvider with ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   String? get forcedLocale { | ||||
|     var fl = prefs?.getString('forcedLocale'); | ||||
|     return supportedLocales | ||||
|             .where((element) => element.key.toLanguageTag() == fl) | ||||
|             .isNotEmpty | ||||
|   Locale? get forcedLocale { | ||||
|     var flSegs = prefs?.getString('forcedLocale')?.split('-'); | ||||
|     var fl = flSegs != null && flSegs.isNotEmpty | ||||
|         ? Locale(flSegs[0], flSegs.length > 1 ? flSegs[1] : null) | ||||
|         : null; | ||||
|     var set = supportedLocales.where((element) => element.key == fl).isNotEmpty | ||||
|         ? fl | ||||
|         : null; | ||||
|     return set; | ||||
|   } | ||||
|  | ||||
|   set forcedLocale(String? fl) { | ||||
|   set forcedLocale(Locale? fl) { | ||||
|     if (fl == null) { | ||||
|       prefs?.remove('forcedLocale'); | ||||
|     } else if (supportedLocales | ||||
|         .where((element) => element.key.toLanguageTag() == fl) | ||||
|         .where((element) => element.key == fl) | ||||
|         .isNotEmpty) { | ||||
|       prefs?.setString('forcedLocale', fl); | ||||
|       prefs?.setString('forcedLocale', fl.toLanguageTag()); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
| @@ -286,9 +296,7 @@ class SettingsProvider with ChangeNotifier { | ||||
|       a.length == b.length && a.union(b).length == a.length; | ||||
|  | ||||
|   void resetLocaleSafe(BuildContext context) { | ||||
|     if (context.supportedLocales | ||||
|         .map((e) => e.languageCode) | ||||
|         .contains(context.deviceLocale.languageCode)) { | ||||
|     if (context.supportedLocales.contains(context.deviceLocale)) { | ||||
|       context.resetLocale(); | ||||
|     } else { | ||||
|       context.setLocale(context.fallbackLocale!); | ||||
| @@ -350,6 +358,15 @@ class SettingsProvider with ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   bool get bgUpdatesWhileChargingOnly { | ||||
|     return prefs?.getBool('bgUpdatesWhileChargingOnly') ?? false; | ||||
|   } | ||||
|  | ||||
|   set bgUpdatesWhileChargingOnly(bool val) { | ||||
|     prefs?.setBool('bgUpdatesWhileChargingOnly', val); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   DateTime get lastCompletedBGCheckTime { | ||||
|     int? temp = prefs?.getInt('lastCompletedBGCheckTime'); | ||||
|     return temp != null | ||||
| @@ -469,4 +486,13 @@ class SettingsProvider with ChangeNotifier { | ||||
|     prefs?.setBool('beforeNewInstallsShareToAppVerifier', val); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   bool get shizukuPretendToBeGooglePlay { | ||||
|     return prefs?.getBool('shizukuPretendToBeGooglePlay') ?? false; | ||||
|   } | ||||
|  | ||||
|   set shizukuPretendToBeGooglePlay(bool val) { | ||||
|     prefs?.setBool('shizukuPretendToBeGooglePlay', val); | ||||
|     notifyListeners(); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -3,16 +3,18 @@ | ||||
|  | ||||
| import 'dart:convert'; | ||||
| import 'dart:io'; | ||||
| import 'package:http/http.dart' as http; | ||||
| import 'dart:typed_data'; | ||||
|  | ||||
| import 'package:device_info_plus/device_info_plus.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/dom.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:http/io_client.dart'; | ||||
| import 'package:obtainium/app_sources/apkmirror.dart'; | ||||
| import 'package:obtainium/app_sources/apkpure.dart'; | ||||
| import 'package:obtainium/app_sources/aptoide.dart'; | ||||
| import 'package:obtainium/app_sources/codeberg.dart'; | ||||
| import 'package:obtainium/app_sources/coolapk.dart'; | ||||
| import 'package:obtainium/app_sources/directAPKLink.dart'; | ||||
| import 'package:obtainium/app_sources/fdroid.dart'; | ||||
| import 'package:obtainium/app_sources/fdroidrepo.dart'; | ||||
| @@ -23,18 +25,16 @@ 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/rustore.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 { | ||||
| @@ -154,11 +154,11 @@ appJSONCompatibilityModifiers(Map<String, dynamic> json) { | ||||
|   if (additionalSettings['autoApkFilterByArch'] == null) { | ||||
|     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; | ||||
|   // GitHub "don't sort" option to new dropdown format | ||||
|   if (additionalSettings['dontSortReleasesList'] == true) { | ||||
|     additionalSettings['sortMethodChoice'] = 'none'; | ||||
|   } | ||||
|   if (source.runtimeType == HTML().runtimeType) { | ||||
|     // HTML key rename | ||||
|     if (originalAdditionalSettings['sortByFileNamesNotLinks'] != null) { | ||||
|       additionalSettings['sortByLastLinkSegment'] = | ||||
| @@ -183,7 +183,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 +200,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 +292,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; | ||||
| @@ -270,6 +336,15 @@ class App { | ||||
|     return overrideName ?? name; | ||||
|   } | ||||
|  | ||||
|   String? get overrideAuthor => | ||||
|       additionalSettings['appAuthor']?.toString().trim().isNotEmpty == true | ||||
|           ? additionalSettings['appAuthor'] | ||||
|           : null; | ||||
|  | ||||
|   String get finalAuthor { | ||||
|     return overrideAuthor ?? author; | ||||
|   } | ||||
|  | ||||
|   App deepCopy() => App( | ||||
|       id, | ||||
|       url, | ||||
| @@ -290,7 +365,14 @@ class App { | ||||
|       otherAssetUrls: otherAssetUrls); | ||||
|  | ||||
|   factory App.fromJson(Map<String, dynamic> json) { | ||||
|     Map<String, dynamic> originalJSON = 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, | ||||
| @@ -398,6 +480,23 @@ List<MapEntry<String, String>> getApkUrlsFromUrls(List<String> urls) => | ||||
|       return MapEntry(apkSegs.isNotEmpty ? apkSegs.last : segments.last, e); | ||||
|     }).toList(); | ||||
|  | ||||
| Future<List<MapEntry<String, String>>> filterApksByArch( | ||||
|     List<MapEntry<String, String>> apkUrls) async { | ||||
|   if (apkUrls.length > 1) { | ||||
|     var abis = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|     for (var abi in abis) { | ||||
|       var urls2 = apkUrls | ||||
|           .where((element) => RegExp('.*$abi.*').hasMatch(element.key)) | ||||
|           .toList(); | ||||
|       if (urls2.isNotEmpty && urls2.length < apkUrls.length) { | ||||
|         apkUrls = urls2; | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   return apkUrls; | ||||
| } | ||||
|  | ||||
| getSourceRegex(List<String> hosts) { | ||||
|   return '(${hosts.join('|').replaceAll('.', '\\.')})'; | ||||
| } | ||||
| @@ -411,9 +510,76 @@ HttpClient createHttpClient(bool insecure) { | ||||
|   return client; | ||||
| } | ||||
|  | ||||
| Future<MapEntry<HttpClient, HttpClientResponse>> sourceRequestStreamResponse( | ||||
|     String method, | ||||
|     String url, | ||||
|     Map<String, String>? requestHeaders, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|     {bool followRedirects = true, | ||||
|     Object? postBody}) async { | ||||
|   var currentUrl = Uri.parse(url); | ||||
|   var redirectCount = 0; | ||||
|   const maxRedirects = 10; | ||||
|   List<Cookie> cookies = []; | ||||
|   while (redirectCount < maxRedirects) { | ||||
|     var httpClient = | ||||
|         createHttpClient(additionalSettings['allowInsecure'] == true); | ||||
|     var request = await httpClient.openUrl(method, currentUrl); | ||||
|     if (requestHeaders != null) { | ||||
|       requestHeaders.forEach((key, value) { | ||||
|         request.headers.set(key, value); | ||||
|       }); | ||||
|     } | ||||
|     request.cookies.addAll(cookies); | ||||
|     request.followRedirects = false; | ||||
|     if (postBody != null) { | ||||
|       request.headers.contentType = ContentType.json; | ||||
|       request.write(jsonEncode(postBody)); | ||||
|     } | ||||
|     final response = await request.close(); | ||||
|  | ||||
|     if (followRedirects && | ||||
|         (response.statusCode >= 300 && response.statusCode <= 399)) { | ||||
|       final location = response.headers.value(HttpHeaders.locationHeader); | ||||
|       if (location != null) { | ||||
|         currentUrl = Uri.parse(ensureAbsoluteUrl(location, currentUrl)); | ||||
|         redirectCount++; | ||||
|         cookies = response.cookies; | ||||
|         httpClient.close(); | ||||
|         continue; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return MapEntry(httpClient, response); | ||||
|   } | ||||
|   throw ObtainiumError('Too many redirects ($maxRedirects)'); | ||||
| } | ||||
|  | ||||
| Future<Response> httpClientResponseStreamToFinalResponse(HttpClient httpClient, | ||||
|     String method, String url, HttpClientResponse response) async { | ||||
|   final bytes = | ||||
|       (await response.fold<BytesBuilder>(BytesBuilder(), (b, d) => b..add(d))) | ||||
|           .toBytes(); | ||||
|  | ||||
|   final headers = <String, String>{}; | ||||
|   response.headers.forEach((name, values) { | ||||
|     headers[name] = values.join(', '); | ||||
|   }); | ||||
|  | ||||
|   httpClient.close(); | ||||
|  | ||||
|   return http.Response.bytes( | ||||
|     bytes, | ||||
|     response.statusCode, | ||||
|     headers: headers, | ||||
|     request: http.Request(method, Uri.parse(url)), | ||||
|   ); | ||||
| } | ||||
|  | ||||
| abstract class AppSource { | ||||
|   List<String> hosts = []; | ||||
|   bool hostChanged = false; | ||||
|   bool hostIdenticalDespiteAnyChange = false; | ||||
|   late String name; | ||||
|   bool enforceTrackOnly = false; | ||||
|   bool changeLogIfAnyIsMarkDown = true; | ||||
| @@ -467,25 +633,16 @@ abstract class AppSource { | ||||
|   Future<Response> sourceRequest( | ||||
|       String url, Map<String, dynamic> additionalSettings, | ||||
|       {bool followRedirects = true, Object? postBody}) async { | ||||
|     var method = postBody == null ? 'GET' : 'POST'; | ||||
|     var requestHeaders = await getRequestHeaders(additionalSettings); | ||||
|     if (requestHeaders != null || followRedirects == false) { | ||||
|       var req = Request(postBody == null ? 'GET' : 'POST', Uri.parse(url)); | ||||
|       req.followRedirects = followRedirects; | ||||
|       if (requestHeaders != null) { | ||||
|         req.headers.addAll(requestHeaders); | ||||
|       } | ||||
|       if (postBody != null) { | ||||
|         req.headers[HttpHeaders.contentTypeHeader] = 'application/json'; | ||||
|         req.body = jsonEncode(postBody); | ||||
|       } | ||||
|       return Response.fromStream(await IOClient( | ||||
|               createHttpClient(additionalSettings['allowInsecure'] == true)) | ||||
|           .send(req)); | ||||
|     } else { | ||||
|       return postBody == null | ||||
|           ? get(Uri.parse(url)) | ||||
|           : post(Uri.parse(url), body: jsonEncode(postBody)); | ||||
|     } | ||||
|     var streamedResponseAndClient = await sourceRequestStreamResponse( | ||||
|         method, url, requestHeaders, additionalSettings, | ||||
|         followRedirects: followRedirects, postBody: postBody); | ||||
|     return await httpClientResponseStreamToFinalResponse( | ||||
|         streamedResponseAndClient.key, | ||||
|         method, | ||||
|         url, | ||||
|         streamedResponseAndClient.value); | ||||
|   } | ||||
|  | ||||
|   void runOnAddAppInputChange(String inputUrl) { | ||||
| @@ -554,6 +711,7 @@ abstract class AppSource { | ||||
|           label: tr('autoApkFilterByArch'), defaultValue: true) | ||||
|     ], | ||||
|     [GeneratedFormTextField('appName', label: tr('appName'), required: false)], | ||||
|     [GeneratedFormTextField('appAuthor', label: tr('author'), required: false)], | ||||
|     [ | ||||
|       GeneratedFormSwitch('shizukuPretendToBeGooglePlay', | ||||
|           label: tr('shizukuPretendToBeGooglePlay'), defaultValue: false) | ||||
| @@ -570,7 +728,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 | ||||
| @@ -628,9 +790,10 @@ abstract class AppSource { | ||||
|       SettingsProvider settingsProvider) async { | ||||
|     Map<String, String> results = {}; | ||||
|     for (var e in sourceConfigSettingFormItems) { | ||||
|       var val = hostChanged | ||||
|       var val = hostChanged && !hostIdenticalDespiteAnyChange | ||||
|           ? additionalSettings[e.key] | ||||
|           : settingsProvider.getSettingString(e.key); | ||||
|           : additionalSettings[e.key] ?? | ||||
|               settingsProvider.getSettingString(e.key); | ||||
|       if (val != null) { | ||||
|         results[e.key] = val; | ||||
|       } | ||||
| @@ -790,11 +953,10 @@ class SourceProvider { | ||||
|         Uptodown(), | ||||
|         HuaweiAppGallery(), | ||||
|         Tencent(), | ||||
|         CoolApk(), | ||||
|         Jenkins(), | ||||
|         APKMirror(), | ||||
|         Signal(), | ||||
|         VLC(), | ||||
|         WhatsApp(), | ||||
|         RuStore(), | ||||
|         TelegramApp(), | ||||
|         NeutronCode(), | ||||
|         DirectAPKLink(), | ||||
| @@ -813,9 +975,14 @@ class SourceProvider { | ||||
|         throw UnsupportedURLError(); | ||||
|       } | ||||
|       var res = srcs.first; | ||||
|       res.hosts = [Uri.parse(url).host]; | ||||
|       var originalHosts = res.hosts; | ||||
|       var newHost = Uri.parse(url).host; | ||||
|       res.hosts = [newHost]; | ||||
|       res.hostChanged = true; | ||||
|       return srcs.first; | ||||
|       if (originalHosts.contains(newHost)) { | ||||
|         res.hostIdenticalDespiteAnyChange = true; | ||||
|       } | ||||
|       return res; | ||||
|     } | ||||
|     AppSource? source; | ||||
|     for (var s in sources.where((element) => element.hosts.isNotEmpty)) { | ||||
| @@ -898,23 +1065,16 @@ class SourceProvider { | ||||
|     if (apk.apkUrls.isEmpty && !trackOnly) { | ||||
|       throw NoAPKError(); | ||||
|     } | ||||
|     if (apk.apkUrls.length > 1 && | ||||
|         additionalSettings['autoApkFilterByArch'] == true) { | ||||
|       var abis = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|       for (var abi in abis) { | ||||
|         var urls2 = apk.apkUrls | ||||
|             .where((element) => RegExp('.*$abi.*').hasMatch(element.key)) | ||||
|             .toList(); | ||||
|         if (urls2.isNotEmpty && urls2.length < apk.apkUrls.length) { | ||||
|           apk.apkUrls = urls2; | ||||
|           break; | ||||
|         } | ||||
|       } | ||||
|     if (additionalSettings['autoApkFilterByArch'] == true) { | ||||
|       apk.apkUrls = await filterApksByArch(apk.apkUrls); | ||||
|     } | ||||
|     var name = currentApp != null ? currentApp.name.trim() : ''; | ||||
|     name = name.isNotEmpty ? name : apk.names.name; | ||||
|     App finalApp = App( | ||||
|         currentApp?.id ?? | ||||
|             ((additionalSettings['appId'] != null) | ||||
|                 ? additionalSettings['appId'] | ||||
|                 : null) ?? | ||||
|             (!trackOnly && | ||||
|                     (!source.appIdInferIsOptional || | ||||
|                         (source.appIdInferIsOptional && inferAppIdIfOptional)) | ||||
|   | ||||
							
								
								
									
										496
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										47
									
								
								pubspec.yaml
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,5 @@ | ||||
| name: obtainium | ||||
| description: Get Android app updates straight from the source. | ||||
|  | ||||
| # The following line prevents the package from being accidentally published to | ||||
| # pub.dev using `flutter pub publish`. This is preferred for private packages. | ||||
| publish_to: 'none' # Remove this line if you wish to publish to pub.dev | ||||
| @@ -17,10 +16,10 @@ 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.25+2282 | ||||
| version: 1.1.52+2309 | ||||
|  | ||||
| environment: | ||||
|   sdk: '>=3.0.0 <4.0.0' | ||||
|   sdk: ^3.6.0 | ||||
|  | ||||
| # Dependencies specify other packages that your package needs in order to work. | ||||
| # To automatically upgrade your package dependencies to the latest versions | ||||
| @@ -32,13 +31,12 @@ dependencies: | ||||
|   flutter: | ||||
|     sdk: flutter | ||||
|  | ||||
|  | ||||
|   # The following adds the Cupertino Icons font to your application. | ||||
|   # Use with the CupertinoIcons class for iOS style icons. | ||||
|   cupertino_icons: ^1.0.5 | ||||
|   path_provider: ^2.0.11 | ||||
|   flutter_fgbg: ^0.6.0 | ||||
|   flutter_local_notifications: ^17.0.0 | ||||
|   flutter_fgbg: ^0.7.1 | ||||
|   flutter_local_notifications: ^18.0.0 | ||||
|   provider: ^6.0.3 | ||||
|   http: ^1.0.0 | ||||
|   webview_flutter: ^4.0.0 | ||||
| @@ -46,17 +44,20 @@ dependencies: | ||||
|   html: ^0.15.0 | ||||
|   shared_preferences: ^2.0.15 | ||||
|   url_launcher: ^6.1.5 | ||||
|   permission_handler: ^11.0.0 | ||||
|   permission_handler: ^12.0.0+1 | ||||
|   fluttertoast: ^8.0.9 | ||||
|   device_info_plus: ^10.0.1 | ||||
|   file_picker: ^8.0.0+1 | ||||
|   device_info_plus: ^11.0.0 | ||||
|   file_picker: ^10.0.0 | ||||
|   animations: ^2.0.4 | ||||
|   android_package_installer: | ||||
|   android_package_installer: # TODO: See if PR will be accepted (dev may not be active), else remove this comment | ||||
|     git: | ||||
|       url: https://github.com/ImranR98/android_package_installer | ||||
|       ref: main | ||||
|   android_package_manager: ^0.7.0 | ||||
|   share_plus: ^10.0.0 | ||||
|   android_package_manager: # TODO: Make PR and switch to upstream | ||||
|     git: | ||||
|       url: https://github.com/ImranR98/android_package_manager | ||||
|       ref: master | ||||
|   share_plus: ^11.0.0 | ||||
|   sqflite: ^2.2.0+3 | ||||
|   easy_localization: ^3.0.1 | ||||
|   android_intent_plus: ^5.0.1 | ||||
| @@ -64,8 +65,12 @@ dependencies: | ||||
|   flutter_archive: ^6.0.0 | ||||
|   hsluv: ^1.1.3 | ||||
|   connectivity_plus: ^6.0.1 | ||||
|   shared_storage: ^0.8.0 | ||||
|   shared_storage: # TODO: Is this maintained? | ||||
|     git: | ||||
|       url: https://github.com/AlexBacich/shared-storage | ||||
|       ref: master | ||||
|   crypto: ^3.0.3 | ||||
|   bcrypt: ^1.1.3 | ||||
|   app_links: ^6.0.1 | ||||
|   background_fetch: ^1.2.1 | ||||
|   equations: ^5.0.2 | ||||
| @@ -76,11 +81,13 @@ dependencies: | ||||
|       ref: master | ||||
|   shizuku_apk_installer: | ||||
|     git: | ||||
|       url: https://github.com/re7gog/shizuku_apk_installer | ||||
|       url: https://github.com/wilver06w/shizuku_apk_installer | ||||
|       ref: master | ||||
|  | ||||
|   markdown: any | ||||
|   flutter_typeahead: ^5.2.0 | ||||
|   battery_plus: ^6.1.0 | ||||
|   flutter_charset_detector: ^5.0.0 | ||||
| dev_dependencies: | ||||
|   flutter_test: | ||||
|     sdk: flutter | ||||
| @@ -109,7 +116,7 @@ flutter: | ||||
|   uses-material-design: true | ||||
|  | ||||
|   # To add assets to your application, add an assets section, like this: | ||||
|   # - assets: | ||||
|   # assets: | ||||
|   #   - images/a_dot_burr.jpeg | ||||
|   #   - images/a_dot_ham.jpeg | ||||
|  | ||||
| @@ -119,10 +126,10 @@ flutter: | ||||
|     - assets/ca/ | ||||
|  | ||||
|   # An image asset can refer to one or more resolution-specific "variants", see | ||||
|   # https://flutter.dev/assets-and-images/#resolution-aware | ||||
|   # https://flutter.dev/to/resolution-aware-images | ||||
|  | ||||
|   # For details regarding adding assets from package dependencies, see | ||||
|   # https://flutter.dev/assets-and-images/#from-packages | ||||
|   # https://flutter.dev/to/asset-from-package | ||||
|  | ||||
|   # To add custom fonts to your application, add a fonts section here, | ||||
|   # in this "flutter" section. Each entry in this list should have a | ||||
| @@ -142,9 +149,9 @@ flutter: | ||||
|   #         weight: 700 | ||||
|   # | ||||
|   # For details regarding fonts from package dependencies, | ||||
|   # see https://flutter.dev/custom-fonts/#from-packages | ||||
|   # see https://flutter.dev/to/font-from-package | ||||
|  | ||||
|   fonts: | ||||
|       - family: Wix-Madefor-Display | ||||
|       - family: Montserrat | ||||
|         fonts: | ||||
|           - asset: assets/fonts/WixMadeforDisplay-Regular.otf | ||||
|           - asset: assets/fonts/Montserrat-Regular.ttf | ||||
							
								
								
									
										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 | ||||