Compare commits
	
		
			242 Commits
		
	
	
		
			v0.9.6-bet
			...
			v0.11.25-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 49affd1bd4 | ||
|  | 202ce4f0d5 | ||
|  | 361a3e1bc2 | ||
|  | f33a26d4f4 | ||
|  | 7aaf56ec8c | ||
|  | ed120016d9 | ||
|  | e8cbac8657 | ||
|  | b66c13d319 | ||
|  | 782d055bc3 | ||
|  | d557746965 | ||
|  | e6b05d50b9 | ||
|  | dea635fa6a | ||
|  | 682026ed0a | ||
|  | 9fe8a200ef | ||
|  | 210100da2b | ||
|  | d52660235b | ||
|  | e386b5ab8a | ||
|  | abf7be222d | ||
|  | 4c5b9304c0 | ||
|  | 4cfe6af044 | ||
|  | 3f0c4068dd | ||
|  | 7981ca29c5 | ||
|  | 187efa8fc5 | ||
|  | cd27ff7f2d | ||
|  | 6f6a25511b | ||
|  | 4e17bbcfd1 | ||
|  | 814e269d1d | ||
|  | 6b7d962b87 | ||
|  | 9fba747802 | ||
|  | c7cd35b6a1 | ||
|  | a8a3fce33a | ||
|  | 3a38cedcf5 | ||
|  | 69ccefcf1a | ||
|  | d3932f317d | ||
|  | 895deeead5 | ||
|  | 4c04af3868 | ||
|  | 07c490bb0e | ||
|  | a081d553bb | ||
|  | 3bc5837999 | ||
|  | 9fbe524818 | ||
|  | c21a9d7292 | ||
|  | 9c6068b270 | ||
|  | cd86d6112b | ||
|  | 1112c79c14 | ||
|  | 08555bac75 | ||
|  | 6db31e2b24 | ||
|  | 48d2532323 | ||
|  | f1fc43a3e7 | ||
|  | 280827d8ec | ||
|  | 05ee0f9c48 | ||
|  | ef06ae289e | ||
|  | bd0e322465 | ||
|  | a93a2411fa | ||
|  | 26e6eef72e | ||
|  | e49a6e311b | ||
|  | 53d3397651 | ||
|  | fe540f5e61 | ||
|  | 234374224b | ||
|  | 83390f648a | ||
|  | 1143b6a546 | ||
|  | 0f3e029312 | ||
|  | c0120f4e40 | ||
|  | a0199f0ceb | ||
|  | 0528936e5a | ||
|  | 4de98b2f36 | ||
|  | dfb5f5596c | ||
|  | 2e706aac47 | ||
|  | 24a600e595 | ||
|  | 1596a44ec5 | ||
|  | 9ee2be76ca | ||
|  | 83b770294d | ||
|  | 2679d5a1aa | ||
|  | e49c09c0ff | ||
|  | c9318ef2b5 | ||
|  | 2e88c8eede | ||
|  | 8648c1bea7 | ||
|  | b22e2bab0c | ||
|  | 57f7bf44c2 | ||
|  | ce526d8d26 | ||
|  | 5f3eeb9971 | ||
|  | e67a6b8627 | ||
|  | f8e99bb0cb | ||
|  | 09b5dd41d3 | ||
|  | b1bd36408c | ||
|  | 54d8dff32f | ||
|  | 7b1416e28e | ||
|  | 926e7b89ce | ||
|  | 43d4f89d61 | ||
|  | 2190da162d | ||
|  | f10bb5ac91 | ||
|  | 8e52f9666d | ||
|  | a8a47bb153 | ||
|  | 728dafcc28 | ||
|  | d53b21906c | ||
|  | d6dcac0f97 | ||
|  | dae5a67652 | ||
|  | 508fcccec9 | ||
|  | cc8a4c3760 | ||
|  | 814e2b7306 | ||
|  | 2e159c9886 | ||
|  | b82d28f2a7 | ||
|  | 3c61735706 | ||
|  | a2879f5bfa | ||
|  | b57f023739 | ||
|  | c376a7abec | ||
|  | 31c6cc3f6f | ||
|  | 8de8438aeb | ||
|  | 2b0225dd5b | ||
|  | f6af3a7998 | ||
|  | bd29d7bc10 | ||
|  | ffb3516a4b | ||
|  | 6a5e7942ee | ||
|  | 859158e84a | ||
|  | 435116e10b | ||
|  | a788d9d7cd | ||
|  | 4be3478b97 | ||
|  | fe0126095a | ||
|  | d5fdf28a98 | ||
|  | f06d245e20 | ||
|  | 2b4f94b407 | ||
|  | 5f7e342e6b | ||
|  | 191776d0d5 | ||
|  | ea81b0e66e | ||
|  | 86131ae3ce | ||
|  | 64ded1d720 | ||
|  | a11c2f1d37 | ||
|  | 890787f87f | ||
|  | c5ff1de950 | ||
|  | 56658abd60 | ||
|  | b60622e2cb | ||
|  | e149f0b225 | ||
|  | d9729f08c0 | ||
|  | eda5c1bac6 | ||
|  | 5574ea870b | ||
|  | 9f03234ac1 | ||
|  | b2503dd43d | ||
|  | e01ca704bc | ||
|  | 6aa4ace8e2 | ||
|  | d762467a31 | ||
|  | b07cce8ecd | ||
|  | 8002a946b2 | ||
|  | fd9aebc5b2 | ||
|  | 1be38d361f | ||
|  | 32c40ae7b3 | ||
|  | 07223d81c7 | ||
|  | 78baee7265 | ||
|  | 348c33dfe9 | ||
|  | c408d70ae6 | ||
|  | 3ae4e7cc8a | ||
|  | dab0f2bb72 | ||
|  | 4baf6bcd3b | ||
|  | db4517aa13 | ||
|  | 55d4d1f978 | ||
|  | f89ac5965f | ||
|  | d5ebaa161f | ||
|  | a4c014a8bf | ||
|  | bbaa42fb01 | ||
|  | 4fe311bc03 | ||
|  | ea68b97ff7 | ||
|  | 6e0f6b528e | ||
|  | a2c227931e | ||
|  | 15ad3bb439 | ||
|  | b03d7fba1a | ||
|  | 31c491d7c5 | ||
|  | 71c80f11f5 | ||
|  | eef4d33431 | ||
|  | d56342e907 | ||
|  | c72c0fdb57 | ||
|  | ffe29009ed | ||
|  | 60e3b68ebd | ||
|  | ee4d0f259f | ||
|  | 0ecfbef0a0 | ||
|  | 1b60e75ca7 | ||
|  | abcfa389e8 | ||
|  | a64bd67ef1 | ||
|  | 4252c2711b | ||
|  | 52913b0450 | ||
|  | 427b0ed8d2 | ||
|  | a85d6d4f08 | ||
|  | 05f712603c | ||
|  | fa2a80e34c | ||
|  | f43e5a2ff1 | ||
|  | b72aa8273e | ||
|  | 520f186e4a | ||
|  | e1e97672cf | ||
|  | 1494bcd013 | ||
|  | 3457a0a12f | ||
|  | b165400a6e | ||
|  | c47bf937f1 | ||
|  | 2e19a8c04c | ||
|  | 05d4da86ec | ||
|  | e9d1b04d54 | ||
|  | cff5334c25 | ||
|  | a55346fc22 | ||
|  | 885df678e5 | ||
|  | bf7b0c5702 | ||
|  | 2972da4609 | ||
|  | b8567af98e | ||
|  | ea62c68b40 | ||
|  | 08a5af0449 | ||
|  | 36f327c16e | ||
|  | 768213cb34 | ||
|  | e888fb7120 | ||
|  | 1fb68dd674 | ||
|  | 5c4bb8f84c | ||
|  | 1c8e759494 | ||
|  | 081c2a07d2 | ||
|  | 02751fe8fa | ||
|  | 95f3362a84 | ||
|  | b68cf5a1be | ||
|  | 4eb7499591 | ||
|  | 98fafe2aa4 | ||
|  | 9bac74aadd | ||
|  | 0a93117bf0 | ||
|  | 451cc41c45 | ||
|  | 3b449d0982 | ||
|  | 1863f55372 | ||
|  | 0c4b8ac79d | ||
|  | e287087753 | ||
|  | 82bcc46d42 | ||
|  | 1f26188ec6 | ||
|  | 794c3e1a81 | ||
|  | 16369b4adf | ||
|  | 8f16f745be | ||
|  | 8ddeb3d776 | ||
|  | 21cf9c98d9 | ||
|  | 358f910d19 | ||
|  | 7a3d74bd05 | ||
|  | 6f27f64699 | ||
|  | 3341fecb68 | ||
|  | d3bce63ca4 | ||
|  | 8aa8b6b698 | ||
|  | 3d6c9bbf98 | ||
|  | 7af0a8628c | ||
|  | 4573ce6bcf | ||
|  | e29d38fa32 | ||
|  | dc82431235 | ||
|  | 424b0028bf | ||
|  | 46fba9e0a4 | ||
|  | b40be7569b | ||
|  | a173be11eb | ||
|  | 0c97b25d99 | 
							
								
								
									
										19
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,4 +1,4 @@ | |||||||
| #  Obtainium | #  Obtainium | ||||||
|  |  | ||||||
| Get Android App Updates Directly From the Source. | Get Android App Updates Directly From the Source. | ||||||
|  |  | ||||||
| @@ -9,6 +9,7 @@ Motivation: [Side Of Burritos - You should use this instead of F-Droid | How to | |||||||
| Currently supported App sources: | Currently supported App sources: | ||||||
| - [GitHub](https://github.com/) | - [GitHub](https://github.com/) | ||||||
| - [GitLab](https://gitlab.com/) | - [GitLab](https://gitlab.com/) | ||||||
|  | - [Codeberg](https://codeberg.org/) | ||||||
| - [F-Droid](https://f-droid.org/) | - [F-Droid](https://f-droid.org/) | ||||||
| - [IzzyOnDroid](https://android.izzysoft.de/) | - [IzzyOnDroid](https://android.izzysoft.de/) | ||||||
| - [Mullvad](https://mullvad.net/en/) | - [Mullvad](https://mullvad.net/en/) | ||||||
| @@ -18,6 +19,20 @@ Currently supported App sources: | |||||||
| - Third Party F-Droid Repos | - Third Party F-Droid Repos | ||||||
|   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` |   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` | ||||||
| - [Steam](https://store.steampowered.com/mobile) | - [Steam](https://store.steampowered.com/mobile) | ||||||
|  | - [Telegram App](https://telegram.org) | ||||||
|  | - [VLC](https://www.videolan.org/vlc/download-android.html) | ||||||
|  | - [Neutron Code](https://neutroncode.com) | ||||||
|  | - "HTML" (Fallback) | ||||||
|  |   - Any other URL that returns an HTML page with links to APK files (if multiple, the last file alphabetically is picked) | ||||||
|  |  | ||||||
|  | ## Installation | ||||||
|  |  | ||||||
|  | [<img src="https://github.com/machiav3lli/oandbackupx/blob/034b226cea5c1b30eb4f6a6f313e4dadcbb0ece4/badge_github.png" | ||||||
|  |     alt="Get it on GitHub" | ||||||
|  |     height="80">](https://github.com/ImranR98/Obtainium/releases) | ||||||
|  | [<img src="https://gitlab.com/IzzyOnDroid/repo/-/raw/master/assets/IzzyOnDroid.png" | ||||||
|  |      alt="Get it on IzzyOnDroid" | ||||||
|  |      height="80">](https://apt.izzysoft.de/fdroid/index/apk/dev.imranr.obtainium) | ||||||
|  |  | ||||||
| ## Limitations | ## Limitations | ||||||
| - App installs happen asynchronously and the success/failure of an install cannot be determined directly. This results in install statuses and versions sometimes being out of sync with the OS until the next launch or until the problem is manually corrected. | - App installs happen asynchronously and the success/failure of an install cannot be determined directly. This results in install statuses and versions sometimes being out of sync with the OS until the next launch or until the problem is manually corrected. | ||||||
| @@ -28,4 +43,4 @@ Currently supported App sources: | |||||||
|  |  | ||||||
| | <img src="./assets/screenshots/1.apps.png" alt="Apps Page" /> | <img src="./assets/screenshots/2.dark_theme.png" alt="Dark Theme" />           | <img src="./assets/screenshots/3.material_you.png" alt="Material You" />    | | | <img src="./assets/screenshots/1.apps.png" alt="Apps Page" /> | <img src="./assets/screenshots/2.dark_theme.png" alt="Dark Theme" />           | <img src="./assets/screenshots/3.material_you.png" alt="Material You" />    | | ||||||
| | ------------------------------------------------------ | ----------------------------------------------------------------------- | -------------------------------------------------------------------- | | | ------------------------------------------------------ | ----------------------------------------------------------------------- | -------------------------------------------------------------------- | | ||||||
| | <img src="./assets/screenshots/4.app.png" alt="App Page" />   | <img src="./assets/screenshots/5.apk_picker.png" alt="Multiple APK Support" /> | <img src="./assets/screenshots/6.apk_install.png" alt="App Installation" /> | | | <img src="./assets/screenshots/4.app.png" alt="App Page" />   | <img src="./assets/screenshots/5.app_opts.png" alt="App Options" /> | <img src="./assets/screenshots/6.app_webview.png" alt="App Web View" /> | | ||||||
|   | |||||||
| @@ -3,7 +3,8 @@ | |||||||
|     <application |     <application | ||||||
|         android:label="Obtainium" |         android:label="Obtainium" | ||||||
|         android:name="${applicationName}" |         android:name="${applicationName}" | ||||||
|         android:icon="@mipmap/ic_launcher"> |         android:icon="@mipmap/ic_launcher" | ||||||
|  |         android:requestLegacyExternalStorage="true"> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".MainActivity" |             android:name=".MainActivity" | ||||||
|             android:exported="true" |             android:exported="true" | ||||||
| @@ -51,4 +52,8 @@ | |||||||
|     <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/> |     <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/> | ||||||
|     <uses-permission android:name="android.permission.WAKE_LOCK"/> |     <uses-permission android:name="android.permission.WAKE_LOCK"/> | ||||||
|     <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/> |     <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/> | ||||||
|  |     <uses-permission android:name="android.permission.REQUEST_DELETE_PACKAGES" /> | ||||||
|  |     <uses-permission | ||||||
|  |         android:name="android.permission.WRITE_EXTERNAL_STORAGE" | ||||||
|  |         android:maxSdkVersion="29"/> | ||||||
| </manifest> | </manifest> | ||||||
| Before Width: | Height: | Size: 4.8 KiB | 
| Before Width: | Height: | Size: 2.8 KiB | 
| Before Width: | Height: | Size: 7.7 KiB | 
| Before Width: | Height: | Size: 15 KiB | 
| Before Width: | Height: | Size: 25 KiB | 
							
								
								
									
										46
									
								
								android/app/src/main/res/drawable/ic_launcher_foreground.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,46 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" xmlns:aapt="http://schemas.android.com/aapt" | ||||||
|  |     android:viewportWidth="142.129" | ||||||
|  |     android:viewportHeight="142.129" | ||||||
|  |     android:width="503.6066dp" | ||||||
|  |     android:height="503.6066dp"> | ||||||
|  |     <group | ||||||
|  |         android:translateX="-30.39437" | ||||||
|  |         android:translateY="-54.68043"> | ||||||
|  |         <path | ||||||
|  |             android:pathData="M109.8808 153.22596c-0.73146 -0.38777 -5.00657 -2.75679 -25.032416 -13.87149 -5.57273 -3.09297 -10.93823 -6.06723 -11.92332 -6.60948 -2.23728 -1.23152 -2.58105 -1.53456 -2.58105 -2.27528 0 -0.3879 0.89293 -2.87231 2.98561 -8.30689 1.64209 -4.2644 3.09426 -8.0014 3.22705 -8.30444 0.3024 -0.69008 0.78972 -1.27621 1.26573 -1.52236 0.44558 -0.23042 11.58052 -4.29685 12.14814 -4.43644 0.61355 -0.1509 1.1428 0.13977 1.45487 0.79901 0.14976 0.31638 0.77213 1.94934 1.38303 3.6288 0.6109 1.67945 1.52036 4.16275 2.02104 5.51844 1.14709 3.10604 1.18992 3.54589 0.3912 4.01771 -0.2117 0.12505 -1.58874 0.66539 -3.06009 1.20075 -1.47136 0.53536 -2.87533 1.08982 -3.11993 1.23213 -0.56422 0.32826 -0.64913 0.83523 -0.20815 1.24273 0.17523 0.16193 3.00434 1.77571 6.28691 3.58618 9.174936 5.06035 8.665596 4.83136 9.277626 4.17097 0.29987 -0.32356 5.78141 -14.266 6.09596 -15.50521 0.1344 -0.5295 0.11969 -0.60308 -0.16695 -0.83519 -0.39165 -0.31714 -0.335 -0.33071 -3.93797 0.9431 -3.56937 1.26192 -3.90926 1.28864 -4.38744 0.34488 -0.25108 -0.49556 -4.095796 -11.05481 -4.334456 -11.90432 -0.15438 -0.5495 0.0344 -1.0717 0.49701 -1.37482 0.19228 -0.12598 2.990116 -1.19935 6.217406 -2.38526 4.78924 -1.75986 6.0081 -2.15842 6.63117 -2.16837 0.8037 -0.0128 0.90917 0.0424 15.64514 8.19599 1.02104 0.56495 1.56579 1.15961 1.56579 1.70925 0 0.21814 -3.6538 9.91011 -8.11957 21.53771 -6.2982 16.39877 -8.19916 21.21114 -8.4744 21.45338 -0.46789 0.41179 -0.8512 0.39392 -1.74794 -0.0815z" | ||||||
|  |             android:strokeWidth="0.139"> | ||||||
|  |             <aapt:attr | ||||||
|  |                 name="android:fillColor"> | ||||||
|  |                 <gradient | ||||||
|  |                     android:startX="76.74697" | ||||||
|  |                     android:startY="113.4246" | ||||||
|  |                     android:endX="110.6445" | ||||||
|  |                     android:endY="152.5006" | ||||||
|  |                     android:tileMode="clamp"> | ||||||
|  |                     <item | ||||||
|  |                         android:color="#9B58DC" | ||||||
|  |                         android:offset="0" /> | ||||||
|  |                     <item | ||||||
|  |                         android:color="#321C92" | ||||||
|  |                         android:offset="1" /> | ||||||
|  |                 </gradient> | ||||||
|  |             </aapt:attr> | ||||||
|  |             <aapt:attr | ||||||
|  |                 name="android:strokeColor"> | ||||||
|  |                 <gradient | ||||||
|  |                     android:startX="76.74697" | ||||||
|  |                     android:startY="113.4246" | ||||||
|  |                     android:endX="110.6445" | ||||||
|  |                     android:endY="152.5006" | ||||||
|  |                     android:tileMode="clamp"> | ||||||
|  |                     <item | ||||||
|  |                         android:color="#9B58DC" | ||||||
|  |                         android:offset="0" /> | ||||||
|  |                     <item | ||||||
|  |                         android:color="#321C92" | ||||||
|  |                         android:offset="1" /> | ||||||
|  |                 </gradient> | ||||||
|  |             </aapt:attr> | ||||||
|  |         </path> | ||||||
|  |     </group> | ||||||
|  | </vector> | ||||||
| Before Width: | Height: | Size: 7.6 KiB After Width: | Height: | Size: 8.1 KiB | 
| @@ -2,4 +2,5 @@ | |||||||
| <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|   <background android:drawable="@color/ic_launcher_background"/> |   <background android:drawable="@color/ic_launcher_background"/> | ||||||
|   <foreground android:drawable="@drawable/ic_launcher_foreground"/> |   <foreground android:drawable="@drawable/ic_launcher_foreground"/> | ||||||
|  |   <monochrome android:drawable="@drawable/ic_launcher_foreground"/> | ||||||
| </adaptive-icon> | </adaptive-icon> | ||||||
|   | |||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|  |   <background android:drawable="@color/ic_launcher_background"/> | ||||||
|  |   <foreground android:drawable="@drawable/ic_launcher_foreground"/> | ||||||
|  |   <monochrome android:drawable="@drawable/ic_launcher_foreground"/> | ||||||
|  | </adaptive-icon> | ||||||
| Before Width: | Height: | Size: 1.7 KiB | 
| Before Width: | Height: | Size: 1.1 KiB | 
| Before Width: | Height: | Size: 2.4 KiB | 
| Before Width: | Height: | Size: 3.9 KiB | 
| Before Width: | Height: | Size: 6.2 KiB | 
							
								
								
									
										30
									
								
								assets/ca/lets-encrypt-r3.pem
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,30 @@ | |||||||
|  | -----BEGIN CERTIFICATE----- | ||||||
|  | MIIFFjCCAv6gAwIBAgIRAJErCErPDBinU/bWLiWnX1owDQYJKoZIhvcNAQELBQAw | ||||||
|  | TzELMAkGA1UEBhMCVVMxKTAnBgNVBAoTIEludGVybmV0IFNlY3VyaXR5IFJlc2Vh | ||||||
|  | cmNoIEdyb3VwMRUwEwYDVQQDEwxJU1JHIFJvb3QgWDEwHhcNMjAwOTA0MDAwMDAw | ||||||
|  | WhcNMjUwOTE1MTYwMDAwWjAyMQswCQYDVQQGEwJVUzEWMBQGA1UEChMNTGV0J3Mg | ||||||
|  | RW5jcnlwdDELMAkGA1UEAxMCUjMwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEK | ||||||
|  | AoIBAQC7AhUozPaglNMPEuyNVZLD+ILxmaZ6QoinXSaqtSu5xUyxr45r+XXIo9cP | ||||||
|  | R5QUVTVXjJ6oojkZ9YI8QqlObvU7wy7bjcCwXPNZOOftz2nwWgsbvsCUJCWH+jdx | ||||||
|  | sxPnHKzhm+/b5DtFUkWWqcFTzjTIUu61ru2P3mBw4qVUq7ZtDpelQDRrK9O8Zutm | ||||||
|  | NHz6a4uPVymZ+DAXXbpyb/uBxa3Shlg9F8fnCbvxK/eG3MHacV3URuPMrSXBiLxg | ||||||
|  | Z3Vms/EY96Jc5lP/Ooi2R6X/ExjqmAl3P51T+c8B5fWmcBcUr2Ok/5mzk53cU6cG | ||||||
|  | /kiFHaFpriV1uxPMUgP17VGhi9sVAgMBAAGjggEIMIIBBDAOBgNVHQ8BAf8EBAMC | ||||||
|  | AYYwHQYDVR0lBBYwFAYIKwYBBQUHAwIGCCsGAQUFBwMBMBIGA1UdEwEB/wQIMAYB | ||||||
|  | Af8CAQAwHQYDVR0OBBYEFBQusxe3WFbLrlAJQOYfr52LFMLGMB8GA1UdIwQYMBaA | ||||||
|  | FHm0WeZ7tuXkAXOACIjIGlj26ZtuMDIGCCsGAQUFBwEBBCYwJDAiBggrBgEFBQcw | ||||||
|  | AoYWaHR0cDovL3gxLmkubGVuY3Iub3JnLzAnBgNVHR8EIDAeMBygGqAYhhZodHRw | ||||||
|  | Oi8veDEuYy5sZW5jci5vcmcvMCIGA1UdIAQbMBkwCAYGZ4EMAQIBMA0GCysGAQQB | ||||||
|  | gt8TAQEBMA0GCSqGSIb3DQEBCwUAA4ICAQCFyk5HPqP3hUSFvNVneLKYY611TR6W | ||||||
|  | PTNlclQtgaDqw+34IL9fzLdwALduO/ZelN7kIJ+m74uyA+eitRY8kc607TkC53wl | ||||||
|  | ikfmZW4/RvTZ8M6UK+5UzhK8jCdLuMGYL6KvzXGRSgi3yLgjewQtCPkIVz6D2QQz | ||||||
|  | CkcheAmCJ8MqyJu5zlzyZMjAvnnAT45tRAxekrsu94sQ4egdRCnbWSDtY7kh+BIm | ||||||
|  | lJNXoB1lBMEKIq4QDUOXoRgffuDghje1WrG9ML+Hbisq/yFOGwXD9RiX8F6sw6W4 | ||||||
|  | avAuvDszue5L3sz85K+EC4Y/wFVDNvZo4TYXao6Z0f+lQKc0t8DQYzk1OXVu8rp2 | ||||||
|  | yJMC6alLbBfODALZvYH7n7do1AZls4I9d1P4jnkDrQoxB3UqQ9hVl3LEKQ73xF1O | ||||||
|  | yK5GhDDX8oVfGKF5u+decIsH4YaTw7mP3GFxJSqv3+0lUFJoi5Lc5da149p90Ids | ||||||
|  | hCExroL1+7mryIkXPeFM5TgO9r0rvZaBFOvV2z0gp35Z0+L4WPlbuEjN/lxPFin+ | ||||||
|  | HlUjr8gRsI3qfJOQFy/9rKIJR0Y/8Omwt/8oTWgy1mdeHmmjk7j1nYsvC9JSQ6Zv | ||||||
|  | MldlTTKB3zhThV1+XWYp6rjd5JW1zbVWEkLNxE7GJThEUG3szgBVGP7pSWTUTsqX | ||||||
|  | nLRbwHOoq7hHwg== | ||||||
|  | -----END CERTIFICATE----- | ||||||
| Before Width: | Height: | Size: 109 KiB | 
							
								
								
									
										78
									
								
								assets/graphics/icon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,78 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||||
|  | <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||||
|  |  | ||||||
|  | <svg | ||||||
|  |    width="142.12897mm" | ||||||
|  |    height="142.12897mm" | ||||||
|  |    viewBox="0 0 142.12897 142.12897" | ||||||
|  |    version="1.1" | ||||||
|  |    id="svg5" | ||||||
|  |    xml:space="preserve" | ||||||
|  |    inkscape:version="1.2.2 (b0a8486541, 2022-12-01)" | ||||||
|  |    sodipodi:docname="icon.svg" | ||||||
|  |    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||||
|  |    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||||
|  |    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||||
|  |    xmlns="http://www.w3.org/2000/svg" | ||||||
|  |    xmlns:svg="http://www.w3.org/2000/svg"><sodipodi:namedview | ||||||
|  |      id="namedview7" | ||||||
|  |      pagecolor="#ffffff" | ||||||
|  |      bordercolor="#000000" | ||||||
|  |      borderopacity="0.25" | ||||||
|  |      inkscape:showpageshadow="2" | ||||||
|  |      inkscape:pageopacity="0.0" | ||||||
|  |      inkscape:pagecheckerboard="0" | ||||||
|  |      inkscape:deskcolor="#d1d1d1" | ||||||
|  |      inkscape:document-units="mm" | ||||||
|  |      showgrid="false" | ||||||
|  |      inkscape:zoom="2.4175295" | ||||||
|  |      inkscape:cx="371.03994" | ||||||
|  |      inkscape:cy="273.62644" | ||||||
|  |      inkscape:window-width="2256" | ||||||
|  |      inkscape:window-height="1427" | ||||||
|  |      inkscape:window-x="0" | ||||||
|  |      inkscape:window-y="0" | ||||||
|  |      inkscape:window-maximized="1" | ||||||
|  |      inkscape:current-layer="layer1" /><defs | ||||||
|  |      id="defs2"><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        id="linearGradient3657"><stop | ||||||
|  |          style="stop-color:#9b58dc;stop-opacity:1;" | ||||||
|  |          offset="0" | ||||||
|  |          id="stop3653" /><stop | ||||||
|  |          style="stop-color:#321c92;stop-opacity:1;" | ||||||
|  |          offset="1" | ||||||
|  |          id="stop3655" /></linearGradient><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        id="linearGradient945"><stop | ||||||
|  |          style="stop-color:#9b58dc;stop-opacity:1;" | ||||||
|  |          offset="0" | ||||||
|  |          id="stop941" /><stop | ||||||
|  |          style="stop-color:#321c92;stop-opacity:1;" | ||||||
|  |          offset="1" | ||||||
|  |          id="stop943" /></linearGradient><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        xlink:href="#linearGradient945" | ||||||
|  |        id="linearGradient947" | ||||||
|  |        x1="76.787094" | ||||||
|  |        y1="113.40435" | ||||||
|  |        x2="110.68458" | ||||||
|  |        y2="152.48038" | ||||||
|  |        gradientUnits="userSpaceOnUse" | ||||||
|  |        gradientTransform="translate(-0.04012535,0.02025786)" /><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        xlink:href="#linearGradient3657" | ||||||
|  |        id="linearGradient3659" | ||||||
|  |        x1="76.787094" | ||||||
|  |        y1="113.40435" | ||||||
|  |        x2="110.68458" | ||||||
|  |        y2="152.48038" | ||||||
|  |        gradientUnits="userSpaceOnUse" | ||||||
|  |        gradientTransform="translate(-0.04012535,0.02025786)" /></defs><g | ||||||
|  |      inkscape:label="Layer 1" | ||||||
|  |      inkscape:groupmode="layer" | ||||||
|  |      id="layer1" | ||||||
|  |      transform="translate(-30.394373,-54.680428)"><path | ||||||
|  |        style="fill:url(#linearGradient3659);fill-opacity:1;stroke:url(#linearGradient947);stroke-width:0.139;stroke-dasharray:none" | ||||||
|  |        d="m 109.8808,153.22596 c -0.73146,-0.38777 -5.00657,-2.75679 -25.032416,-13.87149 -5.57273,-3.09297 -10.93823,-6.06723 -11.92332,-6.60948 -2.23728,-1.23152 -2.58105,-1.53456 -2.58105,-2.27528 0,-0.3879 0.89293,-2.87231 2.98561,-8.30689 1.64209,-4.2644 3.09426,-8.0014 3.22705,-8.30444 0.3024,-0.69008 0.78972,-1.27621 1.26573,-1.52236 0.44558,-0.23042 11.58052,-4.29685 12.14814,-4.43644 0.61355,-0.1509 1.1428,0.13977 1.45487,0.79901 0.14976,0.31638 0.77213,1.94934 1.38303,3.6288 0.6109,1.67945 1.52036,4.16275 2.02104,5.51844 1.14709,3.10604 1.18992,3.54589 0.3912,4.01771 -0.2117,0.12505 -1.58874,0.66539 -3.06009,1.20075 -1.47136,0.53536 -2.87533,1.08982 -3.11993,1.23213 -0.56422,0.32826 -0.64913,0.83523 -0.20815,1.24273 0.17523,0.16193 3.00434,1.77571 6.28691,3.58618 9.174936,5.06035 8.665596,4.83136 9.277626,4.17097 0.29987,-0.32356 5.78141,-14.266 6.09596,-15.50521 0.1344,-0.5295 0.11969,-0.60308 -0.16695,-0.83519 -0.39165,-0.31714 -0.335,-0.33071 -3.93797,0.9431 -3.56937,1.26192 -3.90926,1.28864 -4.38744,0.34488 -0.25108,-0.49556 -4.095796,-11.05481 -4.334456,-11.90432 -0.15438,-0.5495 0.0344,-1.0717 0.49701,-1.37482 0.19228,-0.12598 2.990116,-1.19935 6.217406,-2.38526 4.78924,-1.75986 6.0081,-2.15842 6.63117,-2.16837 0.8037,-0.0128 0.90917,0.0424 15.64514,8.19599 1.02104,0.56495 1.56579,1.15961 1.56579,1.70925 0,0.21814 -3.6538,9.91011 -8.11957,21.53771 -6.2982,16.39877 -8.19916,21.21114 -8.4744,21.45338 -0.46789,0.41179 -0.8512,0.39392 -1.74794,-0.0815 z" | ||||||
|  |        id="path239" /></g></svg> | ||||||
| After Width: | Height: | Size: 4.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/graphics/icon_small.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 7.6 KiB | 
| Before Width: | Height: | Size: 228 KiB After Width: | Height: | Size: 234 KiB | 
| Before Width: | Height: | Size: 162 KiB After Width: | Height: | Size: 238 KiB | 
| Before Width: | Height: | Size: 170 KiB After Width: | Height: | Size: 140 KiB | 
| Before Width: | Height: | Size: 146 KiB After Width: | Height: | Size: 139 KiB | 
| Before Width: | Height: | Size: 188 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/screenshots/5.app_opts.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 118 KiB | 
| Before Width: | Height: | Size: 192 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/screenshots/6.app_webview.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 262 KiB | 
| @@ -74,7 +74,6 @@ | |||||||
|     "changeX": "Ändern {}", |     "changeX": "Ändern {}", | ||||||
|     "installUpdateApps": "Apps installieren/aktualisieren", |     "installUpdateApps": "Apps installieren/aktualisieren", | ||||||
|     "installUpdateSelectedApps": "Ausgewählte Apps installieren/aktualisieren", |     "installUpdateSelectedApps": "Ausgewählte Apps installieren/aktualisieren", | ||||||
|     "onlyWorksWithNonEVDApps": "Funktioniert nur bei Apps, deren Installationsstatus nicht automatisch erkannt werden kann (ungewöhnlich).", |  | ||||||
|     "markXSelectedAppsAsUpdated": "Markiere {} ausgewählte Apps als aktuell?", |     "markXSelectedAppsAsUpdated": "Markiere {} ausgewählte Apps als aktuell?", | ||||||
|     "no": "Nein", |     "no": "Nein", | ||||||
|     "yes": "Ja", |     "yes": "Ja", | ||||||
| @@ -178,7 +177,6 @@ | |||||||
|     "installedVersionX": "Installierte Version: {}", |     "installedVersionX": "Installierte Version: {}", | ||||||
|     "lastUpdateCheckX": "Letzte Aktualisierungsprüfung: {}", |     "lastUpdateCheckX": "Letzte Aktualisierungsprüfung: {}", | ||||||
|     "remove": "Entfernen", |     "remove": "Entfernen", | ||||||
|     "removeAppQuestion": "App entfernen?", |  | ||||||
|     "yesMarkUpdated": "Ja, als aktualisiert markieren", |     "yesMarkUpdated": "Ja, als aktualisiert markieren", | ||||||
|     "fdroid": "F-Droid", |     "fdroid": "F-Droid", | ||||||
|     "appIdOrName": "App ID oder Name", |     "appIdOrName": "App ID oder Name", | ||||||
| @@ -209,6 +207,25 @@ | |||||||
|     "addCategory": "Kategorie hinzufügen", |     "addCategory": "Kategorie hinzufügen", | ||||||
|     "label": "Bezeichnung", |     "label": "Bezeichnung", | ||||||
|     "language": "Sprache", |     "language": "Sprache", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|  |     "storagePermissionDenied": "Speicherberechtigung verweigert", | ||||||
|  |     "selectedCategorizeWarning": "Dadurch werden alle bestehenden Kategorieeinstellungen für die ausgewählten Apps ersetzt.", | ||||||
|  |     "filterAPKsByRegEx": "APKs nach regulärem Ausdruck filtern", | ||||||
|  |     "removeFromObtainium": "Aus Obtainium entfernen", | ||||||
|  |     "uninstallFromDevice": "Vom Gerät deinstallieren", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Funktioniert nur bei Apps mit deaktivierter Versionserkennung.", | ||||||
|  |     "releaseDateAsVersion": "Veröffentlichungsdatum als Version 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", | ||||||
|  |     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", | ||||||
|  |     "versionDetection": "Versionserkennung", | ||||||
|  |     "standardVersionDetection": "Standardversionserkennung", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "App entfernen?", | ||||||
|  |         "other": "App entfernen?" | ||||||
|  |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", |         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", | ||||||
|         "other": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minuten erneut" |         "other": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minuten erneut" | ||||||
| @@ -253,4 +270,4 @@ | |||||||
|         "one": "{} und 1 weitere Anwendung wurden aktualisiert.", |         "one": "{} und 1 weitere Anwendung wurden aktualisiert.", | ||||||
|         "other": "{} und {} weitere Anwendungen wurden aktualisiert." |         "other": "{} und {} weitere Anwendungen wurden aktualisiert." | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -74,7 +74,6 @@ | |||||||
|     "changeX": "Change {}", |     "changeX": "Change {}", | ||||||
|     "installUpdateApps": "Install/Update Apps", |     "installUpdateApps": "Install/Update Apps", | ||||||
|     "installUpdateSelectedApps": "Install/Update Selected Apps", |     "installUpdateSelectedApps": "Install/Update Selected Apps", | ||||||
|     "onlyWorksWithNonEVDApps": "Only works for Apps whose install status cannot be automatically detected (uncommon).", |  | ||||||
|     "markXSelectedAppsAsUpdated": "Mark {} Selected Apps as Updated?", |     "markXSelectedAppsAsUpdated": "Mark {} Selected Apps as Updated?", | ||||||
|     "no": "No", |     "no": "No", | ||||||
|     "yes": "Yes", |     "yes": "Yes", | ||||||
| @@ -178,7 +177,6 @@ | |||||||
|     "installedVersionX": "Installed Version: {}", |     "installedVersionX": "Installed Version: {}", | ||||||
|     "lastUpdateCheckX": "Last Update Check: {}", |     "lastUpdateCheckX": "Last Update Check: {}", | ||||||
|     "remove": "Remove", |     "remove": "Remove", | ||||||
|     "removeAppQuestion": "Remove App?", |  | ||||||
|     "yesMarkUpdated": "Yes, Mark as Updated", |     "yesMarkUpdated": "Yes, Mark as Updated", | ||||||
|     "fdroid": "F-Droid", |     "fdroid": "F-Droid", | ||||||
|     "appIdOrName": "App ID or Name", |     "appIdOrName": "App ID or Name", | ||||||
| @@ -209,6 +207,25 @@ | |||||||
|     "addCategory": "Add Category", |     "addCategory": "Add Category", | ||||||
|     "label": "Label", |     "label": "Label", | ||||||
|     "language": "Language", |     "language": "Language", | ||||||
|  |     "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", | ||||||
|  |     "removeFromObtainium": "Remove from Obtainium", | ||||||
|  |     "uninstallFromDevice": "Uninstall from Device", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.", | ||||||
|  |     "releaseDateAsVersion": "Use Release Date as Version", | ||||||
|  |     "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)", | ||||||
|  |     "versionDetection": "Version Detection", | ||||||
|  |     "standardVersionDetection": "Standard version detection", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "Remove App?", | ||||||
|  |         "other": "Remove Apps?" | ||||||
|  |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Too many requests (rate limited) - try again in {} minute", |         "one": "Too many requests (rate limited) - try again in {} minute", | ||||||
|         "other": "Too many requests (rate limited) - try again in {} minutes" |         "other": "Too many requests (rate limited) - try again in {} minutes" | ||||||
| @@ -253,4 +270,4 @@ | |||||||
|         "one": "{} and 1 more app were updated.", |         "one": "{} and 1 more app were updated.", | ||||||
|         "other": "{} and {} more apps were updated." |         "other": "{} and {} more apps were updated." | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										273
									
								
								assets/translations/fa.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,273 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "آدرس اینترنتی برنامه {} معتبر نیست", | ||||||
|  |     "noReleaseFound": "نسخه مناسبی پیدا نشد", | ||||||
|  |     "noVersionFound": "نمی توان نسخه منتشر شده را تعیین کرد", | ||||||
|  |     "urlMatchesNoSource": "آدرس اینترنتی با منبع شناخته شده مطابقت ندارد", | ||||||
|  |     "cantInstallOlderVersion": "نمی توان نسخه قدیمی یک برنامه را نصب کرد", | ||||||
|  |     "appIdMismatch": "شناسه بسته دانلود شده با شناسه برنامه موجود مطابقت ندارد", | ||||||
|  |     "functionNotImplemented": "این کلاس این تابع را پیاده سازی نکرده است", | ||||||
|  |     "placeholder": "نگهدارنده مکان", | ||||||
|  |     "someErrors": "برخی از خطاها رخ داده است", | ||||||
|  |     "unexpectedError": "خطای غیرمنتظره", | ||||||
|  |     "ok": "باشه", | ||||||
|  |     "and": "و", | ||||||
|  |     "startedBgUpdateTask": "شروع بررسی بروزرسانی BG", | ||||||
|  |     "bgUpdateIgnoreAfterIs": "نادیده گرفتن بروزرسانی BG بعد از {} است", | ||||||
|  |     "startedActualBGUpdateCheck": "بررسی بهروزرسانی واقعی BG آغاز شد", | ||||||
|  |     "bgUpdateTaskFinished": "کار بررسی بهروزرسانی BG تمام شد", | ||||||
|  |     "firstRun": "این اولین اجرای Obtainium است", | ||||||
|  |     "settingUpdateCheckIntervalTo": "تنظیم فاصله بهروزرسانی روی {}", | ||||||
|  |     "githubPATLabel": "توکن دسترسی شخصی گیت هاب(محدودیت نرخ را افزایش میدهد)", | ||||||
|  |     "githubPATHint": "PAT باید در این قالب باشد: username:token", | ||||||
|  |     "githubPATFormat": "username:token", | ||||||
|  |     "githubPATLinkText": "درباره گیتهاب PATs", | ||||||
|  |     "includePrereleases": "شامل نسخه های اولیه", | ||||||
|  |     "fallbackToOlderReleases": "بازگشت به نسخه های قدیمی تر", | ||||||
|  |     "filterReleaseTitlesByRegEx": "عناوین انتشار را با بیان منظم فیلتر کنید", | ||||||
|  |     "invalidRegEx": "عبارت منظم نامعتبر است", | ||||||
|  |     "noDescription": "بدون توضیحات", | ||||||
|  |     "cancel": "لغو", | ||||||
|  |     "continue": "ادامه دهید", | ||||||
|  |     "requiredInBrackets": "(ضروری)", | ||||||
|  |     "dropdownNoOptsError": "خطا: کشویی باید حداقل یک گزینه داشته باشد", | ||||||
|  |     "colour": "رنگ", | ||||||
|  |     "githubStarredRepos": "مخازن ستاره دار گیتهاب", | ||||||
|  |     "uname": "نام کاربری", | ||||||
|  |     "wrongArgNum": "تعداد آرگومان های ارائه شده اشتباه است", | ||||||
|  |     "xIsTrackOnly": "{} فقط ردیابی", | ||||||
|  |     "source": "منبع", | ||||||
|  |     "app": "برنامه", | ||||||
|  |     "appsFromSourceAreTrackOnly": "برنامههای این منبع «فقط ردیابی» هستند", | ||||||
|  |     "youPickedTrackOnly": "شما گزینه ی «فقط ردیابی» را انتخاب کرده اید", | ||||||
|  |     "trackOnlyAppDescription": "برنامه برای به روز رسانی ها ردیابی می شود، اما Obtainium قادر به دانلود یا نصب آن نخواهد بود.", | ||||||
|  |     "cancelled": "لغو شد", | ||||||
|  |     "appAlreadyAdded": "برنامه قبلاً اضافه شده است", | ||||||
|  |     "alreadyUpToDateQuestion": "برنامه از قبل به روز شده است؟", | ||||||
|  |     "addApp": "افزودن برنامه", | ||||||
|  |     "appSourceURL": "آدرس اینترنتی منبع برنامه", | ||||||
|  |     "error": "خطا", | ||||||
|  |     "add": "اضافه کردن", | ||||||
|  |     "searchSomeSourcesLabel": "جستجو (فقط برخی منابع)", | ||||||
|  |     "search": "جستجو کردن", | ||||||
|  |     "additionalOptsFor": "گزینه های اضافی برای {}", | ||||||
|  |     "supportedSourcesBelow": "منابع پشتیبانی شده:", | ||||||
|  |     "trackOnlyInBrackets": "«فقط ردیابی»", | ||||||
|  |     "searchableInBrackets": "(قابل جستجو)", | ||||||
|  |     "appsString": "برنامه ها", | ||||||
|  |     "noApps": "برنامه ای وجود ندارد", | ||||||
|  |     "noAppsForFilter": "برنامه ای برای فیلتر کردن وجود ندارد", | ||||||
|  |     "byX": "توسط {}", | ||||||
|  |     "percentProgress": "پیش رفتن: {}%", | ||||||
|  |     "pleaseWait": "لطفا صبر کنید", | ||||||
|  |     "updateAvailable": "بروزرسانی در دسترس", | ||||||
|  |     "estimateInBracketsShort": "(تخمین)", | ||||||
|  |     "notInstalled": "نصب نشده", | ||||||
|  |     "estimateInBrackets": "(تخمین زدن)", | ||||||
|  |     "selectAll": "انتخاب همه", | ||||||
|  |     "deselectN": "لغو انتخاب {}", | ||||||
|  |     "xWillBeRemovedButRemainInstalled": "{} از Obtainium حذف میشود اما روی دستگاه نصب میماند.", | ||||||
|  |     "removeSelectedAppsQuestion": "برنامه های انتخابی حذف شود؟", | ||||||
|  |     "removeSelectedApps": "حذف برنامه های انتخاب شده", | ||||||
|  |     "updateX": "به روز رسانی {}", | ||||||
|  |     "installX": "نصب {}", | ||||||
|  |     "markXTrackOnlyAsUpdated": "علامت {}\n(فقط ردیابی)\nبروز شده", | ||||||
|  |     "changeX": "تغییر دادن {}", | ||||||
|  |     "installUpdateApps": "نصب/بهروزرسانی برنامهها", | ||||||
|  |     "installUpdateSelectedApps": "برنامههای انتخابی را نصب/بهروزرسانی کنید", | ||||||
|  |     "markXSelectedAppsAsUpdated": "{} برنامه های انتخابی را به عنوان به روز علامت گذاری کنید؟", | ||||||
|  |     "no": "خیر", | ||||||
|  |     "yes": "بله", | ||||||
|  |     "markSelectedAppsUpdated": "برنامه های انتخاب شده را به عنوان به روز علامت گذاری کنید", | ||||||
|  |     "pinToTop": "پین به بالا", | ||||||
|  |     "unpinFromTop": "برداشتن پین از بالا", | ||||||
|  |     "resetInstallStatusForSelectedAppsQuestion": "وضعیت نصب برنامههای انتخابی بازنشانی شود؟", | ||||||
|  |     "installStatusOfXWillBeResetExplanation": "وضعیت نصب برنامههای انتخابشده بازنشانی میشود.\n\nاگر نسخه برنامه نشاندادهشده در Obtainium به دلیل بهروزرسانیهای ناموفق یا مشکلات دیگر نادرست باشد، میتواند کمک کند.", | ||||||
|  |     "shareSelectedAppURLs": "اشتراک گذاری آدرس اینترنتی برنامه های انتخاب شده", | ||||||
|  |     "resetInstallStatus": "بازنشانی وضعیت نصب", | ||||||
|  |     "more": "بیشتر", | ||||||
|  |     "removeOutdatedFilter": "فیلتر برنامه قدیمی را حذف کنید", | ||||||
|  |     "showOutdatedOnly": "فقط برنامه های قدیمی را نشان دهید", | ||||||
|  |     "filter": "فیلتر", | ||||||
|  |     "filterActive": "فیلتر *", | ||||||
|  |     "filterApps": "فیلتر کردن برنامه ها", | ||||||
|  |     "appName": "نام برنامه", | ||||||
|  |     "author": "سازنده", | ||||||
|  |     "upToDateApps": "برنامه های به روز", | ||||||
|  |     "nonInstalledApps": "برنامه های نصب نشده", | ||||||
|  |     "importExport": "وادر کردن/صادر کردن", | ||||||
|  |     "settings": "تنظیمات", | ||||||
|  |     "exportedTo": "صادر کردن به{}", | ||||||
|  |     "obtainiumExport": "صادرکردن Obtainium", | ||||||
|  |     "invalidInput": "ورودی نامعتبر", | ||||||
|  |     "importedX": "وارد شده {}", | ||||||
|  |     "obtainiumImport": "واردکردن Obtainium", | ||||||
|  |     "importFromURLList": "وارد کردن از فهرست آدرس اینترنتی", | ||||||
|  |     "searchQuery": "جستجوی سوال", | ||||||
|  |     "appURLList": "فهرست آدرس اینترنتی برنامه", | ||||||
|  |     "line": "خط", | ||||||
|  |     "searchX": "جستجو {}", | ||||||
|  |     "noResults": "نتیجه ای پیدا نشد", | ||||||
|  |     "importX": "وارد کردن {}", | ||||||
|  |     "importedAppsIdDisclaimer": "ممکن است برنامههای وارد شده به اشتباه بهعنوان \"نصب نشده\" نشان داده شوند.\nبرای رفع این مشکل، آنها را دوباره از طریق Obtainium نصب کنید.\nاین نباید روی دادههای برنامه تأثیر بگذارد.\n\nفقط بر روی آدرس اینترنتی و روشهای وارد کردن شخص ثالث تأثیر میگذارد.", | ||||||
|  |     "importErrors": "خطاهای وارد کردن", | ||||||
|  |     "importedXOfYApps": "{} از {} برنامه وارد شد.", | ||||||
|  |     "followingURLsHadErrors": "آدرس های اینترنتی زیر دارای خطا بودند:", | ||||||
|  |     "okay": "باشه", | ||||||
|  |     "selectURL": "آدرس اینترنتی انتخاب شده", | ||||||
|  |     "selectURLs": "آدرس های اینترنتی انتخاب شده", | ||||||
|  |     "pick": "انتخاب", | ||||||
|  |     "theme": "تم", | ||||||
|  |     "dark": "تاریک", | ||||||
|  |     "light": "روشن", | ||||||
|  |     "followSystem": "هماهنگ با سیستم", | ||||||
|  |     "obtainium": "Obtainium", | ||||||
|  |     "materialYou": "Material You", | ||||||
|  |     "appSortBy": "مرتب سازی برنامه بر اساس", | ||||||
|  |     "authorName": "سازنده/اسم", | ||||||
|  |     "nameAuthor": "اسم/سازنده", | ||||||
|  |     "asAdded": "همانطور که اضافه شد", | ||||||
|  |     "appSortOrder": "ترتیب مرتب سازی برنامه", | ||||||
|  |     "ascending": "صعودی", | ||||||
|  |     "descending": "نزولی", | ||||||
|  |     "bgUpdateCheckInterval": "فاصله بررسی بهروزرسانی در پسزمینه", | ||||||
|  |     "neverManualOnly": "هرگز - فقط دستی", | ||||||
|  |     "appearance": "ظاهر", | ||||||
|  |     "showWebInAppView": "نمایش صفحه وب منبع در نمای برنامه", | ||||||
|  |     "pinUpdates": "بهروزرسانیها را به نمای بالای برنامهها پین کنید", | ||||||
|  |     "updates": "به روز رسانی ها", | ||||||
|  |     "sourceSpecific": "منبع خاص", | ||||||
|  |     "appSource": "منبع برنامه", | ||||||
|  |     "noLogs": "بدون گزارش", | ||||||
|  |     "appLogs": "گزارش های برنامه", | ||||||
|  |     "close": "بستن", | ||||||
|  |     "share": "اشتراک گذاری", | ||||||
|  |     "appNotFound": "برنامه پیدا نشد", | ||||||
|  |     "obtainiumExportHyphenatedLowercase": "صادر کردن-obtainium", | ||||||
|  |     "pickAnAPK": "یک APK انتخاب کنید", | ||||||
|  |     "appHasMoreThanOnePackage": "{} بیش از یک بسته دارد:", | ||||||
|  |     "deviceSupportsXArch": "دستگاه شما از معماری پردازنده {} پشتیبانی میکند", | ||||||
|  |     "deviceSupportsFollowingArchs": "دستگاه شما از معماری های پردازنده زیر پشتیبانی می کند:", | ||||||
|  |     "warning": "اخطار", | ||||||
|  |     "sourceIsXButPackageFromYPrompt": "منبع برنامه \"{}\" است اما بسته انتشار از \"{}\" آمده است. ادامه هید؟", | ||||||
|  |     "updatesAvailable": "بروزرسانی در دسترس ", | ||||||
|  |     "updatesAvailableNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی برای یک یا چند برنامه ردیابی شده توسط Obtainium در دسترس است", | ||||||
|  |     "noNewUpdates": "به روز رسانی جدیدی وجود ندارد.", | ||||||
|  |     "xHasAnUpdate": "{} یک به روز رسانی دارد.", | ||||||
|  |     "appsUpdated": "برنامه ها به روز شدند", | ||||||
|  |     "appsUpdatedNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی یک یا چند برنامه در پس زمینه اعمال شده است", | ||||||
|  |     "xWasUpdatedToY": "{} به {} به روز شد.", | ||||||
|  |     "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": "شناسه یا نام برنامه", | ||||||
|  |     "appWithIdOrNameNotFound": "هیچ برنامه ای با آن شناسه یا نام یافت نشد", | ||||||
|  |     "reposHaveMultipleApps": "مخازن ممکن است شامل چندین برنامه باشد", | ||||||
|  |     "fdroidThirdPartyRepo": "مخازن شخص ثالث F-Droid", | ||||||
|  |     "steam": "Steam", | ||||||
|  |     "steamMobile": "Steam Mobile", | ||||||
|  |     "steamChat": "Steam Chat", | ||||||
|  |     "install": "نصب", | ||||||
|  |     "markInstalled": "علامت گذاری به عنوان نصب شده", | ||||||
|  |     "update": "به روز رسانی", | ||||||
|  |     "markUpdated": "علامت گذاری به روز شد", | ||||||
|  |     "additionalOptions": "گزینه های اضافی", | ||||||
|  |     "disableVersionDetection": "غیرفعال کردن تشخیص نسخه", | ||||||
|  |     "noVersionDetectionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند.", | ||||||
|  |     "downloadingX": "در حال دانلود {}", | ||||||
|  |     "downloadNotifDescription": "کاربر را از پیشرفت دانلود یک برنامه مطلع می کند", | ||||||
|  |     "noAPKFound": "APK پیدا نشد فایل", | ||||||
|  |     "noVersionDetection": "بدون تشخیص نسخه", | ||||||
|  |     "categorize": "دسته بندی کردن", | ||||||
|  |     "categories": "دسته بندی ها", | ||||||
|  |     "category": "دسته بندی", | ||||||
|  |     "noCategory": "بدون دسته بندی", | ||||||
|  |     "noCategories": "بدون دسته بندی ها", | ||||||
|  |     "deleteCategoriesQuestion": "دسته بندی ها حذف شوند؟", | ||||||
|  |     "categoryDeleteWarning": "همه برنامهها در دستههای حذف شده روی دستهبندی نشده تنظیم میشوند.", | ||||||
|  |     "addCategory": "اضافه کردن دسته", | ||||||
|  |     "label": "برچسب", | ||||||
|  |     "language": "زبان", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|  |     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", | ||||||
|  |     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", | ||||||
|  |     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", | ||||||
|  |     "removeFromObtainium": "از Obtainium حذف کنید", | ||||||
|  |     "uninstallFromDevice": "حذف نصب از دستگاه", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "فقط برای برنامههایی کار میکند که تشخیص نسخه غیرفعال است.", | ||||||
|  |     "releaseDateAsVersion": "از تاریخ انتشار به عنوان نسخه استفاده کنید", | ||||||
|  |     "releaseDateAsVersionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند، اما تاریخ انتشار در دسترس است.", | ||||||
|  |     "changes": "تغییرات", | ||||||
|  |     "releaseDate": "تاریخ انتشار", | ||||||
|  |     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", | ||||||
|  |     "versionDetection": "تشخیص نسخه", | ||||||
|  |     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "برنامه حذف شود؟", | ||||||
|  |         "other": "برنامه ها حذف شوند؟" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "درخواستهای بسیار زیاد (نرخ محدود) - {} دقیقه دیگر دوباره امتحان کنید", | ||||||
|  |         "other": "درخواست های بسیار زیاد (نرخ محدود) - بعد از {} دقیقه دوباره امتحان کنید" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "بررسی بهروزرسانی BG با یک {} مواجه شد، یک بررسی مجدد را در {} دقیقه برنامهریزی میکند", | ||||||
|  |         "other": "بررسی بهروزرسانی BG با {} مواجه شد، یک بررسی مجدد را در {} دقیقه برنامهریزی میکند" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "بررسی بهروزرسانی BG پیدا شد {} بهروزرسانی - در صورت نیاز به کاربر اطلاع میدهد", | ||||||
|  |         "other": "بررسی بهروزرسانی BG {} بهروزرسانیهای یافت شده - در صورت نیاز به کاربر اطلاع میدهد" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "برنامه {}", | ||||||
|  |         "other": "{} برنامه ها" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} آدرس اینترنتی", | ||||||
|  |         "other": "{} آدرس های اینترنتی" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} دقیقه", | ||||||
|  |         "other": "{} دقیقه" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} ساعت", | ||||||
|  |         "other": "{} ساعت" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} روز", | ||||||
|  |         "other": "{} روز" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "گزارش {n} پاک شد (قبل از = {پیش از}، بعد = {بعد})", | ||||||
|  |         "other": "{n} گزارش پاک شد (قبل از = {پیش از}، بعد = {بعد})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} و 1 برنامه دیگر بهروزرسانی دارند.", | ||||||
|  |         "other": "{} و {} برنامه دیگر به روز رسانی دارند." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} و 1 برنامه دیگر به روز شدند.", | ||||||
|  |         "other": "{} و {} برنامه دیگر به روز شدند." | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										273
									
								
								assets/translations/fr.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,273 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "URL d'application {} invalide", | ||||||
|  |     "noReleaseFound": "Impossible de trouver une version appropriée", | ||||||
|  |     "noVersionFound": "Impossible de déterminer la version de la version", | ||||||
|  |     "urlMatchesNoSource": "L'URL ne correspond pas à une source connue", | ||||||
|  |     "cantInstallOlderVersion": "Impossible d'installer une ancienne version d'une application", | ||||||
|  |     "appIdMismatch": "L'ID de paquet téléchargé ne correspond pas à l'ID de l'application existante", | ||||||
|  |     "functionNotImplemented": "Cette classe n'a pas implémenté cette fonction", | ||||||
|  |     "placeholder": "Espace réservé", | ||||||
|  |     "someErrors": "Des erreurs se sont produites", | ||||||
|  |     "unexpectedError": "Erreur inattendue", | ||||||
|  |     "ok": "Okay", | ||||||
|  |     "and": "et", | ||||||
|  |     "startedBgUpdateTask": "Démarrage de la tâche de vérification de mise à jour en arrière-plan", | ||||||
|  |     "bgUpdateIgnoreAfterIs": "Mise à jour en arrière-plan est ignoré après  {}", | ||||||
|  |     "startedActualBGUpdateCheck": "Démarrage de la vérification de la mise à jour en arrière-plan", | ||||||
|  |     "bgUpdateTaskFinished": "Tâche de vérification de la mise à jour en arrière-plan terminée", | ||||||
|  |     "firstRun": "Il s'agit de la toute première exécution d'Obtainium", | ||||||
|  |     "settingUpdateCheckIntervalTo": "Définition de l'intervalle de mise à jour sur {}", | ||||||
|  |     "githubPATLabel": "Jeton d'Accès Personnel GitHub (Augmente la limite de débit)", | ||||||
|  |     "githubPATHint": "Le JAP doit être dans ce format : username:token", | ||||||
|  |     "githubPATFormat": "username:token", | ||||||
|  |     "githubPATLinkText": "À propos des JAP GitHub", | ||||||
|  |     "includePrereleases": "Inclure les avant-premières", | ||||||
|  |     "fallbackToOlderReleases": "Retour aux anciennes versions", | ||||||
|  |     "filterReleaseTitlesByRegEx": "Filtrer les titres de version par expression régulière", | ||||||
|  |     "invalidRegEx": "Expression régulière invalide", | ||||||
|  |     "noDescription": "Pas de description", | ||||||
|  |     "cancel": "Annuler", | ||||||
|  |     "continue": "Continuer", | ||||||
|  |     "requiredInBrackets": "(Requis)", | ||||||
|  |     "dropdownNoOptsError": "ERREUR : LE DÉROULEMENT DOIT AVOIR AU MOINS UNE OPT", | ||||||
|  |     "colour": "Couleur", | ||||||
|  |     "githubStarredRepos": "Dépôts étoilés GitHub", | ||||||
|  |     "uname": "Nom d'utilisateur", | ||||||
|  |     "wrongArgNum": "Mauvais nombre d'arguments fournis", | ||||||
|  |     "xIsTrackOnly": "{} est en 'Suivi uniquement'", | ||||||
|  |     "source": "Source", | ||||||
|  |     "app": "Application", | ||||||
|  |     "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": "Application déjà à jour ?", | ||||||
|  |     "addApp": "Ajouter une application", | ||||||
|  |     "appSourceURL": "URL de la source de l'application", | ||||||
|  |     "error": "Erreur", | ||||||
|  |     "add": "Ajoutée", | ||||||
|  |     "searchSomeSourcesLabel": "Rechercher (certaines sources uniquement)", | ||||||
|  |     "search": "Rechercher", | ||||||
|  |     "additionalOptsFor": "Options supplémentaires pour {}", | ||||||
|  |     "supportedSourcesBelow": "Sources prises en charge :", | ||||||
|  |     "trackOnlyInBrackets": "(Suivi uniquement)", | ||||||
|  |     "searchableInBrackets": "(Recherchable)", | ||||||
|  |     "appsString": "Applications", | ||||||
|  |     "noApps": "Aucune application", | ||||||
|  |     "noAppsForFilter": "Aucune application pour le filtre", | ||||||
|  |     "byX": "Par {}", | ||||||
|  |     "percentProgress": "Progrès: {}%", | ||||||
|  |     "pleaseWait": "Veuillez patienter", | ||||||
|  |     "updateAvailable": "Mise à jour disponible", | ||||||
|  |     "estimateInBracketsShort": "(Est.)", | ||||||
|  |     "notInstalled": "Pas installé", | ||||||
|  |     "estimateInBrackets": "(Estimation)", | ||||||
|  |     "selectAll": "Tout sélectionner", | ||||||
|  |     "deselectN": "Déselectionner {}", | ||||||
|  |     "xWillBeRemovedButRemainInstalled": "{} sera supprimé d'Obtainium mais restera installé sur l'appareil.", | ||||||
|  |     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", | ||||||
|  |     "removeSelectedApps": "Supprimer les applications sélectionnées", | ||||||
|  |     "updateX": "Mise à jour {}", | ||||||
|  |     "installX": "Installer {}", | ||||||
|  |     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\nas mis à jour", | ||||||
|  |     "changeX": "Changer {}", | ||||||
|  |     "installUpdateApps": "Installer/Mettre à jour les applications", | ||||||
|  |     "installUpdateSelectedApps": "Installer/Mettre à jour les applications sélectionnées", | ||||||
|  |     "markXSelectedAppsAsUpdated": "Marquer {} les applications sélectionnées comme mises à jour ?", | ||||||
|  |     "no": "Non", | ||||||
|  |     "yes": "Oui", | ||||||
|  |     "markSelectedAppsUpdated": "Marquer les applications sélectionnées comme mises à jour", | ||||||
|  |     "pinToTop": "Épingler en haut", | ||||||
|  |     "unpinFromTop": "Détacher du haut", | ||||||
|  |     "resetInstallStatusForSelectedAppsQuestion": "Réinitialiser l'état d'installation des applications sélectionnées ?", | ||||||
|  |     "installStatusOfXWillBeResetExplanation": "L'état d'installation de toutes les applications sélectionnées sera réinitialisé.\n\nCela peut aider lorsque la version de l'application affichée dans Obtainium est incorrecte en raison d'échecs de mises à jour ou d'autres problèmes.", | ||||||
|  |     "shareSelectedAppURLs": "Partager les URL d'application sélectionnées", | ||||||
|  |     "resetInstallStatus": "Réinitialiser le statut d'installation", | ||||||
|  |     "more": "Plus", | ||||||
|  |     "removeOutdatedFilter": "Supprimer le filtre d'application obsolète", | ||||||
|  |     "showOutdatedOnly": "Afficher uniquement les applications obsolètes", | ||||||
|  |     "filter": "Filtre", | ||||||
|  |     "filterActive": "Filtre *", | ||||||
|  |     "filterApps": "Filtrer les applications", | ||||||
|  |     "appName": "Nom de l'application", | ||||||
|  |     "author": "Auteur", | ||||||
|  |     "upToDateApps": "Applications à jour", | ||||||
|  |     "nonInstalledApps": "Applications non installées", | ||||||
|  |     "importExport": "Importer/Exporter", | ||||||
|  |     "settings": "Paramètres", | ||||||
|  |     "exportedTo": "Exporté vers {}", | ||||||
|  |     "obtainiumExport": "Exportation d'Obtainium", | ||||||
|  |     "invalidInput": "Entrée invalide", | ||||||
|  |     "importedX": "Importé {}", | ||||||
|  |     "obtainiumImport": "Importation d'Obtainium", | ||||||
|  |     "importFromURLList": "Importer à partir de la liste d'URL", | ||||||
|  |     "searchQuery": "Requête de recherche", | ||||||
|  |     "appURLList": "Liste d'URL d'application", | ||||||
|  |     "line": "Queue", | ||||||
|  |     "searchX": "Rechercher {}", | ||||||
|  |     "noResults": "Aucun résultat trouvé", | ||||||
|  |     "importX": "Importer {}", | ||||||
|  |     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher à tort comme \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela ne devrait pas affecter les données de l'application.\n\nN'affecte que les URL et les méthodes d'importation tierces.", | ||||||
|  |     "importErrors": "Erreurs d'importation", | ||||||
|  |     "importedXOfYApps": "{} sur {} applications importées.", | ||||||
|  |     "followingURLsHadErrors": "Les URL suivantes comportaient des erreurs :", | ||||||
|  |     "okay": "Okay", | ||||||
|  |     "selectURL": "Sélectionnez l'URL", | ||||||
|  |     "selectURLs": "Sélectionnez les URLs", | ||||||
|  |     "pick": "Prendre", | ||||||
|  |     "theme": "Thème", | ||||||
|  |     "dark": "Sombre", | ||||||
|  |     "light": "Clair", | ||||||
|  |     "followSystem": "Suivre le système", | ||||||
|  |     "obtainium": "Obtainium", | ||||||
|  |     "materialYou": "Material You", | ||||||
|  |     "appSortBy": "Applications triées par", | ||||||
|  |     "authorName": "Auteur/Nom", | ||||||
|  |     "nameAuthor": "Nom/Auteur", | ||||||
|  |     "asAdded": "Comme ajouté", | ||||||
|  |     "appSortOrder": "Ordre de tri des applications", | ||||||
|  |     "ascending": "Ascendant", | ||||||
|  |     "descending": "Descendanr", | ||||||
|  |     "bgUpdateCheckInterval": "Intervalle de vérification des mises à jour en arrière-plan", | ||||||
|  |     "neverManualOnly": "Jamais - Manuel uniquement", | ||||||
|  |     "appearance": "Apparence", | ||||||
|  |     "showWebInAppView": "Afficher la page Web source dans la vue de l'application", | ||||||
|  |     "pinUpdates": "Épingler les mises à jour dans la vue Top des applications", | ||||||
|  |     "updates": "Mises à jour", | ||||||
|  |     "sourceSpecific": "Spécifique à la source", | ||||||
|  |     "appSource": "Source de l'application", | ||||||
|  |     "noLogs": "Aucun journal", | ||||||
|  |     "appLogs": "Journaux d'application", | ||||||
|  |     "close": "Fermer", | ||||||
|  |     "share": "Partager", | ||||||
|  |     "appNotFound": "Application introuvable", | ||||||
|  |     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||||
|  |     "pickAnAPK": "Choisissez un APK", | ||||||
|  |     "appHasMoreThanOnePackage": "{} a plus d'un paquet :", | ||||||
|  |     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture de processeur {}.", | ||||||
|  |     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivantes :", | ||||||
|  |     "warning": "Avertissement", | ||||||
|  |     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de version provient de '{}'. Continuer?", | ||||||
|  |     "updatesAvailable": "Mises à jour disponibles", | ||||||
|  |     "updatesAvailableNotifDescription": "Avertit l'utilisateur que des mises à jour sont disponibles pour une ou plusieurs applications suivies par Obtainium", | ||||||
|  |     "noNewUpdates": "Aucune nouvelle mise à jour.", | ||||||
|  |     "xHasAnUpdate": "{} a une mise à jour.", | ||||||
|  |     "appsUpdated": "Applications mises à jour", | ||||||
|  |     "appsUpdatedNotifDescription": "Avertit l'utilisateur que les mises à jour d'une ou plusieurs applications ont été appliquées en arrière-plan", | ||||||
|  |     "xWasUpdatedToY": "{} a été mis à jour pour {}.", | ||||||
|  |     "errorCheckingUpdates": "Erreur lors de la vérification des mises à jour", | ||||||
|  |     "errorCheckingUpdatesNotifDescription": "Une notification qui s'affiche lorsque la vérification de la mise à jour en arrière-plan échoue", | ||||||
|  |     "appsRemoved": "Applications supprimées", | ||||||
|  |     "appsRemovedNotifDescription": "Avertit l'utilisateur qu'une ou plusieurs applications ont été supprimées en raison d'erreurs lors de leur chargement", | ||||||
|  |     "xWasRemovedDueToErrorY": "{} a été supprimé en raison de cette erreur : {}", | ||||||
|  |     "completeAppInstallation": "Installation complète de l'application", | ||||||
|  |     "obtainiumMustBeOpenToInstallApps": "Obtainium doit être ouvert pour installer des applications", | ||||||
|  |     "completeAppInstallationNotifDescription": "Demande à l'utilisateur de retourner sur Obtainium pour terminer l'installation d'une application", | ||||||
|  |     "checkingForUpdates": "Vérification des mises à jour", | ||||||
|  |     "checkingForUpdatesNotifDescription": "Notification transitoire qui apparaît lors de la recherche de mises à jour", | ||||||
|  |     "pleaseAllowInstallPerm": "Veuillez autoriser Obtainium à installer des applications", | ||||||
|  |     "trackOnly": "Suivi uniquement", | ||||||
|  |     "errorWithHttpStatusCode": "Erreur {}", | ||||||
|  |     "versionCorrectionDisabled": "Correction de version désactivée (le plugin ne semble pas fonctionner)", | ||||||
|  |     "unknown": "Inconnu", | ||||||
|  |     "none": "Aucun", | ||||||
|  |     "never": "Jamais", | ||||||
|  |     "latestVersionX": "Dernière version: {}", | ||||||
|  |     "installedVersionX": "Version installée : {}", | ||||||
|  |     "lastUpdateCheckX": "Vérification de la dernière mise à jour : {}", | ||||||
|  |     "remove": "Retirer", | ||||||
|  |     "yesMarkUpdated": "Oui, marquer comme mis à jour", | ||||||
|  |     "fdroid": "F-Droid", | ||||||
|  |     "appIdOrName": "ID ou nom de l'application", | ||||||
|  |     "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", | ||||||
|  |     "steam": "Steam", | ||||||
|  |     "steamMobile": "Steam Mobile", | ||||||
|  |     "steamChat": "Steam Chat", | ||||||
|  |     "install": "Installer", | ||||||
|  |     "markInstalled": "Marquer installée", | ||||||
|  |     "update": "Mettre à jour", | ||||||
|  |     "markUpdated": "Marquer à jour", | ||||||
|  |     "additionalOptions": "Options additionelles", | ||||||
|  |     "disableVersionDetection": "Désactiver la détection de version", | ||||||
|  |     "noVersionDetectionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de version ne fonctionne pas correctement.", | ||||||
|  |     "downloadingX": "Téléchargement {}", | ||||||
|  |     "downloadNotifDescription": "Avertit l'utilisateur de la progression du téléchargement d'une application", | ||||||
|  |     "noAPKFound": "Aucun APK trouvé", | ||||||
|  |     "noVersionDetection": "Pas de détection de version", | ||||||
|  |     "categorize": "Catégoriser", | ||||||
|  |     "categories": "Catégories", | ||||||
|  |     "category": "Catégorie", | ||||||
|  |     "noCategory": "No Category", | ||||||
|  |     "noCategories": "Aucune catégorie", | ||||||
|  |     "deleteCategoriesQuestion": "Supprimer les catégories ?", | ||||||
|  |     "categoryDeleteWarning": "Toutes les applications dans les catégories supprimées seront définies sur non catégorisées.", | ||||||
|  |     "addCategory": "Ajouter une catégorie", | ||||||
|  |     "label": "Étiquette", | ||||||
|  |     "language": "Langue", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|  |     "storagePermissionDenied": "Autorisation de stockage refusée", | ||||||
|  |     "selectedCategorizeWarning": "Cela remplacera tous les paramètres de catégorie existants pour les applications sélectionnées.", | ||||||
|  |     "filterAPKsByRegEx": "Filtrer les APK par expression régulière", | ||||||
|  |     "removeFromObtainium": "Supprimer d'Obtainium", | ||||||
|  |     "uninstallFromDevice": "Désinstaller de l'appareil", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Fonctionne uniquement pour les applications avec la détection de version désactivée.", | ||||||
|  |     "releaseDateAsVersion": "Utiliser la date de sortie comme version", | ||||||
|  |     "releaseDateAsVersionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de version ne fonctionne pas correctement, mais une date de sortie est disponible.", | ||||||
|  |     "changes": "Changements", | ||||||
|  |     "releaseDate": "Date de sortie", | ||||||
|  |     "importFromURLsInFile": "Importer à partir d'URL dans un fichier (comme OPML)", | ||||||
|  |     "versionDetection": "Détection des versions", | ||||||
|  |     "standardVersionDetection": "Détection de version standard", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "Supprimer l'application ?", | ||||||
|  |         "other": "Supprimer les applications ?" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "Trop de demandes (taux limité) - réessayez dans {} minute", | ||||||
|  |         "other": "Trop de demandes (taux limité) - réessayez dans {} minutes" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "La vérification de la mise à jour en arrière-plan a rencontré un {}, planifiera une nouvelle tentative de vérification dans {} minute", | ||||||
|  |         "other": "La vérification de la mise à jour en arrière-plan a rencontré un {}, planifiera une nouvelle tentative de vérification dans {} minutes" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "La vérification des mises à jour en arrière-plan trouvée {} mise à jour - avertira l'utilisateur si nécessaire", | ||||||
|  |         "other": "La vérification des mises à jour en arrière-plan a trouvé {} mises à jour - avertira l'utilisateur si nécessaire" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "{} Application", | ||||||
|  |         "other": "{} Applications" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} URL", | ||||||
|  |         "other": "{} URLs" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} Minute", | ||||||
|  |         "other": "{} Minutes" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} Heure", | ||||||
|  |         "other": "{} Heures" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} Jour", | ||||||
|  |         "other": "{} Jours" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "{n} journal effacé (avant = {before}, après = {after})", | ||||||
|  |         "other": "{n} journaux effacés (avant = {before}, après = {after})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} et 1 autre application ont des mises à jour.", | ||||||
|  |         "other": "{} et {} autres applications ont des mises à jour." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} et 1 autre application ont été mises à jour.", | ||||||
|  |         "other": "{} et {} autres applications ont été mises à jour." | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -34,7 +34,7 @@ | |||||||
|     "githubStarredRepos": "GitHub Csillagos Repo-k", |     "githubStarredRepos": "GitHub Csillagos Repo-k", | ||||||
|     "uname": "Felh.név", |     "uname": "Felh.név", | ||||||
|     "wrongArgNum": "Rossz számú argumentumot adott meg", |     "wrongArgNum": "Rossz számú argumentumot adott meg", | ||||||
|     "xIsTrackOnly": "A(z) {} csak nyomkövethető", |     "xIsTrackOnly": "A(z) {} csak nyomonkövethető", | ||||||
|     "source": "Forrás", |     "source": "Forrás", | ||||||
|     "app": "App", |     "app": "App", | ||||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", |     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", | ||||||
| @@ -56,10 +56,10 @@ | |||||||
|     "appsString": "Appok", |     "appsString": "Appok", | ||||||
|     "noApps": "Nincs App", |     "noApps": "Nincs App", | ||||||
|     "noAppsForFilter": "Nincsenek appok a szűrőhöz", |     "noAppsForFilter": "Nincsenek appok a szűrőhöz", | ||||||
|     "byX": "{} által", |     "byX": "Fejlesztő: {}", | ||||||
|     "percentProgress": "Folyamat: {}%", |     "percentProgress": "Folyamat: {}%", | ||||||
|     "pleaseWait": "Kis türelmet", |     "pleaseWait": "Kis türelmet", | ||||||
|     "updateAvailable": "Frissítés elérhető", |     "updateAvailable": "Frissítés érhető el", | ||||||
|     "estimateInBracketsShort": "(Becsült)", |     "estimateInBracketsShort": "(Becsült)", | ||||||
|     "notInstalled": "Nem telepített", |     "notInstalled": "Nem telepített", | ||||||
|     "estimateInBrackets": "(Becslés)", |     "estimateInBrackets": "(Becslés)", | ||||||
| @@ -70,24 +70,23 @@ | |||||||
|     "removeSelectedApps": "Távolítsa el a kiválasztott appokat", |     "removeSelectedApps": "Távolítsa el a kiválasztott appokat", | ||||||
|     "updateX": "Frissítés: {}", |     "updateX": "Frissítés: {}", | ||||||
|     "installX": "Telepítés: {}", |     "installX": "Telepítés: {}", | ||||||
|     "markXTrackOnlyAsUpdated": "Jelölje meg: {}\n(Csak nyomon követhető)\nas Frissítve", |     "markXTrackOnlyAsUpdated": "Jelölje meg: {}\n(Csak nyomon követhető)\nmint Frissített", | ||||||
|     "changeX": "Változás {}", |     "changeX": "Változás {}", | ||||||
|     "installUpdateApps": "Appok telepítése/frissítése", |     "installUpdateApps": "Appok telepítése/frissítése", | ||||||
|     "installUpdateSelectedApps": "Telepítse/frissítse a kiválasztott appokat", |     "installUpdateSelectedApps": "Telepítse/frissítse a kiválasztott appokat", | ||||||
|     "onlyWorksWithNonEVDApps": "Csak azoknál az alkalmazásoknál működik, amelyek telepítési állapota nem észlelhető automatikusan (nem gyakori).", |  | ||||||
|     "markXSelectedAppsAsUpdated": "Megjelöl {} kiválasztott alkalmazást frissítettként?", |     "markXSelectedAppsAsUpdated": "Megjelöl {} kiválasztott alkalmazást frissítettként?", | ||||||
|     "no": "Nem", |     "no": "Nem", | ||||||
|     "yes": "Igen", |     "yes": "Igen", | ||||||
|     "markSelectedAppsUpdated": "Jelölje meg a kiválasztott appokat frissítettként", |     "markSelectedAppsUpdated": "Jelölje meg a kiválasztott appokat frissítettként", | ||||||
|     "pinToTop": "Rögzítés a felülre", |     "pinToTop": "Rögzítés felülre", | ||||||
|     "unpinFromTop": "Eltávolít felülről", |     "unpinFromTop": "Eltávolít felülről", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott appok telepítési állapotát?", |     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott appok telepítési állapotát?", | ||||||
|     "installStatusOfXWillBeResetExplanation": "A kiválasztott appok telepítési állapota visszaáll.\n\nEz akkor segíthet, ha az Obtainiumban megjelenített app verzió hibás, frissítések vagy egyéb problémák miatt.", |     "installStatusOfXWillBeResetExplanation": "A kiválasztott appok telepítési állapota visszaáll.\n\nEz akkor segíthet, ha az Obtainiumban megjelenített app verzió hibás, frissítések vagy egyéb problémák miatt.", | ||||||
|     "shareSelectedAppURLs": "Ossza meg a kiválasztott app URL címeit", |     "shareSelectedAppURLs": "Ossza meg a kiválasztott app URL címeit", | ||||||
|     "resetInstallStatus": "Telepítési állapot visszaállítása", |     "resetInstallStatus": "Telepítési állapot visszaállítása", | ||||||
|     "more": "További", |     "more": "További", | ||||||
|     "removeOutdatedFilter": "Távolítsa el az elavult alkalmazásszűrőt", |     "removeOutdatedFilter": "Távolítsa el az elavult app szűrőt", | ||||||
|     "showOutdatedOnly": "Csak az elavult alkalmazások megjelenítése", |     "showOutdatedOnly": "Csak az elavult appok megjelenítése", | ||||||
|     "filter": "Szűrő", |     "filter": "Szűrő", | ||||||
|     "filterActive": "Szűrő *", |     "filterActive": "Szűrő *", | ||||||
|     "filterApps": "Appok szűrése", |     "filterApps": "Appok szűrése", | ||||||
| @@ -126,11 +125,11 @@ | |||||||
|     "appSortBy": "App rendezés...", |     "appSortBy": "App rendezés...", | ||||||
|     "authorName": "Szerző/Név", |     "authorName": "Szerző/Név", | ||||||
|     "nameAuthor": "Név/Szerző", |     "nameAuthor": "Név/Szerző", | ||||||
|     "asAdded": "Mint hozzáadott", |     "asAdded": "Mint Hozzáadott", | ||||||
|     "appSortOrder": "Appok rendezése", |     "appSortOrder": "Appok rendezése", | ||||||
|     "ascending": "Emelkedő", |     "ascending": "Emelkedő", | ||||||
|     "descending": "Csökkenő", |     "descending": "Csökkenő", | ||||||
|     "bgUpdateCheckInterval": "Háttérfrissítés ellenőrzési időköz", |     "bgUpdateCheckInterval": "Háttérfrissítés ellenőrzés időköze", | ||||||
|     "neverManualOnly": "Soha – csak manuális", |     "neverManualOnly": "Soha – csak manuális", | ||||||
|     "appearance": "Megjelenés", |     "appearance": "Megjelenés", | ||||||
|     "showWebInAppView": "Forrás megjelenítése az Appok nézetben", |     "showWebInAppView": "Forrás megjelenítése az Appok nézetben", | ||||||
| @@ -155,14 +154,14 @@ | |||||||
|     "noNewUpdates": "Nincsenek új frissítések.", |     "noNewUpdates": "Nincsenek új frissítések.", | ||||||
|     "xHasAnUpdate": "A(z) {} frissítést kapott.", |     "xHasAnUpdate": "A(z) {} frissítést kapott.", | ||||||
|     "appsUpdated": "Alkalmazások frissítve", |     "appsUpdated": "Alkalmazások frissítve", | ||||||
|     "appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több app frissítése történt a háttérben", |     "appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy/több app frissítése megtörtént a háttérben", | ||||||
|     "xWasUpdatedToY": "{} frissítve a következőre: {}.", |     "xWasUpdatedToY": "{} frissítve a következőre: {}.", | ||||||
|     "errorCheckingUpdates": "Hiba a frissítések keresésekor", |     "errorCheckingUpdates": "Hiba a frissítések keresésekor", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérbeli frissítések ellenőrzése sikertelen", |     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérbeli frissítések ellenőrzése sikertelen", | ||||||
|     "appsRemoved": "Alkalmazások eltávolítva", |     "appsRemoved": "Alkalmazások eltávolítva", | ||||||
|     "appsRemovedNotifDescription": "Értesíti a felhasználót egy vagy több alkalmazás eltávolításáról a betöltésük során fellépő hibák miatt", |     "appsRemovedNotifDescription": "Értesíti a felhasználót egy vagy több alkalmazás eltávolításáról a betöltésük során fellépő hibák miatt", | ||||||
|     "xWasRemovedDueToErrorY": "A(z) {} a következő hiba miatt lett eltávolítva: {}", |     "xWasRemovedDueToErrorY": "A(z) {} a következő hiba miatt lett eltávolítva: {}", | ||||||
|     "completeAppInstallation": "Teljes alkalmazástelepítés", |     "completeAppInstallation": "Teljes app telepítés", | ||||||
|     "obtainiumMustBeOpenToInstallApps": "Az Obtainiumnak megnyitva kell lennie az alkalmazások telepítéséhez", |     "obtainiumMustBeOpenToInstallApps": "Az Obtainiumnak megnyitva kell lennie az alkalmazások telepítéséhez", | ||||||
|     "completeAppInstallationNotifDescription": "Megkéri a felhasználót, hogy térjen vissza az Obtainiumhoz, hogy befejezze az alkalmazás telepítését", |     "completeAppInstallationNotifDescription": "Megkéri a felhasználót, hogy térjen vissza az Obtainiumhoz, hogy befejezze az alkalmazás telepítését", | ||||||
|     "checkingForUpdates": "Frissítések keresése", |     "checkingForUpdates": "Frissítések keresése", | ||||||
| @@ -178,7 +177,6 @@ | |||||||
|     "installedVersionX": "Telepített verzió: {}", |     "installedVersionX": "Telepített verzió: {}", | ||||||
|     "lastUpdateCheckX": "Frissítés ellenőrizve: {}", |     "lastUpdateCheckX": "Frissítés ellenőrizve: {}", | ||||||
|     "remove": "Eltávolítás", |     "remove": "Eltávolítás", | ||||||
|     "removeAppQuestion": "Eltávolítja az alkalmazást?", |  | ||||||
|     "yesMarkUpdated": "Igen, megjelölés frissítettként", |     "yesMarkUpdated": "Igen, megjelölés frissítettként", | ||||||
|     "fdroid": "F-Droid", |     "fdroid": "F-Droid", | ||||||
|     "appIdOrName": "App ID vagy név", |     "appIdOrName": "App ID vagy név", | ||||||
| @@ -193,12 +191,12 @@ | |||||||
|     "update": "Frissít", |     "update": "Frissít", | ||||||
|     "markUpdated": "Frissítettnek jelöl", |     "markUpdated": "Frissítettnek jelöl", | ||||||
|     "additionalOptions": "További lehetőségek", |     "additionalOptions": "További lehetőségek", | ||||||
|     "disableVersionDetection": "Verzióérzékelés letiltása", |     "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.", |     "noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően.", | ||||||
|     "downloadingX": "{} letöltés", |     "downloadingX": "{} letöltés", | ||||||
|     "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról", |     "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról", | ||||||
|     "noAPKFound": "Nem található APK", |     "noAPKFound": "Nem található APK", | ||||||
|     "noVersionDetection": "Nincs verzióérzékelés", |     "noVersionDetection": "Nincs verzió érzékelés", | ||||||
|     "categorize": "Kategorizálás", |     "categorize": "Kategorizálás", | ||||||
|     "categories": "Kategóriák", |     "categories": "Kategóriák", | ||||||
|     "category": "Kategória", |     "category": "Kategória", | ||||||
| @@ -207,6 +205,26 @@ | |||||||
|     "categoryDeleteWarning": "A(z) {} összes app kategorizálatlan állapotba kerül.", |     "categoryDeleteWarning": "A(z) {} összes app kategorizálatlan állapotba kerül.", | ||||||
|     "addCategory": "Új kategória", |     "addCategory": "Új kategória", | ||||||
|     "label": "Címke", |     "label": "Címke", | ||||||
|  |     "language": "Nyelv", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|  |     "storagePermissionDenied": "Tárhely engedély megtagadva", | ||||||
|  |     "selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.", | ||||||
|  |     "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel", | ||||||
|  |     "removeFromObtainium": "Eltávolítás az Obtainiumból", | ||||||
|  |     "uninstallFromDevice": "Eltávolítás a készülékről", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzióérzékelés le van tiltva.", | ||||||
|  |     "releaseDateAsVersion": "Használja a Kiadás dátumát, mint verziót", | ||||||
|  |     "releaseDateAsVersionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzió érzékelése nem működik megfelelően, de elérhető a kiadás dátuma.", | ||||||
|  |     "changes": "Változtatások", | ||||||
|  |     "releaseDate": "Kiadás dátuma", | ||||||
|  |     "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)", | ||||||
|  |     "versionDetection": "Verzió érzékelés", | ||||||
|  |     "standardVersionDetection": "Alapért. verzió érzékelés", | ||||||
|  |     "groupByCategory": "Csoportosítás Kategória alapján", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "Eltávolítja az alkalmazást?", | ||||||
|  |         "other": "Eltávolítja az alkalmazást?" | ||||||
|  |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva", |         "one": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva", | ||||||
|         "other": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva" |         "other": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva" | ||||||
|   | |||||||
| @@ -56,9 +56,9 @@ | |||||||
|     "appsString": "App", |     "appsString": "App", | ||||||
|     "noApps": "Nessuna App", |     "noApps": "Nessuna App", | ||||||
|     "noAppsForFilter": "Nessuna App per i filtri selezionati", |     "noAppsForFilter": "Nessuna App per i filtri selezionati", | ||||||
|     "byX": "Da {}", |     "byX": "Di {}", | ||||||
|     "percentProgress": "Progresso: {}%", |     "percentProgress": "Progresso: {}%", | ||||||
|     "pleaseWait": "Attendere prego", |     "pleaseWait": "In attesa", | ||||||
|     "updateAvailable": "Aggiornamento disponibile", |     "updateAvailable": "Aggiornamento disponibile", | ||||||
|     "estimateInBracketsShort": "(prev.)", |     "estimateInBracketsShort": "(prev.)", | ||||||
|     "notInstalled": "Non installato", |     "notInstalled": "Non installato", | ||||||
| @@ -74,7 +74,6 @@ | |||||||
|     "changeX": "Modifica {}", |     "changeX": "Modifica {}", | ||||||
|     "installUpdateApps": "Installa/Aggiorna App", |     "installUpdateApps": "Installa/Aggiorna App", | ||||||
|     "installUpdateSelectedApps": "Installa/Aggiorna le App selezionate", |     "installUpdateSelectedApps": "Installa/Aggiorna le App selezionate", | ||||||
|     "onlyWorksWithNonEVDApps": "Funziona solo per le App il cui stato d'installazione non può essere rilevato automaticamente (inconsueto).", |  | ||||||
|     "markXSelectedAppsAsUpdated": "Contrassegnare le {} App selezionate come aggiornate?", |     "markXSelectedAppsAsUpdated": "Contrassegnare le {} App selezionate come aggiornate?", | ||||||
|     "no": "No", |     "no": "No", | ||||||
|     "yes": "Sì", |     "yes": "Sì", | ||||||
| @@ -95,7 +94,7 @@ | |||||||
|     "author": "Autore", |     "author": "Autore", | ||||||
|     "upToDateApps": "App aggiornate", |     "upToDateApps": "App aggiornate", | ||||||
|     "nonInstalledApps": "App non installate", |     "nonInstalledApps": "App non installate", | ||||||
|     "importExport": "Importa - Esporta", |     "importExport": "Importa/Esporta", | ||||||
|     "settings": "Impostazioni", |     "settings": "Impostazioni", | ||||||
|     "exportedTo": "Esportato in {}", |     "exportedTo": "Esportato in {}", | ||||||
|     "obtainiumExport": "Esporta da Obtainium", |     "obtainiumExport": "Esporta da Obtainium", | ||||||
| @@ -178,7 +177,6 @@ | |||||||
|     "installedVersionX": "Versione installata: {}", |     "installedVersionX": "Versione installata: {}", | ||||||
|     "lastUpdateCheckX": "Ultimo controllo degli aggiornamenti: {}", |     "lastUpdateCheckX": "Ultimo controllo degli aggiornamenti: {}", | ||||||
|     "remove": "Rimuovi", |     "remove": "Rimuovi", | ||||||
|     "removeAppQuestion": "Rimuovere l'App?", |  | ||||||
|     "yesMarkUpdated": "Sì, contrassegna come aggiornato", |     "yesMarkUpdated": "Sì, contrassegna come aggiornato", | ||||||
|     "fdroid": "F-Droid", |     "fdroid": "F-Droid", | ||||||
|     "appIdOrName": "ID o nome dell'App", |     "appIdOrName": "ID o nome dell'App", | ||||||
| @@ -209,6 +207,25 @@ | |||||||
|     "addCategory": "Aggiungi categoria", |     "addCategory": "Aggiungi categoria", | ||||||
|     "label": "Etichetta", |     "label": "Etichetta", | ||||||
|     "language": "Lingua", |     "language": "Lingua", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|  |     "storagePermissionDenied": "Accesso ai file non autorizzato", | ||||||
|  |     "selectedCategorizeWarning": "Ciò sostituirà le impostazioni di categoria esistenti per le App selezionate.", | ||||||
|  |     "filterAPKsByRegEx": "Filtra file APK con espressioni regolari", | ||||||
|  |     "removeFromObtainium": "Rimuovi da Obtainium", | ||||||
|  |     "uninstallFromDevice": "Disinstalla dal dispositivo", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Funziona solo per le App con il rilevamento della versione disattivato.", | ||||||
|  |     "releaseDateAsVersion": "Usa data di rilascio come 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", | ||||||
|  |     "importFromURLsInFile": "Importa da URL in file (come OPML)", | ||||||
|  |     "versionDetection": "Rilevamento di versione", | ||||||
|  |     "standardVersionDetection": "Rilevamento di versione standard", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "Rimuovere l'App?", | ||||||
|  |         "other": "Rimuovere le App?" | ||||||
|  |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Troppe richieste (traffico limitato) - riprova tra {} minuto", |         "one": "Troppe richieste (traffico limitato) - riprova tra {} minuto", | ||||||
|         "other": "Troppe richieste (traffico limitato) - riprova tra {} minuti" |         "other": "Troppe richieste (traffico limitato) - riprova tra {} minuti" | ||||||
|   | |||||||
| @@ -7,12 +7,12 @@ | |||||||
|     "appIdMismatch": "ダウンロードしたパッケージのIDが既存のApp IDと一致しません", |     "appIdMismatch": "ダウンロードしたパッケージのIDが既存のApp IDと一致しません", | ||||||
|     "functionNotImplemented": "このクラスはこの機能を実装していません", |     "functionNotImplemented": "このクラスはこの機能を実装していません", | ||||||
|     "placeholder": "プレースホルダー", |     "placeholder": "プレースホルダー", | ||||||
|     "someErrors": "いくつかのエラーが発生しました", |     "someErrors": "何らかのエラーが発生しました", | ||||||
|     "unexpectedError": "予期せぬエラーが発生しました", |     "unexpectedError": "予期せぬエラーが発生しました", | ||||||
|     "ok": "OK", |     "ok": "OK", | ||||||
|     "and": "と", |     "and": "と", | ||||||
|     "startedBgUpdateTask": "バックグラウンドのアップデート確認タスクを開始", |     "startedBgUpdateTask": "バックグラウンドのアップデート確認タスクを開始", | ||||||
|     "bgUpdateIgnoreAfterIs": "Bg update ignoreAfter is  {}", |     "bgUpdateIgnoreAfterIs": "Bg update ignoreAfter is {}", | ||||||
|     "startedActualBGUpdateCheck": "実際のバックグラウンドのアップデート確認を開始", |     "startedActualBGUpdateCheck": "実際のバックグラウンドのアップデート確認を開始", | ||||||
|     "bgUpdateTaskFinished": "バックグラウンドのアップデート確認タスクを終了", |     "bgUpdateTaskFinished": "バックグラウンドのアップデート確認タスクを終了", | ||||||
|     "firstRun": "これがObtainiumの最初の実行です", |     "firstRun": "これがObtainiumの最初の実行です", | ||||||
| @@ -65,16 +65,15 @@ | |||||||
|     "estimateInBrackets": "(推定)", |     "estimateInBrackets": "(推定)", | ||||||
|     "selectAll": "すべて選択", |     "selectAll": "すべて選択", | ||||||
|     "deselectN": "{}件の選択を解除", |     "deselectN": "{}件の選択を解除", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{}はObtainiumから削除されますが、デバイスにはインストールされたままです。", |     "xWillBeRemovedButRemainInstalled": "{} はObtainiumから削除されますが、デバイスにはインストールされたままです。", | ||||||
|     "removeSelectedAppsQuestion": "選択したアプリを削除しますか?", |     "removeSelectedAppsQuestion": "選択したアプリを削除しますか?", | ||||||
|     "removeSelectedApps": "選択したアプリを削除する", |     "removeSelectedApps": "選択したアプリを削除する", | ||||||
|     "updateX": "{}をアップデートする", |     "updateX": "{} をアップデートする", | ||||||
|     "installX": "{}をインストールする", |     "installX": "{} をインストールする", | ||||||
|     "markXTrackOnlyAsUpdated": "{}\n(追跡のみ)\nをアップデート済みとしてマークする", |     "markXTrackOnlyAsUpdated": "{}\n(追跡のみ)\nをアップデート済みとしてマークする", | ||||||
|     "changeX": "{}を変更する", |     "changeX": "{} を変更する", | ||||||
|     "installUpdateApps": "アプリのインストール/アップデート", |     "installUpdateApps": "アプリのインストール/アップデート", | ||||||
|     "installUpdateSelectedApps": "選択したアプリのインストール/アップデート", |     "installUpdateSelectedApps": "選択したアプリのインストール/アップデート", | ||||||
|     "onlyWorksWithNonEVDApps": "インストール状況を自動検出できないアプリ(一般的でないもの)のみ動作します。", |  | ||||||
|     "markXSelectedAppsAsUpdated": "{}個の選択したアプリをアップデート済みとしてマークしますか?", |     "markXSelectedAppsAsUpdated": "{}個の選択したアプリをアップデート済みとしてマークしますか?", | ||||||
|     "no": "いいえ", |     "no": "いいえ", | ||||||
|     "yes": "はい", |     "yes": "はい", | ||||||
| @@ -82,7 +81,7 @@ | |||||||
|     "pinToTop": "トップに固定", |     "pinToTop": "トップに固定", | ||||||
|     "unpinFromTop": "トップから固定解除", |     "unpinFromTop": "トップから固定解除", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "選択したアプリのインストール状態をリセットしますか?", |     "resetInstallStatusForSelectedAppsQuestion": "選択したアプリのインストール状態をリセットしますか?", | ||||||
|     "installStatusOfXWillBeResetExplanation": "選択したアプリのインストール状態がリセットされます。\n\nアップデートに失敗するなどして、Obtainiumに表示されるアプリのバージョンが正しくない場合に役立ちます。", |     "installStatusOfXWillBeResetExplanation": "選択したアプリのインストール状態がリセットされます。\n\nアップデートに失敗した場合など、Obtainiumに表示されるアプリのバージョンが正しくない場合に有効です。", | ||||||
|     "shareSelectedAppURLs": "選択したアプリのURLを共有する", |     "shareSelectedAppURLs": "選択したアプリのURLを共有する", | ||||||
|     "resetInstallStatus": "インストール状態をリセットする", |     "resetInstallStatus": "インストール状態をリセットする", | ||||||
|     "more": "もっと見る", |     "more": "もっと見る", | ||||||
| @@ -108,8 +107,8 @@ | |||||||
|     "line": "行", |     "line": "行", | ||||||
|     "searchX": "{}で検索", |     "searchX": "{}で検索", | ||||||
|     "noResults": "結果は見つかりませんでした", |     "noResults": "結果は見つかりませんでした", | ||||||
|     "importX": "{}をインポートする", |     "importX": "{}をインポート", | ||||||
|     "importedAppsIdDisclaimer": "インポートしたアプリが「未インストール」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響しません。\n\nURLとサードパーティーのインポートメソッドにのみ影響します。", |     "importedAppsIdDisclaimer": "インポートしたアプリが「未インストール」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響しません。\n\nURLとサードパーティのインポートメソッドにのみ影響します。", | ||||||
|     "importErrors": "インポートエラー", |     "importErrors": "インポートエラー", | ||||||
|     "importedXOfYApps": "{} / {} アプリをインポートしました", |     "importedXOfYApps": "{} / {} アプリをインポートしました", | ||||||
|     "followingURLsHadErrors": "以下のURLでエラーが発生しました:", |     "followingURLsHadErrors": "以下のURLでエラーが発生しました:", | ||||||
| @@ -133,7 +132,7 @@ | |||||||
|     "bgUpdateCheckInterval": "バックグラウンドでのアップデート確認の間隔", |     "bgUpdateCheckInterval": "バックグラウンドでのアップデート確認の間隔", | ||||||
|     "neverManualOnly": "手動", |     "neverManualOnly": "手動", | ||||||
|     "appearance": "外観", |     "appearance": "外観", | ||||||
|     "showWebInAppView": "アプリビューにソースウェブページを表示する", |     "showWebInAppView": "アプリページにソースのWebページを表示する", | ||||||
|     "pinUpdates": "アップデートがあるアプリをトップに固定する", |     "pinUpdates": "アップデートがあるアプリをトップに固定する", | ||||||
|     "updates": "アップデート", |     "updates": "アップデート", | ||||||
|     "sourceSpecific": "Github アクセストークン", |     "sourceSpecific": "Github アクセストークン", | ||||||
| @@ -145,23 +144,23 @@ | |||||||
|     "appNotFound": "アプリが見つかりません", |     "appNotFound": "アプリが見つかりません", | ||||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-エクスポート", |     "obtainiumExportHyphenatedLowercase": "obtainium-エクスポート", | ||||||
|     "pickAnAPK": "APKを選択", |     "pickAnAPK": "APKを選択", | ||||||
|     "appHasMoreThanOnePackage": "{}は複数のパッケージが存在します: ", |     "appHasMoreThanOnePackage": "{} は複数のパッケージが存在します: ", | ||||||
|     "deviceSupportsXArch": "お使いのデバイスは{} CPUアーキテクチャに対応しています。", |     "deviceSupportsXArch": "お使いのデバイスは {} CPUアーキテクチャに対応しています。", | ||||||
|     "deviceSupportsFollowingArchs": "お使いのデバイスは、以下のCPUアーキテクチャをサポートしています:", |     "deviceSupportsFollowingArchs": "お使いのデバイスは、以下のCPUアーキテクチャをサポートしています:", | ||||||
|     "warning": "警告", |     "warning": "警告", | ||||||
|     "sourceIsXButPackageFromYPrompt": "アプリのソースは'{}'ですが、リリースパッケージは'{}'から来ています。続行しますか?", |     "sourceIsXButPackageFromYPrompt": "アプリのソースは'{}'ですが、リリースパッケージは'{}'から来ています。続行しますか?", | ||||||
|     "updatesAvailable": "アップデートが利用可能", |     "updatesAvailable": "アップデートが利用可能", | ||||||
|     "updatesAvailableNotifDescription": "Obtainiumが追跡している1つまたは複数のアプリのアップデートが利用可能であることをユーザーに通知する", |     "updatesAvailableNotifDescription": "Obtainiumが追跡している1つまたは複数のアプリのアップデートが利用可能であることをユーザーに通知する", | ||||||
|     "noNewUpdates": "新しいアップデートはありません", |     "noNewUpdates": "新しいアップデートはありません", | ||||||
|     "xHasAnUpdate": "{}のアップデートが利用可能です", |     "xHasAnUpdate": "{} のアップデートが利用可能です", | ||||||
|     "appsUpdated": "アプリをアップデートしました", |     "appsUpdated": "アプリをアップデートしました", | ||||||
|     "appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する", |     "appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する", | ||||||
|     "xWasUpdatedToY": "{}が{}にアップデートされました", |     "xWasUpdatedToY": "{} が {} にアップデートされました", | ||||||
|     "errorCheckingUpdates": "アップデート確認中のエラー", |     "errorCheckingUpdates": "アップデート確認中のエラー", | ||||||
|     "errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデート確認に失敗した際に表示される通知", |     "errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデート確認に失敗した際に表示される通知", | ||||||
|     "appsRemoved": "削除されたアプリ", |     "appsRemoved": "削除されたアプリ", | ||||||
|     "appsRemovedNotifDescription": "アプリの読み込み中にエラーが発生したため、1つまたは複数のアプリが削除されたことをユーザーに通知する", |     "appsRemovedNotifDescription": "アプリの読み込み中にエラーが発生したため、1つまたは複数のアプリが削除されたことをユーザーに通知する", | ||||||
|     "xWasRemovedDueToErrorY": "このエラーのため、{}は削除されました: {}", |     "xWasRemovedDueToErrorY": "このエラーのため、{} は削除されました: {}", | ||||||
|     "completeAppInstallation": "アプリのインストールを完了する", |     "completeAppInstallation": "アプリのインストールを完了する", | ||||||
|     "obtainiumMustBeOpenToInstallApps": "アプリをインストールするにはObtainiumを開いている必要があります。", |     "obtainiumMustBeOpenToInstallApps": "アプリをインストールするにはObtainiumを開いている必要があります。", | ||||||
|     "completeAppInstallationNotifDescription": "アプリのインストールを完了するために、Obtainiumに戻る必要があります。", |     "completeAppInstallationNotifDescription": "アプリのインストールを完了するために、Obtainiumに戻る必要があります。", | ||||||
| @@ -178,13 +177,12 @@ | |||||||
|     "installedVersionX": "インストールされたバージョン: {}", |     "installedVersionX": "インストールされたバージョン: {}", | ||||||
|     "lastUpdateCheckX": "最終アップデート確認: {}", |     "lastUpdateCheckX": "最終アップデート確認: {}", | ||||||
|     "remove": "削除", |     "remove": "削除", | ||||||
|     "removeAppQuestion": "アプリを削除しますか?", |  | ||||||
|     "yesMarkUpdated": "はい、アップデート済みとしてマークします", |     "yesMarkUpdated": "はい、アップデート済みとしてマークします", | ||||||
|     "fdroid": "F-Droid", |     "fdroid": "F-Droid", | ||||||
|     "appIdOrName": "アプリのIDまたは名前", |     "appIdOrName": "アプリのIDまたは名前", | ||||||
|     "appWithIdOrNameNotFound": "そのIDや名前を持つアプリは見つかりませんでした", |     "appWithIdOrNameNotFound": "そのIDや名前を持つアプリは見つかりませんでした", | ||||||
|     "reposHaveMultipleApps": "リポジトリには複数のアプリが含まれることがあります", |     "reposHaveMultipleApps": "リポジトリには複数のアプリが含まれることがあります", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Third-Party Repo", |     "fdroidThirdPartyRepo": "F-Droid サードパーティリポジトリ", | ||||||
|     "steam": "Steam", |     "steam": "Steam", | ||||||
|     "steamMobile": "Steam Mobile", |     "steamMobile": "Steam Mobile", | ||||||
|     "steamChat": "Steam Chat", |     "steamChat": "Steam Chat", | ||||||
| @@ -209,6 +207,25 @@ | |||||||
|     "addCategory": "カテゴリを追加", |     "addCategory": "カテゴリを追加", | ||||||
|     "label": "ラベル", |     "label": "ラベル", | ||||||
|     "language": "言語", |     "language": "言語", | ||||||
|  |     "copiedToClipboard": "クリップボードにコピーしました", | ||||||
|  |     "storagePermissionDenied": "ストレージ権限が拒否されました", | ||||||
|  |     "selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。", | ||||||
|  |     "filterAPKsByRegEx": "正規表現でAPKを絞り込む", | ||||||
|  |     "removeFromObtainium": "Obtainiumから削除する", | ||||||
|  |     "uninstallFromDevice": "デバイスからアンインストールする", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。", | ||||||
|  |     "releaseDateAsVersion": "リリース日をバージョンとして使用する", | ||||||
|  |     "releaseDateAsVersionExplanation": "このオプションは、バージョン検出が正しく機能しないアプリで、リリース日が利用可能な場合にのみ使用する必要があります。", | ||||||
|  |     "changes": "変更点", | ||||||
|  |     "releaseDate": "リリース日", | ||||||
|  |     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", | ||||||
|  |     "versionDetection": "バージョン検出", | ||||||
|  |     "standardVersionDetection": "標準のバージョン検出", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "アプリを削除しますか?", | ||||||
|  |         "other": "アプリを削除しますか?" | ||||||
|  |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "リクエストが多すぎます(レート制限)- {}分後に再試行してください", |         "one": "リクエストが多すぎます(レート制限)- {}分後に再試行してください", | ||||||
|         "other": "リクエストが多すぎます(レート制限)- {}分後に再試行してください" |         "other": "リクエストが多すぎます(レート制限)- {}分後に再試行してください" | ||||||
| @@ -246,11 +263,11 @@ | |||||||
|         "other": "{n}個のログをクリアしました (前 = {before}, 後 = {after})" |         "other": "{n}個のログをクリアしました (前 = {before}, 後 = {after})" | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesAvailable": { |     "xAndNMoreUpdatesAvailable": { | ||||||
|         "one": "{}とさらに{}個のアプリのアップデートが利用可能です", |         "one": "{} とさらに {} 個のアプリのアップデートが利用可能です", | ||||||
|         "other": "{}とさらに{}個のアプリのアップデートが利用可能です" |         "other": "{} とさらに {} 個のアプリのアップデートが利用可能です" | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesInstalled": { |     "xAndNMoreUpdatesInstalled": { | ||||||
|         "one": "{}とさらに{}個のアプリがアップデートされました", |         "one": "{} とさらに {} 個のアプリがアップデートされました", | ||||||
|         "other": "{}とさらに{}個のアプリがアップデートされました" |         "other": "{} とさらに {} 個のアプリがアップデートされました" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ | |||||||
|     "ok": "好的", |     "ok": "好的", | ||||||
|     "and": "和", |     "and": "和", | ||||||
|     "startedBgUpdateTask": "开始后台检查更新任务", |     "startedBgUpdateTask": "开始后台检查更新任务", | ||||||
|     "bgUpdateIgnoreAfterIs": "Bg update ignoreAfter is  {}", |     "bgUpdateIgnoreAfterIs": "下次后台更新检查  {}", | ||||||
|     "startedActualBGUpdateCheck": "后台检查更新已开始", |     "startedActualBGUpdateCheck": "后台检查更新已开始", | ||||||
|     "bgUpdateTaskFinished": "后台检查更新已完成", |     "bgUpdateTaskFinished": "后台检查更新已完成", | ||||||
|     "firstRun": "这是你第一次运行 Obtainium", |     "firstRun": "这是你第一次运行 Obtainium", | ||||||
| @@ -178,7 +178,6 @@ | |||||||
|     "installedVersionX": "已安装: {}", |     "installedVersionX": "已安装: {}", | ||||||
|     "lastUpdateCheckX": "最后检查: {}", |     "lastUpdateCheckX": "最后检查: {}", | ||||||
|     "remove": "删除", |     "remove": "删除", | ||||||
|     "removeAppQuestion": "删除应用?", |  | ||||||
|     "yesMarkUpdated": "'是的,标为已更新", |     "yesMarkUpdated": "'是的,标为已更新", | ||||||
|     "fdroid": "F-Droid", |     "fdroid": "F-Droid", | ||||||
|     "appIdOrName": "应用 ID 或名称", |     "appIdOrName": "应用 ID 或名称", | ||||||
| @@ -199,16 +198,34 @@ | |||||||
|     "downloadNotifDescription": "通知用户下载进度", |     "downloadNotifDescription": "通知用户下载进度", | ||||||
|     "noAPKFound": "未找到安装包", |     "noAPKFound": "未找到安装包", | ||||||
|     "noVersionDetection": "无版本检测", |     "noVersionDetection": "无版本检测", | ||||||
|     "categorize": "Categorize", |     "categorize": "归档", | ||||||
|     "categories": "Categories", |     "categories": "归档", | ||||||
|     "category": "Category", |     "category": "类别", | ||||||
|     "noCategory": "No Category", |     "noCategory": "无类别", | ||||||
|     "noCategories": "No Categories", |     "noCategories": "无类别", | ||||||
|     "deleteCategoriesQuestion": "Delete Categories?", |     "deleteCategoriesQuestion": "删除所有类别?", | ||||||
|     "categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.", |     "categoryDeleteWarning": "所有被删除类别的应用程序将被设置为无类别", | ||||||
|     "addCategory": "Add Category", |     "addCategory": "添加类别", | ||||||
|     "label": "Label", |     "label": "标签", | ||||||
|     "language": "Language", |     "language": "语言", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|  |     "storagePermissionDenied": "存储权限已被拒绝", | ||||||
|  |     "selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别", | ||||||
|  |     "filterAPKsByRegEx": "Filter APKs by Regular Expression", | ||||||
|  |     "removeFromObtainium": "Remove from Obtainium", | ||||||
|  |     "uninstallFromDevice": "Uninstall from Device", | ||||||
|  |     "releaseDateAsVersion": "Use Release Date as Version", | ||||||
|  |     "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)", | ||||||
|  |     "versionDetection": "Version Detection", | ||||||
|  |     "standardVersionDetection": "Standard version detection", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "删除应用?", | ||||||
|  |         "other": "删除应用?" | ||||||
|  |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "请求过多 (API 限制) - 在 {} 分钟后重试", |         "one": "请求过多 (API 限制) - 在 {} 分钟后重试", | ||||||
|         "other": "请求过多 (API 限制) - 在 {} 分钟后重试" |         "other": "请求过多 (API 限制) - 在 {} 分钟后重试" | ||||||
| @@ -253,4 +270,4 @@ | |||||||
|         "one": "{} 和 {} 更多应用已被安装", |         "one": "{} 和 {} 更多应用已被安装", | ||||||
|         "other": "{} 和 {} 更多应用已被安装" |         "other": "{} 和 {} 更多应用已被安装" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,5 +1,9 @@ | |||||||
|  | import 'dart:io'; | ||||||
|  |  | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:html/parser.dart'; | import 'package:html/parser.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| @@ -7,6 +11,23 @@ class APKMirror extends AppSource { | |||||||
|   APKMirror() { |   APKMirror() { | ||||||
|     host = 'apkmirror.com'; |     host = 'apkmirror.com'; | ||||||
|     enforceTrackOnly = true; |     enforceTrackOnly = true; | ||||||
|  |  | ||||||
|  |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('fallbackToOlderReleases', | ||||||
|  |             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||||
|  |       ], | ||||||
|  |       [ | ||||||
|  |         GeneratedFormTextField('filterReleaseTitlesByRegEx', | ||||||
|  |             label: tr('filterReleaseTitlesByRegEx'), | ||||||
|  |             required: false, | ||||||
|  |             additionalValidators: [ | ||||||
|  |               (value) { | ||||||
|  |                 return regExValidator(value); | ||||||
|  |               } | ||||||
|  |             ]) | ||||||
|  |       ] | ||||||
|  |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -28,12 +49,38 @@ class APKMirror extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|  |     bool fallbackToOlderReleases = | ||||||
|  |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|  |     String? regexFilter = | ||||||
|  |         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) | ||||||
|  |                     ?.isNotEmpty == | ||||||
|  |                 true | ||||||
|  |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|  |             : null; | ||||||
|     Response res = await get(Uri.parse('$standardUrl/feed')); |     Response res = await get(Uri.parse('$standardUrl/feed')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       String? titleString = parse(res.body) |       var items = parse(res.body).querySelectorAll('item'); | ||||||
|           .querySelector('item') |       dynamic targetRelease; | ||||||
|           ?.querySelector('title') |       for (int i = 0; i < items.length; i++) { | ||||||
|           ?.innerHtml; |         if (!fallbackToOlderReleases && i > 0) break; | ||||||
|  |         String? nameToFilter = items[i].querySelector('title')?.innerHtml; | ||||||
|  |         if (regexFilter != null && | ||||||
|  |             nameToFilter != null && | ||||||
|  |             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         targetRelease = items[i]; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       String? titleString = targetRelease?.querySelector('title')?.innerHtml; | ||||||
|  |       String? dateString = targetRelease | ||||||
|  |           ?.querySelector('pubDate') | ||||||
|  |           ?.innerHtml | ||||||
|  |           .split(' ') | ||||||
|  |           .sublist(0, 5) | ||||||
|  |           .join(' '); | ||||||
|  |       DateTime? releaseDate = | ||||||
|  |           dateString != null ? HttpDate.parse('$dateString GMT') : null; | ||||||
|       String? version = titleString |       String? version = titleString | ||||||
|           ?.substring(RegExp('[0-9]').firstMatch(titleString)?.start ?? 0, |           ?.substring(RegExp('[0-9]').firstMatch(titleString)?.start ?? 0, | ||||||
|               RegExp(' by ').firstMatch(titleString)?.start ?? 0) |               RegExp(' by ').firstMatch(titleString)?.start ?? 0) | ||||||
| @@ -44,9 +91,10 @@ class APKMirror extends AppSource { | |||||||
|       if (version == null || version.isEmpty) { |       if (version == null || version.isEmpty) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, [], getAppNames(standardUrl)); |       return APKDetails(version, [], getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										156
									
								
								lib/app_sources/codeberg.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,156 @@ | |||||||
|  | import 'dart:convert'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class Codeberg extends AppSource { | ||||||
|  |   Codeberg() { | ||||||
|  |     host = 'codeberg.org'; | ||||||
|  |  | ||||||
|  |     additionalSourceSpecificSettingFormItems = []; | ||||||
|  |  | ||||||
|  |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('includePrereleases', | ||||||
|  |             label: tr('includePrereleases'), defaultValue: false) | ||||||
|  |       ], | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('fallbackToOlderReleases', | ||||||
|  |             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||||
|  |       ], | ||||||
|  |       [ | ||||||
|  |         GeneratedFormTextField('filterReleaseTitlesByRegEx', | ||||||
|  |             label: tr('filterReleaseTitlesByRegEx'), | ||||||
|  |             required: false, | ||||||
|  |             additionalValidators: [ | ||||||
|  |               (value) { | ||||||
|  |                 return regExValidator(value); | ||||||
|  |               } | ||||||
|  |             ]) | ||||||
|  |       ] | ||||||
|  |     ]; | ||||||
|  |  | ||||||
|  |     canSearch = true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+'); | ||||||
|  |     RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase()); | ||||||
|  |     if (match == null) { | ||||||
|  |       throw InvalidURLError(name); | ||||||
|  |     } | ||||||
|  |     return url.substring(0, match.end); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String? changeLogPageFromStandardUrl(String standardUrl) => | ||||||
|  |       '$standardUrl/releases'; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     bool includePrereleases = additionalSettings['includePrereleases'] == true; | ||||||
|  |     bool fallbackToOlderReleases = | ||||||
|  |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|  |     String? regexFilter = | ||||||
|  |         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) | ||||||
|  |                     ?.isNotEmpty == | ||||||
|  |                 true | ||||||
|  |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|  |             : null; | ||||||
|  |     Response res = await get(Uri.parse( | ||||||
|  |         'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var releases = jsonDecode(res.body) as List<dynamic>; | ||||||
|  |  | ||||||
|  |       List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) => | ||||||
|  |           (release['assets'] as List<dynamic>?) | ||||||
|  |               ?.map((e) { | ||||||
|  |                 return e['name'] != null && e['browser_download_url'] != null | ||||||
|  |                     ? MapEntry(e['name'] as String, | ||||||
|  |                         e['browser_download_url'] as String) | ||||||
|  |                     : const MapEntry('', ''); | ||||||
|  |               }) | ||||||
|  |               .where((element) => element.key.toLowerCase().endsWith('.apk')) | ||||||
|  |               .toList() ?? | ||||||
|  |           []; | ||||||
|  |  | ||||||
|  |       dynamic targetRelease; | ||||||
|  |  | ||||||
|  |       for (int i = 0; i < releases.length; i++) { | ||||||
|  |         if (!fallbackToOlderReleases && i > 0) break; | ||||||
|  |         if (!includePrereleases && releases[i]['prerelease'] == true) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         if (releases[i]['draft'] == true) { | ||||||
|  |           // Draft releases not supported | ||||||
|  |         } | ||||||
|  |         var nameToFilter = releases[i]['name'] as String?; | ||||||
|  |         if (nameToFilter == null || nameToFilter.trim().isEmpty) { | ||||||
|  |           // Some leave titles empty so tag is used | ||||||
|  |           nameToFilter = releases[i]['tag_name'] as String; | ||||||
|  |         } | ||||||
|  |         if (regexFilter != null && | ||||||
|  |             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         var apkUrls = getReleaseAPKUrls(releases[i]); | ||||||
|  |         if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         targetRelease = releases[i]; | ||||||
|  |         targetRelease['apkUrls'] = apkUrls; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       if (targetRelease == null) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|  |       } | ||||||
|  |       String? version = targetRelease['tag_name']; | ||||||
|  |       DateTime? releaseDate = targetRelease['published_at'] != null | ||||||
|  |           ? DateTime.parse(targetRelease['published_at']) | ||||||
|  |           : null; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       var changeLog = targetRelease['body'].toString(); | ||||||
|  |       return APKDetails( | ||||||
|  |           version, | ||||||
|  |           targetRelease['apkUrls'] as List<MapEntry<String, String>>, | ||||||
|  |           getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate, | ||||||
|  |           changeLog: changeLog.isEmpty ? null : changeLog); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   AppNames getAppNames(String standardUrl) { | ||||||
|  |     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); | ||||||
|  |     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); | ||||||
|  |     return AppNames(names[0], names[1]); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<Map<String, String>> search(String query) async { | ||||||
|  |     Response res = await get(Uri.parse( | ||||||
|  |         'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       Map<String, String> urlsWithDescriptions = {}; | ||||||
|  |       for (var e in (jsonDecode(res.body)['data'] as List<dynamic>)) { | ||||||
|  |         urlsWithDescriptions.addAll({ | ||||||
|  |           e['html_url'] as String: e['description'] != null | ||||||
|  |               ? e['description'] as String | ||||||
|  |               : tr('noDescription') | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |       return urlsWithDescriptions; | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -14,12 +14,14 @@ class FDroid extends AppSource { | |||||||
|   @override |   @override | ||||||
|   String standardizeURL(String url) { |   String standardizeURL(String url) { | ||||||
|     RegExp standardUrlRegExB = |     RegExp standardUrlRegExB = | ||||||
|         RegExp('^https?://$host/+[^/]+/+packages/+[^/]+'); |         RegExp('^https?://(cloudflare\\.)?$host/+[^/]+/+packages/+[^/]+'); | ||||||
|     RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase()); |     RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase()); | ||||||
|     if (match != null) { |     if (match != null) { | ||||||
|       url = 'https://$host/packages/${Uri.parse(url).pathSegments.last}'; |       url = | ||||||
|  |           'https://${Uri.parse(url.substring(0, match.end)).host}/packages/${Uri.parse(url).pathSegments.last}'; | ||||||
|     } |     } | ||||||
|     RegExp standardUrlRegExA = RegExp('^https?://$host/+packages/+[^/]+'); |     RegExp standardUrlRegExA = | ||||||
|  |         RegExp('^https?://(cloudflare\\.)?$host/+packages/+[^/]+'); | ||||||
|     match = standardUrlRegExA.firstMatch(url.toLowerCase()); |     match = standardUrlRegExA.firstMatch(url.toLowerCase()); | ||||||
|     if (match == null) { |     if (match == null) { | ||||||
|       throw InvalidURLError(name); |       throw InvalidURLError(name); | ||||||
| @@ -27,9 +29,6 @@ class FDroid extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String? tryInferringAppId(String standardUrl, |   String? tryInferringAppId(String standardUrl, | ||||||
|       {Map<String, dynamic> additionalSettings = const {}}) { |       {Map<String, dynamic> additionalSettings = const {}}) { | ||||||
| @@ -51,10 +50,10 @@ class FDroid extends AppSource { | |||||||
|           .where((element) => element['versionName'] == latestVersion) |           .where((element) => element['versionName'] == latestVersion) | ||||||
|           .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk') |           .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk') | ||||||
|           .toList(); |           .toList(); | ||||||
|       return APKDetails(latestVersion, apkUrls, |       return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls), | ||||||
|           AppNames(name, Uri.parse(standardUrl).pathSegments.last)); |           AppNames(name, Uri.parse(standardUrl).pathSegments.last)); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -64,9 +63,10 @@ class FDroid extends AppSource { | |||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     String? appId = tryInferringAppId(standardUrl); |     String? appId = tryInferringAppId(standardUrl); | ||||||
|  |     String host = Uri.parse(standardUrl).host; | ||||||
|     return getAPKUrlsFromFDroidPackagesAPIResponse( |     return getAPKUrlsFromFDroidPackagesAPIResponse( | ||||||
|         await get(Uri.parse('https://f-droid.org/api/v1/packages/$appId')), |         await get(Uri.parse('https://$host/api/v1/packages/$appId')), | ||||||
|         'https://f-droid.org/repo/$appId', |         'https://$host/repo/$appId', | ||||||
|         standardUrl); |         standardUrl); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -69,6 +69,8 @@ class FDroidRepo extends AppSource { | |||||||
|           foundApps[0].querySelector('name')?.innerHtml ?? appIdOrName; |           foundApps[0].querySelector('name')?.innerHtml ?? appIdOrName; | ||||||
|       var releases = foundApps[0].querySelectorAll('package'); |       var releases = foundApps[0].querySelectorAll('package'); | ||||||
|       String? latestVersion = releases[0].querySelector('version')?.innerHtml; |       String? latestVersion = releases[0].querySelector('version')?.innerHtml; | ||||||
|  |       String? added = releases[0].querySelector('added')?.innerHtml; | ||||||
|  |       DateTime? releaseDate = added != null ? DateTime.parse(added) : null; | ||||||
|       if (latestVersion == null) { |       if (latestVersion == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
| @@ -78,9 +80,11 @@ class FDroidRepo extends AppSource { | |||||||
|               element.querySelector('apkname') != null) |               element.querySelector('apkname') != null) | ||||||
|           .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}') |           .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}') | ||||||
|           .toList(); |           .toList(); | ||||||
|       return APKDetails(latestVersion, apkUrls, AppNames(authorName, appName)); |       return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls), | ||||||
|  |           AppNames(authorName, appName), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -15,6 +15,7 @@ class GitHub extends AppSource { | |||||||
|     additionalSourceSpecificSettingFormItems = [ |     additionalSourceSpecificSettingFormItems = [ | ||||||
|       GeneratedFormTextField('github-creds', |       GeneratedFormTextField('github-creds', | ||||||
|           label: tr('githubPATLabel'), |           label: tr('githubPATLabel'), | ||||||
|  |           password: true, | ||||||
|           required: false, |           required: false, | ||||||
|           additionalValidators: [ |           additionalValidators: [ | ||||||
|             (value) { |             (value) { | ||||||
| @@ -64,15 +65,7 @@ class GitHub extends AppSource { | |||||||
|             required: false, |             required: false, | ||||||
|             additionalValidators: [ |             additionalValidators: [ | ||||||
|               (value) { |               (value) { | ||||||
|                 if (value == null || value.isEmpty) { |                 return regExValidator(value); | ||||||
|                   return null; |  | ||||||
|                 } |  | ||||||
|                 try { |  | ||||||
|                   RegExp(value); |  | ||||||
|                 } catch (e) { |  | ||||||
|                   return tr('invalidRegEx'); |  | ||||||
|                 } |  | ||||||
|                 return null; |  | ||||||
|               } |               } | ||||||
|             ]) |             ]) | ||||||
|       ] |       ] | ||||||
| @@ -108,9 +101,9 @@ class GitHub extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases']; |     bool includePrereleases = additionalSettings['includePrereleases'] == true; | ||||||
|     bool fallbackToOlderReleases = |     bool fallbackToOlderReleases = | ||||||
|         additionalSettings['fallbackToOlderReleases']; |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|     String? regexFilter = |     String? regexFilter = | ||||||
|         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) |         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) | ||||||
|                     ?.isNotEmpty == |                     ?.isNotEmpty == | ||||||
| @@ -118,7 +111,7 @@ class GitHub extends AppSource { | |||||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|             : null; |             : null; | ||||||
|     Response res = await get(Uri.parse( |     Response res = await get(Uri.parse( | ||||||
|         'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases')); |         'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var releases = jsonDecode(res.body) as List<dynamic>; |       var releases = jsonDecode(res.body) as List<dynamic>; | ||||||
|  |  | ||||||
| @@ -140,10 +133,13 @@ class GitHub extends AppSource { | |||||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { |         if (!includePrereleases && releases[i]['prerelease'] == true) { | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|  |         var nameToFilter = releases[i]['name'] as String?; | ||||||
|  |         if (nameToFilter == null || nameToFilter.trim().isEmpty) { | ||||||
|  |           // Some leave titles empty so tag is used | ||||||
|  |           nameToFilter = releases[i]['tag_name'] as String; | ||||||
|  |         } | ||||||
|         if (regexFilter != null && |         if (regexFilter != null && | ||||||
|             !RegExp(regexFilter) |             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { | ||||||
|                 .hasMatch((releases[i]['name'] as String).trim())) { |  | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|         var apkUrls = getReleaseAPKUrls(releases[i]); |         var apkUrls = getReleaseAPKUrls(releases[i]); | ||||||
| @@ -158,11 +154,19 @@ class GitHub extends AppSource { | |||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String? version = targetRelease['tag_name']; |       String? version = targetRelease['tag_name']; | ||||||
|  |       DateTime? releaseDate = targetRelease['published_at'] != null | ||||||
|  |           ? DateTime.parse(targetRelease['published_at']) | ||||||
|  |           : null; | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |       var changeLog = targetRelease['body'].toString(); | ||||||
|           getAppNames(standardUrl)); |       return APKDetails( | ||||||
|  |           version, | ||||||
|  |           getApkUrlsFromUrls(targetRelease['apkUrls'] as List<String>), | ||||||
|  |           getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate, | ||||||
|  |           changeLog: changeLog.isEmpty ? null : changeLog); | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       rateLimitErrorCheck(res); | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
| @@ -183,9 +187,11 @@ class GitHub extends AppSource { | |||||||
|       Map<String, String> urlsWithDescriptions = {}; |       Map<String, String> urlsWithDescriptions = {}; | ||||||
|       for (var e in (jsonDecode(res.body)['items'] as List<dynamic>)) { |       for (var e in (jsonDecode(res.body)['items'] as List<dynamic>)) { | ||||||
|         urlsWithDescriptions.addAll({ |         urlsWithDescriptions.addAll({ | ||||||
|           e['html_url'] as String: e['description'] != null |           e['html_url'] as String: | ||||||
|               ? e['description'] as String |               ((e['archived'] == true ? '[ARCHIVED] ' : '') + | ||||||
|               : tr('noDescription') |                   (e['description'] != null | ||||||
|  |                       ? e['description'] as String | ||||||
|  |                       : tr('noDescription'))) | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|       return urlsWithDescriptions; |       return urlsWithDescriptions; | ||||||
|   | |||||||
| @@ -54,12 +54,17 @@ class GitLab extends AppSource { | |||||||
|       var entryId = entry?.querySelector('id')?.innerHtml; |       var entryId = entry?.querySelector('id')?.innerHtml; | ||||||
|       var version = |       var version = | ||||||
|           entryId == null ? null : Uri.parse(entryId).pathSegments.last; |           entryId == null ? null : Uri.parse(entryId).pathSegments.last; | ||||||
|  |       var releaseDateString = entry?.querySelector('updated')?.innerHtml; | ||||||
|  |       DateTime? releaseDate = | ||||||
|  |           releaseDateString != null ? DateTime.parse(releaseDateString) : null; | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl)); |       return APKDetails(version, getApkUrlsFromUrls(apkUrls), | ||||||
|  |           GitHub().getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										51
									
								
								lib/app_sources/html.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | |||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class HTML extends AppSource { | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     return url; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     var uri = Uri.parse(standardUrl); | ||||||
|  |     Response res = await get(uri); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       List<String> links = parse(res.body) | ||||||
|  |           .querySelectorAll('a') | ||||||
|  |           .map((element) => element.attributes['href'] ?? '') | ||||||
|  |           .where((element) => element.toLowerCase().endsWith('.apk')) | ||||||
|  |           .toList(); | ||||||
|  |       links.sort((a, b) => a.split('/').last.compareTo(b.split('/').last)); | ||||||
|  |       if (additionalSettings['apkFilterRegEx'] != null) { | ||||||
|  |         var reg = RegExp(additionalSettings['apkFilterRegEx']); | ||||||
|  |         links = links.where((element) => reg.hasMatch(element)).toList(); | ||||||
|  |       } | ||||||
|  |       if (links.isEmpty) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|  |       } | ||||||
|  |       var rel = links.last; | ||||||
|  |       var apkName = rel.split('/').last; | ||||||
|  |       var version = apkName.substring(0, apkName.length - 4); | ||||||
|  |       List<String> apkUrls = [rel] | ||||||
|  |           .map((e) => e.toLowerCase().startsWith('http://') || | ||||||
|  |                   e.toLowerCase().startsWith('https://') | ||||||
|  |               ? e | ||||||
|  |               : e.startsWith('/') | ||||||
|  |                   ? '${uri.origin}/$e' | ||||||
|  |                   : '${uri.origin}/${uri.path}/$e') | ||||||
|  |           .toList(); | ||||||
|  |       return APKDetails( | ||||||
|  |           version, getApkUrlsFromUrls(apkUrls), AppNames(uri.host, tr('app'))); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -18,9 +18,6 @@ class IzzyOnDroid extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String? tryInferringAppId(String standardUrl, |   String? tryInferringAppId(String standardUrl, | ||||||
|       {Map<String, dynamic> additionalSettings = const {}}) { |       {Map<String, dynamic> additionalSettings = const {}}) { | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| import 'package:html/parser.dart'; | import 'package:html/parser.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| @@ -29,21 +30,39 @@ class Mullvad extends AppSource { | |||||||
|   ) async { |   ) async { | ||||||
|     Response res = await get(Uri.parse('$standardUrl/en/download/android')); |     Response res = await get(Uri.parse('$standardUrl/en/download/android')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var version = parse(res.body) |       var versions = parse(res.body) | ||||||
|           .querySelector('p.subtitle.is-6') |           .querySelectorAll('p') | ||||||
|           ?.querySelector('a') |           .map((e) => e.innerHtml) | ||||||
|           ?.attributes['href'] |           .where((p) => p.contains('Latest version: ')) | ||||||
|           ?.split('/') |           .map((e) { | ||||||
|           .last; |             var match = RegExp('[0-9]+(\\.[0-9]+)*').firstMatch(e); | ||||||
|       if (version == null) { |             if (match == null) { | ||||||
|  |               return ''; | ||||||
|  |             } else { | ||||||
|  |               return e.substring(match.start, match.end); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .where((element) => element.isNotEmpty) | ||||||
|  |           .toList(); | ||||||
|  |       if (versions.isEmpty) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|  |       String? changeLog; | ||||||
|  |       try { | ||||||
|  |         changeLog = (await GitHub().getLatestAPKDetails( | ||||||
|  |                 'https://github.com/mullvad/mullvadvpn-app', | ||||||
|  |                 {'fallbackToOlderReleases': true})) | ||||||
|  |             .changeLog; | ||||||
|  |       } catch (e) { | ||||||
|  |         // Ignore | ||||||
|  |       } | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, |           versions[0], | ||||||
|           ['https://mullvad.net/download/app/apk/latest'], |           getApkUrlsFromUrls(['https://mullvad.net/download/app/apk/latest']), | ||||||
|           AppNames(name, 'Mullvad-VPN')); |           AppNames(name, 'Mullvad-VPN'), | ||||||
|  |           changeLog: changeLog); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										111
									
								
								lib/app_sources/neutroncode.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,111 @@ | |||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class NeutronCode extends AppSource { | ||||||
|  |   NeutronCode() { | ||||||
|  |     host = 'neutroncode.com'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     RegExp standardUrlRegEx = RegExp('^https?://$host/downloads/file/[^/]+'); | ||||||
|  |     RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase()); | ||||||
|  |     if (match == null) { | ||||||
|  |       throw InvalidURLError(name); | ||||||
|  |     } | ||||||
|  |     return url.substring(0, match.end); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String? changeLogPageFromStandardUrl(String standardUrl) => standardUrl; | ||||||
|  |  | ||||||
|  |   String monthNameToNumberString(String s) { | ||||||
|  |     switch (s.toLowerCase()) { | ||||||
|  |       case 'january': | ||||||
|  |         return '01'; | ||||||
|  |       case 'february': | ||||||
|  |         return '02'; | ||||||
|  |       case 'march': | ||||||
|  |         return '03'; | ||||||
|  |       case 'april': | ||||||
|  |         return '04'; | ||||||
|  |       case 'may': | ||||||
|  |         return '05'; | ||||||
|  |       case 'june': | ||||||
|  |         return '06'; | ||||||
|  |       case 'july': | ||||||
|  |         return '07'; | ||||||
|  |       case 'august': | ||||||
|  |         return '08'; | ||||||
|  |       case 'september': | ||||||
|  |         return '09'; | ||||||
|  |       case 'october': | ||||||
|  |         return '10'; | ||||||
|  |       case 'november': | ||||||
|  |         return '11'; | ||||||
|  |       case 'december': | ||||||
|  |         return '12'; | ||||||
|  |       default: | ||||||
|  |         throw ArgumentError('Invalid month name: $s'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   customDateParse(String dateString) { | ||||||
|  |     List<String> parts = dateString.split(' '); | ||||||
|  |     if (parts.length != 3) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |     String result = ''; | ||||||
|  |     for (var s in parts.reversed) { | ||||||
|  |       try { | ||||||
|  |         try { | ||||||
|  |           int.parse(s); | ||||||
|  |           result += '$s-'; | ||||||
|  |         } catch (e) { | ||||||
|  |           result += '${monthNameToNumberString(s)}-'; | ||||||
|  |         } | ||||||
|  |       } catch (e) { | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return result.substring(0, result.length - 1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     Response res = await get(Uri.parse(standardUrl)); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var http = parse(res.body); | ||||||
|  |       var name = http.querySelector('.pd-title')?.innerHtml; | ||||||
|  |       var filename = http.querySelector('.pd-filename .pd-float')?.innerHtml; | ||||||
|  |       if (filename == null) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|  |       } | ||||||
|  |       var version = | ||||||
|  |           http.querySelector('.pd-version-txt')?.nextElementSibling?.innerHtml; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String? apkUrl = 'https://$host/download/$filename'; | ||||||
|  |       var dateStringOriginal = | ||||||
|  |           http.querySelector('.pd-date-txt')?.nextElementSibling?.innerHtml; | ||||||
|  |       var dateString = dateStringOriginal != null | ||||||
|  |           ? (customDateParse(dateStringOriginal)) | ||||||
|  |           : null; | ||||||
|  |       var changeLogElements = http.querySelectorAll('.pd-fdesc p'); | ||||||
|  |       return APKDetails(version, getApkUrlsFromUrls([apkUrl]), | ||||||
|  |           AppNames(runtimeType.toString(), name ?? standardUrl.split('/').last), | ||||||
|  |           releaseDate: dateString != null ? DateTime.parse(dateString) : null, | ||||||
|  |           changeLog: changeLogElements.isNotEmpty | ||||||
|  |               ? changeLogElements.last.innerHtml | ||||||
|  |               : null); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -13,9 +13,6 @@ class Signal extends AppSource { | |||||||
|     return 'https://$host'; |     return 'https://$host'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -31,9 +28,10 @@ class Signal extends AppSource { | |||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, apkUrls, AppNames(name, 'Signal')); |       return APKDetails( | ||||||
|  |           version, getApkUrlsFromUrls(apkUrls), AppNames(name, 'Signal')); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -18,9 +18,6 @@ class SourceForge extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -53,11 +50,11 @@ class SourceForge extends AppSource { | |||||||
|               .toList(); |               .toList(); | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, |           version, | ||||||
|           apkUrlList, |           getApkUrlsFromUrls(apkUrlList), | ||||||
|           AppNames( |           AppNames( | ||||||
|               name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1))); |               name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1))); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,7 +10,10 @@ class SteamMobile extends AppSource { | |||||||
|     host = 'store.steampowered.com'; |     host = 'store.steampowered.com'; | ||||||
|     name = tr('steam'); |     name = tr('steam'); | ||||||
|     additionalSourceAppSpecificSettingFormItems = [ |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|       [GeneratedFormDropdown('app', apks.entries.toList(), label: tr('app'))] |       [ | ||||||
|  |         GeneratedFormDropdown('app', apks.entries.toList(), | ||||||
|  |             label: tr('app'), defaultValue: apks.entries.toList()[0].key) | ||||||
|  |       ] | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -21,9 +24,6 @@ class SteamMobile extends AppSource { | |||||||
|     return 'https://$host'; |     return 'https://$host'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -35,7 +35,8 @@ class SteamMobile extends AppSource { | |||||||
|       if (apkNamePrefix == null) { |       if (apkNamePrefix == null) { | ||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String apkInURLRegexPattern = '/$apkNamePrefix-[^/]+\\.apk\$'; |       String apkInURLRegexPattern = | ||||||
|  |           '/$apkNamePrefix-([0-9]+\\.)*[0-9]+\\.apk\$'; | ||||||
|       var links = parse(res.body) |       var links = parse(res.body) | ||||||
|           .querySelectorAll('a') |           .querySelectorAll('a') | ||||||
|           .map((e) => e.attributes['href'] ?? '') |           .map((e) => e.attributes['href'] ?? '') | ||||||
| @@ -52,9 +53,10 @@ class SteamMobile extends AppSource { | |||||||
|       var version = links[0].substring( |       var version = links[0].substring( | ||||||
|           versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4); |           versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4); | ||||||
|       var apkUrls = [links[0]]; |       var apkUrls = [links[0]]; | ||||||
|       return APKDetails(version, apkUrls, AppNames(name, apks[apkNamePrefix]!)); |       return APKDetails(version, getApkUrlsFromUrls(apkUrls), | ||||||
|  |           AppNames(name, apks[apkNamePrefix]!)); | ||||||
|     } else { |     } else { | ||||||
|       throw NoReleasesError(); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										41
									
								
								lib/app_sources/telegramapp.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,41 @@ | |||||||
|  | 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 TelegramApp extends AppSource { | ||||||
|  |   TelegramApp() { | ||||||
|  |     host = 'telegram.org'; | ||||||
|  |     name = 'Telegram ${tr('app')}'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     return 'https://$host'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     Response res = await get(Uri.parse('https://t.me/s/TAndroidAPK')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var http = parse(res.body); | ||||||
|  |       var messages = | ||||||
|  |           http.querySelectorAll('.tgme_widget_message_text.js-message_text'); | ||||||
|  |       var version = messages.isNotEmpty | ||||||
|  |           ? messages.last.innerHtml.split('\n').first.trim().split(' ').first | ||||||
|  |           : null; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String? apkUrl = 'https://telegram.org/dl/android/apk'; | ||||||
|  |       return APKDetails(version, getApkUrlsFromUrls([apkUrl]), | ||||||
|  |           AppNames('Telegram', 'Telegram')); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										63
									
								
								lib/app_sources/vlc.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,63 @@ | |||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/app_sources/html.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class VLC extends AppSource { | ||||||
|  |   VLC() { | ||||||
|  |     host = 'videolan.org'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     return 'https://$host'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @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/'; | ||||||
|  |       Response res2 = await get(Uri.parse(targetUrl)); | ||||||
|  |       String mirrorDwBase = | ||||||
|  |           'https://plug-mirror.rcac.purdue.edu/vlc/vlc-android/$version/'; | ||||||
|  |       List<String> apkUrls = []; | ||||||
|  |       if (res2.statusCode == 200) { | ||||||
|  |         apkUrls = parse(res2.body) | ||||||
|  |             .querySelectorAll('a') | ||||||
|  |             .map((e) => e.attributes['href']) | ||||||
|  |             .where((h) => | ||||||
|  |                 h != null && h.isNotEmpty && h.toLowerCase().endsWith('.apk')) | ||||||
|  |             .map((e) => mirrorDwBase + e!) | ||||||
|  |             .toList(); | ||||||
|  |       } else { | ||||||
|  |         throw getObtainiumHttpError(res2); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       return APKDetails( | ||||||
|  |           version, getApkUrlsFromUrls(apkUrls), AppNames('VideoLAN', 'VLC')); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										75
									
								
								lib/app_sources/whatsapp.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,75 @@ | |||||||
|  | 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() { | ||||||
|  |     host = 'whatsapp.com'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     return 'https://$host'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<String> apkUrlPrefetchModifier(String apkUrl) async { | ||||||
|  |     Response res = await get(Uri.parse('https://www.whatsapp.com/android')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var targetLinks = parse(res.body) | ||||||
|  |           .querySelectorAll('a') | ||||||
|  |           .map((e) => e.attributes['href']) | ||||||
|  |           .where((e) => e != null) | ||||||
|  |           .where((e) => | ||||||
|  |               e!.contains('scontent.whatsapp.net') && | ||||||
|  |               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 { | ||||||
|  |     Response res = await get(Uri.parse('https://www.whatsapp.com/android')); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var targetElements = parse(res.body) | ||||||
|  |           .querySelectorAll('p') | ||||||
|  |           .where((element) => element.innerHtml.contains('Version ')) | ||||||
|  |           .toList(); | ||||||
|  |       if (targetElements.isEmpty) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       var vLines = targetElements[0] | ||||||
|  |           .innerHtml | ||||||
|  |           .split('\n') | ||||||
|  |           .where((element) => element.contains('Version ')) | ||||||
|  |           .toList(); | ||||||
|  |       if (vLines.isEmpty) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       var versionMatch = RegExp('[0-9]+(\\.[0-9]+)+').firstMatch(vLines[0]); | ||||||
|  |       if (versionMatch == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String version = | ||||||
|  |           vLines[0].substring(versionMatch.start, versionMatch.end); | ||||||
|  |       return APKDetails( | ||||||
|  |           version, | ||||||
|  |           getApkUrlsFromUrls([ | ||||||
|  |             'https://www.whatsapp.com/android?v=$version&=thisIsaPlaceholder&a=realURLPrefetchedAtDownloadTime' | ||||||
|  |           ]), | ||||||
|  |           AppNames('Meta', 'WhatsApp')); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -3,7 +3,6 @@ import 'dart:math'; | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; |  | ||||||
|  |  | ||||||
| abstract class GeneratedFormItem { | abstract class GeneratedFormItem { | ||||||
|   late String key; |   late String key; | ||||||
| @@ -24,6 +23,7 @@ class GeneratedFormTextField extends GeneratedFormItem { | |||||||
|   late bool required; |   late bool required; | ||||||
|   late int max; |   late int max; | ||||||
|   late String? hint; |   late String? hint; | ||||||
|  |   late bool password; | ||||||
|  |  | ||||||
|   GeneratedFormTextField(String key, |   GeneratedFormTextField(String key, | ||||||
|       {String label = 'Input', |       {String label = 'Input', | ||||||
| @@ -32,7 +32,8 @@ class GeneratedFormTextField extends GeneratedFormItem { | |||||||
|       List<String? Function(String? value)> additionalValidators = const [], |       List<String? Function(String? value)> additionalValidators = const [], | ||||||
|       this.required = true, |       this.required = true, | ||||||
|       this.max = 1, |       this.max = 1, | ||||||
|       this.hint}) |       this.hint, | ||||||
|  |       this.password = false}) | ||||||
|       : super(key, |       : super(key, | ||||||
|             label: label, |             label: label, | ||||||
|             belowWidgets: belowWidgets, |             belowWidgets: belowWidgets, | ||||||
| @@ -47,6 +48,7 @@ class GeneratedFormTextField extends GeneratedFormItem { | |||||||
|  |  | ||||||
| class GeneratedFormDropdown extends GeneratedFormItem { | class GeneratedFormDropdown extends GeneratedFormItem { | ||||||
|   late List<MapEntry<String, String>>? opts; |   late List<MapEntry<String, String>>? opts; | ||||||
|  |   List<String>? disabledOptKeys; | ||||||
|  |  | ||||||
|   GeneratedFormDropdown( |   GeneratedFormDropdown( | ||||||
|     String key, |     String key, | ||||||
| @@ -54,6 +56,7 @@ class GeneratedFormDropdown extends GeneratedFormItem { | |||||||
|     String label = 'Input', |     String label = 'Input', | ||||||
|     List<Widget> belowWidgets = const [], |     List<Widget> belowWidgets = const [], | ||||||
|     String defaultValue = '', |     String defaultValue = '', | ||||||
|  |     this.disabledOptKeys, | ||||||
|     List<String? Function(String? value)> additionalValidators = const [], |     List<String? Function(String? value)> additionalValidators = const [], | ||||||
|   }) : super(key, |   }) : super(key, | ||||||
|             label: label, |             label: label, | ||||||
| @@ -91,6 +94,7 @@ class GeneratedFormTagInput extends GeneratedFormItem { | |||||||
|   late bool singleSelect; |   late bool singleSelect; | ||||||
|   late WrapAlignment alignment; |   late WrapAlignment alignment; | ||||||
|   late String emptyMessage; |   late String emptyMessage; | ||||||
|  |   late bool showLabelWhenNotEmpty; | ||||||
|   GeneratedFormTagInput(String key, |   GeneratedFormTagInput(String key, | ||||||
|       {String label = 'Input', |       {String label = 'Input', | ||||||
|       List<Widget> belowWidgets = const [], |       List<Widget> belowWidgets = const [], | ||||||
| @@ -100,7 +104,8 @@ class GeneratedFormTagInput extends GeneratedFormItem { | |||||||
|       this.deleteConfirmationMessage, |       this.deleteConfirmationMessage, | ||||||
|       this.singleSelect = false, |       this.singleSelect = false, | ||||||
|       this.alignment = WrapAlignment.start, |       this.alignment = WrapAlignment.start, | ||||||
|       this.emptyMessage = 'Input'}) |       this.emptyMessage = 'Input', | ||||||
|  |       this.showLabelWhenNotEmpty = true}) | ||||||
|       : super(key, |       : super(key, | ||||||
|             label: label, |             label: label, | ||||||
|             belowWidgets: belowWidgets, |             belowWidgets: belowWidgets, | ||||||
| @@ -127,11 +132,27 @@ class GeneratedForm extends StatefulWidget { | |||||||
|   State<GeneratedForm> createState() => _GeneratedFormState(); |   State<GeneratedForm> createState() => _GeneratedFormState(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Generates a random light color | ||||||
|  | // Courtesy of ChatGPT 😭 (with a bugfix 🥳) | ||||||
|  | Color generateRandomLightColor() { | ||||||
|  |   // Create a random number generator | ||||||
|  |   final Random random = Random(); | ||||||
|  |  | ||||||
|  |   // Generate random hue, saturation, and value values | ||||||
|  |   final double hue = random.nextDouble() * 360; | ||||||
|  |   final double saturation = 0.5 + random.nextDouble() * 0.5; | ||||||
|  |   final double value = 0.9 + random.nextDouble() * 0.1; | ||||||
|  |  | ||||||
|  |   // Create a HSV color with the random values | ||||||
|  |   return HSVColor.fromAHSV(1.0, hue, saturation, value).toColor(); | ||||||
|  | } | ||||||
|  |  | ||||||
| class _GeneratedFormState extends State<GeneratedForm> { | class _GeneratedFormState extends State<GeneratedForm> { | ||||||
|   final _formKey = GlobalKey<FormState>(); |   final _formKey = GlobalKey<FormState>(); | ||||||
|   Map<String, dynamic> values = {}; |   Map<String, dynamic> values = {}; | ||||||
|   late List<List<Widget>> formInputs; |   late List<List<Widget>> formInputs; | ||||||
|   List<List<Widget>> rows = []; |   List<List<Widget>> rows = []; | ||||||
|  |   String? initKey; | ||||||
|  |  | ||||||
|   // If any value changes, call this to update the parent with value and validity |   // If any value changes, call this to update the parent with value and validity | ||||||
|   void someValueChanged({bool isBuilding = false}) { |   void someValueChanged({bool isBuilding = false}) { | ||||||
| @@ -140,39 +161,21 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|     for (int r = 0; r < widget.items.length; r++) { |     for (int r = 0; r < widget.items.length; r++) { | ||||||
|       for (int i = 0; i < widget.items[r].length; i++) { |       for (int i = 0; i < widget.items[r].length; i++) { | ||||||
|         if (formInputs[r][i] is TextFormField) { |         if (formInputs[r][i] is TextFormField) { | ||||||
|           valid = valid && |           var fieldState = | ||||||
|               ((formInputs[r][i].key as GlobalKey<FormFieldState>) |               (formInputs[r][i].key as GlobalKey<FormFieldState>).currentState; | ||||||
|                       .currentState |           if (fieldState != null) { | ||||||
|                       ?.isValid ?? |             valid = valid && fieldState.isValid; | ||||||
|                   false); |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     widget.onValueChanges(returnValues, valid, isBuilding); |     widget.onValueChanges(returnValues, valid, isBuilding); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Generates a random light color |   initForm() { | ||||||
| // Courtesy of ChatGPT 😭 (with a bugfix 🥳) |     initKey = widget.key.toString(); | ||||||
|   Color generateRandomLightColor() { |  | ||||||
|     // Create a random number generator |  | ||||||
|     final Random random = Random(); |  | ||||||
|  |  | ||||||
|     // Generate random hue, saturation, and value values |  | ||||||
|     final double hue = random.nextDouble() * 360; |  | ||||||
|     final double saturation = 0.5 + random.nextDouble() * 0.5; |  | ||||||
|     final double value = 0.9 + random.nextDouble() * 0.1; |  | ||||||
|  |  | ||||||
|     // Create a HSV color with the random values |  | ||||||
|     return HSVColor.fromAHSV(1.0, hue, saturation, value).toColor(); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   void initState() { |  | ||||||
|     super.initState(); |  | ||||||
|  |  | ||||||
|     // Initialize form values as all empty |     // Initialize form values as all empty | ||||||
|     values.clear(); |     values.clear(); | ||||||
|     int j = 0; |  | ||||||
|     for (var row in widget.items) { |     for (var row in widget.items) { | ||||||
|       for (var e in row) { |       for (var e in row) { | ||||||
|         values[e.key] = e.defaultValue; |         values[e.key] = e.defaultValue; | ||||||
| @@ -186,6 +189,9 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|         if (formItem is GeneratedFormTextField) { |         if (formItem is GeneratedFormTextField) { | ||||||
|           final formFieldKey = GlobalKey<FormFieldState>(); |           final formFieldKey = GlobalKey<FormFieldState>(); | ||||||
|           return TextFormField( |           return TextFormField( | ||||||
|  |             obscureText: formItem.password, | ||||||
|  |             autocorrect: !formItem.password, | ||||||
|  |             enableSuggestions: !formItem.password, | ||||||
|             key: formFieldKey, |             key: formFieldKey, | ||||||
|             initialValue: values[formItem.key], |             initialValue: values[formItem.key], | ||||||
|             autovalidateMode: AutovalidateMode.onUserInteraction, |             autovalidateMode: AutovalidateMode.onUserInteraction, | ||||||
| @@ -221,10 +227,15 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|           return DropdownButtonFormField( |           return DropdownButtonFormField( | ||||||
|               decoration: InputDecoration(labelText: formItem.label), |               decoration: InputDecoration(labelText: formItem.label), | ||||||
|               value: values[formItem.key], |               value: values[formItem.key], | ||||||
|               items: formItem.opts! |               items: formItem.opts!.map((e2) { | ||||||
|                   .map((e2) => |                 var enabled = | ||||||
|                       DropdownMenuItem(value: e2.key, child: Text(e2.value))) |                     formItem.disabledOptKeys?.contains(e2.key) != true; | ||||||
|                   .toList(), |                 return DropdownMenuItem( | ||||||
|  |                     value: e2.key, | ||||||
|  |                     enabled: enabled, | ||||||
|  |                     child: Opacity( | ||||||
|  |                         opacity: enabled ? 1 : 0.5, child: Text(e2.value))); | ||||||
|  |               }).toList(), | ||||||
|               onChanged: (value) { |               onChanged: (value) { | ||||||
|                 setState(() { |                 setState(() { | ||||||
|                   values[formItem.key] = value ?? formItem.opts!.first.key; |                   values[formItem.key] = value ?? formItem.opts!.first.key; | ||||||
| @@ -239,8 +250,17 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|     someValueChanged(isBuilding: true); |     someValueChanged(isBuilding: true); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void initState() { | ||||||
|  |     super.initState(); | ||||||
|  |     initForm(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|  |     if (widget.key.toString() != initKey) { | ||||||
|  |       initForm(); | ||||||
|  |     } | ||||||
|     for (var r = 0; r < formInputs.length; r++) { |     for (var r = 0; r < formInputs.length; r++) { | ||||||
|       for (var e = 0; e < formInputs[r].length; e++) { |       for (var e = 0; e < formInputs[r].length; e++) { | ||||||
|         if (widget.items[r][e] is GeneratedFormSwitch) { |         if (widget.items[r][e] is GeneratedFormSwitch) { | ||||||
| @@ -259,157 +279,185 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|             ], |             ], | ||||||
|           ); |           ); | ||||||
|         } else if (widget.items[r][e] is GeneratedFormTagInput) { |         } else if (widget.items[r][e] is GeneratedFormTagInput) { | ||||||
|           formInputs[r][e] = Wrap( |           formInputs[r][e] = | ||||||
|             alignment: (widget.items[r][e] as GeneratedFormTagInput).alignment, |               Column(crossAxisAlignment: CrossAxisAlignment.stretch, children: [ | ||||||
|             crossAxisAlignment: WrapCrossAlignment.center, |             if ((values[widget.items[r][e].key] | ||||||
|             children: [ |                             as Map<String, MapEntry<int, bool>>?) | ||||||
|               (values[widget.items[r][e].key] |                         ?.isNotEmpty == | ||||||
|                               as Map<String, MapEntry<int, bool>>?) |                     true && | ||||||
|                           ?.isEmpty == |                 (widget.items[r][e] as GeneratedFormTagInput) | ||||||
|                       true |                     .showLabelWhenNotEmpty) | ||||||
|                   ? Text( |               Column( | ||||||
|                       (widget.items[r][e] as GeneratedFormTagInput) |                 crossAxisAlignment: | ||||||
|                           .emptyMessage, |                     (widget.items[r][e] as GeneratedFormTagInput).alignment == | ||||||
|                       style: const TextStyle(fontWeight: FontWeight.bold), |                             WrapAlignment.center | ||||||
|                     ) |                         ? CrossAxisAlignment.center | ||||||
|                   : const SizedBox.shrink(), |                         : CrossAxisAlignment.stretch, | ||||||
|               ...(values[widget.items[r][e].key] |                 children: [ | ||||||
|                           as Map<String, MapEntry<int, bool>>?) |                   Text(widget.items[r][e].label), | ||||||
|                       ?.entries |                   const SizedBox( | ||||||
|                       .map((e2) { |                     height: 8, | ||||||
|                     return Padding( |                   ), | ||||||
|                         padding: const EdgeInsets.symmetric(horizontal: 4), |                 ], | ||||||
|                         child: ChoiceChip( |               ), | ||||||
|                           label: Text(e2.key), |             Wrap( | ||||||
|                           backgroundColor: Color(e2.value.key).withAlpha(50), |               alignment: | ||||||
|                           selectedColor: Color(e2.value.key), |                   (widget.items[r][e] as GeneratedFormTagInput).alignment, | ||||||
|                           visualDensity: VisualDensity.compact, |               crossAxisAlignment: WrapCrossAlignment.center, | ||||||
|                           selected: e2.value.value, |               children: [ | ||||||
|                           onSelected: (value) { |                 (values[widget.items[r][e].key] | ||||||
|                             setState(() { |                                 as Map<String, MapEntry<int, bool>>?) | ||||||
|                               (values[widget.items[r][e].key] as Map<String, |                             ?.isEmpty == | ||||||
|                                       MapEntry<int, bool>>)[e2.key] = |                         true | ||||||
|                                   MapEntry( |                     ? Text( | ||||||
|                                       (values[widget.items[r][e].key] as Map< |                         (widget.items[r][e] as GeneratedFormTagInput) | ||||||
|                                               String, |                             .emptyMessage, | ||||||
|                                               MapEntry<int, bool>>)[e2.key]! |                       ) | ||||||
|                                           .key, |                     : const SizedBox.shrink(), | ||||||
|                                       value); |                 ...(values[widget.items[r][e].key] | ||||||
|                               if ((widget.items[r][e] as GeneratedFormTagInput) |                             as Map<String, MapEntry<int, bool>>?) | ||||||
|                                       .singleSelect && |                         ?.entries | ||||||
|                                   value == true) { |                         .map((e2) { | ||||||
|                                 for (var key in (values[widget.items[r][e].key] |                       return Padding( | ||||||
|                                         as Map<String, MapEntry<int, bool>>) |                           padding: const EdgeInsets.symmetric(horizontal: 4), | ||||||
|                                     .keys) { |                           child: ChoiceChip( | ||||||
|                                   if (key != e2.key) { |                             label: Text(e2.key), | ||||||
|                                     (values[widget.items[r][e].key] as Map< |                             backgroundColor: Color(e2.value.key).withAlpha(50), | ||||||
|                                         String, |                             selectedColor: Color(e2.value.key), | ||||||
|                                         MapEntry<int, |                             visualDensity: VisualDensity.compact, | ||||||
|                                             bool>>)[key] = MapEntry( |                             selected: e2.value.value, | ||||||
|  |                             onSelected: (value) { | ||||||
|  |                               setState(() { | ||||||
|  |                                 (values[widget.items[r][e].key] as Map<String, | ||||||
|  |                                         MapEntry<int, bool>>)[e2.key] = | ||||||
|  |                                     MapEntry( | ||||||
|                                         (values[widget.items[r][e].key] as Map< |                                         (values[widget.items[r][e].key] as Map< | ||||||
|                                                 String, |                                                 String, | ||||||
|                                                 MapEntry<int, bool>>)[key]! |                                                 MapEntry<int, bool>>)[e2.key]! | ||||||
|                                             .key, |                                             .key, | ||||||
|                                         false); |                                         value); | ||||||
|  |                                 if ((widget.items[r][e] | ||||||
|  |                                             as GeneratedFormTagInput) | ||||||
|  |                                         .singleSelect && | ||||||
|  |                                     value == true) { | ||||||
|  |                                   for (var key in (values[ | ||||||
|  |                                               widget.items[r][e].key] | ||||||
|  |                                           as Map<String, MapEntry<int, bool>>) | ||||||
|  |                                       .keys) { | ||||||
|  |                                     if (key != e2.key) { | ||||||
|  |                                       (values[widget.items[r][e].key] as Map< | ||||||
|  |                                               String, | ||||||
|  |                                               MapEntry<int, bool>>)[key] = | ||||||
|  |                                           MapEntry( | ||||||
|  |                                               (values[widget.items[r][e].key] | ||||||
|  |                                                       as Map< | ||||||
|  |                                                           String, | ||||||
|  |                                                           MapEntry<int, | ||||||
|  |                                                               bool>>)[key]! | ||||||
|  |                                                   .key, | ||||||
|  |                                               false); | ||||||
|  |                                     } | ||||||
|                                   } |                                   } | ||||||
|                                 } |                                 } | ||||||
|                               } |                                 someValueChanged(); | ||||||
|                               someValueChanged(); |                               }); | ||||||
|                             }); |                             }, | ||||||
|  |                           )); | ||||||
|  |                     }) ?? | ||||||
|  |                     [const SizedBox.shrink()], | ||||||
|  |                 (values[widget.items[r][e].key] | ||||||
|  |                                 as Map<String, MapEntry<int, bool>>?) | ||||||
|  |                             ?.values | ||||||
|  |                             .where((e) => e.value) | ||||||
|  |                             .isNotEmpty == | ||||||
|  |                         true | ||||||
|  |                     ? Padding( | ||||||
|  |                         padding: const EdgeInsets.symmetric(horizontal: 4), | ||||||
|  |                         child: IconButton( | ||||||
|  |                           onPressed: () { | ||||||
|  |                             fn() { | ||||||
|  |                               setState(() { | ||||||
|  |                                 var temp = values[widget.items[r][e].key] | ||||||
|  |                                     as Map<String, MapEntry<int, bool>>; | ||||||
|  |                                 temp.removeWhere((key, value) => value.value); | ||||||
|  |                                 values[widget.items[r][e].key] = temp; | ||||||
|  |                                 someValueChanged(); | ||||||
|  |                               }); | ||||||
|  |                             } | ||||||
|  |  | ||||||
|  |                             if ((widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                                     .deleteConfirmationMessage != | ||||||
|  |                                 null) { | ||||||
|  |                               var message = | ||||||
|  |                                   (widget.items[r][e] as GeneratedFormTagInput) | ||||||
|  |                                       .deleteConfirmationMessage!; | ||||||
|  |                               showDialog<Map<String, dynamic>?>( | ||||||
|  |                                   context: context, | ||||||
|  |                                   builder: (BuildContext ctx) { | ||||||
|  |                                     return GeneratedFormModal( | ||||||
|  |                                         title: message.key, | ||||||
|  |                                         message: message.value, | ||||||
|  |                                         items: const []); | ||||||
|  |                                   }).then((value) { | ||||||
|  |                                 if (value != null) { | ||||||
|  |                                   fn(); | ||||||
|  |                                 } | ||||||
|  |                               }); | ||||||
|  |                             } else { | ||||||
|  |                               fn(); | ||||||
|  |                             } | ||||||
|                           }, |                           }, | ||||||
|                         )); |                           icon: const Icon(Icons.remove), | ||||||
|                   }) ?? |                           visualDensity: VisualDensity.compact, | ||||||
|                   [const SizedBox.shrink()], |                           tooltip: tr('remove'), | ||||||
|               (values[widget.items[r][e].key] |                         )) | ||||||
|                               as Map<String, MapEntry<int, bool>>?) |                     : const SizedBox.shrink(), | ||||||
|                           ?.values |                 Padding( | ||||||
|                           .where((e) => e.value) |                     padding: const EdgeInsets.symmetric(horizontal: 4), | ||||||
|                           .isNotEmpty == |                     child: IconButton( | ||||||
|                       true |                       onPressed: () { | ||||||
|                   ? Padding( |                         showDialog<Map<String, dynamic>?>( | ||||||
|                       padding: const EdgeInsets.symmetric(horizontal: 4), |                             context: context, | ||||||
|                       child: IconButton( |                             builder: (BuildContext ctx) { | ||||||
|                         onPressed: () { |                               return GeneratedFormModal( | ||||||
|                           fn() { |                                   title: widget.items[r][e].label, | ||||||
|  |                                   items: [ | ||||||
|  |                                     [ | ||||||
|  |                                       GeneratedFormTextField('label', | ||||||
|  |                                           label: tr('label')) | ||||||
|  |                                     ] | ||||||
|  |                                   ]); | ||||||
|  |                             }).then((value) { | ||||||
|  |                           String? label = value?['label']; | ||||||
|  |                           if (label != null) { | ||||||
|                             setState(() { |                             setState(() { | ||||||
|                               var temp = values[widget.items[r][e].key] |                               var temp = values[widget.items[r][e].key] | ||||||
|                                   as Map<String, MapEntry<int, bool>>; |                                   as Map<String, MapEntry<int, bool>>?; | ||||||
|                               temp.removeWhere((key, value) => value.value); |                               temp ??= {}; | ||||||
|                               values[widget.items[r][e].key] = temp; |                               if (temp[label] == null) { | ||||||
|                               someValueChanged(); |                                 var singleSelect = (widget.items[r][e] | ||||||
|                             }); |                                         as GeneratedFormTagInput) | ||||||
|                           } |                                     .singleSelect; | ||||||
|  |                                 var someSelected = temp.entries | ||||||
|                           if ((widget.items[r][e] as GeneratedFormTagInput) |                                     .where((element) => element.value.value) | ||||||
|                                   .deleteConfirmationMessage != |                                     .isNotEmpty; | ||||||
|                               null) { |                                 temp[label] = MapEntry( | ||||||
|                             var message = |                                     generateRandomLightColor().value, | ||||||
|                                 (widget.items[r][e] as GeneratedFormTagInput) |                                     !(someSelected && singleSelect)); | ||||||
|                                     .deleteConfirmationMessage!; |                                 values[widget.items[r][e].key] = temp; | ||||||
|                             showDialog<Map<String, dynamic>?>( |                                 someValueChanged(); | ||||||
|                                 context: context, |  | ||||||
|                                 builder: (BuildContext ctx) { |  | ||||||
|                                   return GeneratedFormModal( |  | ||||||
|                                       title: message.key, |  | ||||||
|                                       message: message.value, |  | ||||||
|                                       items: const []); |  | ||||||
|                                 }).then((value) { |  | ||||||
|                               if (value != null) { |  | ||||||
|                                 fn(); |  | ||||||
|                               } |                               } | ||||||
|                             }); |                             }); | ||||||
|                           } else { |  | ||||||
|                             fn(); |  | ||||||
|                           } |                           } | ||||||
|                         }, |                         }); | ||||||
|                         icon: const Icon(Icons.remove), |                       }, | ||||||
|                         visualDensity: VisualDensity.compact, |                       icon: const Icon(Icons.add), | ||||||
|                         tooltip: tr('remove'), |                       visualDensity: VisualDensity.compact, | ||||||
|                       )) |                       tooltip: tr('add'), | ||||||
|                   : const SizedBox.shrink(), |                     )), | ||||||
|               Padding( |               ], | ||||||
|                   padding: const EdgeInsets.symmetric(horizontal: 4), |             ) | ||||||
|                   child: IconButton( |           ]); | ||||||
|                     onPressed: () { |  | ||||||
|                       showDialog<Map<String, dynamic>?>( |  | ||||||
|                           context: context, |  | ||||||
|                           builder: (BuildContext ctx) { |  | ||||||
|                             return GeneratedFormModal( |  | ||||||
|                                 title: widget.items[r][e].label, |  | ||||||
|                                 items: [ |  | ||||||
|                                   [ |  | ||||||
|                                     GeneratedFormTextField('label', |  | ||||||
|                                         label: tr('label')) |  | ||||||
|                                   ] |  | ||||||
|                                 ]); |  | ||||||
|                           }).then((value) { |  | ||||||
|                         String? label = value?['label']; |  | ||||||
|                         if (label != null) { |  | ||||||
|                           setState(() { |  | ||||||
|                             var temp = values[widget.items[r][e].key] |  | ||||||
|                                 as Map<String, MapEntry<int, bool>>?; |  | ||||||
|                             temp ??= {}; |  | ||||||
|                             var singleSelect = |  | ||||||
|                                 (widget.items[r][e] as GeneratedFormTagInput) |  | ||||||
|                                     .singleSelect; |  | ||||||
|                             var someSelected = temp.entries |  | ||||||
|                                 .where((element) => element.value.value) |  | ||||||
|                                 .isNotEmpty; |  | ||||||
|                             temp[label] = MapEntry( |  | ||||||
|                                 generateRandomLightColor().value, |  | ||||||
|                                 !(someSelected && singleSelect)); |  | ||||||
|                             values[widget.items[r][e].key] = temp; |  | ||||||
|                             someValueChanged(); |  | ||||||
|                           }); |  | ||||||
|                         } |  | ||||||
|                       }); |  | ||||||
|                     }, |  | ||||||
|                     icon: const Icon(Icons.add), |  | ||||||
|                     visualDensity: VisualDensity.compact, |  | ||||||
|                     tooltip: tr('add'), |  | ||||||
|                   )), |  | ||||||
|             ], |  | ||||||
|           ); |  | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -419,10 +467,9 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|       if (rowInputs.key > 0) { |       if (rowInputs.key > 0) { | ||||||
|         rows.add([ |         rows.add([ | ||||||
|           SizedBox( |           SizedBox( | ||||||
|             height: widget.items[rowInputs.key][0] is GeneratedFormSwitch && |             height: widget.items[rowInputs.key - 1][0] is GeneratedFormSwitch | ||||||
|                     widget.items[rowInputs.key - 1][0] is! GeneratedFormSwitch |                 ? 8 | ||||||
|                 ? 25 |                 : 25, | ||||||
|                 : 8, |  | ||||||
|           ) |           ) | ||||||
|         ]); |         ]); | ||||||
|       } |       } | ||||||
| @@ -436,6 +483,7 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|         rowItems.add(Expanded( |         rowItems.add(Expanded( | ||||||
|             child: Column( |             child: Column( | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.stretch, |                 crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |                 mainAxisSize: MainAxisSize.min, | ||||||
|                 children: [ |                 children: [ | ||||||
|               rowInput.value, |               rowInput.value, | ||||||
|               ...widget.items[rowInputs.key][rowInput.key].belowWidgets |               ...widget.items[rowInputs.key][rowInput.key].belowWidgets | ||||||
|   | |||||||
| @@ -9,12 +9,16 @@ class GeneratedFormModal extends StatefulWidget { | |||||||
|       required this.title, |       required this.title, | ||||||
|       required this.items, |       required this.items, | ||||||
|       this.initValid = false, |       this.initValid = false, | ||||||
|       this.message = ''}); |       this.message = '', | ||||||
|  |       this.additionalWidgets = const [], | ||||||
|  |       this.singleNullReturnButton}); | ||||||
|  |  | ||||||
|   final String title; |   final String title; | ||||||
|   final String message; |   final String message; | ||||||
|   final List<List<GeneratedFormItem>> items; |   final List<List<GeneratedFormItem>> items; | ||||||
|   final bool initValid; |   final bool initValid; | ||||||
|  |   final List<Widget> additionalWidgets; | ||||||
|  |   final String? singleNullReturnButton; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   State<GeneratedFormModal> createState() => _GeneratedFormModalState(); |   State<GeneratedFormModal> createState() => _GeneratedFormModalState(); | ||||||
| @@ -54,24 +58,29 @@ class _GeneratedFormModalState extends State<GeneratedFormModal> { | |||||||
|                   this.valid = valid; |                   this.valid = valid; | ||||||
|                 }); |                 }); | ||||||
|               } |               } | ||||||
|             }) |             }), | ||||||
|  |         if (widget.additionalWidgets.isNotEmpty) ...widget.additionalWidgets | ||||||
|       ]), |       ]), | ||||||
|       actions: [ |       actions: [ | ||||||
|         TextButton( |         TextButton( | ||||||
|             onPressed: () { |             onPressed: () { | ||||||
|               Navigator.of(context).pop(null); |               Navigator.of(context).pop(null); | ||||||
|             }, |             }, | ||||||
|             child: Text(tr('cancel'))), |             child: Text(widget.singleNullReturnButton == null | ||||||
|         TextButton( |                 ? tr('cancel') | ||||||
|             onPressed: !valid |                 : widget.singleNullReturnButton!)), | ||||||
|                 ? null |         widget.singleNullReturnButton == null | ||||||
|                 : () { |             ? TextButton( | ||||||
|                     if (valid) { |                 onPressed: !valid | ||||||
|                       HapticFeedback.selectionClick(); |                     ? null | ||||||
|                       Navigator.of(context).pop(values); |                     : () { | ||||||
|                     } |                         if (valid) { | ||||||
|                   }, |                           HapticFeedback.selectionClick(); | ||||||
|             child: Text(tr('continue'))) |                           Navigator.of(context).pop(values); | ||||||
|  |                         } | ||||||
|  |                       }, | ||||||
|  |                 child: Text(tr('continue'))) | ||||||
|  |             : const SizedBox.shrink() | ||||||
|       ], |       ], | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -13,13 +13,10 @@ class ObtainiumError { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| class RateLimitError { | class RateLimitError extends ObtainiumError { | ||||||
|   late int remainingMinutes; |   late int remainingMinutes; | ||||||
|   RateLimitError(this.remainingMinutes); |   RateLimitError(this.remainingMinutes) | ||||||
|  |       : super(plural('tooManyRequestsTryAgainInMinutes', remainingMinutes)); | ||||||
|   @override |  | ||||||
|   String toString() => |  | ||||||
|       plural('tooManyRequestsTryAgainInMinutes', remainingMinutes); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| class InvalidURLError extends ObtainiumError { | class InvalidURLError extends ObtainiumError { | ||||||
| @@ -32,7 +29,7 @@ class NoReleasesError extends ObtainiumError { | |||||||
| } | } | ||||||
|  |  | ||||||
| class NoAPKError extends ObtainiumError { | class NoAPKError extends ObtainiumError { | ||||||
|   NoAPKError() : super(tr('noReleaseFound')); |   NoAPKError() : super(tr('noAPKFound')); | ||||||
| } | } | ||||||
|  |  | ||||||
| class NoVersionError extends ObtainiumError { | class NoVersionError extends ObtainiumError { | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart'; | |||||||
| // ignore: implementation_imports | // ignore: implementation_imports | ||||||
| import 'package:easy_localization/src/localization.dart'; | import 'package:easy_localization/src/localization.dart'; | ||||||
|  |  | ||||||
| const String currentVersion = '0.9.6'; | const String currentVersion = '0.11.25'; | ||||||
| const String currentReleaseTag = | const String currentReleaseTag = | ||||||
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES |     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES | ||||||
|  |  | ||||||
| @@ -33,7 +33,9 @@ const supportedLocales = [ | |||||||
|   Locale('it'), |   Locale('it'), | ||||||
|   Locale('ja'), |   Locale('ja'), | ||||||
|   Locale('hu'), |   Locale('hu'), | ||||||
|   Locale('de') |   Locale('de'), | ||||||
|  |   Locale('fa'), | ||||||
|  |   Locale('fr') | ||||||
| ]; | ]; | ||||||
| const fallbackLocale = Locale('en'); | const fallbackLocale = Locale('en'); | ||||||
| const localeDir = 'assets/translations'; | const localeDir = 'assets/translations'; | ||||||
| @@ -145,6 +147,14 @@ Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async { | |||||||
|  |  | ||||||
| void main() async { | void main() async { | ||||||
|   WidgetsFlutterBinding.ensureInitialized(); |   WidgetsFlutterBinding.ensureInitialized(); | ||||||
|  |   try { | ||||||
|  |     ByteData data = | ||||||
|  |         await PlatformAssetBundle().load('assets/ca/lets-encrypt-r3.pem'); | ||||||
|  |     SecurityContext.defaultContext | ||||||
|  |         .setTrustedCertificatesBytes(data.buffer.asUint8List()); | ||||||
|  |   } catch (e) { | ||||||
|  |     // Already added, do nothing (see #375) | ||||||
|  |   } | ||||||
|   await EasyLocalization.ensureInitialized(); |   await EasyLocalization.ensureInitialized(); | ||||||
|   if ((await DeviceInfoPlugin().androidInfo).version.sdkInt >= 29) { |   if ((await DeviceInfoPlugin().androidInfo).version.sdkInt >= 29) { | ||||||
|     SystemChrome.setSystemUIOverlayStyle( |     SystemChrome.setSystemUIOverlayStyle( | ||||||
| @@ -208,7 +218,15 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|               {'includePrereleases': true}, |               {'includePrereleases': true}, | ||||||
|               null, |               null, | ||||||
|               false) |               false) | ||||||
|         ]); |         ], onlyIfExists: false); | ||||||
|  |       } | ||||||
|  |       if (!supportedLocales | ||||||
|  |               .map((e) => e.languageCode) | ||||||
|  |               .contains(context.locale.languageCode) || | ||||||
|  |           settingsProvider.forcedLocale == null && | ||||||
|  |               context.deviceLocale.languageCode != | ||||||
|  |                   context.locale.languageCode) { | ||||||
|  |         settingsProvider.resetLocaleSafe(context); | ||||||
|       } |       } | ||||||
|       // Register the background update task according to the user's setting |       // Register the background update task according to the user's setting | ||||||
|       if (existingUpdateInterval != settingsProvider.updateInterval) { |       if (existingUpdateInterval != settingsProvider.updateInterval) { | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ import 'package:obtainium/custom_errors.dart'; | |||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
| import 'package:obtainium/pages/app.dart'; | import 'package:obtainium/pages/app.dart'; | ||||||
| import 'package:obtainium/pages/import_export.dart'; | import 'package:obtainium/pages/import_export.dart'; | ||||||
|  | import 'package:obtainium/pages/settings.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| @@ -23,96 +24,103 @@ class AddAppPage extends StatefulWidget { | |||||||
|  |  | ||||||
| class _AddAppPageState extends State<AddAppPage> { | class _AddAppPageState extends State<AddAppPage> { | ||||||
|   bool gettingAppInfo = false; |   bool gettingAppInfo = false; | ||||||
|  |   bool searching = false; | ||||||
|  |  | ||||||
|   String userInput = ''; |   String userInput = ''; | ||||||
|   String searchQuery = ''; |   String searchQuery = ''; | ||||||
|   AppSource? pickedSource; |   AppSource? pickedSource; | ||||||
|   Map<String, dynamic> additionalSettings = {}; |   Map<String, dynamic> additionalSettings = {}; | ||||||
|   bool additionalSettingsValid = true; |   bool additionalSettingsValid = true; | ||||||
|  |   List<String> pickedCategories = []; | ||||||
|  |   int searchnum = 0; | ||||||
|  |   SourceProvider sourceProvider = SourceProvider(); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |  | ||||||
|     AppsProvider appsProvider = context.read<AppsProvider>(); |     AppsProvider appsProvider = context.read<AppsProvider>(); | ||||||
|  |  | ||||||
|     changeUserInput(String input, bool valid, bool isBuilding) { |     bool doingSomething = gettingAppInfo || searching; | ||||||
|       userInput = input; |  | ||||||
|       fn() { |  | ||||||
|         var source = valid ? sourceProvider.getSource(userInput) : null; |  | ||||||
|         if (pickedSource.runtimeType != source.runtimeType) { |  | ||||||
|           pickedSource = source; |  | ||||||
|           additionalSettings = source != null |  | ||||||
|               ? getDefaultValuesFromFormItems( |  | ||||||
|                   source.combinedAppSpecificSettingFormItems) |  | ||||||
|               : {}; |  | ||||||
|           additionalSettingsValid = source != null |  | ||||||
|               ? !sourceProvider.ifRequiredAppSpecificSettingsExist(source) |  | ||||||
|               : true; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (isBuilding) { |     changeUserInput(String input, bool valid, bool isBuilding, | ||||||
|         fn(); |         {bool isSearch = false}) { | ||||||
|       } else { |       userInput = input; | ||||||
|  |       if (!isBuilding) { | ||||||
|         setState(() { |         setState(() { | ||||||
|           fn(); |           if (isSearch) { | ||||||
|  |             searchnum++; | ||||||
|  |           } | ||||||
|  |           var source = valid ? sourceProvider.getSource(userInput) : null; | ||||||
|  |           if (pickedSource.runtimeType != source.runtimeType) { | ||||||
|  |             pickedSource = source; | ||||||
|  |             additionalSettings = source != null | ||||||
|  |                 ? getDefaultValuesFromFormItems( | ||||||
|  |                     source.combinedAppSpecificSettingFormItems) | ||||||
|  |                 : {}; | ||||||
|  |             additionalSettingsValid = source != null | ||||||
|  |                 ? !sourceProvider.ifRequiredAppSpecificSettingsExist(source) | ||||||
|  |                 : true; | ||||||
|  |           } | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     getTrackOnlyConfirmationIfNeeded(bool userPickedTrackOnly) async { | ||||||
|  |       return (!((userPickedTrackOnly || pickedSource!.enforceTrackOnly) && | ||||||
|  |           // ignore: use_build_context_synchronously | ||||||
|  |           await showDialog( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return GeneratedFormModal( | ||||||
|  |                       title: tr('xIsTrackOnly', args: [ | ||||||
|  |                         pickedSource!.enforceTrackOnly | ||||||
|  |                             ? tr('source') | ||||||
|  |                             : tr('app') | ||||||
|  |                       ]), | ||||||
|  |                       items: const [], | ||||||
|  |                       message: | ||||||
|  |                           '${pickedSource!.enforceTrackOnly ? tr('appsFromSourceAreTrackOnly') : tr('youPickedTrackOnly')}\n\n${tr('trackOnlyAppDescription')}', | ||||||
|  |                     ); | ||||||
|  |                   }) == | ||||||
|  |               null)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     getReleaseDateAsVersionConfirmationIfNeeded( | ||||||
|  |         bool userPickedTrackOnly) async { | ||||||
|  |       return (!(additionalSettings['versionDetection'] == | ||||||
|  |               'releaseDateAsVersion' && | ||||||
|  |           // ignore: use_build_context_synchronously | ||||||
|  |           await showDialog( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return GeneratedFormModal( | ||||||
|  |                       title: tr('releaseDateAsVersion'), | ||||||
|  |                       items: const [], | ||||||
|  |                       message: tr('releaseDateAsVersionExplanation'), | ||||||
|  |                     ); | ||||||
|  |                   }) == | ||||||
|  |               null)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     addApp({bool resetUserInputAfter = false}) async { |     addApp({bool resetUserInputAfter = false}) async { | ||||||
|       setState(() { |       setState(() { | ||||||
|         gettingAppInfo = true; |         gettingAppInfo = true; | ||||||
|       }); |       }); | ||||||
|       var settingsProvider = context.read<SettingsProvider>(); |       try { | ||||||
|       () async { |         var settingsProvider = context.read<SettingsProvider>(); | ||||||
|         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; |         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; | ||||||
|         var userPickedNoVersionDetection = |         App? app; | ||||||
|             additionalSettings['noVersionDetection'] == true; |         if ((await getTrackOnlyConfirmationIfNeeded(userPickedTrackOnly)) && | ||||||
|         var cont = true; |             (await getReleaseDateAsVersionConfirmationIfNeeded( | ||||||
|         if ((userPickedTrackOnly || pickedSource!.enforceTrackOnly) && |                 userPickedTrackOnly))) { | ||||||
|             await showDialog( |  | ||||||
|                     context: context, |  | ||||||
|                     builder: (BuildContext ctx) { |  | ||||||
|                       return GeneratedFormModal( |  | ||||||
|                         title: tr('xIsTrackOnly', args: [ |  | ||||||
|                           pickedSource!.enforceTrackOnly |  | ||||||
|                               ? tr('source') |  | ||||||
|                               : tr('app') |  | ||||||
|                         ]), |  | ||||||
|                         items: const [], |  | ||||||
|                         message: |  | ||||||
|                             '${pickedSource!.enforceTrackOnly ? tr('appsFromSourceAreTrackOnly') : tr('youPickedTrackOnly')}\n\n${tr('trackOnlyAppDescription')}', |  | ||||||
|                       ); |  | ||||||
|                     }) == |  | ||||||
|                 null) { |  | ||||||
|           cont = false; |  | ||||||
|         } |  | ||||||
|         if (userPickedNoVersionDetection && |  | ||||||
|             await showDialog( |  | ||||||
|                     context: context, |  | ||||||
|                     builder: (BuildContext ctx) { |  | ||||||
|                       return GeneratedFormModal( |  | ||||||
|                         title: tr('disableVersionDetection'), |  | ||||||
|                         items: const [], |  | ||||||
|                         message: tr('noVersionDetectionExplanation'), |  | ||||||
|                       ); |  | ||||||
|                     }) == |  | ||||||
|                 null) { |  | ||||||
|           cont = false; |  | ||||||
|         } |  | ||||||
|         if (cont) { |  | ||||||
|           HapticFeedback.selectionClick(); |  | ||||||
|           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; |           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; | ||||||
|           App app = await sourceProvider.getApp( |           app = await sourceProvider.getApp( | ||||||
|               pickedSource!, userInput, additionalSettings, |               pickedSource!, userInput, additionalSettings, | ||||||
|               trackOnlyOverride: trackOnly, |               trackOnlyOverride: trackOnly); | ||||||
|               noVersionDetectionOverride: userPickedNoVersionDetection); |  | ||||||
|           if (!trackOnly) { |           if (!trackOnly) { | ||||||
|             await settingsProvider.getInstallPermission(); |             await settingsProvider.getInstallPermission(); | ||||||
|           } |           } | ||||||
|           // Only download the APK here if you need to for the package ID |           // Only download the APK here if you need to for the package ID | ||||||
|           if (sourceProvider.isTempId(app.id) && |           if (sourceProvider.isTempId(app) && | ||||||
|               app.additionalSettings['trackOnly'] != true) { |               app.additionalSettings['trackOnly'] != true) { | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
|             var apkUrl = await appsProvider.confirmApkUrl(app, context); |             var apkUrl = await appsProvider.confirmApkUrl(app, context); | ||||||
| @@ -131,240 +139,254 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|           if (app.additionalSettings['trackOnly'] == true) { |           if (app.additionalSettings['trackOnly'] == true) { | ||||||
|             app.installedVersion = app.latestVersion; |             app.installedVersion = app.latestVersion; | ||||||
|           } |           } | ||||||
|           await appsProvider.saveApps([app]); |           app.categories = pickedCategories; | ||||||
|  |           await appsProvider.saveApps([app], onlyIfExists: false); | ||||||
|           return app; |  | ||||||
|         } |         } | ||||||
|       }() |  | ||||||
|           .then((app) { |  | ||||||
|         if (app != null) { |         if (app != null) { | ||||||
|           Navigator.push(context, |           Navigator.push(globalNavigatorKey.currentContext ?? context, | ||||||
|               MaterialPageRoute(builder: (context) => AppPage(appId: app.id))); |               MaterialPageRoute(builder: (context) => AppPage(appId: app!.id))); | ||||||
|         } |         } | ||||||
|       }).catchError((e) { |       } catch (e) { | ||||||
|         showError(e, context); |         showError(e, context); | ||||||
|       }).whenComplete(() { |       } finally { | ||||||
|         setState(() { |         setState(() { | ||||||
|           gettingAppInfo = false; |           gettingAppInfo = false; | ||||||
|           if (resetUserInputAfter) { |           if (resetUserInputAfter) { | ||||||
|             changeUserInput('', false, true); |             changeUserInput('', false, true); | ||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|       }); |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     Widget getUrlInputRow() => Row( | ||||||
|  |           children: [ | ||||||
|  |             Expanded( | ||||||
|  |                 child: GeneratedForm( | ||||||
|  |                     key: Key(searchnum.toString()), | ||||||
|  |                     items: [ | ||||||
|  |                       [ | ||||||
|  |                         GeneratedFormTextField('appSourceURL', | ||||||
|  |                             label: tr('appSourceURL'), | ||||||
|  |                             defaultValue: userInput, | ||||||
|  |                             additionalValidators: [ | ||||||
|  |                               (value) { | ||||||
|  |                                 try { | ||||||
|  |                                   sourceProvider | ||||||
|  |                                       .getSource(value ?? '') | ||||||
|  |                                       .standardizeURL( | ||||||
|  |                                           preStandardizeUrl(value ?? '')); | ||||||
|  |                                 } catch (e) { | ||||||
|  |                                   return e is String | ||||||
|  |                                       ? e | ||||||
|  |                                       : e is ObtainiumError | ||||||
|  |                                           ? e.toString() | ||||||
|  |                                           : tr('error'); | ||||||
|  |                                 } | ||||||
|  |                                 return null; | ||||||
|  |                               } | ||||||
|  |                             ]) | ||||||
|  |                       ] | ||||||
|  |                     ], | ||||||
|  |                     onValueChanges: (values, valid, isBuilding) { | ||||||
|  |                       changeUserInput( | ||||||
|  |                           values['appSourceURL']!, valid, isBuilding); | ||||||
|  |                     })), | ||||||
|  |             const SizedBox( | ||||||
|  |               width: 16, | ||||||
|  |             ), | ||||||
|  |             gettingAppInfo | ||||||
|  |                 ? const CircularProgressIndicator() | ||||||
|  |                 : ElevatedButton( | ||||||
|  |                     onPressed: doingSomething || | ||||||
|  |                             pickedSource == null || | ||||||
|  |                             (pickedSource!.combinedAppSpecificSettingFormItems | ||||||
|  |                                     .isNotEmpty && | ||||||
|  |                                 !additionalSettingsValid) | ||||||
|  |                         ? null | ||||||
|  |                         : () { | ||||||
|  |                             HapticFeedback.selectionClick(); | ||||||
|  |                             addApp(); | ||||||
|  |                           }, | ||||||
|  |                     child: Text(tr('add'))) | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     runSearch() async { | ||||||
|  |       setState(() { | ||||||
|  |         searching = true; | ||||||
|  |       }); | ||||||
|  |       try { | ||||||
|  |         var results = await Future.wait(sourceProvider.sources | ||||||
|  |             .where((e) => e.canSearch) | ||||||
|  |             .map((e) => e.search(searchQuery))); | ||||||
|  |  | ||||||
|  |         // .then((results) async { | ||||||
|  |         // Interleave results instead of simple reduce | ||||||
|  |         Map<String, String> res = {}; | ||||||
|  |         var si = 0; | ||||||
|  |         var done = false; | ||||||
|  |         while (!done) { | ||||||
|  |           done = true; | ||||||
|  |           for (var r in results) { | ||||||
|  |             if (r.length > si) { | ||||||
|  |               done = false; | ||||||
|  |               res.addEntries([r.entries.elementAt(si)]); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |           si++; | ||||||
|  |         } | ||||||
|  |         List<String>? selectedUrls = res.isEmpty | ||||||
|  |             ? [] | ||||||
|  |             // ignore: use_build_context_synchronously | ||||||
|  |             : await showDialog<List<String>?>( | ||||||
|  |                 context: context, | ||||||
|  |                 builder: (BuildContext ctx) { | ||||||
|  |                   return UrlSelectionModal( | ||||||
|  |                     urlsWithDescriptions: res, | ||||||
|  |                     selectedByDefault: false, | ||||||
|  |                     onlyOneSelectionAllowed: true, | ||||||
|  |                   ); | ||||||
|  |                 }); | ||||||
|  |         if (selectedUrls != null && selectedUrls.isNotEmpty) { | ||||||
|  |           changeUserInput(selectedUrls[0], true, false, isSearch: true); | ||||||
|  |         } | ||||||
|  |       } catch (e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       } finally { | ||||||
|  |         setState(() { | ||||||
|  |           searching = false; | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     bool shouldShowSearchBar() => | ||||||
|  |         sourceProvider.sources.where((e) => e.canSearch).isNotEmpty && | ||||||
|  |         pickedSource == null && | ||||||
|  |         userInput.isEmpty; | ||||||
|  |  | ||||||
|  |     Widget getSearchBarRow() => Row( | ||||||
|  |           children: [ | ||||||
|  |             Expanded( | ||||||
|  |               child: GeneratedForm( | ||||||
|  |                   items: [ | ||||||
|  |                     [ | ||||||
|  |                       GeneratedFormTextField('searchSomeSources', | ||||||
|  |                           label: tr('searchSomeSourcesLabel'), required: false), | ||||||
|  |                     ] | ||||||
|  |                   ], | ||||||
|  |                   onValueChanges: (values, valid, isBuilding) { | ||||||
|  |                     if (values.isNotEmpty && valid && !isBuilding) { | ||||||
|  |                       setState(() { | ||||||
|  |                         searchQuery = values['searchSomeSources']!.trim(); | ||||||
|  |                       }); | ||||||
|  |                     } | ||||||
|  |                   }), | ||||||
|  |             ), | ||||||
|  |             const SizedBox( | ||||||
|  |               width: 16, | ||||||
|  |             ), | ||||||
|  |             ElevatedButton( | ||||||
|  |                 onPressed: searchQuery.isEmpty || doingSomething | ||||||
|  |                     ? null | ||||||
|  |                     : () { | ||||||
|  |                         runSearch(); | ||||||
|  |                       }, | ||||||
|  |                 child: Text(tr('search'))) | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     Widget getAdditionalOptsCol() => Column( | ||||||
|  |           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |           children: [ | ||||||
|  |             const Divider( | ||||||
|  |               height: 64, | ||||||
|  |             ), | ||||||
|  |             Text( | ||||||
|  |                 tr('additionalOptsFor', | ||||||
|  |                     args: [pickedSource?.name ?? tr('source')]), | ||||||
|  |                 style: TextStyle(color: Theme.of(context).colorScheme.primary)), | ||||||
|  |             const SizedBox( | ||||||
|  |               height: 16, | ||||||
|  |             ), | ||||||
|  |             GeneratedForm( | ||||||
|  |                 key: Key(pickedSource.runtimeType.toString()), | ||||||
|  |                 items: pickedSource!.combinedAppSpecificSettingFormItems, | ||||||
|  |                 onValueChanges: (values, valid, isBuilding) { | ||||||
|  |                   if (!isBuilding) { | ||||||
|  |                     setState(() { | ||||||
|  |                       additionalSettings = values; | ||||||
|  |                       additionalSettingsValid = valid; | ||||||
|  |                     }); | ||||||
|  |                   } | ||||||
|  |                 }), | ||||||
|  |             Column( | ||||||
|  |               children: [ | ||||||
|  |                 const SizedBox( | ||||||
|  |                   height: 16, | ||||||
|  |                 ), | ||||||
|  |                 CategoryEditorSelector( | ||||||
|  |                     alignment: WrapAlignment.start, | ||||||
|  |                     onSelected: (categories) { | ||||||
|  |                       pickedCategories = categories; | ||||||
|  |                     }), | ||||||
|  |               ], | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     Widget getSourcesListWidget() => Column( | ||||||
|  |             crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|  |             mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |             children: [ | ||||||
|  |               const SizedBox( | ||||||
|  |                 height: 48, | ||||||
|  |               ), | ||||||
|  |               Text( | ||||||
|  |                 tr('supportedSourcesBelow'), | ||||||
|  |               ), | ||||||
|  |               const SizedBox( | ||||||
|  |                 height: 8, | ||||||
|  |               ), | ||||||
|  |               ...sourceProvider.sources | ||||||
|  |                   .map((e) => GestureDetector( | ||||||
|  |                       onTap: e.host != null | ||||||
|  |                           ? () { | ||||||
|  |                               launchUrlString('https://${e.host}', | ||||||
|  |                                   mode: LaunchMode.externalApplication); | ||||||
|  |                             } | ||||||
|  |                           : null, | ||||||
|  |                       child: Text( | ||||||
|  |                         '${e.name}${e.enforceTrackOnly ? ' ${tr('trackOnlyInBrackets')}' : ''}${e.canSearch ? ' ${tr('searchableInBrackets')}' : ''}', | ||||||
|  |                         style: TextStyle( | ||||||
|  |                             decoration: e.host != null | ||||||
|  |                                 ? TextDecoration.underline | ||||||
|  |                                 : TextDecoration.none, | ||||||
|  |                             fontStyle: FontStyle.italic), | ||||||
|  |                       ))) | ||||||
|  |                   .toList() | ||||||
|  |             ]); | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|         backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|         body: CustomScrollView(slivers: <Widget>[ |         body: CustomScrollView(shrinkWrap: true, slivers: <Widget>[ | ||||||
|           CustomAppBar(title: tr('addApp')), |           CustomAppBar(title: tr('addApp')), | ||||||
|           SliverFillRemaining( |           SliverToBoxAdapter( | ||||||
|             child: Padding( |             child: Padding( | ||||||
|                 padding: const EdgeInsets.all(16), |                 padding: const EdgeInsets.all(16), | ||||||
|                 child: Column( |                 child: Column( | ||||||
|  |                     mainAxisSize: MainAxisSize.min, | ||||||
|                     crossAxisAlignment: CrossAxisAlignment.stretch, |                     crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|                     children: [ |                     children: [ | ||||||
|                       Row( |                       getUrlInputRow(), | ||||||
|                         children: [ |                       if (shouldShowSearchBar()) | ||||||
|                           Expanded( |  | ||||||
|                               child: GeneratedForm( |  | ||||||
|                                   items: [ |  | ||||||
|                                 [ |  | ||||||
|                                   GeneratedFormTextField('appSourceURL', |  | ||||||
|                                       label: tr('appSourceURL'), |  | ||||||
|                                       additionalValidators: [ |  | ||||||
|                                         (value) { |  | ||||||
|                                           try { |  | ||||||
|                                             sourceProvider |  | ||||||
|                                                 .getSource(value ?? '') |  | ||||||
|                                                 .standardizeURL( |  | ||||||
|                                                     preStandardizeUrl( |  | ||||||
|                                                         value ?? '')); |  | ||||||
|                                           } catch (e) { |  | ||||||
|                                             return e is String |  | ||||||
|                                                 ? e |  | ||||||
|                                                 : e is ObtainiumError |  | ||||||
|                                                     ? e.toString() |  | ||||||
|                                                     : tr('error'); |  | ||||||
|                                           } |  | ||||||
|                                           return null; |  | ||||||
|                                         } |  | ||||||
|                                       ]) |  | ||||||
|                                 ] |  | ||||||
|                               ], |  | ||||||
|                                   onValueChanges: (values, valid, isBuilding) { |  | ||||||
|                                     changeUserInput(values['appSourceURL']!, |  | ||||||
|                                         valid, isBuilding); |  | ||||||
|                                   })), |  | ||||||
|                           const SizedBox( |  | ||||||
|                             width: 16, |  | ||||||
|                           ), |  | ||||||
|                           gettingAppInfo |  | ||||||
|                               ? const CircularProgressIndicator() |  | ||||||
|                               : ElevatedButton( |  | ||||||
|                                   onPressed: gettingAppInfo || |  | ||||||
|                                           pickedSource == null || |  | ||||||
|                                           (pickedSource! |  | ||||||
|                                                   .combinedAppSpecificSettingFormItems |  | ||||||
|                                                   .isNotEmpty && |  | ||||||
|                                               !additionalSettingsValid) |  | ||||||
|                                       ? null |  | ||||||
|                                       : addApp, |  | ||||||
|                                   child: Text(tr('add'))) |  | ||||||
|                         ], |  | ||||||
|                       ), |  | ||||||
|                       if (sourceProvider.sources |  | ||||||
|                               .where((e) => e.canSearch) |  | ||||||
|                               .isNotEmpty && |  | ||||||
|                           pickedSource == null && |  | ||||||
|                           userInput.isEmpty) |  | ||||||
|                         const SizedBox( |                         const SizedBox( | ||||||
|                           height: 16, |                           height: 16, | ||||||
|                         ), |                         ), | ||||||
|                       if (sourceProvider.sources |                       if (shouldShowSearchBar()) getSearchBarRow(), | ||||||
|                               .where((e) => e.canSearch) |                       if (pickedSource != null) | ||||||
|                               .isNotEmpty && |                         getAdditionalOptsCol() | ||||||
|                           pickedSource == null && |  | ||||||
|                           userInput.isEmpty) |  | ||||||
|                         Row( |  | ||||||
|                           children: [ |  | ||||||
|                             Expanded( |  | ||||||
|                               child: GeneratedForm( |  | ||||||
|                                   items: [ |  | ||||||
|                                     [ |  | ||||||
|                                       GeneratedFormTextField( |  | ||||||
|                                           'searchSomeSources', |  | ||||||
|                                           label: tr('searchSomeSourcesLabel'), |  | ||||||
|                                           required: false), |  | ||||||
|                                     ] |  | ||||||
|                                   ], |  | ||||||
|                                   onValueChanges: (values, valid, isBuilding) { |  | ||||||
|                                     if (values.isNotEmpty && valid) { |  | ||||||
|                                       setState(() { |  | ||||||
|                                         searchQuery = |  | ||||||
|                                             values['searchSomeSources']!.trim(); |  | ||||||
|                                       }); |  | ||||||
|                                     } |  | ||||||
|                                   }), |  | ||||||
|                             ), |  | ||||||
|                             const SizedBox( |  | ||||||
|                               width: 16, |  | ||||||
|                             ), |  | ||||||
|                             ElevatedButton( |  | ||||||
|                                 onPressed: searchQuery.isEmpty || gettingAppInfo |  | ||||||
|                                     ? null |  | ||||||
|                                     : () { |  | ||||||
|                                         Future.wait(sourceProvider.sources |  | ||||||
|                                                 .where((e) => e.canSearch) |  | ||||||
|                                                 .map((e) => |  | ||||||
|                                                     e.search(searchQuery))) |  | ||||||
|                                             .then((results) async { |  | ||||||
|                                           // Interleave results instead of simple reduce |  | ||||||
|                                           Map<String, String> res = {}; |  | ||||||
|                                           var si = 0; |  | ||||||
|                                           var done = false; |  | ||||||
|                                           while (!done) { |  | ||||||
|                                             done = true; |  | ||||||
|                                             for (var r in results) { |  | ||||||
|                                               if (r.length > si) { |  | ||||||
|                                                 done = false; |  | ||||||
|                                                 res.addEntries( |  | ||||||
|                                                     [r.entries.elementAt(si)]); |  | ||||||
|                                               } |  | ||||||
|                                             } |  | ||||||
|                                             si++; |  | ||||||
|                                           } |  | ||||||
|                                           List<String>? selectedUrls = res |  | ||||||
|                                                   .isEmpty |  | ||||||
|                                               ? [] |  | ||||||
|                                               : await showDialog<List<String>?>( |  | ||||||
|                                                   context: context, |  | ||||||
|                                                   builder: (BuildContext ctx) { |  | ||||||
|                                                     return UrlSelectionModal( |  | ||||||
|                                                       urlsWithDescriptions: res, |  | ||||||
|                                                       selectedByDefault: false, |  | ||||||
|                                                       onlyOneSelectionAllowed: |  | ||||||
|                                                           true, |  | ||||||
|                                                     ); |  | ||||||
|                                                   }); |  | ||||||
|                                           if (selectedUrls != null && |  | ||||||
|                                               selectedUrls.isNotEmpty) { |  | ||||||
|                                             changeUserInput( |  | ||||||
|                                                 selectedUrls[0], true, true); |  | ||||||
|                                             addApp(resetUserInputAfter: true); |  | ||||||
|                                           } |  | ||||||
|                                         }).catchError((e) { |  | ||||||
|                                           showError(e, context); |  | ||||||
|                                         }); |  | ||||||
|                                       }, |  | ||||||
|                                 child: Text(tr('search'))) |  | ||||||
|                           ], |  | ||||||
|                         ), |  | ||||||
|                       if (pickedSource != null && |  | ||||||
|                           (pickedSource! |  | ||||||
|                               .combinedAppSpecificSettingFormItems.isNotEmpty)) |  | ||||||
|                         Column( |  | ||||||
|                           crossAxisAlignment: CrossAxisAlignment.stretch, |  | ||||||
|                           children: [ |  | ||||||
|                             const Divider( |  | ||||||
|                               height: 64, |  | ||||||
|                             ), |  | ||||||
|                             Text( |  | ||||||
|                                 tr('additionalOptsFor', |  | ||||||
|                                     args: [pickedSource?.name ?? tr('source')]), |  | ||||||
|                                 style: TextStyle( |  | ||||||
|                                     color: |  | ||||||
|                                         Theme.of(context).colorScheme.primary)), |  | ||||||
|                             const SizedBox( |  | ||||||
|                               height: 16, |  | ||||||
|                             ), |  | ||||||
|                             GeneratedForm( |  | ||||||
|                                 items: pickedSource! |  | ||||||
|                                     .combinedAppSpecificSettingFormItems, |  | ||||||
|                                 onValueChanges: (values, valid, isBuilding) { |  | ||||||
|                                   if (!isBuilding) { |  | ||||||
|                                     setState(() { |  | ||||||
|                                       additionalSettings = values; |  | ||||||
|                                       additionalSettingsValid = valid; |  | ||||||
|                                     }); |  | ||||||
|                                   } |  | ||||||
|                                 }), |  | ||||||
|                           ], |  | ||||||
|                         ) |  | ||||||
|                       else |                       else | ||||||
|                         Expanded( |                         getSourcesListWidget(), | ||||||
|                             child: Column( |  | ||||||
|                                 crossAxisAlignment: CrossAxisAlignment.center, |  | ||||||
|                                 mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|                                 children: [ |  | ||||||
|                               const SizedBox( |  | ||||||
|                                 height: 48, |  | ||||||
|                               ), |  | ||||||
|                               Text( |  | ||||||
|                                 tr('supportedSourcesBelow'), |  | ||||||
|                               ), |  | ||||||
|                               const SizedBox( |  | ||||||
|                                 height: 8, |  | ||||||
|                               ), |  | ||||||
|                               ...sourceProvider.sources |  | ||||||
|                                   .map((e) => GestureDetector( |  | ||||||
|                                       onTap: e.host != null |  | ||||||
|                                           ? () { |  | ||||||
|                                               launchUrlString( |  | ||||||
|                                                   'https://${e.host}', |  | ||||||
|                                                   mode: LaunchMode |  | ||||||
|                                                       .externalApplication); |  | ||||||
|                                             } |  | ||||||
|                                           : null, |  | ||||||
|                                       child: Text( |  | ||||||
|                                         '${e.name}${e.enforceTrackOnly ? ' ${tr('trackOnlyInBrackets')}' : ''}${e.canSearch ? ' ${tr('searchableInBrackets')}' : ''}', |  | ||||||
|                                         style: TextStyle( |  | ||||||
|                                             decoration: e.host != null |  | ||||||
|                                                 ? TextDecoration.underline |  | ||||||
|                                                 : TextDecoration.none, |  | ||||||
|                                             fontStyle: FontStyle.italic), |  | ||||||
|                                       ))) |  | ||||||
|                                   .toList() |  | ||||||
|                             ])), |  | ||||||
|                       const SizedBox( |                       const SizedBox( | ||||||
|                         height: 8, |                         height: 8, | ||||||
|                       ), |                       ), | ||||||
|   | |||||||
| @@ -35,355 +35,420 @@ class _AppPageState extends State<AppPage> { | |||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     var categories = settingsProvider.categories; |     bool areDownloadsRunning = appsProvider.areDownloadsRunning(); | ||||||
|  |  | ||||||
|     var sourceProvider = SourceProvider(); |     var sourceProvider = SourceProvider(); | ||||||
|     AppInMemory? app = appsProvider.apps[widget.appId]; |     AppInMemory? app = appsProvider.apps[widget.appId]; | ||||||
|     var source = app != null ? sourceProvider.getSource(app.app.url) : null; |     var source = app != null ? sourceProvider.getSource(app.app.url) : null; | ||||||
|     if (!appsProvider.areDownloadsRunning() && prevApp == null && app != null) { |     if (!areDownloadsRunning && prevApp == null && app != null) { | ||||||
|       prevApp = app; |       prevApp = app; | ||||||
|       getUpdate(app.app.id); |       getUpdate(app.app.id); | ||||||
|     } |     } | ||||||
|     var trackOnly = app?.app.additionalSettings['trackOnly'] == true; |     var trackOnly = app?.app.additionalSettings['trackOnly'] == true; | ||||||
|     return Scaffold( |  | ||||||
|       appBar: settingsProvider.showAppWebpage ? AppBar() : null, |     bool isVersionDetectionStandard = | ||||||
|       backgroundColor: Theme.of(context).colorScheme.surface, |         app?.app.additionalSettings['versionDetection'] == | ||||||
|       body: RefreshIndicator( |             'standardVersionDetection'; | ||||||
|           child: settingsProvider.showAppWebpage |  | ||||||
|               ? app != null |     getInfoColumn() => Column( | ||||||
|                   ? WebViewWidget( |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                       controller: WebViewController() |           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|                         ..setJavaScriptMode(JavaScriptMode.unrestricted) |           children: [ | ||||||
|                         ..setBackgroundColor( |             GestureDetector( | ||||||
|                             Theme.of(context).colorScheme.background) |                 onTap: () { | ||||||
|                         ..setJavaScriptMode(JavaScriptMode.unrestricted) |                   if (app?.app.url != null) { | ||||||
|                         ..setNavigationDelegate( |                     launchUrlString(app?.app.url ?? '', | ||||||
|                           NavigationDelegate( |                         mode: LaunchMode.externalApplication); | ||||||
|                             onWebResourceError: (WebResourceError error) { |                   } | ||||||
|                               if (error.isForMainFrame == true) { |                 }, | ||||||
|                                 showError( |                 onLongPress: () { | ||||||
|                                     ObtainiumError(error.description, |                   Clipboard.setData(ClipboardData(text: app?.app.url ?? '')); | ||||||
|                                         unexpected: true), |                   ScaffoldMessenger.of(context).showSnackBar(SnackBar( | ||||||
|                                     context); |                     content: Text(tr('copiedToClipboard')), | ||||||
|                               } |                   )); | ||||||
|                             }, |                 }, | ||||||
|                           ), |                 child: Text( | ||||||
|                         ) |                   app?.app.url ?? '', | ||||||
|                         ..loadRequest(Uri.parse(app.app.url))) |                   textAlign: TextAlign.center, | ||||||
|                   : Container() |                   style: const TextStyle( | ||||||
|               : CustomScrollView( |                       decoration: TextDecoration.underline, | ||||||
|                   slivers: [ |                       fontStyle: FontStyle.italic, | ||||||
|                     SliverFillRemaining( |                       fontSize: 12), | ||||||
|                         child: Column( |                 )), | ||||||
|                       mainAxisAlignment: MainAxisAlignment.center, |             const SizedBox( | ||||||
|                       crossAxisAlignment: CrossAxisAlignment.stretch, |               height: 32, | ||||||
|                       children: [ |             ), | ||||||
|                         app?.installedInfo != null |             Text( | ||||||
|                             ? Row( |               tr('latestVersionX', | ||||||
|                                 mainAxisAlignment: MainAxisAlignment.center, |                   args: [app?.app.latestVersion ?? tr('unknown')]), | ||||||
|                                 children: [ |               textAlign: TextAlign.center, | ||||||
|                                     Image.memory( |               style: Theme.of(context).textTheme.bodyLarge, | ||||||
|                                       app!.installedInfo!.icon!, |             ), | ||||||
|                                       height: 150, |             Text( | ||||||
|                                       gaplessPlayback: true, |               '${tr('installedVersionX', args: [ | ||||||
|                                     ) |                     app?.app.installedVersion ?? tr('none') | ||||||
|                                   ]) |                   ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [ | ||||||
|                             : Container(), |                       tr('app') | ||||||
|                         const SizedBox( |                     ])}' : ''}', | ||||||
|                           height: 25, |               textAlign: TextAlign.center, | ||||||
|                         ), |               style: Theme.of(context).textTheme.bodyLarge, | ||||||
|                         Text( |             ), | ||||||
|                           app?.installedInfo?.name ?? |             if (app?.app.installedVersion != null && | ||||||
|                               app?.app.name ?? |                 !isVersionDetectionStandard) | ||||||
|                               tr('app'), |               Column( | ||||||
|                           textAlign: TextAlign.center, |                 children: [ | ||||||
|                           style: Theme.of(context).textTheme.displayLarge, |                   const SizedBox( | ||||||
|                         ), |                     height: 4, | ||||||
|                         Text( |                   ), | ||||||
|                           tr('byX', args: [app?.app.author ?? tr('unknown')]), |                   Text( | ||||||
|                           textAlign: TextAlign.center, |                     tr('noVersionDetection'), | ||||||
|                           style: Theme.of(context).textTheme.headlineMedium, |                     style: Theme.of(context).textTheme.labelSmall, | ||||||
|                         ), |                   ) | ||||||
|                         const SizedBox( |                 ], | ||||||
|                           height: 32, |               ), | ||||||
|                         ), |             const SizedBox( | ||||||
|                         GestureDetector( |               height: 32, | ||||||
|                             onTap: () { |             ), | ||||||
|                               if (app?.app.url != null) { |             Text( | ||||||
|                                 launchUrlString(app?.app.url ?? '', |               tr('lastUpdateCheckX', args: [ | ||||||
|                                     mode: LaunchMode.externalApplication); |                 app?.app.lastUpdateCheck == null | ||||||
|                               } |                     ? tr('never') | ||||||
|                             }, |                     : '\n${app?.app.lastUpdateCheck?.toLocal()}' | ||||||
|                             child: Text( |               ]), | ||||||
|                               app?.app.url ?? '', |               textAlign: TextAlign.center, | ||||||
|                               textAlign: TextAlign.center, |               style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12), | ||||||
|                               style: const TextStyle( |             ), | ||||||
|                                   decoration: TextDecoration.underline, |             const SizedBox( | ||||||
|                                   fontStyle: FontStyle.italic, |               height: 48, | ||||||
|                                   fontSize: 12), |             ), | ||||||
|                             )), |             CategoryEditorSelector( | ||||||
|                         const SizedBox( |                 alignment: WrapAlignment.center, | ||||||
|                           height: 32, |                 preselected: app?.app.categories != null | ||||||
|                         ), |                     ? app!.app.categories.toSet() | ||||||
|                         Text( |                     : {}, | ||||||
|                           tr('latestVersionX', |                 onSelected: (categories) { | ||||||
|                               args: [app?.app.latestVersion ?? tr('unknown')]), |                   if (app != null) { | ||||||
|                           textAlign: TextAlign.center, |                     app.app.categories = categories; | ||||||
|                           style: Theme.of(context).textTheme.bodyLarge, |                     appsProvider.saveApps([app.app]); | ||||||
|                         ), |                   } | ||||||
|                         Text( |                 }), | ||||||
|                           '${tr('installedVersionX', args: [ |           ], | ||||||
|                                 app?.app.installedVersion ?? tr('none') |         ); | ||||||
|                               ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [ |  | ||||||
|                                   tr('app') |     getFullInfoColumn() => Column( | ||||||
|                                 ])}' : ''}', |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                           textAlign: TextAlign.center, |           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|                           style: Theme.of(context).textTheme.bodyLarge, |           children: [ | ||||||
|                         ), |             const SizedBox(height: 125), | ||||||
|                         const SizedBox( |             app?.installedInfo != null | ||||||
|                           height: 32, |                 ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [ | ||||||
|                         ), |                     Image.memory( | ||||||
|                         Text( |                       app!.installedInfo!.icon!, | ||||||
|                           tr('lastUpdateCheckX', args: [ |                       height: 150, | ||||||
|                             app?.app.lastUpdateCheck == null |                       gaplessPlayback: true, | ||||||
|                                 ? tr('never') |                     ) | ||||||
|                                 : '\n${app?.app.lastUpdateCheck?.toLocal()}' |                   ]) | ||||||
|                           ]), |                 : Container(), | ||||||
|                           textAlign: TextAlign.center, |             const SizedBox( | ||||||
|                           style: const TextStyle( |               height: 25, | ||||||
|                               fontStyle: FontStyle.italic, fontSize: 12), |             ), | ||||||
|                         ), |             Text( | ||||||
|                         const SizedBox( |               app?.app.name ?? tr('app'), | ||||||
|                           height: 48, |               textAlign: TextAlign.center, | ||||||
|                         ), |               style: Theme.of(context).textTheme.displayLarge, | ||||||
|                         CategoryEditorSelector( |             ), | ||||||
|                             alignment: WrapAlignment.center, |             Text( | ||||||
|                             singleSelect: true, |               tr('byX', args: [app?.app.author ?? tr('unknown')]), | ||||||
|                             preselected: app?.app.category != null |               textAlign: TextAlign.center, | ||||||
|                                 ? {app!.app.category!} |               style: Theme.of(context).textTheme.headlineMedium, | ||||||
|                                 : {}, |             ), | ||||||
|                             onSelected: (categories) { |             const SizedBox( | ||||||
|                               if (app != null) { |               height: 8, | ||||||
|                                 app.app.category = categories.isNotEmpty |             ), | ||||||
|                                     ? categories[0] |             Text( | ||||||
|                                     : null; |               app?.app.id ?? '', | ||||||
|                                 appsProvider.saveApps([app.app]); |               textAlign: TextAlign.center, | ||||||
|                               } |               style: Theme.of(context).textTheme.labelSmall, | ||||||
|                             }) |             ), | ||||||
|                       ], |             app?.app.releaseDate == null | ||||||
|                     )), |                 ? const SizedBox.shrink() | ||||||
|                   ], |                 : Text( | ||||||
|  |                     app!.app.releaseDate.toString(), | ||||||
|  |                     textAlign: TextAlign.center, | ||||||
|  |                     style: Theme.of(context).textTheme.labelSmall, | ||||||
|  |                   ), | ||||||
|  |             const SizedBox( | ||||||
|  |               height: 32, | ||||||
|  |             ), | ||||||
|  |             getInfoColumn(), | ||||||
|  |             const SizedBox(height: 150) | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     getAppWebView() => app != null | ||||||
|  |         ? WebViewWidget( | ||||||
|  |             controller: WebViewController() | ||||||
|  |               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||||
|  |               ..setBackgroundColor(Theme.of(context).colorScheme.background) | ||||||
|  |               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||||
|  |               ..setNavigationDelegate( | ||||||
|  |                 NavigationDelegate( | ||||||
|  |                   onWebResourceError: (WebResourceError error) { | ||||||
|  |                     if (error.isForMainFrame == true) { | ||||||
|  |                       showError( | ||||||
|  |                           ObtainiumError(error.description, unexpected: true), | ||||||
|  |                           context); | ||||||
|  |                     } | ||||||
|  |                   }, | ||||||
|                 ), |                 ), | ||||||
|           onRefresh: () async { |               ) | ||||||
|             if (app != null) { |               ..loadRequest(Uri.parse(app.app.url))) | ||||||
|               getUpdate(app.app.id); |         : Container(); | ||||||
|  |  | ||||||
|  |     showMarkUpdatedDialog() { | ||||||
|  |       return showDialog( | ||||||
|  |           context: context, | ||||||
|  |           builder: (BuildContext ctx) { | ||||||
|  |             return AlertDialog( | ||||||
|  |               title: Text(tr('alreadyUpToDateQuestion')), | ||||||
|  |               actions: [ | ||||||
|  |                 TextButton( | ||||||
|  |                     onPressed: () { | ||||||
|  |                       Navigator.of(context).pop(); | ||||||
|  |                     }, | ||||||
|  |                     child: Text(tr('no'))), | ||||||
|  |                 TextButton( | ||||||
|  |                     onPressed: () { | ||||||
|  |                       HapticFeedback.selectionClick(); | ||||||
|  |                       var updatedApp = app?.app; | ||||||
|  |                       if (updatedApp != null) { | ||||||
|  |                         updatedApp.installedVersion = updatedApp.latestVersion; | ||||||
|  |                         appsProvider.saveApps([updatedApp]); | ||||||
|  |                       } | ||||||
|  |                       Navigator.of(context).pop(); | ||||||
|  |                     }, | ||||||
|  |                     child: Text(tr('yesMarkUpdated'))) | ||||||
|  |               ], | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     showAdditionalOptionsDialog() async { | ||||||
|  |       return await showDialog<Map<String, dynamic>?>( | ||||||
|  |           context: context, | ||||||
|  |           builder: (BuildContext ctx) { | ||||||
|  |             var items = | ||||||
|  |                 (source?.combinedAppSpecificSettingFormItems ?? []).map((row) { | ||||||
|  |               row = row.map((e) { | ||||||
|  |                 if (app?.app.additionalSettings[e.key] != null) { | ||||||
|  |                   e.defaultValue = app?.app.additionalSettings[e.key]; | ||||||
|  |                 } | ||||||
|  |                 return e; | ||||||
|  |               }).toList(); | ||||||
|  |               return row; | ||||||
|  |             }).toList(); | ||||||
|  |  | ||||||
|  |             items = items.map((row) { | ||||||
|  |               row = row.map((e) { | ||||||
|  |                 if (e.key == 'versionDetection' && e is GeneratedFormDropdown) { | ||||||
|  |                   e.disabledOptKeys ??= []; | ||||||
|  |                   if (app?.app.installedVersion != null && | ||||||
|  |                       app?.app.additionalSettings['versionDetection'] != | ||||||
|  |                           'releaseDateAsVersion' && | ||||||
|  |                       !appsProvider.isVersionDetectionPossible(app)) { | ||||||
|  |                     e.disabledOptKeys!.add('standardVersionDetection'); | ||||||
|  |                   } | ||||||
|  |                   if (app?.app.releaseDate == null) { | ||||||
|  |                     e.disabledOptKeys!.add('releaseDateAsVersion'); | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |                 return e; | ||||||
|  |               }).toList(); | ||||||
|  |               return row; | ||||||
|  |             }).toList(); | ||||||
|  |  | ||||||
|  |             return GeneratedFormModal( | ||||||
|  |               title: tr('additionalOptions'), | ||||||
|  |               items: items, | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     handleAdditionalOptionChanges(Map<String, dynamic>? values) { | ||||||
|  |       if (app != null && values != null) { | ||||||
|  |         Map<String, dynamic> originalSettings = app.app.additionalSettings; | ||||||
|  |         app.app.additionalSettings = values; | ||||||
|  |         if (source?.enforceTrackOnly == true) { | ||||||
|  |           app.app.additionalSettings['trackOnly'] = true; | ||||||
|  |           // ignore: use_build_context_synchronously | ||||||
|  |           showError(tr('appsFromSourceAreTrackOnly'), context); | ||||||
|  |         } | ||||||
|  |         if (app.app.additionalSettings['versionDetection'] == | ||||||
|  |             'releaseDateAsVersion') { | ||||||
|  |           if (originalSettings['versionDetection'] != 'releaseDateAsVersion') { | ||||||
|  |             if (app.app.releaseDate != null) { | ||||||
|  |               bool isUpdated = | ||||||
|  |                   app.app.installedVersion == app.app.latestVersion; | ||||||
|  |               app.app.latestVersion = | ||||||
|  |                   app.app.releaseDate!.microsecondsSinceEpoch.toString(); | ||||||
|  |               if (isUpdated) { | ||||||
|  |                 app.app.installedVersion = app.app.latestVersion; | ||||||
|  |               } | ||||||
|             } |             } | ||||||
|           }), |           } | ||||||
|       bottomSheet: Padding( |         } else if (originalSettings['versionDetection'] == | ||||||
|           padding: EdgeInsets.fromLTRB( |             'releaseDateAsVersion') { | ||||||
|               0, 0, 0, MediaQuery.of(context).padding.bottom), |           app.app.installedVersion = | ||||||
|           child: Column( |               app.installedInfo?.versionName ?? app.app.installedVersion; | ||||||
|             mainAxisSize: MainAxisSize.min, |         } | ||||||
|             children: [ |         appsProvider.saveApps([app.app]).then((value) { | ||||||
|               Padding( |           getUpdate(app.app.id); | ||||||
|                   padding: const EdgeInsets.fromLTRB(16, 8, 16, 0), |         }); | ||||||
|                   child: Row( |       } | ||||||
|                       mainAxisAlignment: MainAxisAlignment.spaceEvenly, |     } | ||||||
|                       children: [ |  | ||||||
|                         if (app?.app.installedVersion != null && |     getInstallOrUpdateButton() => TextButton( | ||||||
|                             !trackOnly && |         onPressed: (app?.app.installedVersion == null || | ||||||
|                             app?.app.installedVersion != app?.app.latestVersion) |                     app?.app.installedVersion != app?.app.latestVersion) && | ||||||
|                           IconButton( |                 !areDownloadsRunning | ||||||
|                               onPressed: app?.downloadProgress != null |             ? () async { | ||||||
|                                   ? null |                 try { | ||||||
|                                   : () { |                   HapticFeedback.heavyImpact(); | ||||||
|                                       showDialog( |                   if (app?.app.additionalSettings['trackOnly'] != true) { | ||||||
|                                           context: context, |                     await settingsProvider.getInstallPermission(); | ||||||
|                                           builder: (BuildContext ctx) { |                   } | ||||||
|                                             return AlertDialog( |                   var res = await appsProvider.downloadAndInstallLatestApps( | ||||||
|                                               title: Text(tr( |                       [app!.app.id], globalNavigatorKey.currentContext); | ||||||
|                                                   'alreadyUpToDateQuestion')), |                   if (res.isNotEmpty && mounted) { | ||||||
|                                               content: Text( |                     Navigator.of(context).pop(); | ||||||
|                                                   tr('onlyWorksWithNonEVDApps'), |                   } | ||||||
|                                                   style: const TextStyle( |                 } catch (e) { | ||||||
|                                                       fontWeight: |                   showError(e, context); | ||||||
|                                                           FontWeight.bold, |                 } | ||||||
|                                                       fontStyle: |               } | ||||||
|                                                           FontStyle.italic)), |             : null, | ||||||
|                                               actions: [ |         child: Text(app?.app.installedVersion == null | ||||||
|                                                 TextButton( |             ? !trackOnly | ||||||
|                                                     onPressed: () { |                 ? tr('install') | ||||||
|                                                       Navigator.of(context) |                 : tr('markInstalled') | ||||||
|                                                           .pop(); |             : !trackOnly | ||||||
|                                                     }, |                 ? tr('update') | ||||||
|                                                     child: Text(tr('no'))), |                 : tr('markUpdated'))); | ||||||
|                                                 TextButton( |  | ||||||
|                                                     onPressed: () { |     getBottomSheetMenu() => Padding( | ||||||
|                                                       HapticFeedback |         padding: | ||||||
|                                                           .selectionClick(); |             EdgeInsets.fromLTRB(0, 0, 0, MediaQuery.of(context).padding.bottom), | ||||||
|                                                       var updatedApp = app?.app; |         child: Column( | ||||||
|                                                       if (updatedApp != null) { |           mainAxisSize: MainAxisSize.min, | ||||||
|                                                         updatedApp |           children: [ | ||||||
|                                                                 .installedVersion = |             Padding( | ||||||
|                                                             updatedApp |                 padding: const EdgeInsets.fromLTRB(16, 8, 16, 0), | ||||||
|                                                                 .latestVersion; |                 child: Row( | ||||||
|                                                         appsProvider.saveApps( |                     mainAxisAlignment: MainAxisAlignment.spaceEvenly, | ||||||
|                                                             [updatedApp]); |                     children: [ | ||||||
|                                                       } |                       if (app?.app.installedVersion != null && | ||||||
|                                                       Navigator.of(context) |                           app?.app.installedVersion != app?.app.latestVersion && | ||||||
|                                                           .pop(); |                           !isVersionDetectionStandard && | ||||||
|                                                     }, |                           !trackOnly) | ||||||
|                                                     child: Text( |                         IconButton( | ||||||
|                                                         tr('yesMarkUpdated'))) |                             onPressed: app?.downloadProgress != null | ||||||
|                                               ], |                                 ? null | ||||||
|                                             ); |                                 : showMarkUpdatedDialog, | ||||||
|                                           }); |                             tooltip: tr('markUpdated'), | ||||||
|                                     }, |                             icon: const Icon(Icons.done)), | ||||||
|                               tooltip: tr('markUpdated'), |                       if (source != null && | ||||||
|                               icon: const Icon(Icons.done)), |                           source.combinedAppSpecificSettingFormItems.isNotEmpty) | ||||||
|                         if (source != null && |                         IconButton( | ||||||
|                             source |                             onPressed: app?.downloadProgress != null | ||||||
|                                 .combinedAppSpecificSettingFormItems.isNotEmpty) |                                 ? null | ||||||
|                           IconButton( |                                 : () async { | ||||||
|                               onPressed: app?.downloadProgress != null |                                     var values = | ||||||
|                                   ? null |                                         await showAdditionalOptionsDialog(); | ||||||
|                                   : () { |                                     handleAdditionalOptionChanges(values); | ||||||
|                                       showDialog<Map<String, dynamic>?>( |                                   }, | ||||||
|                                           context: context, |                             tooltip: tr('additionalOptions'), | ||||||
|                                           builder: (BuildContext ctx) { |                             icon: const Icon(Icons.edit)), | ||||||
|                                             var items = source |                       if (app != null && app.installedInfo != null) | ||||||
|                                                 .combinedAppSpecificSettingFormItems |                         IconButton( | ||||||
|                                                 .map((row) { |                           onPressed: () { | ||||||
|                                               row.map((e) { |                             appsProvider.openAppSettings(app.app.id); | ||||||
|                                                 if (app?.app.additionalSettings[ |                           }, | ||||||
|                                                         e.key] != |                           icon: const Icon(Icons.settings), | ||||||
|                                                     null) { |                           tooltip: tr('settings'), | ||||||
|                                                   e.defaultValue = app?.app |  | ||||||
|                                                           .additionalSettings[ |  | ||||||
|                                                       e.key]; |  | ||||||
|                                                 } |  | ||||||
|                                                 return e; |  | ||||||
|                                               }).toList(); |  | ||||||
|                                               return row; |  | ||||||
|                                             }).toList(); |  | ||||||
|                                             return GeneratedFormModal( |  | ||||||
|                                                 title: tr('additionalOptions'), |  | ||||||
|                                                 items: items); |  | ||||||
|                                           }).then((values) { |  | ||||||
|                                         if (app != null && values != null) { |  | ||||||
|                                           var changedApp = app.app; |  | ||||||
|                                           changedApp.additionalSettings = |  | ||||||
|                                               values; |  | ||||||
|                                           if (source.enforceTrackOnly) { |  | ||||||
|                                             changedApp.additionalSettings[ |  | ||||||
|                                                 'trackOnly'] = true; |  | ||||||
|                                             showError( |  | ||||||
|                                                 tr('appsFromSourceAreTrackOnly'), |  | ||||||
|                                                 context); |  | ||||||
|                                           } |  | ||||||
|                                           appsProvider.saveApps( |  | ||||||
|                                               [changedApp]).then((value) { |  | ||||||
|                                             getUpdate(changedApp.id); |  | ||||||
|                                           }); |  | ||||||
|                                         } |  | ||||||
|                                       }); |  | ||||||
|                                     }, |  | ||||||
|                               tooltip: tr('additionalOptions'), |  | ||||||
|                               icon: const Icon(Icons.settings)), |  | ||||||
|                         const SizedBox(width: 16.0), |  | ||||||
|                         Expanded( |  | ||||||
|                             child: ElevatedButton( |  | ||||||
|                                 onPressed: (app?.app.installedVersion == null || |  | ||||||
|                                             app?.app.installedVersion != |  | ||||||
|                                                 app?.app.latestVersion) && |  | ||||||
|                                         !appsProvider.areDownloadsRunning() |  | ||||||
|                                     ? () { |  | ||||||
|                                         HapticFeedback.heavyImpact(); |  | ||||||
|                                         () async { |  | ||||||
|                                           if (app?.app.additionalSettings[ |  | ||||||
|                                                   'trackOnly'] != |  | ||||||
|                                               true) { |  | ||||||
|                                             await settingsProvider |  | ||||||
|                                                 .getInstallPermission(); |  | ||||||
|                                           } |  | ||||||
|                                         }() |  | ||||||
|                                             .then((value) { |  | ||||||
|                                           appsProvider |  | ||||||
|                                               .downloadAndInstallLatestApps( |  | ||||||
|                                                   [app!.app.id], |  | ||||||
|                                                   globalNavigatorKey |  | ||||||
|                                                       .currentContext).then( |  | ||||||
|                                                   (res) { |  | ||||||
|                                             if (res.isNotEmpty && mounted) { |  | ||||||
|                                               Navigator.of(context).pop(); |  | ||||||
|                                             } |  | ||||||
|                                           }); |  | ||||||
|                                         }).catchError((e) { |  | ||||||
|                                           showError(e, context); |  | ||||||
|                                         }); |  | ||||||
|                                       } |  | ||||||
|                                     : null, |  | ||||||
|                                 child: Text(app?.app.installedVersion == null |  | ||||||
|                                     ? !trackOnly |  | ||||||
|                                         ? tr('install') |  | ||||||
|                                         : tr('markInstalled') |  | ||||||
|                                     : !trackOnly |  | ||||||
|                                         ? tr('update') |  | ||||||
|                                         : tr('markUpdated')))), |  | ||||||
|                         const SizedBox(width: 16.0), |  | ||||||
|                         ElevatedButton( |  | ||||||
|                           onPressed: app?.downloadProgress != null |  | ||||||
|                               ? null |  | ||||||
|                               : () { |  | ||||||
|                                   showDialog( |  | ||||||
|                                       context: context, |  | ||||||
|                                       builder: (BuildContext ctx) { |  | ||||||
|                                         return AlertDialog( |  | ||||||
|                                           title: Text(tr('removeAppQuestion')), |  | ||||||
|                                           content: Text(tr( |  | ||||||
|                                               'xWillBeRemovedButRemainInstalled', |  | ||||||
|                                               args: [ |  | ||||||
|                                                 app?.installedInfo?.name ?? |  | ||||||
|                                                     app?.app.name ?? |  | ||||||
|                                                     tr('app') |  | ||||||
|                                               ])), |  | ||||||
|                                           actions: [ |  | ||||||
|                                             TextButton( |  | ||||||
|                                                 onPressed: () { |  | ||||||
|                                                   HapticFeedback |  | ||||||
|                                                       .selectionClick(); |  | ||||||
|                                                   appsProvider.removeApps( |  | ||||||
|                                                       [app!.app.id]).then((_) { |  | ||||||
|                                                     int count = 0; |  | ||||||
|                                                     Navigator.of(context) |  | ||||||
|                                                         .popUntil((_) => |  | ||||||
|                                                             count++ >= 2); |  | ||||||
|                                                   }); |  | ||||||
|                                                 }, |  | ||||||
|                                                 child: Text(tr('remove'))), |  | ||||||
|                                             TextButton( |  | ||||||
|                                                 onPressed: () { |  | ||||||
|                                                   Navigator.of(context).pop(); |  | ||||||
|                                                 }, |  | ||||||
|                                                 child: Text(tr('cancel'))) |  | ||||||
|                                           ], |  | ||||||
|                                         ); |  | ||||||
|                                       }); |  | ||||||
|                                 }, |  | ||||||
|                           style: TextButton.styleFrom( |  | ||||||
|                               foregroundColor: |  | ||||||
|                                   Theme.of(context).colorScheme.error, |  | ||||||
|                               surfaceTintColor: |  | ||||||
|                                   Theme.of(context).colorScheme.error), |  | ||||||
|                           child: Text(tr('remove')), |  | ||||||
|                         ), |                         ), | ||||||
|                       ])), |                       if (app != null && settingsProvider.showAppWebpage) | ||||||
|               if (app?.downloadProgress != null) |                         IconButton( | ||||||
|                 Padding( |                             onPressed: () { | ||||||
|                     padding: const EdgeInsets.fromLTRB(0, 8, 0, 0), |                               showDialog( | ||||||
|                     child: LinearProgressIndicator( |                                   context: context, | ||||||
|                         value: app!.downloadProgress! / 100)) |                                   builder: (BuildContext ctx) { | ||||||
|             ], |                                     return AlertDialog( | ||||||
|           )), |                                       scrollable: true, | ||||||
|     ); |                                       content: getInfoColumn(), | ||||||
|  |                                       title: Text( | ||||||
|  |                                           '${app.app.name} ${tr('byX', args: [ | ||||||
|  |                                             app.app.author | ||||||
|  |                                           ])}'), | ||||||
|  |                                       actions: [ | ||||||
|  |                                         TextButton( | ||||||
|  |                                             onPressed: () { | ||||||
|  |                                               Navigator.of(context).pop(); | ||||||
|  |                                             }, | ||||||
|  |                                             child: Text(tr('continue'))) | ||||||
|  |                                       ], | ||||||
|  |                                     ); | ||||||
|  |                                   }); | ||||||
|  |                             }, | ||||||
|  |                             icon: const Icon(Icons.more_horiz), | ||||||
|  |                             tooltip: tr('more')), | ||||||
|  |                       const SizedBox(width: 16.0), | ||||||
|  |                       Expanded(child: getInstallOrUpdateButton()), | ||||||
|  |                       const SizedBox(width: 16.0), | ||||||
|  |                       Expanded( | ||||||
|  |                           child: TextButton( | ||||||
|  |                         onPressed: app?.downloadProgress != null | ||||||
|  |                             ? null | ||||||
|  |                             : () { | ||||||
|  |                                 appsProvider.removeAppsWithModal( | ||||||
|  |                                     context, [app!.app]).then((value) { | ||||||
|  |                                   if (value == true) { | ||||||
|  |                                     Navigator.of(context).pop(); | ||||||
|  |                                   } | ||||||
|  |                                 }); | ||||||
|  |                               }, | ||||||
|  |                         style: TextButton.styleFrom( | ||||||
|  |                             foregroundColor: | ||||||
|  |                                 Theme.of(context).colorScheme.error, | ||||||
|  |                             surfaceTintColor: | ||||||
|  |                                 Theme.of(context).colorScheme.error), | ||||||
|  |                         child: Text(tr('remove')), | ||||||
|  |                       )), | ||||||
|  |                     ])), | ||||||
|  |             if (app?.downloadProgress != null) | ||||||
|  |               Padding( | ||||||
|  |                   padding: const EdgeInsets.fromLTRB(0, 8, 0, 0), | ||||||
|  |                   child: LinearProgressIndicator( | ||||||
|  |                       value: app!.downloadProgress! / 100)) | ||||||
|  |           ], | ||||||
|  |         )); | ||||||
|  |  | ||||||
|  |     return Scaffold( | ||||||
|  |         appBar: settingsProvider.showAppWebpage ? AppBar() : null, | ||||||
|  |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|  |         body: RefreshIndicator( | ||||||
|  |             child: settingsProvider.showAppWebpage | ||||||
|  |                 ? getAppWebView() | ||||||
|  |                 : CustomScrollView( | ||||||
|  |                     slivers: [ | ||||||
|  |                       SliverToBoxAdapter( | ||||||
|  |                           child: Column(children: [getFullInfoColumn()])), | ||||||
|  |                     ], | ||||||
|  |                   ), | ||||||
|  |             onRefresh: () async { | ||||||
|  |               if (app != null) { | ||||||
|  |                 getUpdate(app.app.id); | ||||||
|  |               } | ||||||
|  |             }), | ||||||
|  |         bottomSheet: getBottomSheetMenu()); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1480
									
								
								lib/pages/apps.dart
									
									
									
									
									
								
							
							
						
						| @@ -63,21 +63,29 @@ class _HomePageState extends State<HomePage> { | |||||||
|                 .map((e) => |                 .map((e) => | ||||||
|                     NavigationDestination(icon: Icon(e.icon), label: e.title)) |                     NavigationDestination(icon: Icon(e.icon), label: e.title)) | ||||||
|                 .toList(), |                 .toList(), | ||||||
|             onDestinationSelected: (int index) { |             onDestinationSelected: (int index) async { | ||||||
|               HapticFeedback.selectionClick(); |               HapticFeedback.selectionClick(); | ||||||
|               setState(() { |               if (index == 0) { | ||||||
|                 if (index == 0) { |                 while ((pages[0].widget.key as GlobalKey<AppsPageState>) | ||||||
|  |                         .currentState != | ||||||
|  |                     null) { | ||||||
|  |                   // Avoid duplicate GlobalKey error | ||||||
|  |                   await Future.delayed(const Duration(microseconds: 1)); | ||||||
|  |                 } | ||||||
|  |                 setState(() { | ||||||
|                   selectedIndexHistory.clear(); |                   selectedIndexHistory.clear(); | ||||||
|                 } else if (selectedIndexHistory.isEmpty || |                 }); | ||||||
|                     (selectedIndexHistory.isNotEmpty && |               } else if (selectedIndexHistory.isEmpty || | ||||||
|                         selectedIndexHistory.last != index)) { |                   (selectedIndexHistory.isNotEmpty && | ||||||
|  |                       selectedIndexHistory.last != index)) { | ||||||
|  |                 setState(() { | ||||||
|                   int existingInd = selectedIndexHistory.indexOf(index); |                   int existingInd = selectedIndexHistory.indexOf(index); | ||||||
|                   if (existingInd >= 0) { |                   if (existingInd >= 0) { | ||||||
|                     selectedIndexHistory.removeAt(existingInd); |                     selectedIndexHistory.removeAt(existingInd); | ||||||
|                   } |                   } | ||||||
|                   selectedIndexHistory.add(index); |                   selectedIndexHistory.add(index); | ||||||
|                 } |                 }); | ||||||
|               }); |               } | ||||||
|             }, |             }, | ||||||
|             selectedIndex: |             selectedIndex: | ||||||
|                 selectedIndexHistory.isEmpty ? 0 : selectedIndexHistory.last, |                 selectedIndexHistory.isEmpty ? 0 : selectedIndexHistory.last, | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import 'package:obtainium/components/generated_form.dart'; | |||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
|  | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
| import 'package:file_picker/file_picker.dart'; | import 'package:file_picker/file_picker.dart'; | ||||||
| @@ -28,6 +29,8 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |     SourceProvider sourceProvider = SourceProvider(); | ||||||
|     var appsProvider = context.read<AppsProvider>(); |     var appsProvider = context.read<AppsProvider>(); | ||||||
|  |     var settingsProvider = context.read<SettingsProvider>(); | ||||||
|  |  | ||||||
|     var outlineButtonStyle = ButtonStyle( |     var outlineButtonStyle = ButtonStyle( | ||||||
|       shape: MaterialStateProperty.all( |       shape: MaterialStateProperty.all( | ||||||
|         StadiumBorder( |         StadiumBorder( | ||||||
| @@ -39,6 +42,253 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     urlListImport({String? initValue, bool overrideInitValid = false}) { | ||||||
|  |       showDialog<Map<String, dynamic>?>( | ||||||
|  |           context: context, | ||||||
|  |           builder: (BuildContext ctx) { | ||||||
|  |             return GeneratedFormModal( | ||||||
|  |               initValid: overrideInitValid, | ||||||
|  |               title: tr('importFromURLList'), | ||||||
|  |               items: [ | ||||||
|  |                 [ | ||||||
|  |                   GeneratedFormTextField('appURLList', | ||||||
|  |                       defaultValue: initValue ?? '', | ||||||
|  |                       label: tr('appURLList'), | ||||||
|  |                       max: 7, | ||||||
|  |                       additionalValidators: [ | ||||||
|  |                         (dynamic value) { | ||||||
|  |                           if (value != null && value.isNotEmpty) { | ||||||
|  |                             var lines = value.trim().split('\n'); | ||||||
|  |                             for (int i = 0; i < lines.length; i++) { | ||||||
|  |                               try { | ||||||
|  |                                 sourceProvider.getSource(lines[i]); | ||||||
|  |                               } catch (e) { | ||||||
|  |                                 return '${tr('line')} ${i + 1}: $e'; | ||||||
|  |                               } | ||||||
|  |                             } | ||||||
|  |                           } | ||||||
|  |                           return null; | ||||||
|  |                         } | ||||||
|  |                       ]) | ||||||
|  |                 ] | ||||||
|  |               ], | ||||||
|  |             ); | ||||||
|  |           }).then((values) { | ||||||
|  |         if (values != null) { | ||||||
|  |           var urls = (values['appURLList'] as String).split('\n'); | ||||||
|  |           setState(() { | ||||||
|  |             importInProgress = true; | ||||||
|  |           }); | ||||||
|  |           appsProvider.addAppsByURL(urls).then((errors) { | ||||||
|  |             if (errors.isEmpty) { | ||||||
|  |               showError(tr('importedX', args: [plural('apps', urls.length)]), | ||||||
|  |                   context); | ||||||
|  |             } else { | ||||||
|  |               showDialog( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return ImportErrorDialog( | ||||||
|  |                         urlsLength: urls.length, errors: errors); | ||||||
|  |                   }); | ||||||
|  |             } | ||||||
|  |           }).catchError((e) { | ||||||
|  |             showError(e, context); | ||||||
|  |           }).whenComplete(() { | ||||||
|  |             setState(() { | ||||||
|  |               importInProgress = false; | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runObtainiumExport() { | ||||||
|  |       HapticFeedback.selectionClick(); | ||||||
|  |       appsProvider.exportApps().then((String path) { | ||||||
|  |         showError(tr('exportedTo', args: [path]), context); | ||||||
|  |       }).catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runObtainiumImport() { | ||||||
|  |       HapticFeedback.selectionClick(); | ||||||
|  |       FilePicker.platform.pickFiles().then((result) { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = true; | ||||||
|  |         }); | ||||||
|  |         if (result != null) { | ||||||
|  |           String data = File(result.files.single.path!).readAsStringSync(); | ||||||
|  |           try { | ||||||
|  |             jsonDecode(data); | ||||||
|  |           } catch (e) { | ||||||
|  |             throw ObtainiumError(tr('invalidInput')); | ||||||
|  |           } | ||||||
|  |           appsProvider.importApps(data).then((value) { | ||||||
|  |             var cats = settingsProvider.categories; | ||||||
|  |             appsProvider.apps.forEach((key, value) { | ||||||
|  |               for (var c in value.app.categories) { | ||||||
|  |                 if (!cats.containsKey(c)) { | ||||||
|  |                   cats[c] = generateRandomLightColor().value; | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |             appsProvider.addMissingCategories(settingsProvider); | ||||||
|  |             showError(tr('importedX', args: [plural('apps', value)]), context); | ||||||
|  |           }); | ||||||
|  |         } else { | ||||||
|  |           // User canceled the picker | ||||||
|  |         } | ||||||
|  |       }).catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }).whenComplete(() { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = false; | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runUrlImport() { | ||||||
|  |       FilePicker.platform.pickFiles().then((result) { | ||||||
|  |         if (result != null) { | ||||||
|  |           urlListImport( | ||||||
|  |               overrideInitValid: true, | ||||||
|  |               initValue: RegExp('https?://[^"]+') | ||||||
|  |                   .allMatches( | ||||||
|  |                       File(result.files.single.path!).readAsStringSync()) | ||||||
|  |                   .map((e) => e.input.substring(e.start, e.end)) | ||||||
|  |                   .toSet() | ||||||
|  |                   .toList() | ||||||
|  |                   .where((url) { | ||||||
|  |                 try { | ||||||
|  |                   sourceProvider.getSource(url); | ||||||
|  |                   return true; | ||||||
|  |                 } catch (e) { | ||||||
|  |                   return false; | ||||||
|  |                 } | ||||||
|  |               }).join('\n')); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runSourceSearch(AppSource source) { | ||||||
|  |       () async { | ||||||
|  |         var values = await showDialog<Map<String, dynamic>?>( | ||||||
|  |             context: context, | ||||||
|  |             builder: (BuildContext ctx) { | ||||||
|  |               return GeneratedFormModal( | ||||||
|  |                 title: tr('searchX', args: [source.name]), | ||||||
|  |                 items: [ | ||||||
|  |                   [ | ||||||
|  |                     GeneratedFormTextField('searchQuery', | ||||||
|  |                         label: tr('searchQuery')) | ||||||
|  |                   ] | ||||||
|  |                 ], | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |         if (values != null && | ||||||
|  |             (values['searchQuery'] as String?)?.isNotEmpty == true) { | ||||||
|  |           setState(() { | ||||||
|  |             importInProgress = true; | ||||||
|  |           }); | ||||||
|  |           var urlsWithDescriptions = | ||||||
|  |               await source.search(values['searchQuery'] as String); | ||||||
|  |           if (urlsWithDescriptions.isNotEmpty) { | ||||||
|  |             var selectedUrls = | ||||||
|  |                 // ignore: use_build_context_synchronously | ||||||
|  |                 await showDialog<List<String>?>( | ||||||
|  |                     context: context, | ||||||
|  |                     builder: (BuildContext ctx) { | ||||||
|  |                       return UrlSelectionModal( | ||||||
|  |                         urlsWithDescriptions: urlsWithDescriptions, | ||||||
|  |                         selectedByDefault: false, | ||||||
|  |                       ); | ||||||
|  |                     }); | ||||||
|  |             if (selectedUrls != null && selectedUrls.isNotEmpty) { | ||||||
|  |               var errors = await appsProvider.addAppsByURL(selectedUrls); | ||||||
|  |               if (errors.isEmpty) { | ||||||
|  |                 // ignore: use_build_context_synchronously | ||||||
|  |                 showError( | ||||||
|  |                     tr('importedX', args: [plural('app', selectedUrls.length)]), | ||||||
|  |                     context); | ||||||
|  |               } else { | ||||||
|  |                 // ignore: use_build_context_synchronously | ||||||
|  |                 showDialog( | ||||||
|  |                     context: context, | ||||||
|  |                     builder: (BuildContext ctx) { | ||||||
|  |                       return ImportErrorDialog( | ||||||
|  |                           urlsLength: selectedUrls.length, errors: errors); | ||||||
|  |                     }); | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } else { | ||||||
|  |             throw ObtainiumError(tr('noResults')); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }() | ||||||
|  |           .catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }).whenComplete(() { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = false; | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runMassSourceImport(MassAppUrlSource source) { | ||||||
|  |       () async { | ||||||
|  |         var values = await showDialog<Map<String, dynamic>?>( | ||||||
|  |             context: context, | ||||||
|  |             builder: (BuildContext ctx) { | ||||||
|  |               return GeneratedFormModal( | ||||||
|  |                 title: tr('importX', args: [source.name]), | ||||||
|  |                 items: source.requiredArgs | ||||||
|  |                     .map((e) => [GeneratedFormTextField(e, label: e)]) | ||||||
|  |                     .toList(), | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |         if (values != null) { | ||||||
|  |           setState(() { | ||||||
|  |             importInProgress = true; | ||||||
|  |           }); | ||||||
|  |           var urlsWithDescriptions = await source.getUrlsWithDescriptions( | ||||||
|  |               values.values.map((e) => e.toString()).toList()); | ||||||
|  |           var selectedUrls = | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               await showDialog<List<String>?>( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return UrlSelectionModal( | ||||||
|  |                         urlsWithDescriptions: urlsWithDescriptions); | ||||||
|  |                   }); | ||||||
|  |           if (selectedUrls != null) { | ||||||
|  |             var errors = await appsProvider.addAppsByURL(selectedUrls); | ||||||
|  |             if (errors.isEmpty) { | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               showError( | ||||||
|  |                   tr('importedX', args: [plural('app', selectedUrls.length)]), | ||||||
|  |                   context); | ||||||
|  |             } else { | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               showDialog( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return ImportErrorDialog( | ||||||
|  |                         urlsLength: selectedUrls.length, errors: errors); | ||||||
|  |                   }); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }() | ||||||
|  |           .catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }).whenComplete(() { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = false; | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|         backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|         body: CustomScrollView(slivers: <Widget>[ |         body: CustomScrollView(slivers: <Widget>[ | ||||||
| @@ -58,16 +308,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                   onPressed: appsProvider.apps.isEmpty || |                                   onPressed: appsProvider.apps.isEmpty || | ||||||
|                                           importInProgress |                                           importInProgress | ||||||
|                                       ? null |                                       ? null | ||||||
|                                       : () { |                                       : runObtainiumExport, | ||||||
|                                           HapticFeedback.selectionClick(); |  | ||||||
|                                           appsProvider |  | ||||||
|                                               .exportApps() |  | ||||||
|                                               .then((String path) { |  | ||||||
|                                             showError( |  | ||||||
|                                                 tr('exportedTo', args: [path]), |  | ||||||
|                                                 context); |  | ||||||
|                                           }); |  | ||||||
|                                         }, |  | ||||||
|                                   child: Text(tr('obtainiumExport')))), |                                   child: Text(tr('obtainiumExport')))), | ||||||
|                           const SizedBox( |                           const SizedBox( | ||||||
|                             width: 16, |                             width: 16, | ||||||
| @@ -77,44 +318,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                   style: outlineButtonStyle, |                                   style: outlineButtonStyle, | ||||||
|                                   onPressed: importInProgress |                                   onPressed: importInProgress | ||||||
|                                       ? null |                                       ? null | ||||||
|                                       : () { |                                       : runObtainiumImport, | ||||||
|                                           HapticFeedback.selectionClick(); |  | ||||||
|                                           FilePicker.platform |  | ||||||
|                                               .pickFiles() |  | ||||||
|                                               .then((result) { |  | ||||||
|                                             setState(() { |  | ||||||
|                                               importInProgress = true; |  | ||||||
|                                             }); |  | ||||||
|                                             if (result != null) { |  | ||||||
|                                               String data = File( |  | ||||||
|                                                       result.files.single.path!) |  | ||||||
|                                                   .readAsStringSync(); |  | ||||||
|                                               try { |  | ||||||
|                                                 jsonDecode(data); |  | ||||||
|                                               } catch (e) { |  | ||||||
|                                                 throw ObtainiumError( |  | ||||||
|                                                     tr('invalidInput')); |  | ||||||
|                                               } |  | ||||||
|                                               appsProvider |  | ||||||
|                                                   .importApps(data) |  | ||||||
|                                                   .then((value) { |  | ||||||
|                                                 showError( |  | ||||||
|                                                     tr('importedX', args: [ |  | ||||||
|                                                       plural('apps', value) |  | ||||||
|                                                     ]), |  | ||||||
|                                                     context); |  | ||||||
|                                               }); |  | ||||||
|                                             } else { |  | ||||||
|                                               // User canceled the picker |  | ||||||
|                                             } |  | ||||||
|                                           }).catchError((e) { |  | ||||||
|                                             showError(e, context); |  | ||||||
|                                           }).whenComplete(() { |  | ||||||
|                                             setState(() { |  | ||||||
|                                               importInProgress = false; |  | ||||||
|                                             }); |  | ||||||
|                                           }); |  | ||||||
|                                         }, |  | ||||||
|                                   child: Text(tr('obtainiumImport')))) |                                   child: Text(tr('obtainiumImport')))) | ||||||
|                         ], |                         ], | ||||||
|                       ), |                       ), | ||||||
| @@ -131,88 +335,26 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                           ], |                           ], | ||||||
|                         ) |                         ) | ||||||
|                       else |                       else | ||||||
|                         const Divider( |                         Column( | ||||||
|                           height: 32, |                           children: [ | ||||||
|  |                             const Divider( | ||||||
|  |                               height: 32, | ||||||
|  |                             ), | ||||||
|  |                             TextButton( | ||||||
|  |                                 onPressed: | ||||||
|  |                                     importInProgress ? null : urlListImport, | ||||||
|  |                                 child: Text( | ||||||
|  |                                   tr('importFromURLList'), | ||||||
|  |                                 )), | ||||||
|  |                             const SizedBox(height: 8), | ||||||
|  |                             TextButton( | ||||||
|  |                                 onPressed: | ||||||
|  |                                     importInProgress ? null : runUrlImport, | ||||||
|  |                                 child: Text( | ||||||
|  |                                   tr('importFromURLsInFile'), | ||||||
|  |                                 )), | ||||||
|  |                           ], | ||||||
|                         ), |                         ), | ||||||
|                       TextButton( |  | ||||||
|                           onPressed: importInProgress |  | ||||||
|                               ? null |  | ||||||
|                               : () { |  | ||||||
|                                   showDialog<Map<String, dynamic>?>( |  | ||||||
|                                       context: context, |  | ||||||
|                                       builder: (BuildContext ctx) { |  | ||||||
|                                         return GeneratedFormModal( |  | ||||||
|                                           title: tr('importFromURLList'), |  | ||||||
|                                           items: [ |  | ||||||
|                                             [ |  | ||||||
|                                               GeneratedFormTextField( |  | ||||||
|                                                   'appURLList', |  | ||||||
|                                                   label: tr('appURLList'), |  | ||||||
|                                                   max: 7, |  | ||||||
|                                                   additionalValidators: [ |  | ||||||
|                                                     (dynamic value) { |  | ||||||
|                                                       if (value != null && |  | ||||||
|                                                           value.isNotEmpty) { |  | ||||||
|                                                         var lines = value |  | ||||||
|                                                             .trim() |  | ||||||
|                                                             .split('\n'); |  | ||||||
|                                                         for (int i = 0; |  | ||||||
|                                                             i < lines.length; |  | ||||||
|                                                             i++) { |  | ||||||
|                                                           try { |  | ||||||
|                                                             sourceProvider |  | ||||||
|                                                                 .getSource( |  | ||||||
|                                                                     lines[i]); |  | ||||||
|                                                           } catch (e) { |  | ||||||
|                                                             return '${tr('line')} ${i + 1}: $e'; |  | ||||||
|                                                           } |  | ||||||
|                                                         } |  | ||||||
|                                                       } |  | ||||||
|                                                       return null; |  | ||||||
|                                                     } |  | ||||||
|                                                   ]) |  | ||||||
|                                             ] |  | ||||||
|                                           ], |  | ||||||
|                                         ); |  | ||||||
|                                       }).then((values) { |  | ||||||
|                                     if (values != null) { |  | ||||||
|                                       var urls = |  | ||||||
|                                           (values['appURLList'] as String) |  | ||||||
|                                               .split('\n'); |  | ||||||
|                                       setState(() { |  | ||||||
|                                         importInProgress = true; |  | ||||||
|                                       }); |  | ||||||
|                                       appsProvider |  | ||||||
|                                           .addAppsByURL(urls) |  | ||||||
|                                           .then((errors) { |  | ||||||
|                                         if (errors.isEmpty) { |  | ||||||
|                                           showError( |  | ||||||
|                                               tr('importedX', args: [ |  | ||||||
|                                                 plural('apps', urls.length) |  | ||||||
|                                               ]), |  | ||||||
|                                               context); |  | ||||||
|                                         } else { |  | ||||||
|                                           showDialog( |  | ||||||
|                                               context: context, |  | ||||||
|                                               builder: (BuildContext ctx) { |  | ||||||
|                                                 return ImportErrorDialog( |  | ||||||
|                                                     urlsLength: urls.length, |  | ||||||
|                                                     errors: errors); |  | ||||||
|                                               }); |  | ||||||
|                                         } |  | ||||||
|                                       }).catchError((e) { |  | ||||||
|                                         showError(e, context); |  | ||||||
|                                       }).whenComplete(() { |  | ||||||
|                                         setState(() { |  | ||||||
|                                           importInProgress = false; |  | ||||||
|                                         }); |  | ||||||
|                                       }); |  | ||||||
|                                     } |  | ||||||
|                                   }); |  | ||||||
|                                 }, |  | ||||||
|                           child: Text( |  | ||||||
|                             tr('importFromURLList'), |  | ||||||
|                           )), |  | ||||||
|                       ...sourceProvider.sources |                       ...sourceProvider.sources | ||||||
|                           .where((element) => element.canSearch) |                           .where((element) => element.canSearch) | ||||||
|                           .map((source) => Column( |                           .map((source) => Column( | ||||||
| @@ -224,104 +366,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                         onPressed: importInProgress |                                         onPressed: importInProgress | ||||||
|                                             ? null |                                             ? null | ||||||
|                                             : () { |                                             : () { | ||||||
|                                                 () async { |                                                 runSourceSearch(source); | ||||||
|                                                   var values = await showDialog< |  | ||||||
|                                                           Map<String, |  | ||||||
|                                                               dynamic>?>( |  | ||||||
|                                                       context: context, |  | ||||||
|                                                       builder: |  | ||||||
|                                                           (BuildContext ctx) { |  | ||||||
|                                                         return GeneratedFormModal( |  | ||||||
|                                                           title: tr('searchX', |  | ||||||
|                                                               args: [ |  | ||||||
|                                                                 source.name |  | ||||||
|                                                               ]), |  | ||||||
|                                                           items: [ |  | ||||||
|                                                             [ |  | ||||||
|                                                               GeneratedFormTextField( |  | ||||||
|                                                                   'searchQuery', |  | ||||||
|                                                                   label: tr( |  | ||||||
|                                                                       'searchQuery')) |  | ||||||
|                                                             ] |  | ||||||
|                                                           ], |  | ||||||
|                                                         ); |  | ||||||
|                                                       }); |  | ||||||
|                                                   if (values != null && |  | ||||||
|                                                       (values['searchQuery'] |  | ||||||
|                                                                   as String?) |  | ||||||
|                                                               ?.isNotEmpty == |  | ||||||
|                                                           true) { |  | ||||||
|                                                     setState(() { |  | ||||||
|                                                       importInProgress = true; |  | ||||||
|                                                     }); |  | ||||||
|                                                     var urlsWithDescriptions = |  | ||||||
|                                                         await source.search( |  | ||||||
|                                                             values['searchQuery'] |  | ||||||
|                                                                 as String); |  | ||||||
|                                                     if (urlsWithDescriptions |  | ||||||
|                                                         .isNotEmpty) { |  | ||||||
|                                                       var selectedUrls = |  | ||||||
|                                                           await showDialog< |  | ||||||
|                                                                   List< |  | ||||||
|                                                                       String>?>( |  | ||||||
|                                                               context: context, |  | ||||||
|                                                               builder: |  | ||||||
|                                                                   (BuildContext |  | ||||||
|                                                                       ctx) { |  | ||||||
|                                                                 return UrlSelectionModal( |  | ||||||
|                                                                   urlsWithDescriptions: |  | ||||||
|                                                                       urlsWithDescriptions, |  | ||||||
|                                                                   selectedByDefault: |  | ||||||
|                                                                       false, |  | ||||||
|                                                                 ); |  | ||||||
|                                                               }); |  | ||||||
|                                                       if (selectedUrls != |  | ||||||
|                                                               null && |  | ||||||
|                                                           selectedUrls |  | ||||||
|                                                               .isNotEmpty) { |  | ||||||
|                                                         var errors = |  | ||||||
|                                                             await appsProvider |  | ||||||
|                                                                 .addAppsByURL( |  | ||||||
|                                                                     selectedUrls); |  | ||||||
|                                                         if (errors.isEmpty) { |  | ||||||
|                                                           // ignore: use_build_context_synchronously |  | ||||||
|                                                           showError( |  | ||||||
|                                                               tr('importedX', |  | ||||||
|                                                                   args: [ |  | ||||||
|                                                                     plural( |  | ||||||
|                                                                         'app', |  | ||||||
|                                                                         selectedUrls |  | ||||||
|                                                                             .length) |  | ||||||
|                                                                   ]), |  | ||||||
|                                                               context); |  | ||||||
|                                                         } else { |  | ||||||
|                                                           showDialog( |  | ||||||
|                                                               context: context, |  | ||||||
|                                                               builder: |  | ||||||
|                                                                   (BuildContext |  | ||||||
|                                                                       ctx) { |  | ||||||
|                                                                 return ImportErrorDialog( |  | ||||||
|                                                                     urlsLength: |  | ||||||
|                                                                         selectedUrls |  | ||||||
|                                                                             .length, |  | ||||||
|                                                                     errors: |  | ||||||
|                                                                         errors); |  | ||||||
|                                                               }); |  | ||||||
|                                                         } |  | ||||||
|                                                       } |  | ||||||
|                                                     } else { |  | ||||||
|                                                       throw ObtainiumError( |  | ||||||
|                                                           tr('noResults')); |  | ||||||
|                                                     } |  | ||||||
|                                                   } |  | ||||||
|                                                 }() |  | ||||||
|                                                     .catchError((e) { |  | ||||||
|                                                   showError(e, context); |  | ||||||
|                                                 }).whenComplete(() { |  | ||||||
|                                                   setState(() { |  | ||||||
|                                                     importInProgress = false; |  | ||||||
|                                                   }); |  | ||||||
|                                                 }); |  | ||||||
|                                               }, |                                               }, | ||||||
|                                         child: Text( |                                         child: Text( | ||||||
|                                             tr('searchX', args: [source.name]))) |                                             tr('searchX', args: [source.name]))) | ||||||
| @@ -337,86 +382,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                         onPressed: importInProgress |                                         onPressed: importInProgress | ||||||
|                                             ? null |                                             ? null | ||||||
|                                             : () { |                                             : () { | ||||||
|                                                 () async { |                                                 runMassSourceImport(source); | ||||||
|                                                   var values = await showDialog( |  | ||||||
|                                                       context: context, |  | ||||||
|                                                       builder: |  | ||||||
|                                                           (BuildContext ctx) { |  | ||||||
|                                                         return GeneratedFormModal( |  | ||||||
|                                                           title: tr('importX', |  | ||||||
|                                                               args: [ |  | ||||||
|                                                                 source.name |  | ||||||
|                                                               ]), |  | ||||||
|                                                           items: |  | ||||||
|                                                               source |  | ||||||
|                                                                   .requiredArgs |  | ||||||
|                                                                   .map( |  | ||||||
|                                                                       (e) => [ |  | ||||||
|                                                                             GeneratedFormTextField(e, |  | ||||||
|                                                                                 label: e) |  | ||||||
|                                                                           ]) |  | ||||||
|                                                                   .toList(), |  | ||||||
|                                                         ); |  | ||||||
|                                                       }); |  | ||||||
|                                                   if (values != null) { |  | ||||||
|                                                     setState(() { |  | ||||||
|                                                       importInProgress = true; |  | ||||||
|                                                     }); |  | ||||||
|                                                     var urlsWithDescriptions = |  | ||||||
|                                                         await source |  | ||||||
|                                                             .getUrlsWithDescriptions( |  | ||||||
|                                                                 values); |  | ||||||
|                                                     var selectedUrls = |  | ||||||
|                                                         await showDialog< |  | ||||||
|                                                                 List<String>?>( |  | ||||||
|                                                             context: context, |  | ||||||
|                                                             builder: |  | ||||||
|                                                                 (BuildContext |  | ||||||
|                                                                     ctx) { |  | ||||||
|                                                               return UrlSelectionModal( |  | ||||||
|                                                                   urlsWithDescriptions: |  | ||||||
|                                                                       urlsWithDescriptions); |  | ||||||
|                                                             }); |  | ||||||
|                                                     if (selectedUrls != null) { |  | ||||||
|                                                       var errors = |  | ||||||
|                                                           await appsProvider |  | ||||||
|                                                               .addAppsByURL( |  | ||||||
|                                                                   selectedUrls); |  | ||||||
|                                                       if (errors.isEmpty) { |  | ||||||
|                                                         // ignore: use_build_context_synchronously |  | ||||||
|                                                         showError( |  | ||||||
|                                                             tr('importedX', |  | ||||||
|                                                                 args: [ |  | ||||||
|                                                                   plural( |  | ||||||
|                                                                       'app', |  | ||||||
|                                                                       selectedUrls |  | ||||||
|                                                                           .length) |  | ||||||
|                                                                 ]), |  | ||||||
|                                                             context); |  | ||||||
|                                                       } else { |  | ||||||
|                                                         showDialog( |  | ||||||
|                                                             context: context, |  | ||||||
|                                                             builder: |  | ||||||
|                                                                 (BuildContext |  | ||||||
|                                                                     ctx) { |  | ||||||
|                                                               return ImportErrorDialog( |  | ||||||
|                                                                   urlsLength: |  | ||||||
|                                                                       selectedUrls |  | ||||||
|                                                                           .length, |  | ||||||
|                                                                   errors: |  | ||||||
|                                                                       errors); |  | ||||||
|                                                             }); |  | ||||||
|                                                       } |  | ||||||
|                                                     } |  | ||||||
|                                                   } |  | ||||||
|                                                 }() |  | ||||||
|                                                     .catchError((e) { |  | ||||||
|                                                   showError(e, context); |  | ||||||
|                                                 }).whenComplete(() { |  | ||||||
|                                                   setState(() { |  | ||||||
|                                                     importInProgress = false; |  | ||||||
|                                                   }); |  | ||||||
|                                                 }); |  | ||||||
|                                               }, |                                               }, | ||||||
|                                         child: Text( |                                         child: Text( | ||||||
|                                             tr('importX', args: [source.name]))) |                                             tr('importX', args: [source.name]))) | ||||||
| @@ -540,18 +506,22 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|           widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')), |           widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')), | ||||||
|       content: Column(children: [ |       content: Column(children: [ | ||||||
|         ...urlWithDescriptionSelections.keys.map((urlWithD) { |         ...urlWithDescriptionSelections.keys.map((urlWithD) { | ||||||
|  |           select(bool? value) { | ||||||
|  |             setState(() { | ||||||
|  |               value ??= false; | ||||||
|  |               if (value! && widget.onlyOneSelectionAllowed) { | ||||||
|  |                 selectOnlyOne(urlWithD.key); | ||||||
|  |               } else { | ||||||
|  |                 urlWithDescriptionSelections[urlWithD] = value!; | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |  | ||||||
|           return Row(children: [ |           return Row(children: [ | ||||||
|             Checkbox( |             Checkbox( | ||||||
|                 value: urlWithDescriptionSelections[urlWithD], |                 value: urlWithDescriptionSelections[urlWithD], | ||||||
|                 onChanged: (value) { |                 onChanged: (value) { | ||||||
|                   setState(() { |                   select(value); | ||||||
|                     value ??= false; |  | ||||||
|                     if (value! && widget.onlyOneSelectionAllowed) { |  | ||||||
|                       selectOnlyOne(urlWithD.key); |  | ||||||
|                     } else { |  | ||||||
|                       urlWithDescriptionSelections[urlWithD] = value!; |  | ||||||
|                     } |  | ||||||
|                   }); |  | ||||||
|                 }), |                 }), | ||||||
|             const SizedBox( |             const SizedBox( | ||||||
|               width: 8, |               width: 8, | ||||||
| @@ -575,12 +545,17 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|                           const TextStyle(decoration: TextDecoration.underline), |                           const TextStyle(decoration: TextDecoration.underline), | ||||||
|                       textAlign: TextAlign.start, |                       textAlign: TextAlign.start, | ||||||
|                     )), |                     )), | ||||||
|                 Text( |                 GestureDetector( | ||||||
|                   urlWithD.value.length > 128 |                   onTap: () { | ||||||
|                       ? '${urlWithD.value.substring(0, 128)}...' |                     select(!(urlWithDescriptionSelections[urlWithD] ?? false)); | ||||||
|                       : urlWithD.value, |                   }, | ||||||
|                   style: const TextStyle( |                   child: Text( | ||||||
|                       fontStyle: FontStyle.italic, fontSize: 12), |                     urlWithD.value.length > 128 | ||||||
|  |                         ? '${urlWithD.value.substring(0, 128)}...' | ||||||
|  |                         : urlWithD.value, | ||||||
|  |                     style: const TextStyle( | ||||||
|  |                         fontStyle: FontStyle.italic, fontSize: 12), | ||||||
|  |                   ), | ||||||
|                 ), |                 ), | ||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
|                   height: 8, |                   height: 8, | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:obtainium/components/custom_app_bar.dart'; | import 'package:obtainium/components/custom_app_bar.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.dart'; |  | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| @@ -89,6 +88,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|         }); |         }); | ||||||
|  |  | ||||||
|     var sortDropdown = DropdownButtonFormField( |     var sortDropdown = DropdownButtonFormField( | ||||||
|  |         isExpanded: true, | ||||||
|         decoration: InputDecoration(labelText: tr('appSortBy')), |         decoration: InputDecoration(labelText: tr('appSortBy')), | ||||||
|         value: settingsProvider.sortColumn, |         value: settingsProvider.sortColumn, | ||||||
|         items: [ |         items: [ | ||||||
| @@ -103,6 +103,10 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|           DropdownMenuItem( |           DropdownMenuItem( | ||||||
|             value: SortColumnSettings.added, |             value: SortColumnSettings.added, | ||||||
|             child: Text(tr('asAdded')), |             child: Text(tr('asAdded')), | ||||||
|  |           ), | ||||||
|  |           DropdownMenuItem( | ||||||
|  |             value: SortColumnSettings.releaseDate, | ||||||
|  |             child: Text(tr('releaseDate')), | ||||||
|           ) |           ) | ||||||
|         ], |         ], | ||||||
|         onChanged: (value) { |         onChanged: (value) { | ||||||
| @@ -112,6 +116,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|         }); |         }); | ||||||
|  |  | ||||||
|     var orderDropdown = DropdownButtonFormField( |     var orderDropdown = DropdownButtonFormField( | ||||||
|  |         isExpanded: true, | ||||||
|         decoration: InputDecoration(labelText: tr('appSortOrder')), |         decoration: InputDecoration(labelText: tr('appSortOrder')), | ||||||
|         value: settingsProvider.sortOrder, |         value: settingsProvider.sortOrder, | ||||||
|         items: [ |         items: [ | ||||||
| @@ -148,7 +153,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|           if (value != null) { |           if (value != null) { | ||||||
|             context.setLocale(Locale(value)); |             context.setLocale(Locale(value)); | ||||||
|           } else { |           } else { | ||||||
|             context.resetLocale(); |             settingsProvider.resetLocaleSafe(context); | ||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
| @@ -185,7 +190,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|               return [e]; |               return [e]; | ||||||
|             }).toList(), |             }).toList(), | ||||||
|             onValueChanges: (values, valid, isBuilding) { |             onValueChanges: (values, valid, isBuilding) { | ||||||
|               if (valid) { |               if (valid && !isBuilding) { | ||||||
|                 values.forEach((key, value) { |                 values.forEach((key, value) { | ||||||
|                   settingsProvider.setSettingString(key, value); |                   settingsProvider.setSettingString(key, value); | ||||||
|                 }); |                 }); | ||||||
| @@ -258,6 +263,18 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                     }) |                                     }) | ||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|  |                             height16, | ||||||
|  |                             Row( | ||||||
|  |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|  |                               children: [ | ||||||
|  |                                 Text(tr('groupByCategory')), | ||||||
|  |                                 Switch( | ||||||
|  |                                     value: settingsProvider.groupByCategory, | ||||||
|  |                                     onChanged: (value) { | ||||||
|  |                                       settingsProvider.groupByCategory = value; | ||||||
|  |                                     }) | ||||||
|  |                               ], | ||||||
|  |                             ), | ||||||
|                             const Divider( |                             const Divider( | ||||||
|                               height: 16, |                               height: 16, | ||||||
|                             ), |                             ), | ||||||
| @@ -286,7 +303,9 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                   color: Theme.of(context).colorScheme.primary), |                                   color: Theme.of(context).colorScheme.primary), | ||||||
|                             ), |                             ), | ||||||
|                             height16, |                             height16, | ||||||
|                             const CategoryEditorSelector() |                             const CategoryEditorSelector( | ||||||
|  |                               showLabelWhenNotEmpty: false, | ||||||
|  |                             ) | ||||||
|                           ], |                           ], | ||||||
|                         ))), |                         ))), | ||||||
|           SliverToBoxAdapter( |           SliverToBoxAdapter( | ||||||
| @@ -407,12 +426,14 @@ class CategoryEditorSelector extends StatefulWidget { | |||||||
|   final bool singleSelect; |   final bool singleSelect; | ||||||
|   final Set<String> preselected; |   final Set<String> preselected; | ||||||
|   final WrapAlignment alignment; |   final WrapAlignment alignment; | ||||||
|  |   final bool showLabelWhenNotEmpty; | ||||||
|   const CategoryEditorSelector( |   const CategoryEditorSelector( | ||||||
|       {super.key, |       {super.key, | ||||||
|       this.onSelected, |       this.onSelected, | ||||||
|       this.singleSelect = false, |       this.singleSelect = false, | ||||||
|       this.preselected = const {}, |       this.preselected = const {}, | ||||||
|       this.alignment = WrapAlignment.start}); |       this.alignment = WrapAlignment.start, | ||||||
|  |       this.showLabelWhenNotEmpty = true}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   State<CategoryEditorSelector> createState() => _CategoryEditorSelectorState(); |   State<CategoryEditorSelector> createState() => _CategoryEditorSelectorState(); | ||||||
| @@ -424,6 +445,7 @@ class _CategoryEditorSelectorState extends State<CategoryEditorSelector> { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     var settingsProvider = context.watch<SettingsProvider>(); |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|  |     var appsProvider = context.watch<AppsProvider>(); | ||||||
|     storedValues = settingsProvider.categories.map((key, value) => MapEntry( |     storedValues = settingsProvider.categories.map((key, value) => MapEntry( | ||||||
|         key, |         key, | ||||||
|         MapEntry(value, |         MapEntry(value, | ||||||
| @@ -432,22 +454,24 @@ class _CategoryEditorSelectorState extends State<CategoryEditorSelector> { | |||||||
|         items: [ |         items: [ | ||||||
|           [ |           [ | ||||||
|             GeneratedFormTagInput('categories', |             GeneratedFormTagInput('categories', | ||||||
|                 label: tr('category'), |                 label: tr('categories'), | ||||||
|                 emptyMessage: tr('noCategories'), |                 emptyMessage: tr('noCategories'), | ||||||
|                 defaultValue: storedValues, |                 defaultValue: storedValues, | ||||||
|                 alignment: widget.alignment, |                 alignment: widget.alignment, | ||||||
|                 deleteConfirmationMessage: MapEntry( |                 deleteConfirmationMessage: MapEntry( | ||||||
|                     tr('deleteCategoriesQuestion'), |                     tr('deleteCategoriesQuestion'), | ||||||
|                     tr('categoryDeleteWarning')), |                     tr('categoryDeleteWarning')), | ||||||
|                 singleSelect: widget.singleSelect) |                 singleSelect: widget.singleSelect, | ||||||
|  |                 showLabelWhenNotEmpty: widget.showLabelWhenNotEmpty) | ||||||
|           ] |           ] | ||||||
|         ], |         ], | ||||||
|         onValueChanges: ((values, valid, isBuilding) { |         onValueChanges: ((values, valid, isBuilding) { | ||||||
|           if (!isBuilding) { |           if (!isBuilding) { | ||||||
|             storedValues = |             storedValues = | ||||||
|                 values['categories'] as Map<String, MapEntry<int, bool>>; |                 values['categories'] as Map<String, MapEntry<int, bool>>; | ||||||
|             settingsProvider.categories = |             settingsProvider.setCategories( | ||||||
|                 storedValues.map((key, value) => MapEntry(key, value.key)); |                 storedValues.map((key, value) => MapEntry(key, value.key)), | ||||||
|  |                 appsProvider: appsProvider); | ||||||
|             if (widget.onSelected != null) { |             if (widget.onSelected != null) { | ||||||
|               widget.onSelected!(storedValues.keys |               widget.onSelected!(storedValues.keys | ||||||
|                   .where((k) => storedValues[k]!.value) |                   .where((k) => storedValues[k]!.value) | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import 'dart:async'; | |||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
|  |  | ||||||
|  | import 'package:android_intent_plus/flag.dart'; | ||||||
| import 'package:device_info_plus/device_info_plus.dart'; | import 'package:device_info_plus/device_info_plus.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| @@ -12,16 +13,20 @@ import 'package:flutter/services.dart'; | |||||||
| import 'package:install_plugin_v2/install_plugin_v2.dart'; | import 'package:install_plugin_v2/install_plugin_v2.dart'; | ||||||
| import 'package:installed_apps/app_info.dart'; | import 'package:installed_apps/app_info.dart'; | ||||||
| import 'package:installed_apps/installed_apps.dart'; | import 'package:installed_apps/installed_apps.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/logs_provider.dart'; | import 'package:obtainium/providers/logs_provider.dart'; | ||||||
| import 'package:obtainium/providers/notifications_provider.dart'; | import 'package:obtainium/providers/notifications_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| import 'package:package_archive_info/package_archive_info.dart'; | import 'package:package_archive_info/package_archive_info.dart'; | ||||||
|  | import 'package:permission_handler/permission_handler.dart'; | ||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
| import 'package:path_provider/path_provider.dart'; | import 'package:path_provider/path_provider.dart'; | ||||||
| import 'package:flutter_fgbg/flutter_fgbg.dart'; | import 'package:flutter_fgbg/flutter_fgbg.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:android_intent_plus/android_intent.dart'; | ||||||
|  |  | ||||||
| class AppInMemory { | class AppInMemory { | ||||||
|   late App app; |   late App app; | ||||||
| @@ -68,6 +73,18 @@ List<String> generateStandardVersionRegExStrings() { | |||||||
| List<String> standardVersionRegExStrings = | List<String> standardVersionRegExStrings = | ||||||
|     generateStandardVersionRegExStrings(); |     generateStandardVersionRegExStrings(); | ||||||
|  |  | ||||||
|  | Set<String> findStandardFormatsForVersion(String version, bool strict) { | ||||||
|  |   // If !strict, even a substring match is valid | ||||||
|  |   Set<String> results = {}; | ||||||
|  |   for (var pattern in standardVersionRegExStrings) { | ||||||
|  |     if (RegExp('${strict ? '^' : ''}$pattern${strict ? '\$' : ''}') | ||||||
|  |         .hasMatch(version)) { | ||||||
|  |       results.add(pattern); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return results; | ||||||
|  | } | ||||||
|  |  | ||||||
| class AppsProvider with ChangeNotifier { | class AppsProvider with ChangeNotifier { | ||||||
|   // In memory App state (should always be kept in sync with local storage versions) |   // In memory App state (should always be kept in sync with local storage versions) | ||||||
|   Map<String, AppInMemory> apps = {}; |   Map<String, AppInMemory> apps = {}; | ||||||
| @@ -140,56 +157,68 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { |   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { | ||||||
|     var fileName = |  | ||||||
|         '${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'; |  | ||||||
|     String downloadUrl = await SourceProvider() |  | ||||||
|         .getSource(app.url) |  | ||||||
|         .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex]); |  | ||||||
|     NotificationsProvider? notificationsProvider = |     NotificationsProvider? notificationsProvider = | ||||||
|         context?.read<NotificationsProvider>(); |         context?.read<NotificationsProvider>(); | ||||||
|     var notif = DownloadNotification(app.name, 100); |     var notifId = DownloadNotification(app.name, 0).id; | ||||||
|     notificationsProvider?.cancel(notif.id); |     if (apps[app.id] != null) { | ||||||
|     int? prevProg; |       apps[app.id]!.downloadProgress = 0; | ||||||
|     File downloadedFile = |       notifyListeners(); | ||||||
|         await downloadFile(downloadUrl, fileName, (double? progress) { |     } | ||||||
|       int? prog = progress?.ceil(); |     try { | ||||||
|  |       var fileName = | ||||||
|  |           '${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'; | ||||||
|  |       String downloadUrl = await SourceProvider() | ||||||
|  |           .getSource(app.url) | ||||||
|  |           .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value); | ||||||
|  |       var notif = DownloadNotification(app.name, 100); | ||||||
|  |       notificationsProvider?.cancel(notif.id); | ||||||
|  |       int? prevProg; | ||||||
|  |       File downloadedFile = | ||||||
|  |           await downloadFile(downloadUrl, fileName, (double? progress) { | ||||||
|  |         int? prog = progress?.ceil(); | ||||||
|  |         if (apps[app.id] != null) { | ||||||
|  |           apps[app.id]!.downloadProgress = progress; | ||||||
|  |           notifyListeners(); | ||||||
|  |         } | ||||||
|  |         notif = DownloadNotification(app.name, prog ?? 100); | ||||||
|  |         if (prog != null && prevProg != prog) { | ||||||
|  |           notificationsProvider?.notify(notif); | ||||||
|  |         } | ||||||
|  |         prevProg = prog; | ||||||
|  |       }); | ||||||
|  |       // Delete older versions of the APK if any | ||||||
|  |       for (var file in downloadedFile.parent.listSync()) { | ||||||
|  |         var fn = file.path.split('/').last; | ||||||
|  |         if (fn.startsWith('${app.id}-') && | ||||||
|  |             fn.endsWith('.apk') && | ||||||
|  |             fn != fileName) { | ||||||
|  |           file.delete(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       // If the APK package ID is different from the App ID, it is either new (using a placeholder ID) or the ID has changed | ||||||
|  |       // The former case should be handled (give the App its real ID), the latter is a security issue | ||||||
|  |       var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path); | ||||||
|  |       if (app.id != newInfo.packageName) { | ||||||
|  |         if (apps[app.id] != null && !SourceProvider().isTempId(app)) { | ||||||
|  |           throw IDChangedError(); | ||||||
|  |         } | ||||||
|  |         var originalAppId = app.id; | ||||||
|  |         app.id = newInfo.packageName; | ||||||
|  |         downloadedFile = downloadedFile.renameSync( | ||||||
|  |             '${downloadedFile.parent.path}/${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'); | ||||||
|  |         if (apps[originalAppId] != null) { | ||||||
|  |           await removeApps([originalAppId]); | ||||||
|  |           await saveApps([app]); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       return DownloadedApk(app.id, downloadedFile); | ||||||
|  |     } finally { | ||||||
|  |       notificationsProvider?.cancel(notifId); | ||||||
|       if (apps[app.id] != null) { |       if (apps[app.id] != null) { | ||||||
|         apps[app.id]!.downloadProgress = progress; |         apps[app.id]!.downloadProgress = null; | ||||||
|         notifyListeners(); |         notifyListeners(); | ||||||
|       } |       } | ||||||
|       notif = DownloadNotification(app.name, prog ?? 100); |  | ||||||
|       if (prog != null && prevProg != prog) { |  | ||||||
|         notificationsProvider?.notify(notif); |  | ||||||
|       } |  | ||||||
|       prevProg = prog; |  | ||||||
|     }); |  | ||||||
|     notificationsProvider?.cancel(notif.id); |  | ||||||
|     // Delete older versions of the APK if any |  | ||||||
|     for (var file in downloadedFile.parent.listSync()) { |  | ||||||
|       var fn = file.path.split('/').last; |  | ||||||
|       if (fn.startsWith('${app.id}-') && |  | ||||||
|           fn.endsWith('.apk') && |  | ||||||
|           fn != fileName) { |  | ||||||
|         file.delete(); |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|     // If the APK package ID is different from the App ID, it is either new (using a placeholder ID) or the ID has changed |  | ||||||
|     // The former case should be handled (give the App its real ID), the latter is a security issue |  | ||||||
|     var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path); |  | ||||||
|     if (app.id != newInfo.packageName) { |  | ||||||
|       if (apps[app.id] != null && !SourceProvider().isTempId(app.id)) { |  | ||||||
|         throw IDChangedError(); |  | ||||||
|       } |  | ||||||
|       var originalAppId = app.id; |  | ||||||
|       app.id = newInfo.packageName; |  | ||||||
|       downloadedFile = downloadedFile.renameSync( |  | ||||||
|           '${downloadedFile.parent.path}/${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'); |  | ||||||
|       if (apps[originalAppId] != null) { |  | ||||||
|         await removeApps([originalAppId]); |  | ||||||
|         await saveApps([app]); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     return DownloadedApk(app.id, downloadedFile); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool areDownloadsRunning() => apps.values |   bool areDownloadsRunning() => apps.values | ||||||
| @@ -246,9 +275,10 @@ class AppsProvider with ChangeNotifier { | |||||||
|         !(await canDowngradeApps())) { |         !(await canDowngradeApps())) { | ||||||
|       throw DowngradeError(); |       throw DowngradeError(); | ||||||
|     } |     } | ||||||
|     if (appInfo == null || |     await InstallPlugin.installApk(file.file.path, obtainiumId); | ||||||
|         int.parse(newInfo.buildNumber) > appInfo.versionCode!) { |     if (file.appId == obtainiumId) { | ||||||
|       await InstallPlugin.installApk(file.file.path, 'dev.imranr.obtainium'); |       // Obtainium prompt should be lowest | ||||||
|  |       await Future.delayed(const Duration(milliseconds: 500)); | ||||||
|     } |     } | ||||||
|     apps[file.appId]!.app.installedVersion = |     apps[file.appId]!.app.installedVersion = | ||||||
|         apps[file.appId]!.app.latestVersion; |         apps[file.appId]!.app.latestVersion; | ||||||
| @@ -257,13 +287,24 @@ class AppsProvider with ChangeNotifier { | |||||||
|         attemptToCorrectInstallStatus: false); |         attemptToCorrectInstallStatus: false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<String?> confirmApkUrl(App app, BuildContext? context) async { |   void uninstallApp(String appId) async { | ||||||
|  |     var intent = AndroidIntent( | ||||||
|  |         action: 'android.intent.action.DELETE', | ||||||
|  |         data: 'package:$appId', | ||||||
|  |         flags: <int>[Flag.FLAG_ACTIVITY_NEW_TASK], | ||||||
|  |         package: 'vnd.android.package-archive'); | ||||||
|  |     await intent.launch(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<MapEntry<String, String>?> confirmApkUrl( | ||||||
|  |       App app, BuildContext? context) async { | ||||||
|     // If the App has more than one APK, the user should pick one (if context provided) |     // If the App has more than one APK, the user should pick one (if context provided) | ||||||
|     String? apkUrl = app.apkUrls[app.preferredApkIndex]; |     MapEntry<String, String>? apkUrl = app.apkUrls[app.preferredApkIndex]; | ||||||
|     // get device supported architecture |     // get device supported architecture | ||||||
|     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; |     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||||
|  |  | ||||||
|     if (app.apkUrls.length > 1 && context != null) { |     if (app.apkUrls.length > 1 && context != null) { | ||||||
|  |       // ignore: use_build_context_synchronously | ||||||
|       apkUrl = await showDialog( |       apkUrl = await showDialog( | ||||||
|           context: context, |           context: context, | ||||||
|           builder: (BuildContext ctx) { |           builder: (BuildContext ctx) { | ||||||
| @@ -281,13 +322,14 @@ class AppsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|     // If the picked APK comes from an origin different from the source, get user confirmation (if context provided) |     // If the picked APK comes from an origin different from the source, get user confirmation (if context provided) | ||||||
|     if (apkUrl != null && |     if (apkUrl != null && | ||||||
|         getHost(apkUrl) != getHost(app.url) && |         getHost(apkUrl.value) != getHost(app.url) && | ||||||
|         context != null) { |         context != null) { | ||||||
|  |       // ignore: use_build_context_synchronously | ||||||
|       if (await showDialog( |       if (await showDialog( | ||||||
|               context: context, |               context: context, | ||||||
|               builder: (BuildContext ctx) { |               builder: (BuildContext ctx) { | ||||||
|                 return APKOriginWarningDialog( |                 return APKOriginWarningDialog( | ||||||
|                     sourceUrl: app.url, apkUrl: apkUrl!); |                     sourceUrl: app.url, apkUrl: apkUrl!.value); | ||||||
|               }) != |               }) != | ||||||
|           true) { |           true) { | ||||||
|         apkUrl = null; |         apkUrl = null; | ||||||
| @@ -312,7 +354,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|       if (apps[id] == null) { |       if (apps[id] == null) { | ||||||
|         throw ObtainiumError(tr('appNotFound')); |         throw ObtainiumError(tr('appNotFound')); | ||||||
|       } |       } | ||||||
|       String? apkUrl; |       MapEntry<String, String>? apkUrl; | ||||||
|       var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true; |       var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true; | ||||||
|       if (!trackOnly) { |       if (!trackOnly) { | ||||||
|         apkUrl = await confirmApkUrl(apps[id]!.app, context); |         apkUrl = await confirmApkUrl(apps[id]!.app, context); | ||||||
| @@ -440,100 +482,120 @@ class AppsProvider with ChangeNotifier { | |||||||
|     } catch (e) { |     } catch (e) { | ||||||
|       // |       // | ||||||
|     } |     } | ||||||
|     if (!res) { |  | ||||||
|       logs.add(tr('versionCorrectionDisabled')); |  | ||||||
|     } |  | ||||||
|     return res; |     return res; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // If the App says it is installed but installedInfo is null, set it to not installed |   bool isVersionDetectionPossible(AppInMemory? app) { | ||||||
|   // If there is any other mismatch between installedInfo and installedVersion, try reconciling them intelligently |     return app?.app.additionalSettings['trackOnly'] != true && | ||||||
|   // If that fails, just set it to the actual version string (all we can do at that point) |         app?.app.additionalSettings['versionDetection'] != | ||||||
|   // Don't save changes, just return the object if changes were made (else null) |             'releaseDateAsVersion' && | ||||||
|  |         app?.installedInfo?.versionName != null && | ||||||
|  |         app?.app.installedVersion != null && | ||||||
|  |         reconcileVersionDifferences( | ||||||
|  |                 app!.installedInfo!.versionName!, app.app.installedVersion!) != | ||||||
|  |             null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Given an App and it's on-device info... | ||||||
|  |   // Reconcile unexpected differences between its reported installed version, real installed version, and reported latest version | ||||||
|   App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) { |   App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) { | ||||||
|     var modded = false; |     var modded = false; | ||||||
|     var trackOnly = app.additionalSettings['trackOnly'] == true; |     var trackOnly = app.additionalSettings['trackOnly'] == true; | ||||||
|     var noVersionDetection = |     var noVersionDetection = app.additionalSettings['versionDetection'] != | ||||||
|         app.additionalSettings['noVersionDetection'] == true; |         'standardVersionDetection'; | ||||||
|  |     // FIRST, COMPARE THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE ONE IS NULL | ||||||
|     if (installedInfo == null && app.installedVersion != null && !trackOnly) { |     if (installedInfo == null && app.installedVersion != null && !trackOnly) { | ||||||
|  |       // App says it's installed but isn't really (and isn't track only) - set to not installed | ||||||
|       app.installedVersion = null; |       app.installedVersion = null; | ||||||
|       modded = true; |       modded = true; | ||||||
|     } else if (installedInfo?.versionName != null && |     } else if (installedInfo?.versionName != null && | ||||||
|         app.installedVersion == null) { |         app.installedVersion == null) { | ||||||
|  |       // App says it's not installed but really is - set to installed and use real package versionName | ||||||
|       app.installedVersion = installedInfo!.versionName; |       app.installedVersion = installedInfo!.versionName; | ||||||
|       modded = true; |       modded = true; | ||||||
|     } else if (installedInfo?.versionName != null && |     } | ||||||
|  |     // SECOND, RECONCILE DIFFERENCES BETWEEN THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE NEITHER IS NULL | ||||||
|  |     if (installedInfo?.versionName != null && | ||||||
|         installedInfo!.versionName != app.installedVersion && |         installedInfo!.versionName != app.installedVersion && | ||||||
|         !noVersionDetection) { |         !noVersionDetection) { | ||||||
|       String? correctedInstalledVersion = reconcileRealAndInternalVersions( |       // App's reported version and real version don't match (and it uses standard version detection) | ||||||
|  |       // If they share a standard format (and are still different under it), update the reported version accordingly | ||||||
|  |       var correctedInstalledVersion = reconcileVersionDifferences( | ||||||
|           installedInfo.versionName!, app.installedVersion!); |           installedInfo.versionName!, app.installedVersion!); | ||||||
|       if (correctedInstalledVersion != null) { |       if (correctedInstalledVersion?.key == false) { | ||||||
|         app.installedVersion = correctedInstalledVersion; |         app.installedVersion = correctedInstalledVersion!.value; | ||||||
|         modded = true; |         modded = true; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |     // THIRD, RECONCILE THE APP'S REPORTED INSTALLED AND LATEST VERSIONS | ||||||
|     if (app.installedVersion != null && |     if (app.installedVersion != null && | ||||||
|         app.installedVersion != app.latestVersion && |         app.installedVersion != app.latestVersion && | ||||||
|         !noVersionDetection) { |         !noVersionDetection) { | ||||||
|       app.installedVersion = reconcileRealAndInternalVersions( |       // App's reported installed and latest versions don't match (and it uses standard version detection) | ||||||
|               app.installedVersion!, app.latestVersion, |       // If they share a standard format, make sure the App's reported installed version uses that format | ||||||
|               matchMode: true) ?? |       var correctedInstalledVersion = | ||||||
|           app.installedVersion; |           reconcileVersionDifferences(app.installedVersion!, app.latestVersion); | ||||||
|  |       if (correctedInstalledVersion?.key == true) { | ||||||
|  |         app.installedVersion = correctedInstalledVersion!.value; | ||||||
|  |         modded = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     // FOURTH, DISABLE VERSION DETECTION IF ENABLED AND THE REPORTED/REAL INSTALLED VERSIONS ARE NOT STANDARDIZED | ||||||
|  |     if (installedInfo != null && | ||||||
|  |         app.additionalSettings['versionDetection'] == | ||||||
|  |             'standardVersionDetection' && | ||||||
|  |         !isVersionDetectionPossible(AppInMemory(app, null, installedInfo))) { | ||||||
|  |       app.additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|  |       logs.add('Could not reconcile version formats for: ${app.id}'); | ||||||
|       modded = true; |       modded = true; | ||||||
|     } |     } | ||||||
|  |     // if (app.installedVersion != null && | ||||||
|  |     //     app.additionalSettings['versionDetection'] == | ||||||
|  |     //         'standardVersionDetection') { | ||||||
|  |     //   var correctedInstalledVersion = | ||||||
|  |     //       reconcileVersionDifferences(app.installedVersion!, app.latestVersion); | ||||||
|  |     //   if (correctedInstalledVersion == null) { | ||||||
|  |     //     app.additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|  |     //     logs.add('Could not reconcile version formats for: ${app.id}'); | ||||||
|  |     //     modded = true; | ||||||
|  |     //   } | ||||||
|  |     // } | ||||||
|  |  | ||||||
|     return modded ? app : null; |     return modded ? app : null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   String? reconcileRealAndInternalVersions( |   MapEntry<bool, String>? reconcileVersionDifferences( | ||||||
|       String realVersion, String internalVersion, |       String templateVersion, String comparisonVersion) { | ||||||
|       {bool matchMode = false}) { |     // Returns null if the versions don't share a common standard format | ||||||
|     // 1. If one or both of these can't be converted to a "standard" format, return null (leave as is) |     // Returns <true, comparisonVersion> if they share a common format and are equal | ||||||
|     // 2. If both have a "standard" format under which they are equal, return null (leave as is) |     // Returns <false, templateVersion> if they share a common format but are not equal | ||||||
|     // 3. If both have a "standard" format in common but are unequal, return realVersion (this means it was changed externally) |     // templateVersion must fully match a standard format, while comparisonVersion can have a substring match | ||||||
|     // If in matchMode, the outcomes of rules 2 and 3 are reversed, and the "real" version is not matched strictly |     var templateVersionFormats = | ||||||
|     // Matchmode to be used when comparing internal install version and internal latest version |         findStandardFormatsForVersion(templateVersion, true); | ||||||
|  |     var comparisonVersionFormats = | ||||||
|     bool doStringsMatchUnderRegEx( |         findStandardFormatsForVersion(comparisonVersion, false); | ||||||
|         String pattern, String value1, String value2) { |  | ||||||
|       var r = RegExp(pattern); |  | ||||||
|       var m1 = r.firstMatch(value1); |  | ||||||
|       var m2 = r.firstMatch(value2); |  | ||||||
|       return m1 != null && m2 != null |  | ||||||
|           ? value1.substring(m1.start, m1.end) == |  | ||||||
|               value2.substring(m2.start, m2.end) |  | ||||||
|           : false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     Set<String> findStandardFormatsForVersion(String version, bool strict) { |  | ||||||
|       Set<String> results = {}; |  | ||||||
|       for (var pattern in standardVersionRegExStrings) { |  | ||||||
|         if (RegExp('${strict ? '^' : ''}$pattern${strict ? '\$' : ''}') |  | ||||||
|             .hasMatch(version)) { |  | ||||||
|           results.add(pattern); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|       return results; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     var realStandardVersionFormats = |  | ||||||
|         findStandardFormatsForVersion(realVersion, true); |  | ||||||
|     var internalStandardVersionFormats = |  | ||||||
|         findStandardFormatsForVersion(internalVersion, false); |  | ||||||
|     var commonStandardFormats = |     var commonStandardFormats = | ||||||
|         realStandardVersionFormats.intersection(internalStandardVersionFormats); |         templateVersionFormats.intersection(comparisonVersionFormats); | ||||||
|     if (commonStandardFormats.isEmpty) { |     if (commonStandardFormats.isEmpty) { | ||||||
|       return null; // Incompatible; no "enhanced detection" |       return null; | ||||||
|     } |     } | ||||||
|     for (String pattern in commonStandardFormats) { |     for (String pattern in commonStandardFormats) { | ||||||
|       if (doStringsMatchUnderRegEx(pattern, internalVersion, realVersion)) { |       if (doStringsMatchUnderRegEx( | ||||||
|         return matchMode |           pattern, comparisonVersion, templateVersion)) { | ||||||
|             ? internalVersion |         return MapEntry(true, comparisonVersion); | ||||||
|             : null; // Enhanced detection says no change |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return matchMode |     return MapEntry(false, templateVersion); | ||||||
|         ? null |   } | ||||||
|         : realVersion; // Enhanced detection says something changed |  | ||||||
|  |   bool doStringsMatchUnderRegEx(String pattern, String value1, String value2) { | ||||||
|  |     var r = RegExp(pattern); | ||||||
|  |     var m1 = r.firstMatch(value1); | ||||||
|  |     var m2 = r.firstMatch(value2); | ||||||
|  |     return m1 != null && m2 != null | ||||||
|  |         ? value1.substring(m1.start, m1.end) == | ||||||
|  |             value2.substring(m2.start, m2.end) | ||||||
|  |         : false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> loadApps() async { |   Future<void> loadApps() async { | ||||||
| @@ -545,7 +607,21 @@ class AppsProvider with ChangeNotifier { | |||||||
|     List<App> newApps = (await getAppsDir()) |     List<App> newApps = (await getAppsDir()) | ||||||
|         .listSync() |         .listSync() | ||||||
|         .where((item) => item.path.toLowerCase().endsWith('.json')) |         .where((item) => item.path.toLowerCase().endsWith('.json')) | ||||||
|         .map((e) => App.fromJson(jsonDecode(File(e.path).readAsStringSync()))) |         .map((e) { | ||||||
|  |           try { | ||||||
|  |             return App.fromJson(jsonDecode(File(e.path).readAsStringSync())); | ||||||
|  |           } catch (err) { | ||||||
|  |             if (err is FormatException) { | ||||||
|  |               logs.add('Corrupt JSON when loading App (will be ignored): $e'); | ||||||
|  |               e.renameSync('${e.path}.corrupt'); | ||||||
|  |               return App( | ||||||
|  |                   '', '', '', '', '', '', [], 0, {}, DateTime.now(), false); | ||||||
|  |             } else { | ||||||
|  |               rethrow; | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .where((element) => element.id.isNotEmpty) | ||||||
|         .toList(); |         .toList(); | ||||||
|     var idsToDelete = apps.values |     var idsToDelete = apps.values | ||||||
|         .map((e) => e.app.id) |         .map((e) => e.app.id) | ||||||
| @@ -588,20 +664,30 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> saveApps(List<App> apps, |   Future<void> saveApps(List<App> apps, | ||||||
|       {bool attemptToCorrectInstallStatus = true}) async { |       {bool attemptToCorrectInstallStatus = true, | ||||||
|  |       bool onlyIfExists = true}) async { | ||||||
|     attemptToCorrectInstallStatus = |     attemptToCorrectInstallStatus = | ||||||
|         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); |         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); | ||||||
|     for (var app in apps) { |     for (var app in apps) { | ||||||
|       AppInfo? info = await getInstalledInfo(app.id); |       AppInfo? info = await getInstalledInfo(app.id); | ||||||
|       app.name = info?.name ?? app.name; |       app.name = info?.name ?? app.name; | ||||||
|  |       if (app.additionalSettings['appName']?.toString().isNotEmpty == true) { | ||||||
|  |         app.name = app.additionalSettings['appName'].toString().trim(); | ||||||
|  |       } | ||||||
|       if (attemptToCorrectInstallStatus) { |       if (attemptToCorrectInstallStatus) { | ||||||
|         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; |         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; | ||||||
|       } |       } | ||||||
|       File('${(await getAppsDir()).path}/${app.id}.json') |       File('${(await getAppsDir()).path}/${app.id}.json') | ||||||
|           .writeAsStringSync(jsonEncode(app.toJson())); |           .writeAsStringSync(jsonEncode(app.toJson())); | ||||||
|       this.apps.update( |       try { | ||||||
|           app.id, (value) => AppInMemory(app, value.downloadProgress, info), |         this.apps.update( | ||||||
|           ifAbsent: () => AppInMemory(app, null, info)); |             app.id, (value) => AppInMemory(app, value.downloadProgress, info), | ||||||
|  |             ifAbsent: onlyIfExists ? null : () => AppInMemory(app, null, info)); | ||||||
|  |       } catch (e) { | ||||||
|  |         if (e is! ArgumentError || e.name != 'key') { | ||||||
|  |           rethrow; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
| @@ -621,6 +707,72 @@ class AppsProvider with ChangeNotifier { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   Future<bool> removeAppsWithModal(BuildContext context, List<App> apps) async { | ||||||
|  |     var showUninstallOption = apps | ||||||
|  |         .where((a) => | ||||||
|  |             a.installedVersion != null && | ||||||
|  |             a.additionalSettings['trackOnly'] != true) | ||||||
|  |         .isNotEmpty; | ||||||
|  |     var values = await showDialog( | ||||||
|  |         context: context, | ||||||
|  |         builder: (BuildContext ctx) { | ||||||
|  |           return GeneratedFormModal( | ||||||
|  |             title: plural('removeAppQuestion', apps.length), | ||||||
|  |             items: !showUninstallOption | ||||||
|  |                 ? [] | ||||||
|  |                 : [ | ||||||
|  |                     [ | ||||||
|  |                       GeneratedFormSwitch('rmAppEntry', | ||||||
|  |                           label: tr('removeFromObtainium'), defaultValue: true) | ||||||
|  |                     ], | ||||||
|  |                     [ | ||||||
|  |                       GeneratedFormSwitch('uninstallApp', | ||||||
|  |                           label: tr('uninstallFromDevice')) | ||||||
|  |                     ] | ||||||
|  |                   ], | ||||||
|  |             initValid: true, | ||||||
|  |           ); | ||||||
|  |         }); | ||||||
|  |     if (values != null) { | ||||||
|  |       bool uninstall = values['uninstallApp'] == true && showUninstallOption; | ||||||
|  |       bool remove = values['rmAppEntry'] == true || !showUninstallOption; | ||||||
|  |       if (uninstall) { | ||||||
|  |         for (var i = 0; i < apps.length; i++) { | ||||||
|  |           if (apps[i].installedVersion != null) { | ||||||
|  |             uninstallApp(apps[i].id); | ||||||
|  |             apps[i].installedVersion = null; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         await saveApps(apps, attemptToCorrectInstallStatus: false); | ||||||
|  |       } | ||||||
|  |       if (remove) { | ||||||
|  |         await removeApps(apps.map((e) => e.id).toList()); | ||||||
|  |       } | ||||||
|  |       return uninstall || remove; | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<void> openAppSettings(String appId) async { | ||||||
|  |     final AndroidIntent intent = AndroidIntent( | ||||||
|  |       action: 'action_application_details_settings', | ||||||
|  |       data: 'package:$appId', | ||||||
|  |     ); | ||||||
|  |     await intent.launch(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   addMissingCategories(SettingsProvider settingsProvider) { | ||||||
|  |     var cats = settingsProvider.categories; | ||||||
|  |     apps.forEach((key, value) { | ||||||
|  |       for (var c in value.app.categories) { | ||||||
|  |         if (!cats.containsKey(c)) { | ||||||
|  |           cats[c] = generateRandomLightColor().value; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     settingsProvider.setCategories(cats, appsProvider: this); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   Future<App?> checkUpdate(String appId) async { |   Future<App?> checkUpdate(String appId) async { | ||||||
|     App? currentApp = apps[appId]!.app; |     App? currentApp = apps[appId]!.app; | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |     SourceProvider sourceProvider = SourceProvider(); | ||||||
| @@ -700,9 +852,23 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<String> exportApps() async { |   Future<String> exportApps() async { | ||||||
|  |     if ((await DeviceInfoPlugin().androidInfo).version.sdkInt <= 29) { | ||||||
|  |       if (await Permission.storage.isDenied) { | ||||||
|  |         await Permission.storage.request(); | ||||||
|  |       } | ||||||
|  |       if (await Permission.storage.isDenied) { | ||||||
|  |         throw ObtainiumError(tr('storagePermissionDenied')); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|     Directory? exportDir = Directory('/storage/emulated/0/Download'); |     Directory? exportDir = Directory('/storage/emulated/0/Download'); | ||||||
|     String path = 'Downloads'; // TODO: See if hardcoding this can be avoided |     String path = 'Downloads'; // TODO: See if hardcoding this can be avoided | ||||||
|     if (!exportDir.existsSync()) { |     var downloadsAccessible = false; | ||||||
|  |     try { | ||||||
|  |       downloadsAccessible = exportDir.existsSync(); | ||||||
|  |     } catch (e) { | ||||||
|  |       logs.add('Error accessing Downloads (will use fallback): $e'); | ||||||
|  |     } | ||||||
|  |     if (!downloadsAccessible) { | ||||||
|       exportDir = await getExternalStorageDirectory(); |       exportDir = await getExternalStorageDirectory(); | ||||||
|       path = exportDir!.path; |       path = exportDir!.path; | ||||||
|     } |     } | ||||||
| @@ -725,7 +891,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|         a.installedVersion = apps[a.id]?.app.installedVersion; |         a.installedVersion = apps[a.id]?.app.installedVersion; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     await saveApps(importedApps); |     await saveApps(importedApps, onlyIfExists: false); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|     return importedApps.length; |     return importedApps.length; | ||||||
|   } |   } | ||||||
| @@ -745,7 +911,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|       if (apps.containsKey(app.id)) { |       if (apps.containsKey(app.id)) { | ||||||
|         errorsMap.addAll({app.id: tr('appAlreadyAdded')}); |         errorsMap.addAll({app.id: tr('appAlreadyAdded')}); | ||||||
|       } else { |       } else { | ||||||
|         await saveApps([app]); |         await saveApps([app], onlyIfExists: false); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     List<List<String>> errors = |     List<List<String>> errors = | ||||||
| @@ -758,7 +924,7 @@ class APKPicker extends StatefulWidget { | |||||||
|   const APKPicker({super.key, required this.app, this.initVal, this.archs}); |   const APKPicker({super.key, required this.app, this.initVal, this.archs}); | ||||||
|  |  | ||||||
|   final App app; |   final App app; | ||||||
|   final String? initVal; |   final MapEntry<String, String>? initVal; | ||||||
|   final List<String>? archs; |   final List<String>? archs; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -766,7 +932,7 @@ class APKPicker extends StatefulWidget { | |||||||
| } | } | ||||||
|  |  | ||||||
| class _APKPickerState extends State<APKPicker> { | class _APKPickerState extends State<APKPicker> { | ||||||
|   String? apkUrl; |   MapEntry<String, String>? apkUrl; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
| @@ -779,15 +945,13 @@ class _APKPickerState extends State<APKPicker> { | |||||||
|         const SizedBox(height: 16), |         const SizedBox(height: 16), | ||||||
|         ...widget.app.apkUrls.map( |         ...widget.app.apkUrls.map( | ||||||
|           (u) => RadioListTile<String>( |           (u) => RadioListTile<String>( | ||||||
|               title: Text(Uri.parse(u) |               title: Text(u.key), | ||||||
|                   .pathSegments |               value: u.value, | ||||||
|                   .where((element) => element.isNotEmpty) |               groupValue: apkUrl!.value, | ||||||
|                   .last), |  | ||||||
|               value: u, |  | ||||||
|               groupValue: apkUrl, |  | ||||||
|               onChanged: (String? val) { |               onChanged: (String? val) { | ||||||
|                 setState(() { |                 setState(() { | ||||||
|                   apkUrl = val; |                   apkUrl = | ||||||
|  |                       widget.app.apkUrls.where((e) => e.value == val).first; | ||||||
|                 }); |                 }); | ||||||
|               }), |               }), | ||||||
|         ), |         ), | ||||||
|   | |||||||
| @@ -6,8 +6,9 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:fluttertoast/fluttertoast.dart'; | import 'package:fluttertoast/fluttertoast.dart'; | ||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; |  | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
|  | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:permission_handler/permission_handler.dart'; | import 'package:permission_handler/permission_handler.dart'; | ||||||
| import 'package:shared_preferences/shared_preferences.dart'; | import 'package:shared_preferences/shared_preferences.dart'; | ||||||
|  |  | ||||||
| @@ -18,7 +19,7 @@ enum ThemeSettings { system, light, dark } | |||||||
|  |  | ||||||
| enum ColourSettings { basic, materialYou } | enum ColourSettings { basic, materialYou } | ||||||
|  |  | ||||||
| enum SortColumnSettings { added, nameAuthor, authorName } | enum SortColumnSettings { added, nameAuthor, authorName, releaseDate } | ||||||
|  |  | ||||||
| enum SortOrderSettings { ascending, descending } | enum SortOrderSettings { ascending, descending } | ||||||
|  |  | ||||||
| @@ -140,6 +141,15 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool get groupByCategory { | ||||||
|  |     return prefs?.getBool('groupByCategory') ?? false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set groupByCategory(bool show) { | ||||||
|  |     prefs?.setBool('groupByCategory', show); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   String? getSettingString(String settingId) { |   String? getSettingString(String settingId) { | ||||||
|     return prefs?.getString(settingId); |     return prefs?.getString(settingId); | ||||||
|   } |   } | ||||||
| @@ -152,20 +162,26 @@ class SettingsProvider with ChangeNotifier { | |||||||
|   Map<String, int> get categories => |   Map<String, int> get categories => | ||||||
|       Map<String, int>.from(jsonDecode(prefs?.getString('categories') ?? '{}')); |       Map<String, int>.from(jsonDecode(prefs?.getString('categories') ?? '{}')); | ||||||
|  |  | ||||||
|   set categories(Map<String, int> cats) { |   void setCategories(Map<String, int> cats, {AppsProvider? appsProvider}) { | ||||||
|  |     if (appsProvider != null) { | ||||||
|  |       List<App> changedApps = appsProvider.apps.values | ||||||
|  |           .map((a) { | ||||||
|  |             var n1 = a.app.categories.length; | ||||||
|  |             a.app.categories.removeWhere((c) => !cats.keys.contains(c)); | ||||||
|  |             return n1 > a.app.categories.length ? a.app : null; | ||||||
|  |           }) | ||||||
|  |           .where((element) => element != null) | ||||||
|  |           .map((e) => e as App) | ||||||
|  |           .toList(); | ||||||
|  |       if (changedApps.isNotEmpty) { | ||||||
|  |         appsProvider.saveApps(changedApps, | ||||||
|  |             attemptToCorrectInstallStatus: false); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|     prefs?.setString('categories', jsonEncode(cats)); |     prefs?.setString('categories', jsonEncode(cats)); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getCategoryFormItem({String initCategory = ''}) => GeneratedFormDropdown( |  | ||||||
|       'category', |  | ||||||
|       label: tr('category'), |  | ||||||
|       [ |  | ||||||
|         MapEntry('', tr('noCategory')), |  | ||||||
|         ...categories.entries.map((e) => MapEntry(e.key, e.key)).toList() |  | ||||||
|       ], |  | ||||||
|       defaultValue: initCategory); |  | ||||||
|  |  | ||||||
|   String? get forcedLocale { |   String? get forcedLocale { | ||||||
|     var fl = prefs?.getString('forcedLocale'); |     var fl = prefs?.getString('forcedLocale'); | ||||||
|     return supportedLocales |     return supportedLocales | ||||||
| @@ -185,4 +201,18 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     } |     } | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool setEqual(Set<String> a, Set<String> b) => | ||||||
|  |       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)) { | ||||||
|  |       context.resetLocale(); | ||||||
|  |     } else { | ||||||
|  |       context.setLocale(context.fallbackLocale!); | ||||||
|  |       context.deleteSaveLocale(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,19 +7,23 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:html/dom.dart'; | import 'package:html/dom.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
| import 'package:obtainium/app_sources/apkmirror.dart'; | import 'package:obtainium/app_sources/apkmirror.dart'; | ||||||
|  | import 'package:obtainium/app_sources/codeberg.dart'; | ||||||
| import 'package:obtainium/app_sources/fdroid.dart'; | import 'package:obtainium/app_sources/fdroid.dart'; | ||||||
| import 'package:obtainium/app_sources/fdroidrepo.dart'; | import 'package:obtainium/app_sources/fdroidrepo.dart'; | ||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/app_sources/gitlab.dart'; | import 'package:obtainium/app_sources/gitlab.dart'; | ||||||
| import 'package:obtainium/app_sources/izzyondroid.dart'; | import 'package:obtainium/app_sources/izzyondroid.dart'; | ||||||
|  | import 'package:obtainium/app_sources/html.dart'; | ||||||
| import 'package:obtainium/app_sources/mullvad.dart'; | import 'package:obtainium/app_sources/mullvad.dart'; | ||||||
|  | import 'package:obtainium/app_sources/neutroncode.dart'; | ||||||
| import 'package:obtainium/app_sources/signal.dart'; | import 'package:obtainium/app_sources/signal.dart'; | ||||||
| import 'package:obtainium/app_sources/sourceforge.dart'; | import 'package:obtainium/app_sources/sourceforge.dart'; | ||||||
| import 'package:obtainium/app_sources/steammobile.dart'; | import 'package:obtainium/app_sources/steammobile.dart'; | ||||||
|  | import 'package:obtainium/app_sources/telegramapp.dart'; | ||||||
|  | import 'package:obtainium/app_sources/vlc.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/mass_app_sources/githubstars.dart'; | import 'package:obtainium/mass_app_sources/githubstars.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; |  | ||||||
|  |  | ||||||
| class AppNames { | class AppNames { | ||||||
|   late String author; |   late String author; | ||||||
| @@ -30,10 +34,13 @@ class AppNames { | |||||||
|  |  | ||||||
| class APKDetails { | class APKDetails { | ||||||
|   late String version; |   late String version; | ||||||
|   late List<String> apkUrls; |   late List<MapEntry<String, String>> apkUrls; | ||||||
|   late AppNames names; |   late AppNames names; | ||||||
|  |   late DateTime? releaseDate; | ||||||
|  |   late String? changeLog; | ||||||
|  |  | ||||||
|   APKDetails(this.version, this.apkUrls, this.names); |   APKDetails(this.version, this.apkUrls, this.names, | ||||||
|  |       {this.releaseDate, this.changeLog}); | ||||||
| } | } | ||||||
|  |  | ||||||
| class App { | class App { | ||||||
| @@ -43,12 +50,14 @@ class App { | |||||||
|   late String name; |   late String name; | ||||||
|   String? installedVersion; |   String? installedVersion; | ||||||
|   late String latestVersion; |   late String latestVersion; | ||||||
|   List<String> apkUrls = []; |   List<MapEntry<String, String>> apkUrls = []; | ||||||
|   late int preferredApkIndex; |   late int preferredApkIndex; | ||||||
|   late Map<String, dynamic> additionalSettings; |   late Map<String, dynamic> additionalSettings; | ||||||
|   late DateTime? lastUpdateCheck; |   late DateTime? lastUpdateCheck; | ||||||
|   bool pinned = false; |   bool pinned = false; | ||||||
|   String? category; |   List<String> categories; | ||||||
|  |   late DateTime? releaseDate; | ||||||
|  |   late String? changeLog; | ||||||
|   App( |   App( | ||||||
|       this.id, |       this.id, | ||||||
|       this.url, |       this.url, | ||||||
| @@ -61,7 +70,9 @@ class App { | |||||||
|       this.additionalSettings, |       this.additionalSettings, | ||||||
|       this.lastUpdateCheck, |       this.lastUpdateCheck, | ||||||
|       this.pinned, |       this.pinned, | ||||||
|       {this.category}); |       {this.categories = const [], | ||||||
|  |       this.releaseDate, | ||||||
|  |       this.changeLog}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
| @@ -96,6 +107,20 @@ class App { | |||||||
|       additionalSettings['noVersionDetection'] = |       additionalSettings['noVersionDetection'] = | ||||||
|           json['noVersionDetection'] == 'true' || json['trackOnly'] == true; |           json['noVersionDetection'] == 'true' || json['trackOnly'] == true; | ||||||
|     } |     } | ||||||
|  |     // Convert bool style version detection options to dropdown style | ||||||
|  |     if (additionalSettings['noVersionDetection'] == true) { | ||||||
|  |       additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|  |       if (additionalSettings['releaseDateAsVersion'] == true) { | ||||||
|  |         additionalSettings['versionDetection'] = 'releaseDateAsVersion'; | ||||||
|  |         additionalSettings.remove('releaseDateAsVersion'); | ||||||
|  |       } | ||||||
|  |       if (additionalSettings['noVersionDetection'] != null) { | ||||||
|  |         additionalSettings.remove('noVersionDetection'); | ||||||
|  |       } | ||||||
|  |       if (additionalSettings['releaseDateAsVersion'] != null) { | ||||||
|  |         additionalSettings.remove('releaseDateAsVersion'); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|     // Ensure additionalSettings are correctly typed |     // Ensure additionalSettings are correctly typed | ||||||
|     for (var item in formItems) { |     for (var item in formItems) { | ||||||
|       if (additionalSettings[item.key] != null) { |       if (additionalSettings[item.key] != null) { | ||||||
| @@ -103,6 +128,24 @@ class App { | |||||||
|             item.ensureType(additionalSettings[item.key]); |             item.ensureType(additionalSettings[item.key]); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |     int preferredApkIndex = json['preferredApkIndex'] == null | ||||||
|  |         ? 0 | ||||||
|  |         : json['preferredApkIndex'] as int; | ||||||
|  |     if (preferredApkIndex < 0) { | ||||||
|  |       preferredApkIndex = 0; | ||||||
|  |     } | ||||||
|  |     // apkUrls can either be old list or new named list apkUrls | ||||||
|  |     List<MapEntry<String, String>> apkUrls = []; | ||||||
|  |     if (json['apkUrls'] != null) { | ||||||
|  |       var apkUrlJson = jsonDecode(json['apkUrls']); | ||||||
|  |       try { | ||||||
|  |         apkUrls = getApkUrlsFromUrls(List<String>.from(apkUrlJson)); | ||||||
|  |       } catch (e) { | ||||||
|  |         apkUrls = List<dynamic>.from(apkUrlJson) | ||||||
|  |             .map((e) => MapEntry(e[0] as String, e[1] as String)) | ||||||
|  |             .toList(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|     return App( |     return App( | ||||||
|         json['id'] as String, |         json['id'] as String, | ||||||
|         json['url'] as String, |         json['url'] as String, | ||||||
| @@ -112,18 +155,25 @@ class App { | |||||||
|             ? null |             ? null | ||||||
|             : json['installedVersion'] as String, |             : json['installedVersion'] as String, | ||||||
|         json['latestVersion'] as String, |         json['latestVersion'] as String, | ||||||
|         json['apkUrls'] == null |         apkUrls, | ||||||
|             ? [] |         preferredApkIndex, | ||||||
|             : List<String>.from(jsonDecode(json['apkUrls'])), |  | ||||||
|         json['preferredApkIndex'] == null |  | ||||||
|             ? 0 |  | ||||||
|             : json['preferredApkIndex'] as int, |  | ||||||
|         additionalSettings, |         additionalSettings, | ||||||
|         json['lastUpdateCheck'] == null |         json['lastUpdateCheck'] == null | ||||||
|             ? null |             ? null | ||||||
|             : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), |             : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), | ||||||
|         json['pinned'] ?? false, |         json['pinned'] ?? false, | ||||||
|         category: json['category']); |         categories: json['categories'] != null | ||||||
|  |             ? (json['categories'] as List<dynamic>) | ||||||
|  |                 .map((e) => e.toString()) | ||||||
|  |                 .toList() | ||||||
|  |             : json['category'] != null | ||||||
|  |                 ? [json['category'] as String] | ||||||
|  |                 : [], | ||||||
|  |         releaseDate: json['releaseDate'] == null | ||||||
|  |             ? null | ||||||
|  |             : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']), | ||||||
|  |         changeLog: | ||||||
|  |             json['changeLog'] == null ? null : json['changeLog'] as String); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> toJson() => { |   Map<String, dynamic> toJson() => { | ||||||
| @@ -133,17 +183,23 @@ class App { | |||||||
|         'name': name, |         'name': name, | ||||||
|         'installedVersion': installedVersion, |         'installedVersion': installedVersion, | ||||||
|         'latestVersion': latestVersion, |         'latestVersion': latestVersion, | ||||||
|         'apkUrls': jsonEncode(apkUrls), |         'apkUrls': jsonEncode(apkUrls.map((e) => [e.key, e.value]).toList()), | ||||||
|         'preferredApkIndex': preferredApkIndex, |         'preferredApkIndex': preferredApkIndex, | ||||||
|         'additionalSettings': jsonEncode(additionalSettings), |         'additionalSettings': jsonEncode(additionalSettings), | ||||||
|         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, |         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, | ||||||
|         'pinned': pinned, |         'pinned': pinned, | ||||||
|         'category': category |         'categories': categories, | ||||||
|  |         'releaseDate': releaseDate?.microsecondsSinceEpoch, | ||||||
|  |         'changeLog': changeLog | ||||||
|       }; |       }; | ||||||
| } | } | ||||||
|  |  | ||||||
| // Ensure the input is starts with HTTPS and has no WWW | // Ensure the input is starts with HTTPS and has no WWW | ||||||
| preStandardizeUrl(String url) { | preStandardizeUrl(String url) { | ||||||
|  |   var firstDotIndex = url.indexOf('.'); | ||||||
|  |   if (!(firstDotIndex >= 0 && firstDotIndex != url.length - 1)) { | ||||||
|  |     throw UnsupportedURLError(); | ||||||
|  |   } | ||||||
|   if (url.toLowerCase().indexOf('http://') != 0 && |   if (url.toLowerCase().indexOf('http://') != 0 && | ||||||
|       url.toLowerCase().indexOf('https://') != 0) { |       url.toLowerCase().indexOf('https://') != 0) { | ||||||
|     url = 'https://$url'; |     url = 'https://$url'; | ||||||
| @@ -179,10 +235,16 @@ Map<String, dynamic> getDefaultValuesFromFormItems( | |||||||
|       .reduce((value, element) => [...value, ...element])); |       .reduce((value, element) => [...value, ...element])); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | getApkUrlsFromUrls(List<String> urls) => urls | ||||||
|  |     .map((e) => | ||||||
|  |         MapEntry(e.split('/').where((el) => el.trim().isNotEmpty).last, e)) | ||||||
|  |     .toList(); | ||||||
|  |  | ||||||
| class AppSource { | class AppSource { | ||||||
|   String? host; |   String? host; | ||||||
|   late String name; |   late String name; | ||||||
|   bool enforceTrackOnly = false; |   bool enforceTrackOnly = false; | ||||||
|  |   bool changeLogIfAnyIsMarkDown = true; | ||||||
|  |  | ||||||
|   AppSource() { |   AppSource() { | ||||||
|     name = runtimeType.toString(); |     name = runtimeType.toString(); | ||||||
| @@ -210,7 +272,29 @@ class AppSource { | |||||||
|         label: tr('trackOnly'), |         label: tr('trackOnly'), | ||||||
|       ) |       ) | ||||||
|     ], |     ], | ||||||
|     [GeneratedFormSwitch('noVersionDetection', label: tr('noVersionDetection'))] |     [ | ||||||
|  |       GeneratedFormDropdown( | ||||||
|  |           'versionDetection', | ||||||
|  |           [ | ||||||
|  |             MapEntry( | ||||||
|  |                 'standardVersionDetection', tr('standardVersionDetection')), | ||||||
|  |             MapEntry('releaseDateAsVersion', tr('releaseDateAsVersion')), | ||||||
|  |             MapEntry('noVersionDetection', tr('noVersionDetection')) | ||||||
|  |           ], | ||||||
|  |           label: tr('versionDetection'), | ||||||
|  |           defaultValue: 'standardVersionDetection') | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       GeneratedFormTextField('apkFilterRegEx', | ||||||
|  |           label: tr('filterAPKsByRegEx'), | ||||||
|  |           required: false, | ||||||
|  |           additionalValidators: [ | ||||||
|  |             (value) { | ||||||
|  |               return regExValidator(value); | ||||||
|  |             } | ||||||
|  |           ]) | ||||||
|  |     ], | ||||||
|  |     [GeneratedFormTextField('appName', label: tr('appName'), required: false)] | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   // Previous 2 variables combined into one at runtime for convenient usage |   // Previous 2 variables combined into one at runtime for convenient usage | ||||||
| @@ -254,19 +338,37 @@ abstract class MassAppUrlSource { | |||||||
|   Future<Map<String, String>> getUrlsWithDescriptions(List<String> args); |   Future<Map<String, String>> getUrlsWithDescriptions(List<String> args); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | regExValidator(String? value) { | ||||||
|  |   if (value == null || value.isEmpty) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |   try { | ||||||
|  |     RegExp(value); | ||||||
|  |   } catch (e) { | ||||||
|  |     return tr('invalidRegEx'); | ||||||
|  |   } | ||||||
|  |   return null; | ||||||
|  | } | ||||||
|  |  | ||||||
| class SourceProvider { | class SourceProvider { | ||||||
|   // Add more source classes here so they are available via the service |   // Add more source classes here so they are available via the service | ||||||
|   List<AppSource> sources = [ |   List<AppSource> sources = [ | ||||||
|     GitHub(), |     GitHub(), | ||||||
|     GitLab(), |     GitLab(), | ||||||
|  |     Codeberg(), | ||||||
|     FDroid(), |     FDroid(), | ||||||
|     IzzyOnDroid(), |     IzzyOnDroid(), | ||||||
|     Mullvad(), |     FDroidRepo(), | ||||||
|     Signal(), |  | ||||||
|     SourceForge(), |     SourceForge(), | ||||||
|     APKMirror(), |     APKMirror(), | ||||||
|     FDroidRepo(), |     Mullvad(), | ||||||
|     SteamMobile() |     Signal(), | ||||||
|  |     VLC(), | ||||||
|  |     // WhatsApp(), // As of 2023-03-20 this is unusable as the version on the webpage is months out of date | ||||||
|  |     TelegramApp(), | ||||||
|  |     SteamMobile(), | ||||||
|  |     NeutronCode(), | ||||||
|  |     HTML() // This should ALWAYS be the last option as they are tried in order | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   // Add more mass url source classes here so they are available via the service |   // Add more mass url source classes here so they are available via the service | ||||||
| @@ -276,7 +378,7 @@ class SourceProvider { | |||||||
|     url = preStandardizeUrl(url); |     url = preStandardizeUrl(url); | ||||||
|     AppSource? source; |     AppSource? source; | ||||||
|     for (var s in sources.where((element) => element.host != null)) { |     for (var s in sources.where((element) => element.host != null)) { | ||||||
|       if (url.contains('://${s.host}')) { |       if (RegExp('://(.+\\.)?${s.host}').hasMatch(url)) { | ||||||
|         source = s; |         source = s; | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
| @@ -309,62 +411,60 @@ class SourceProvider { | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   String generateTempID(AppNames names, AppSource source) => |   String generateTempID( | ||||||
|       '${names.author.toLowerCase()}_${names.name.toLowerCase()}_${source.host}'; |           String standardUrl, Map<String, dynamic> additionalSettings) => | ||||||
|  |       (standardUrl + additionalSettings.toString()).hashCode.toString(); | ||||||
|  |  | ||||||
|   bool isTempId(String id) { |   bool isTempId(App app) { | ||||||
|     List<String> parts = id.split('_'); |     // return app.id == generateTempID(app.url, app.additionalSettings); | ||||||
|     if (parts.length < 3) { |     return RegExp('^[0-9]+\$').hasMatch(app.id); | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|     for (int i = 0; i < parts.length - 1; i++) { |  | ||||||
|       if (RegExp('.*[A-Z].*').hasMatch(parts[i])) { |  | ||||||
|         // TODO: Look into RegEx for non-Latin characters |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     return true; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<App> getApp( |   Future<App> getApp( | ||||||
|       AppSource source, String url, Map<String, dynamic> additionalSettings, |       AppSource source, String url, Map<String, dynamic> additionalSettings, | ||||||
|       {App? currentApp, |       {App? currentApp, bool trackOnlyOverride = false}) async { | ||||||
|       bool trackOnlyOverride = false, |     if (trackOnlyOverride || source.enforceTrackOnly) { | ||||||
|       noVersionDetectionOverride = false}) async { |  | ||||||
|     if (trackOnlyOverride) { |  | ||||||
|       additionalSettings['trackOnly'] = true; |       additionalSettings['trackOnly'] = true; | ||||||
|     } |     } | ||||||
|     if (noVersionDetectionOverride) { |     var trackOnly = additionalSettings['trackOnly'] == true; | ||||||
|       additionalSettings['noVersionDetection'] = true; |  | ||||||
|     } |  | ||||||
|     var trackOnly = currentApp?.additionalSettings['trackOnly'] == true; |  | ||||||
|     String standardUrl = source.standardizeURL(preStandardizeUrl(url)); |     String standardUrl = source.standardizeURL(preStandardizeUrl(url)); | ||||||
|     APKDetails apk = |     APKDetails apk = | ||||||
|         await source.getLatestAPKDetails(standardUrl, additionalSettings); |         await source.getLatestAPKDetails(standardUrl, additionalSettings); | ||||||
|  |     if (additionalSettings['versionDetection'] == 'releaseDateAsVersion' && | ||||||
|  |         apk.releaseDate != null) { | ||||||
|  |       apk.version = apk.releaseDate!.microsecondsSinceEpoch.toString(); | ||||||
|  |     } | ||||||
|  |     if (additionalSettings['apkFilterRegEx'] != null) { | ||||||
|  |       var reg = RegExp(additionalSettings['apkFilterRegEx']); | ||||||
|  |       apk.apkUrls = | ||||||
|  |           apk.apkUrls.where((element) => reg.hasMatch(element.key)).toList(); | ||||||
|  |     } | ||||||
|     if (apk.apkUrls.isEmpty && !trackOnly) { |     if (apk.apkUrls.isEmpty && !trackOnly) { | ||||||
|       throw NoAPKError(); |       throw NoAPKError(); | ||||||
|     } |     } | ||||||
|     String apkVersion = apk.version.replaceAll('/', '-'); |     String apkVersion = apk.version.replaceAll('/', '-'); | ||||||
|     var name = currentApp?.name.trim() ?? |     var name = currentApp != null ? currentApp.name.trim() : ''; | ||||||
|         apk.names.name[0].toUpperCase() + apk.names.name.substring(1); |     name = name.isNotEmpty | ||||||
|  |         ? name | ||||||
|  |         : apk.names.name[0].toUpperCase() + apk.names.name.substring(1); | ||||||
|     return App( |     return App( | ||||||
|         currentApp?.id ?? |         currentApp?.id ?? | ||||||
|             source.tryInferringAppId(standardUrl, |             source.tryInferringAppId(standardUrl, | ||||||
|                 additionalSettings: additionalSettings) ?? |                 additionalSettings: additionalSettings) ?? | ||||||
|             generateTempID(apk.names, source), |             generateTempID(standardUrl, additionalSettings), | ||||||
|         standardUrl, |         standardUrl, | ||||||
|         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), |         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), | ||||||
|         name.trim().isNotEmpty |         name, | ||||||
|             ? name |  | ||||||
|             : apk.names.name[0].toUpperCase() + apk.names.name.substring(1), |  | ||||||
|         currentApp?.installedVersion, |         currentApp?.installedVersion, | ||||||
|         apkVersion, |         apkVersion, | ||||||
|         apk.apkUrls, |         apk.apkUrls, | ||||||
|         apk.apkUrls.length - 1, |         apk.apkUrls.length - 1 >= 0 ? apk.apkUrls.length - 1 : 0, | ||||||
|         additionalSettings, |         additionalSettings, | ||||||
|         DateTime.now(), |         DateTime.now(), | ||||||
|         currentApp?.pinned ?? false, |         currentApp?.pinned ?? false, | ||||||
|         category: currentApp?.category); |         categories: currentApp?.categories ?? const [], | ||||||
|  |         releaseDate: apk.releaseDate, | ||||||
|  |         changeLog: apk.changeLog); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Returns errors in [results, errors] instead of throwing them |   // Returns errors in [results, errors] instead of throwing them | ||||||
|   | |||||||
							
								
								
									
										521
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										13
									
								
								pubspec.yaml
									
									
									
									
									
								
							
							
						
						| @@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev | |||||||
| # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | ||||||
| # In Windows, build-name is used as the major, minor, and patch parts | # In Windows, build-name is used as the major, minor, and patch parts | ||||||
| # of the product and file versions while build-number is used as the build suffix. | # of the product and file versions while build-number is used as the build suffix. | ||||||
| version: 0.9.6+94 # When changing this, update the tag in main() accordingly | version: 0.11.25+147 # When changing this, update the tag in main() accordingly | ||||||
|  |  | ||||||
| environment: | environment: | ||||||
|   sdk: '>=2.18.2 <3.0.0' |   sdk: '>=2.18.2 <3.0.0' | ||||||
| @@ -58,12 +58,13 @@ dependencies: | |||||||
|   android_alarm_manager_plus: ^2.1.0 |   android_alarm_manager_plus: ^2.1.0 | ||||||
|   sqflite: ^2.2.0+3 |   sqflite: ^2.2.0+3 | ||||||
|   easy_localization: ^3.0.1 |   easy_localization: ^3.0.1 | ||||||
|  |   android_intent_plus: ^3.1.5 | ||||||
|  |   flutter_markdown: ^0.6.14 | ||||||
|  |  | ||||||
|  |  | ||||||
| dev_dependencies: | dev_dependencies: | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     sdk: flutter |     sdk: flutter | ||||||
|   flutter_launcher_icons: ^0.11.0 |  | ||||||
|  |  | ||||||
|   # The "flutter_lints" package below contains a set of recommended lints to |   # The "flutter_lints" package below contains a set of recommended lints to | ||||||
|   # encourage good coding practices. The lint set provided by the package is |   # encourage good coding practices. The lint set provided by the package is | ||||||
| @@ -72,12 +73,6 @@ dev_dependencies: | |||||||
|   # rules and activating additional ones. |   # rules and activating additional ones. | ||||||
|   flutter_lints: ^2.0.1 |   flutter_lints: ^2.0.1 | ||||||
|  |  | ||||||
| flutter_icons: |  | ||||||
|   android: true |  | ||||||
|   image_path: "assets/graphics/icon.png" |  | ||||||
|   adaptive_icon_background: "#FFFFFF" |  | ||||||
|   adaptive_icon_foreground: "assets/graphics/icon.png" |  | ||||||
|  |  | ||||||
| # For information on the generic Dart part of this file, see the | # For information on the generic Dart part of this file, see the | ||||||
| # following page: https://dart.dev/tools/pub/pubspec | # following page: https://dart.dev/tools/pub/pubspec | ||||||
|  |  | ||||||
| @@ -96,6 +91,8 @@ flutter: | |||||||
|    |    | ||||||
|   assets: |   assets: | ||||||
|     - assets/translations/ |     - assets/translations/ | ||||||
|  |     - assets/graphics/ | ||||||
|  |     - assets/ca/ | ||||||
|  |  | ||||||
|   # An image asset can refer to one or more resolution-specific "variants", see |   # An image asset can refer to one or more resolution-specific "variants", see | ||||||
|   # https://flutter.dev/assets-and-images/#resolution-aware |   # https://flutter.dev/assets-and-images/#resolution-aware | ||||||
|   | |||||||