Compare commits
	
		
			135 Commits
		
	
	
		
			v0.11.0-be
			...
			v0.11.30-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					537628f378 | ||
| 
						 | 
					c92d76df98 | ||
| 
						 | 
					b6959e1a8b | ||
| 
						 | 
					1bf648da60 | ||
| 
						 | 
					6a1275e9e4 | ||
| 
						 | 
					df242b91ad | ||
| 
						 | 
					7ea75325bb | ||
| 
						 | 
					0704dfe2ee | ||
| 
						 | 
					6275cbf114 | ||
| 
						 | 
					36b8ef6782 | ||
| 
						 | 
					d274b9a428 | ||
| 
						 | 
					1c2980d1ac | ||
| 
						 | 
					8f0aac057e | ||
| 
						 | 
					e929920a48 | ||
| 
						 | 
					8ed254c7dd | ||
| 
						 | 
					46a00836df | ||
| 
						 | 
					f144ffdded | ||
| 
						 | 
					d597d569e2 | ||
| 
						 | 
					b62475de87 | ||
| 
						 | 
					334ac8d3d6 | ||
| 
						 | 
					9193788356 | ||
| 
						 | 
					8f75ddd43f | ||
| 
						 | 
					a2edc86bfa | ||
| 
						 | 
					0804e680b2 | ||
| 
						 | 
					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 | 
@@ -19,6 +19,9 @@ Currently supported App sources:
 | 
			
		||||
- Third Party F-Droid Repos
 | 
			
		||||
  - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo`
 | 
			
		||||
- [Steam](https://store.steampowered.com/mobile)
 | 
			
		||||
- [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)
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
| 
		 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>
 | 
			
		||||
@@ -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  | 
@@ -207,16 +207,22 @@
 | 
			
		||||
    "addCategory": "Kategorie hinzufügen",
 | 
			
		||||
    "label": "Bezeichnung",
 | 
			
		||||
    "language": "Sprache",
 | 
			
		||||
    "storagePermissionDenied": "Storage permission denied",
 | 
			
		||||
    "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.",
 | 
			
		||||
    "filterAPKsByRegEx": "Filter APKs by Regular Expression",
 | 
			
		||||
    "removeFromObtainium": "Remove from Obtainium",
 | 
			
		||||
    "uninstallFromDevice": "Uninstall from Device",
 | 
			
		||||
    "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.",
 | 
			
		||||
    "useReleaseDateAsVersion": "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",
 | 
			
		||||
    "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": "Nach Kategorie gruppieren",
 | 
			
		||||
    "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "App entfernen?",
 | 
			
		||||
        "other": "App entfernen?"
 | 
			
		||||
@@ -265,4 +271,4 @@
 | 
			
		||||
        "one": "{} und 1 weitere Anwendung wurden aktualisiert.",
 | 
			
		||||
        "other": "{} und {} weitere Anwendungen wurden aktualisiert."
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -207,16 +207,22 @@
 | 
			
		||||
    "addCategory": "Add Category",
 | 
			
		||||
    "label": "Label",
 | 
			
		||||
    "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.",
 | 
			
		||||
    "useReleaseDateAsVersion": "Use Release Date as Version",
 | 
			
		||||
    "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",
 | 
			
		||||
    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "Remove App?",
 | 
			
		||||
        "other": "Remove Apps?"
 | 
			
		||||
@@ -265,4 +271,4 @@
 | 
			
		||||
        "one": "{} and 1 more app were updated.",
 | 
			
		||||
        "other": "{} and {} more apps were updated."
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -207,16 +207,22 @@
 | 
			
		||||
    "addCategory": "اضافه کردن دسته",
 | 
			
		||||
    "label": "برچسب",
 | 
			
		||||
    "language": "زبان",
 | 
			
		||||
    "copiedToClipboard": "در کلیپ بورد کپی شد",
 | 
			
		||||
    "storagePermissionDenied": "مجوز ذخیره سازی رد شد",
 | 
			
		||||
    "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.",
 | 
			
		||||
    "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید",
 | 
			
		||||
    "removeFromObtainium": "از Obtainium حذف کنید",
 | 
			
		||||
    "uninstallFromDevice": "حذف نصب از دستگاه",
 | 
			
		||||
    "onlyWorksWithNonVersionDetectApps": "فقط برای برنامههایی کار میکند که تشخیص نسخه غیرفعال است.",
 | 
			
		||||
    "useReleaseDateAsVersion": "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",
 | 
			
		||||
    "releaseDateAsVersion": "از تاریخ انتشار به عنوان نسخه استفاده کنید",
 | 
			
		||||
    "releaseDateAsVersionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند، اما تاریخ انتشار در دسترس است.",
 | 
			
		||||
    "changes": "تغییرات",
 | 
			
		||||
    "releaseDate": "تاریخ انتشار",
 | 
			
		||||
    "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)",
 | 
			
		||||
    "versionDetection": "تشخیص نسخه",
 | 
			
		||||
    "standardVersionDetection": "تشخیص نسخه استاندارد",
 | 
			
		||||
    "groupByCategory": "گروه بر اساس دسته",
 | 
			
		||||
    "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "برنامه حذف شود؟",
 | 
			
		||||
        "other": "برنامه ها حذف شوند؟"
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										274
									
								
								assets/translations/fr.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,274 @@
 | 
			
		||||
{
 | 
			
		||||
    "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",
 | 
			
		||||
    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
			
		||||
    "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",
 | 
			
		||||
    "uname": "Felh.név",
 | 
			
		||||
    "wrongArgNum": "Rossz számú argumentumot adott meg",
 | 
			
		||||
    "xIsTrackOnly": "A(z) {} csak nyomkövethető",
 | 
			
		||||
    "xIsTrackOnly": "A(z) {} csak nyomonkövethető",
 | 
			
		||||
    "source": "Forrás",
 | 
			
		||||
    "app": "App",
 | 
			
		||||
    "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.",
 | 
			
		||||
@@ -56,7 +56,7 @@
 | 
			
		||||
    "appsString": "Appok",
 | 
			
		||||
    "noApps": "Nincs App",
 | 
			
		||||
    "noAppsForFilter": "Nincsenek appok a szűrőhöz",
 | 
			
		||||
    "byX": "{} által",
 | 
			
		||||
    "byX": "Fejlesztő: {}",
 | 
			
		||||
    "percentProgress": "Folyamat: {}%",
 | 
			
		||||
    "pleaseWait": "Kis türelmet",
 | 
			
		||||
    "updateAvailable": "Frissítés érhető el",
 | 
			
		||||
@@ -78,7 +78,7 @@
 | 
			
		||||
    "no": "Nem",
 | 
			
		||||
    "yes": "Igen",
 | 
			
		||||
    "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",
 | 
			
		||||
    "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.",
 | 
			
		||||
@@ -191,7 +191,7 @@
 | 
			
		||||
    "update": "Frissít",
 | 
			
		||||
    "markUpdated": "Frissítettnek jelöl",
 | 
			
		||||
    "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.",
 | 
			
		||||
    "downloadingX": "{} letöltés",
 | 
			
		||||
    "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról",
 | 
			
		||||
@@ -206,16 +206,22 @@
 | 
			
		||||
    "addCategory": "Új kategória",
 | 
			
		||||
    "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.",
 | 
			
		||||
    "useReleaseDateAsVersion": "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",
 | 
			
		||||
    "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",
 | 
			
		||||
    "autoApkFilterByArch": "Ha lehetséges, próbálja CPU architektúra szerint szűrni az APK-kat",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "Eltávolítja az alkalmazást?",
 | 
			
		||||
        "other": "Eltávolítja az alkalmazást?"
 | 
			
		||||
 
 | 
			
		||||
@@ -56,9 +56,9 @@
 | 
			
		||||
    "appsString": "App",
 | 
			
		||||
    "noApps": "Nessuna App",
 | 
			
		||||
    "noAppsForFilter": "Nessuna App per i filtri selezionati",
 | 
			
		||||
    "byX": "Da {}",
 | 
			
		||||
    "byX": "Di {}",
 | 
			
		||||
    "percentProgress": "Progresso: {}%",
 | 
			
		||||
    "pleaseWait": "Attendere prego",
 | 
			
		||||
    "pleaseWait": "In attesa",
 | 
			
		||||
    "updateAvailable": "Aggiornamento disponibile",
 | 
			
		||||
    "estimateInBracketsShort": "(prev.)",
 | 
			
		||||
    "notInstalled": "Non installato",
 | 
			
		||||
@@ -94,7 +94,7 @@
 | 
			
		||||
    "author": "Autore",
 | 
			
		||||
    "upToDateApps": "App aggiornate",
 | 
			
		||||
    "nonInstalledApps": "App non installate",
 | 
			
		||||
    "importExport": "Importa - Esporta",
 | 
			
		||||
    "importExport": "Importa/Esporta",
 | 
			
		||||
    "settings": "Impostazioni",
 | 
			
		||||
    "exportedTo": "Esportato in {}",
 | 
			
		||||
    "obtainiumExport": "Esporta da Obtainium",
 | 
			
		||||
@@ -207,16 +207,22 @@
 | 
			
		||||
    "addCategory": "Aggiungi categoria",
 | 
			
		||||
    "label": "Etichetta",
 | 
			
		||||
    "language": "Lingua",
 | 
			
		||||
    "copiedToClipboard": "Copiato negli appunti",
 | 
			
		||||
    "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.",
 | 
			
		||||
    "useReleaseDateAsVersion": "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",
 | 
			
		||||
    "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": "Raggruppa per categoria",
 | 
			
		||||
    "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "Rimuovere l'App?",
 | 
			
		||||
        "other": "Rimuovere le App?"
 | 
			
		||||
 
 | 
			
		||||
@@ -107,7 +107,7 @@
 | 
			
		||||
    "line": "行",
 | 
			
		||||
    "searchX": "{}で検索",
 | 
			
		||||
    "noResults": "結果は見つかりませんでした",
 | 
			
		||||
    "importX": "{}をインポートする",
 | 
			
		||||
    "importX": "{}をインポート",
 | 
			
		||||
    "importedAppsIdDisclaimer": "インポートしたアプリが「未インストール」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響しません。\n\nURLとサードパーティのインポートメソッドにのみ影響します。",
 | 
			
		||||
    "importErrors": "インポートエラー",
 | 
			
		||||
    "importedXOfYApps": "{} / {} アプリをインポートしました",
 | 
			
		||||
@@ -207,16 +207,22 @@
 | 
			
		||||
    "addCategory": "カテゴリを追加",
 | 
			
		||||
    "label": "ラベル",
 | 
			
		||||
    "language": "言語",
 | 
			
		||||
    "copiedToClipboard": "クリップボードにコピーしました",
 | 
			
		||||
    "storagePermissionDenied": "ストレージ権限が拒否されました",
 | 
			
		||||
    "selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。",
 | 
			
		||||
    "filterAPKsByRegEx": "正規表現でAPKを絞り込む",
 | 
			
		||||
    "removeFromObtainium": "Obtainiumから削除する",
 | 
			
		||||
    "uninstallFromDevice": "デバイスからアンインストールする",
 | 
			
		||||
    "onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。",
 | 
			
		||||
    "useReleaseDateAsVersion": "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",
 | 
			
		||||
    "releaseDateAsVersion": "リリース日をバージョンとして使用する",
 | 
			
		||||
    "releaseDateAsVersionExplanation": "このオプションは、バージョン検出が正しく機能しないアプリで、リリース日が利用可能な場合にのみ使用する必要があります。",
 | 
			
		||||
    "changes": "変更点",
 | 
			
		||||
    "releaseDate": "リリース日",
 | 
			
		||||
    "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート",
 | 
			
		||||
    "versionDetection": "バージョン検出",
 | 
			
		||||
    "standardVersionDetection": "標準のバージョン検出",
 | 
			
		||||
    "groupByCategory": "カテゴリ別にグループ化する",
 | 
			
		||||
    "autoApkFilterByArch": "可能であれば,CPUアーキテクチャによるAPKのフィルタリングを試みる",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "アプリを削除しますか?",
 | 
			
		||||
        "other": "アプリを削除しますか?"
 | 
			
		||||
@@ -265,4 +271,4 @@
 | 
			
		||||
        "one": "{} とさらに {} 個のアプリがアップデートされました",
 | 
			
		||||
        "other": "{} とさらに {} 個のアプリがアップデートされました"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -208,15 +208,21 @@
 | 
			
		||||
    "addCategory": "添加类别",
 | 
			
		||||
    "label": "标签",
 | 
			
		||||
    "language": "语言",
 | 
			
		||||
    "copiedToClipboard": "Copied to Clipboard",
 | 
			
		||||
    "storagePermissionDenied": "存储权限已被拒绝",
 | 
			
		||||
    "selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别",
 | 
			
		||||
    "filterAPKsByRegEx": "Filter APKs by Regular Expression",
 | 
			
		||||
    "removeFromObtainium": "Remove from Obtainium",
 | 
			
		||||
    "uninstallFromDevice": "Uninstall from Device",
 | 
			
		||||
    "useReleaseDateAsVersion": "Use Release Date as Version",
 | 
			
		||||
    "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",
 | 
			
		||||
    "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "删除应用?",
 | 
			
		||||
        "other": "删除应用?"
 | 
			
		||||
@@ -265,4 +271,4 @@
 | 
			
		||||
        "one": "{} 和 {} 更多应用已被安装",
 | 
			
		||||
        "other": "{} 和 {} 更多应用已被安装"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,9 @@
 | 
			
		||||
import 'dart:io';
 | 
			
		||||
 | 
			
		||||
import 'package:easy_localization/easy_localization.dart';
 | 
			
		||||
import 'package:html/parser.dart';
 | 
			
		||||
import 'package:http/http.dart';
 | 
			
		||||
import 'package:obtainium/components/generated_form.dart';
 | 
			
		||||
import 'package:obtainium/custom_errors.dart';
 | 
			
		||||
import 'package:obtainium/providers/source_provider.dart';
 | 
			
		||||
 | 
			
		||||
@@ -9,6 +11,23 @@ class APKMirror extends AppSource {
 | 
			
		||||
  APKMirror() {
 | 
			
		||||
    host = 'apkmirror.com';
 | 
			
		||||
    enforceTrackOnly = true;
 | 
			
		||||
 | 
			
		||||
    additionalSourceAppSpecificSettingFormItems = [
 | 
			
		||||
      [
 | 
			
		||||
        GeneratedFormSwitch('fallbackToOlderReleases',
 | 
			
		||||
            label: tr('fallbackToOlderReleases'), defaultValue: true)
 | 
			
		||||
      ],
 | 
			
		||||
      [
 | 
			
		||||
        GeneratedFormTextField('filterReleaseTitlesByRegEx',
 | 
			
		||||
            label: tr('filterReleaseTitlesByRegEx'),
 | 
			
		||||
            required: false,
 | 
			
		||||
            additionalValidators: [
 | 
			
		||||
              (value) {
 | 
			
		||||
                return regExValidator(value);
 | 
			
		||||
              }
 | 
			
		||||
            ])
 | 
			
		||||
      ]
 | 
			
		||||
    ];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
@@ -30,11 +49,31 @@ class APKMirror extends AppSource {
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
    Map<String, dynamic> additionalSettings,
 | 
			
		||||
  ) 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'));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      var item = parse(res.body).querySelector('item');
 | 
			
		||||
      String? titleString = item?.querySelector('title')?.innerHtml;
 | 
			
		||||
      String? dateString = item
 | 
			
		||||
      var items = parse(res.body).querySelectorAll('item');
 | 
			
		||||
      dynamic targetRelease;
 | 
			
		||||
      for (int i = 0; i < items.length; i++) {
 | 
			
		||||
        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(' ')
 | 
			
		||||
 
 | 
			
		||||
@@ -68,7 +68,7 @@ class Codeberg extends AppSource {
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      var releases = jsonDecode(res.body) as List<dynamic>;
 | 
			
		||||
 | 
			
		||||
      List<String> getReleaseAPKUrls(dynamic release) =>
 | 
			
		||||
      List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) =>
 | 
			
		||||
          (release['assets'] as List<dynamic>?)
 | 
			
		||||
              ?.map((e) {
 | 
			
		||||
                return e['name'] != null && e['browser_download_url'] != null
 | 
			
		||||
@@ -77,15 +77,15 @@ class Codeberg extends AppSource {
 | 
			
		||||
                    : const MapEntry('', '');
 | 
			
		||||
              })
 | 
			
		||||
              .where((element) => element.key.toLowerCase().endsWith('.apk'))
 | 
			
		||||
              .map((e) => e.value)
 | 
			
		||||
              .toList() ??
 | 
			
		||||
          [];
 | 
			
		||||
 | 
			
		||||
      dynamic targetRelease;
 | 
			
		||||
 | 
			
		||||
      var prerrelsSkipped = 0;
 | 
			
		||||
      for (int i = 0; i < releases.length; i++) {
 | 
			
		||||
        if (!fallbackToOlderReleases && i > 0) break;
 | 
			
		||||
        if (!fallbackToOlderReleases && i > prerrelsSkipped) break;
 | 
			
		||||
        if (!includePrereleases && releases[i]['prerelease'] == true) {
 | 
			
		||||
          prerrelsSkipped++;
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        if (releases[i]['draft'] == true) {
 | 
			
		||||
@@ -118,9 +118,13 @@ class Codeberg extends AppSource {
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      return APKDetails(version, targetRelease['apkUrls'] as List<String>,
 | 
			
		||||
      var changeLog = targetRelease['body'].toString();
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version,
 | 
			
		||||
          targetRelease['apkUrls'] as List<MapEntry<String, String>>,
 | 
			
		||||
          getAppNames(standardUrl),
 | 
			
		||||
          releaseDate: releaseDate);
 | 
			
		||||
          releaseDate: releaseDate,
 | 
			
		||||
          changeLog: changeLog.isEmpty ? null : changeLog);
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -14,12 +14,14 @@ class FDroid extends AppSource {
 | 
			
		||||
  @override
 | 
			
		||||
  String standardizeURL(String url) {
 | 
			
		||||
    RegExp standardUrlRegExB =
 | 
			
		||||
        RegExp('^https?://$host/+[^/]+/+packages/+[^/]+');
 | 
			
		||||
        RegExp('^https?://(cloudflare\\.)?$host/+[^/]+/+packages/+[^/]+');
 | 
			
		||||
    RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase());
 | 
			
		||||
    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());
 | 
			
		||||
    if (match == null) {
 | 
			
		||||
      throw InvalidURLError(name);
 | 
			
		||||
@@ -27,9 +29,6 @@ class FDroid extends AppSource {
 | 
			
		||||
    return url.substring(0, match.end);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) => null;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? tryInferringAppId(String standardUrl,
 | 
			
		||||
      {Map<String, dynamic> additionalSettings = const {}}) {
 | 
			
		||||
@@ -51,7 +50,7 @@ class FDroid extends AppSource {
 | 
			
		||||
          .where((element) => element['versionName'] == latestVersion)
 | 
			
		||||
          .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk')
 | 
			
		||||
          .toList();
 | 
			
		||||
      return APKDetails(latestVersion, apkUrls,
 | 
			
		||||
      return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls),
 | 
			
		||||
          AppNames(name, Uri.parse(standardUrl).pathSegments.last));
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
@@ -64,9 +63,10 @@ class FDroid extends AppSource {
 | 
			
		||||
    Map<String, dynamic> additionalSettings,
 | 
			
		||||
  ) async {
 | 
			
		||||
    String? appId = tryInferringAppId(standardUrl);
 | 
			
		||||
    String host = Uri.parse(standardUrl).host;
 | 
			
		||||
    return getAPKUrlsFromFDroidPackagesAPIResponse(
 | 
			
		||||
        await get(Uri.parse('https://f-droid.org/api/v1/packages/$appId')),
 | 
			
		||||
        'https://f-droid.org/repo/$appId',
 | 
			
		||||
        await get(Uri.parse('https://$host/api/v1/packages/$appId')),
 | 
			
		||||
        'https://$host/repo/$appId',
 | 
			
		||||
        standardUrl);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -80,7 +80,8 @@ class FDroidRepo extends AppSource {
 | 
			
		||||
              element.querySelector('apkname') != null)
 | 
			
		||||
          .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}')
 | 
			
		||||
          .toList();
 | 
			
		||||
      return APKDetails(latestVersion, apkUrls, AppNames(authorName, appName),
 | 
			
		||||
      return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls),
 | 
			
		||||
          AppNames(authorName, appName),
 | 
			
		||||
          releaseDate: releaseDate);
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
 
 | 
			
		||||
@@ -127,10 +127,11 @@ class GitHub extends AppSource {
 | 
			
		||||
          [];
 | 
			
		||||
 | 
			
		||||
      dynamic targetRelease;
 | 
			
		||||
 | 
			
		||||
      var prerrelsSkipped = 0;
 | 
			
		||||
      for (int i = 0; i < releases.length; i++) {
 | 
			
		||||
        if (!fallbackToOlderReleases && i > 0) break;
 | 
			
		||||
        if (!fallbackToOlderReleases && i > prerrelsSkipped) break;
 | 
			
		||||
        if (!includePrereleases && releases[i]['prerelease'] == true) {
 | 
			
		||||
          prerrelsSkipped++;
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        var nameToFilter = releases[i]['name'] as String?;
 | 
			
		||||
@@ -160,9 +161,13 @@ class GitHub extends AppSource {
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      return APKDetails(version, targetRelease['apkUrls'] as List<String>,
 | 
			
		||||
      var changeLog = targetRelease['body'].toString();
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version,
 | 
			
		||||
          getApkUrlsFromUrls(targetRelease['apkUrls'] as List<String>),
 | 
			
		||||
          getAppNames(standardUrl),
 | 
			
		||||
          releaseDate: releaseDate);
 | 
			
		||||
          releaseDate: releaseDate,
 | 
			
		||||
          changeLog: changeLog.isEmpty ? null : changeLog);
 | 
			
		||||
    } else {
 | 
			
		||||
      rateLimitErrorCheck(res);
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
@@ -183,9 +188,11 @@ class GitHub extends AppSource {
 | 
			
		||||
      Map<String, String> urlsWithDescriptions = {};
 | 
			
		||||
      for (var e in (jsonDecode(res.body)['items'] as List<dynamic>)) {
 | 
			
		||||
        urlsWithDescriptions.addAll({
 | 
			
		||||
          e['html_url'] as String: e['description'] != null
 | 
			
		||||
              ? e['description'] as String
 | 
			
		||||
              : tr('noDescription')
 | 
			
		||||
          e['html_url'] as String:
 | 
			
		||||
              ((e['archived'] == true ? '[ARCHIVED] ' : '') +
 | 
			
		||||
                  (e['description'] != null
 | 
			
		||||
                      ? e['description'] as String
 | 
			
		||||
                      : tr('noDescription')))
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
      return urlsWithDescriptions;
 | 
			
		||||
 
 | 
			
		||||
@@ -60,7 +60,8 @@ class GitLab extends AppSource {
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl),
 | 
			
		||||
      return APKDetails(version, getApkUrlsFromUrls(apkUrls),
 | 
			
		||||
          GitHub().getAppNames(standardUrl),
 | 
			
		||||
          releaseDate: releaseDate);
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
 
 | 
			
		||||
@@ -10,9 +10,6 @@ class HTML extends AppSource {
 | 
			
		||||
    return url;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) => null;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetails(
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
@@ -37,15 +34,22 @@ class HTML extends AppSource {
 | 
			
		||||
      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, apkUrls, AppNames(uri.host, tr('app')));
 | 
			
		||||
      List<String> apkUrls = [rel].map((e) {
 | 
			
		||||
        try {
 | 
			
		||||
          Uri.parse(e).origin;
 | 
			
		||||
          return e;
 | 
			
		||||
        } catch (err) {
 | 
			
		||||
          // is relative
 | 
			
		||||
        }
 | 
			
		||||
        var currPathSegments = uri.path.split('/');
 | 
			
		||||
        if (e.startsWith('/') || currPathSegments.isEmpty) {
 | 
			
		||||
          return '${uri.origin}/$e';
 | 
			
		||||
        } else {
 | 
			
		||||
          return '${uri.origin}/${currPathSegments.sublist(0, currPathSegments.length - 1).join('/')}/$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);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) => null;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? tryInferringAppId(String standardUrl,
 | 
			
		||||
      {Map<String, dynamic> additionalSettings = const {}}) {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,6 @@
 | 
			
		||||
import 'package:html/parser.dart';
 | 
			
		||||
import 'package:http/http.dart';
 | 
			
		||||
import 'package:obtainium/app_sources/github.dart';
 | 
			
		||||
import 'package:obtainium/custom_errors.dart';
 | 
			
		||||
import 'package:obtainium/providers/source_provider.dart';
 | 
			
		||||
 | 
			
		||||
@@ -29,19 +30,37 @@ class Mullvad extends AppSource {
 | 
			
		||||
  ) async {
 | 
			
		||||
    Response res = await get(Uri.parse('$standardUrl/en/download/android'));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      var version = parse(res.body)
 | 
			
		||||
          .querySelector('p.subtitle.is-6')
 | 
			
		||||
          ?.querySelector('a')
 | 
			
		||||
          ?.attributes['href']
 | 
			
		||||
          ?.split('/')
 | 
			
		||||
          .last;
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
      var versions = parse(res.body)
 | 
			
		||||
          .querySelectorAll('p')
 | 
			
		||||
          .map((e) => e.innerHtml)
 | 
			
		||||
          .where((p) => p.contains('Latest version: '))
 | 
			
		||||
          .map((e) {
 | 
			
		||||
            var match = RegExp('[0-9]+(\\.[0-9]+)*').firstMatch(e);
 | 
			
		||||
            if (match == null) {
 | 
			
		||||
              return '';
 | 
			
		||||
            } else {
 | 
			
		||||
              return e.substring(match.start, match.end);
 | 
			
		||||
            }
 | 
			
		||||
          })
 | 
			
		||||
          .where((element) => element.isNotEmpty)
 | 
			
		||||
          .toList();
 | 
			
		||||
      if (versions.isEmpty) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      String? changeLog;
 | 
			
		||||
      try {
 | 
			
		||||
        changeLog = (await GitHub().getLatestAPKDetails(
 | 
			
		||||
                'https://github.com/mullvad/mullvadvpn-app',
 | 
			
		||||
                {'fallbackToOlderReleases': true}))
 | 
			
		||||
            .changeLog;
 | 
			
		||||
      } catch (e) {
 | 
			
		||||
        // Ignore
 | 
			
		||||
      }
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version,
 | 
			
		||||
          ['https://mullvad.net/download/app/apk/latest'],
 | 
			
		||||
          AppNames(name, 'Mullvad-VPN'));
 | 
			
		||||
          versions[0],
 | 
			
		||||
          getApkUrlsFromUrls(['https://mullvad.net/download/app/apk/latest']),
 | 
			
		||||
          AppNames(name, 'Mullvad-VPN'),
 | 
			
		||||
          changeLog: changeLog);
 | 
			
		||||
    } else {
 | 
			
		||||
      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';
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) => null;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetails(
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
@@ -31,7 +28,8 @@ class Signal extends AppSource {
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      return APKDetails(version, apkUrls, AppNames(name, 'Signal'));
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version, getApkUrlsFromUrls(apkUrls), AppNames(name, 'Signal'));
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -18,9 +18,6 @@ class SourceForge extends AppSource {
 | 
			
		||||
    return url.substring(0, match.end);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) => null;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetails(
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
@@ -53,7 +50,7 @@ class SourceForge extends AppSource {
 | 
			
		||||
              .toList();
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version,
 | 
			
		||||
          apkUrlList,
 | 
			
		||||
          getApkUrlsFromUrls(apkUrlList),
 | 
			
		||||
          AppNames(
 | 
			
		||||
              name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1)));
 | 
			
		||||
    } else {
 | 
			
		||||
 
 | 
			
		||||
@@ -24,9 +24,6 @@ class SteamMobile extends AppSource {
 | 
			
		||||
    return 'https://$host';
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) => null;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetails(
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
@@ -56,7 +53,8 @@ class SteamMobile extends AppSource {
 | 
			
		||||
      var version = links[0].substring(
 | 
			
		||||
          versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4);
 | 
			
		||||
      var apkUrls = [links[0]];
 | 
			
		||||
      return APKDetails(version, apkUrls, AppNames(name, apks[apkNamePrefix]!));
 | 
			
		||||
      return APKDetails(version, getApkUrlsFromUrls(apkUrls),
 | 
			
		||||
          AppNames(name, apks[apkNamePrefix]!));
 | 
			
		||||
    } else {
 | 
			
		||||
      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);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -48,6 +48,7 @@ class GeneratedFormTextField extends GeneratedFormItem {
 | 
			
		||||
 | 
			
		||||
class GeneratedFormDropdown extends GeneratedFormItem {
 | 
			
		||||
  late List<MapEntry<String, String>>? opts;
 | 
			
		||||
  List<String>? disabledOptKeys;
 | 
			
		||||
 | 
			
		||||
  GeneratedFormDropdown(
 | 
			
		||||
    String key,
 | 
			
		||||
@@ -55,6 +56,7 @@ class GeneratedFormDropdown extends GeneratedFormItem {
 | 
			
		||||
    String label = 'Input',
 | 
			
		||||
    List<Widget> belowWidgets = const [],
 | 
			
		||||
    String defaultValue = '',
 | 
			
		||||
    this.disabledOptKeys,
 | 
			
		||||
    List<String? Function(String? value)> additionalValidators = const [],
 | 
			
		||||
  }) : super(key,
 | 
			
		||||
            label: label,
 | 
			
		||||
@@ -225,10 +227,15 @@ class _GeneratedFormState extends State<GeneratedForm> {
 | 
			
		||||
          return DropdownButtonFormField(
 | 
			
		||||
              decoration: InputDecoration(labelText: formItem.label),
 | 
			
		||||
              value: values[formItem.key],
 | 
			
		||||
              items: formItem.opts!
 | 
			
		||||
                  .map((e2) =>
 | 
			
		||||
                      DropdownMenuItem(value: e2.key, child: Text(e2.value)))
 | 
			
		||||
                  .toList(),
 | 
			
		||||
              items: formItem.opts!.map((e2) {
 | 
			
		||||
                var enabled =
 | 
			
		||||
                    formItem.disabledOptKeys?.contains(e2.key) != true;
 | 
			
		||||
                return DropdownMenuItem(
 | 
			
		||||
                    value: e2.key,
 | 
			
		||||
                    enabled: enabled,
 | 
			
		||||
                    child: Opacity(
 | 
			
		||||
                        opacity: enabled ? 1 : 0.5, child: Text(e2.value)));
 | 
			
		||||
              }).toList(),
 | 
			
		||||
              onChanged: (value) {
 | 
			
		||||
                setState(() {
 | 
			
		||||
                  values[formItem.key] = value ?? formItem.opts!.first.key;
 | 
			
		||||
@@ -260,7 +267,10 @@ class _GeneratedFormState extends State<GeneratedForm> {
 | 
			
		||||
          formInputs[r][e] = Row(
 | 
			
		||||
            mainAxisAlignment: MainAxisAlignment.spaceBetween,
 | 
			
		||||
            children: [
 | 
			
		||||
              Text(widget.items[r][e].label),
 | 
			
		||||
              Flexible(child: Text(widget.items[r][e].label)),
 | 
			
		||||
              const SizedBox(
 | 
			
		||||
                width: 8,
 | 
			
		||||
              ),
 | 
			
		||||
              Switch(
 | 
			
		||||
                  value: values[widget.items[r][e].key],
 | 
			
		||||
                  onChanged: (value) {
 | 
			
		||||
@@ -460,10 +470,9 @@ class _GeneratedFormState extends State<GeneratedForm> {
 | 
			
		||||
      if (rowInputs.key > 0) {
 | 
			
		||||
        rows.add([
 | 
			
		||||
          SizedBox(
 | 
			
		||||
            height: widget.items[rowInputs.key][0] is GeneratedFormSwitch &&
 | 
			
		||||
                    widget.items[rowInputs.key - 1][0] is! GeneratedFormSwitch
 | 
			
		||||
                ? 25
 | 
			
		||||
                : 8,
 | 
			
		||||
            height: widget.items[rowInputs.key - 1][0] is GeneratedFormSwitch
 | 
			
		||||
                ? 8
 | 
			
		||||
                : 25,
 | 
			
		||||
          )
 | 
			
		||||
        ]);
 | 
			
		||||
      }
 | 
			
		||||
@@ -477,6 +486,7 @@ class _GeneratedFormState extends State<GeneratedForm> {
 | 
			
		||||
        rowItems.add(Expanded(
 | 
			
		||||
            child: Column(
 | 
			
		||||
                crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
			
		||||
                mainAxisSize: MainAxisSize.min,
 | 
			
		||||
                children: [
 | 
			
		||||
              rowInput.value,
 | 
			
		||||
              ...widget.items[rowInputs.key][rowInput.key].belowWidgets
 | 
			
		||||
 
 | 
			
		||||
@@ -21,7 +21,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart';
 | 
			
		||||
// ignore: implementation_imports
 | 
			
		||||
import 'package:easy_localization/src/localization.dart';
 | 
			
		||||
 | 
			
		||||
const String currentVersion = '0.11.0';
 | 
			
		||||
const String currentVersion = '0.11.30';
 | 
			
		||||
const String currentReleaseTag =
 | 
			
		||||
    'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES
 | 
			
		||||
 | 
			
		||||
@@ -34,7 +34,8 @@ const supportedLocales = [
 | 
			
		||||
  Locale('ja'),
 | 
			
		||||
  Locale('hu'),
 | 
			
		||||
  Locale('de'),
 | 
			
		||||
  Locale('fa')
 | 
			
		||||
  Locale('fa'),
 | 
			
		||||
  Locale('fr')
 | 
			
		||||
];
 | 
			
		||||
const fallbackLocale = Locale('en');
 | 
			
		||||
const localeDir = 'assets/translations';
 | 
			
		||||
@@ -146,6 +147,14 @@ Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async {
 | 
			
		||||
 | 
			
		||||
void main() async {
 | 
			
		||||
  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();
 | 
			
		||||
  if ((await DeviceInfoPlugin().androidInfo).version.sdkInt >= 29) {
 | 
			
		||||
    SystemChrome.setSystemUIOverlayStyle(
 | 
			
		||||
@@ -209,7 +218,15 @@ class _ObtainiumState extends State<Obtainium> {
 | 
			
		||||
              {'includePrereleases': true},
 | 
			
		||||
              null,
 | 
			
		||||
              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
 | 
			
		||||
      if (existingUpdateInterval != settingsProvider.updateInterval) {
 | 
			
		||||
 
 | 
			
		||||
@@ -33,10 +33,10 @@ class _AddAppPageState extends State<AddAppPage> {
 | 
			
		||||
  bool additionalSettingsValid = true;
 | 
			
		||||
  List<String> pickedCategories = [];
 | 
			
		||||
  int searchnum = 0;
 | 
			
		||||
  SourceProvider sourceProvider = SourceProvider();
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Widget build(BuildContext context) {
 | 
			
		||||
    SourceProvider sourceProvider = SourceProvider();
 | 
			
		||||
    AppsProvider appsProvider = context.read<AppsProvider>();
 | 
			
		||||
 | 
			
		||||
    bool doingSomething = gettingAppInfo || searching;
 | 
			
		||||
@@ -64,74 +64,58 @@ class _AddAppPageState extends State<AddAppPage> {
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    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 {
 | 
			
		||||
      setState(() {
 | 
			
		||||
        gettingAppInfo = true;
 | 
			
		||||
      });
 | 
			
		||||
      var settingsProvider = context.read<SettingsProvider>();
 | 
			
		||||
      () async {
 | 
			
		||||
      try {
 | 
			
		||||
        var settingsProvider = context.read<SettingsProvider>();
 | 
			
		||||
        var userPickedTrackOnly = additionalSettings['trackOnly'] == true;
 | 
			
		||||
        var userPickedNoVersionDetection =
 | 
			
		||||
            additionalSettings['noVersionDetection'] == true;
 | 
			
		||||
        var userPickedReleaseDateAsVersion =
 | 
			
		||||
            additionalSettings['releaseDateAsVersion'] == true;
 | 
			
		||||
        var cont = true;
 | 
			
		||||
        if ((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) {
 | 
			
		||||
          cont = false;
 | 
			
		||||
        }
 | 
			
		||||
        if (userPickedReleaseDateAsVersion && // ignore: use_build_context_synchronously
 | 
			
		||||
            // ignore: use_build_context_synchronously
 | 
			
		||||
            await showDialog(
 | 
			
		||||
                    context: context,
 | 
			
		||||
                    builder: (BuildContext ctx) {
 | 
			
		||||
                      return GeneratedFormModal(
 | 
			
		||||
                        title: tr('useReleaseDateAsVersion'),
 | 
			
		||||
                        items: const [],
 | 
			
		||||
                        message: tr('releaseDateAsVersionExplanation'),
 | 
			
		||||
                      );
 | 
			
		||||
                    }) ==
 | 
			
		||||
                null) {
 | 
			
		||||
          cont = false;
 | 
			
		||||
        }
 | 
			
		||||
        if (!userPickedReleaseDateAsVersion &&
 | 
			
		||||
            userPickedNoVersionDetection &&
 | 
			
		||||
            // ignore: use_build_context_synchronously
 | 
			
		||||
            await showDialog(
 | 
			
		||||
                    context: context,
 | 
			
		||||
                    builder: (BuildContext ctx) {
 | 
			
		||||
                      return GeneratedFormModal(
 | 
			
		||||
                        title: tr('disableVersionDetection'),
 | 
			
		||||
                        items: const [],
 | 
			
		||||
                        message: tr('noVersionDetectionExplanation'),
 | 
			
		||||
                      );
 | 
			
		||||
                    }) ==
 | 
			
		||||
                null) {
 | 
			
		||||
          cont = false;
 | 
			
		||||
        }
 | 
			
		||||
        if (cont) {
 | 
			
		||||
          HapticFeedback.selectionClick();
 | 
			
		||||
        App? app;
 | 
			
		||||
        if ((await getTrackOnlyConfirmationIfNeeded(userPickedTrackOnly)) &&
 | 
			
		||||
            (await getReleaseDateAsVersionConfirmationIfNeeded(
 | 
			
		||||
                userPickedTrackOnly))) {
 | 
			
		||||
          var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly;
 | 
			
		||||
          App app = await sourceProvider.getApp(
 | 
			
		||||
          app = await sourceProvider.getApp(
 | 
			
		||||
              pickedSource!, userInput, additionalSettings,
 | 
			
		||||
              trackOnlyOverride: trackOnly,
 | 
			
		||||
              noVersionDetectionOverride: userPickedNoVersionDetection,
 | 
			
		||||
              releaseDateAsVersionOverride: userPickedReleaseDateAsVersion);
 | 
			
		||||
              trackOnlyOverride: trackOnly);
 | 
			
		||||
          if (!trackOnly) {
 | 
			
		||||
            await settingsProvider.getInstallPermission();
 | 
			
		||||
          }
 | 
			
		||||
@@ -156,262 +140,253 @@ class _AddAppPageState extends State<AddAppPage> {
 | 
			
		||||
            app.installedVersion = app.latestVersion;
 | 
			
		||||
          }
 | 
			
		||||
          app.categories = pickedCategories;
 | 
			
		||||
          await appsProvider.saveApps([app]);
 | 
			
		||||
 | 
			
		||||
          return app;
 | 
			
		||||
          await appsProvider.saveApps([app], onlyIfExists: false);
 | 
			
		||||
        }
 | 
			
		||||
      }()
 | 
			
		||||
          .then((app) {
 | 
			
		||||
        if (app != null) {
 | 
			
		||||
          Navigator.push(context,
 | 
			
		||||
              MaterialPageRoute(builder: (context) => AppPage(appId: app.id)));
 | 
			
		||||
          Navigator.push(globalNavigatorKey.currentContext ?? context,
 | 
			
		||||
              MaterialPageRoute(builder: (context) => AppPage(appId: app!.id)));
 | 
			
		||||
        }
 | 
			
		||||
      }).catchError((e) {
 | 
			
		||||
      } catch (e) {
 | 
			
		||||
        showError(e, context);
 | 
			
		||||
      }).whenComplete(() {
 | 
			
		||||
      } finally {
 | 
			
		||||
        setState(() {
 | 
			
		||||
          gettingAppInfo = false;
 | 
			
		||||
          if (resetUserInputAfter) {
 | 
			
		||||
            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(
 | 
			
		||||
        backgroundColor: Theme.of(context).colorScheme.surface,
 | 
			
		||||
        body: CustomScrollView(slivers: <Widget>[
 | 
			
		||||
        body: CustomScrollView(shrinkWrap: true, slivers: <Widget>[
 | 
			
		||||
          CustomAppBar(title: tr('addApp')),
 | 
			
		||||
          SliverFillRemaining(
 | 
			
		||||
          SliverToBoxAdapter(
 | 
			
		||||
            child: Padding(
 | 
			
		||||
                padding: const EdgeInsets.all(16),
 | 
			
		||||
                child: Column(
 | 
			
		||||
                    mainAxisSize: MainAxisSize.min,
 | 
			
		||||
                    crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
			
		||||
                    children: [
 | 
			
		||||
                      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
 | 
			
		||||
                                      : addApp,
 | 
			
		||||
                                  child: Text(tr('add')))
 | 
			
		||||
                        ],
 | 
			
		||||
                      ),
 | 
			
		||||
                      if (sourceProvider.sources
 | 
			
		||||
                              .where((e) => e.canSearch)
 | 
			
		||||
                              .isNotEmpty &&
 | 
			
		||||
                          pickedSource == null &&
 | 
			
		||||
                          userInput.isEmpty)
 | 
			
		||||
                      getUrlInputRow(),
 | 
			
		||||
                      if (shouldShowSearchBar())
 | 
			
		||||
                        const SizedBox(
 | 
			
		||||
                          height: 16,
 | 
			
		||||
                        ),
 | 
			
		||||
                      if (sourceProvider.sources
 | 
			
		||||
                              .where((e) => e.canSearch)
 | 
			
		||||
                              .isNotEmpty &&
 | 
			
		||||
                          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 &&
 | 
			
		||||
                                        !isBuilding) {
 | 
			
		||||
                                      setState(() {
 | 
			
		||||
                                        searchQuery =
 | 
			
		||||
                                            values['searchSomeSources']!.trim();
 | 
			
		||||
                                      });
 | 
			
		||||
                                    }
 | 
			
		||||
                                  }),
 | 
			
		||||
                            ),
 | 
			
		||||
                            const SizedBox(
 | 
			
		||||
                              width: 16,
 | 
			
		||||
                            ),
 | 
			
		||||
                            ElevatedButton(
 | 
			
		||||
                                onPressed: searchQuery.isEmpty || doingSomething
 | 
			
		||||
                                    ? null
 | 
			
		||||
                                    : () {
 | 
			
		||||
                                        setState(() {
 | 
			
		||||
                                          searching = true;
 | 
			
		||||
                                        });
 | 
			
		||||
                                        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, false,
 | 
			
		||||
                                                isSearch: true);
 | 
			
		||||
                                          }
 | 
			
		||||
                                        }).catchError((e) {
 | 
			
		||||
                                          showError(e, context);
 | 
			
		||||
                                        }).whenComplete(() {
 | 
			
		||||
                                          setState(() {
 | 
			
		||||
                                            searching = false;
 | 
			
		||||
                                          });
 | 
			
		||||
                                        });
 | 
			
		||||
                                      },
 | 
			
		||||
                                child: Text(tr('search')))
 | 
			
		||||
                          ],
 | 
			
		||||
                        ),
 | 
			
		||||
                      if (shouldShowSearchBar()) getSearchBarRow(),
 | 
			
		||||
                      if (pickedSource != null)
 | 
			
		||||
                        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;
 | 
			
		||||
                                    }),
 | 
			
		||||
                              ],
 | 
			
		||||
                            ),
 | 
			
		||||
                          ],
 | 
			
		||||
                        )
 | 
			
		||||
                        getAdditionalOptsCol()
 | 
			
		||||
                      else
 | 
			
		||||
                        Expanded(
 | 
			
		||||
                            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()
 | 
			
		||||
                            ])),
 | 
			
		||||
                        getSourcesListWidget(),
 | 
			
		||||
                      const SizedBox(
 | 
			
		||||
                        height: 8,
 | 
			
		||||
                      ),
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,7 @@
 | 
			
		||||
import 'package:easy_localization/easy_localization.dart';
 | 
			
		||||
import 'package:flutter/material.dart';
 | 
			
		||||
import 'package:flutter/services.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/main.dart';
 | 
			
		||||
@@ -34,411 +35,420 @@ class _AppPageState extends State<AppPage> {
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    bool areDownloadsRunning = appsProvider.areDownloadsRunning();
 | 
			
		||||
 | 
			
		||||
    var sourceProvider = SourceProvider();
 | 
			
		||||
    AppInMemory? app = appsProvider.apps[widget.appId];
 | 
			
		||||
    AppInMemory? app = appsProvider.apps[widget.appId]?.deepCopy();
 | 
			
		||||
    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;
 | 
			
		||||
      getUpdate(app.app.id);
 | 
			
		||||
    }
 | 
			
		||||
    var trackOnly = app?.app.additionalSettings['trackOnly'] == true;
 | 
			
		||||
    var noVersionDetection =
 | 
			
		||||
        app?.app.additionalSettings['noVersionDetection'] == true;
 | 
			
		||||
 | 
			
		||||
    var infoColumn = Column(
 | 
			
		||||
      mainAxisAlignment: MainAxisAlignment.center,
 | 
			
		||||
      crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
			
		||||
      children: [
 | 
			
		||||
        GestureDetector(
 | 
			
		||||
            onTap: () {
 | 
			
		||||
              if (app?.app.url != null) {
 | 
			
		||||
                launchUrlString(app?.app.url ?? '',
 | 
			
		||||
                    mode: LaunchMode.externalApplication);
 | 
			
		||||
              }
 | 
			
		||||
            },
 | 
			
		||||
            child: Text(
 | 
			
		||||
              app?.app.url ?? '',
 | 
			
		||||
    bool isVersionDetectionStandard =
 | 
			
		||||
        app?.app.additionalSettings['versionDetection'] ==
 | 
			
		||||
            'standardVersionDetection';
 | 
			
		||||
 | 
			
		||||
    getInfoColumn() => Column(
 | 
			
		||||
          mainAxisAlignment: MainAxisAlignment.center,
 | 
			
		||||
          crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
			
		||||
          children: [
 | 
			
		||||
            GestureDetector(
 | 
			
		||||
                onTap: () {
 | 
			
		||||
                  if (app?.app.url != null) {
 | 
			
		||||
                    launchUrlString(app?.app.url ?? '',
 | 
			
		||||
                        mode: LaunchMode.externalApplication);
 | 
			
		||||
                  }
 | 
			
		||||
                },
 | 
			
		||||
                onLongPress: () {
 | 
			
		||||
                  Clipboard.setData(ClipboardData(text: app?.app.url ?? ''));
 | 
			
		||||
                  ScaffoldMessenger.of(context).showSnackBar(SnackBar(
 | 
			
		||||
                    content: Text(tr('copiedToClipboard')),
 | 
			
		||||
                  ));
 | 
			
		||||
                },
 | 
			
		||||
                child: Text(
 | 
			
		||||
                  app?.app.url ?? '',
 | 
			
		||||
                  textAlign: TextAlign.center,
 | 
			
		||||
                  style: const TextStyle(
 | 
			
		||||
                      decoration: TextDecoration.underline,
 | 
			
		||||
                      fontStyle: FontStyle.italic,
 | 
			
		||||
                      fontSize: 12),
 | 
			
		||||
                )),
 | 
			
		||||
            const SizedBox(
 | 
			
		||||
              height: 32,
 | 
			
		||||
            ),
 | 
			
		||||
            Text(
 | 
			
		||||
              tr('latestVersionX',
 | 
			
		||||
                  args: [app?.app.latestVersion ?? tr('unknown')]),
 | 
			
		||||
              textAlign: TextAlign.center,
 | 
			
		||||
              style: const TextStyle(
 | 
			
		||||
                  decoration: TextDecoration.underline,
 | 
			
		||||
                  fontStyle: FontStyle.italic,
 | 
			
		||||
                  fontSize: 12),
 | 
			
		||||
            )),
 | 
			
		||||
        const SizedBox(
 | 
			
		||||
          height: 32,
 | 
			
		||||
        ),
 | 
			
		||||
        Text(
 | 
			
		||||
          tr('latestVersionX', args: [app?.app.latestVersion ?? tr('unknown')]),
 | 
			
		||||
          textAlign: TextAlign.center,
 | 
			
		||||
          style: Theme.of(context).textTheme.bodyLarge,
 | 
			
		||||
        ),
 | 
			
		||||
        Text(
 | 
			
		||||
          '${tr('installedVersionX', args: [
 | 
			
		||||
                app?.app.installedVersion ?? tr('none')
 | 
			
		||||
              ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [
 | 
			
		||||
                  tr('app')
 | 
			
		||||
                ])}' : ''}',
 | 
			
		||||
          textAlign: TextAlign.center,
 | 
			
		||||
          style: Theme.of(context).textTheme.bodyLarge,
 | 
			
		||||
        ),
 | 
			
		||||
        const SizedBox(
 | 
			
		||||
          height: 32,
 | 
			
		||||
        ),
 | 
			
		||||
        Text(
 | 
			
		||||
          tr('lastUpdateCheckX', args: [
 | 
			
		||||
            app?.app.lastUpdateCheck == null
 | 
			
		||||
                ? tr('never')
 | 
			
		||||
                : '\n${app?.app.lastUpdateCheck?.toLocal()}'
 | 
			
		||||
          ]),
 | 
			
		||||
          textAlign: TextAlign.center,
 | 
			
		||||
          style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12),
 | 
			
		||||
        ),
 | 
			
		||||
        const SizedBox(
 | 
			
		||||
          height: 48,
 | 
			
		||||
        ),
 | 
			
		||||
        CategoryEditorSelector(
 | 
			
		||||
            alignment: WrapAlignment.center,
 | 
			
		||||
            preselected:
 | 
			
		||||
                app?.app.categories != null ? app!.app.categories.toSet() : {},
 | 
			
		||||
            onSelected: (categories) {
 | 
			
		||||
              if (app != null) {
 | 
			
		||||
                app.app.categories = categories;
 | 
			
		||||
                appsProvider.saveApps([app.app]);
 | 
			
		||||
              }
 | 
			
		||||
            }),
 | 
			
		||||
      ],
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    var fullInfoColumn = Column(
 | 
			
		||||
      mainAxisAlignment: MainAxisAlignment.center,
 | 
			
		||||
      crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
			
		||||
      children: [
 | 
			
		||||
        const SizedBox(height: 125),
 | 
			
		||||
        app?.installedInfo != null
 | 
			
		||||
            ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [
 | 
			
		||||
                Image.memory(
 | 
			
		||||
                  app!.installedInfo!.icon!,
 | 
			
		||||
                  height: 150,
 | 
			
		||||
                  gaplessPlayback: true,
 | 
			
		||||
                )
 | 
			
		||||
              ])
 | 
			
		||||
            : Container(),
 | 
			
		||||
        const SizedBox(
 | 
			
		||||
          height: 25,
 | 
			
		||||
        ),
 | 
			
		||||
        Text(
 | 
			
		||||
          app?.installedInfo?.name ?? app?.app.name ?? tr('app'),
 | 
			
		||||
          textAlign: TextAlign.center,
 | 
			
		||||
          style: Theme.of(context).textTheme.displayLarge,
 | 
			
		||||
        ),
 | 
			
		||||
        Text(
 | 
			
		||||
          tr('byX', args: [app?.app.author ?? tr('unknown')]),
 | 
			
		||||
          textAlign: TextAlign.center,
 | 
			
		||||
          style: Theme.of(context).textTheme.headlineMedium,
 | 
			
		||||
        ),
 | 
			
		||||
        const SizedBox(
 | 
			
		||||
          height: 8,
 | 
			
		||||
        ),
 | 
			
		||||
        Text(
 | 
			
		||||
          app?.app.id ?? '',
 | 
			
		||||
          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,
 | 
			
		||||
              style: Theme.of(context).textTheme.bodyLarge,
 | 
			
		||||
            ),
 | 
			
		||||
            Text(
 | 
			
		||||
              '${tr('installedVersionX', args: [
 | 
			
		||||
                    app?.app.installedVersion ?? tr('none')
 | 
			
		||||
                  ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [
 | 
			
		||||
                      tr('app')
 | 
			
		||||
                    ])}' : ''}',
 | 
			
		||||
              textAlign: TextAlign.center,
 | 
			
		||||
              style: Theme.of(context).textTheme.bodyLarge,
 | 
			
		||||
            ),
 | 
			
		||||
            if (app?.app.installedVersion != null &&
 | 
			
		||||
                !isVersionDetectionStandard)
 | 
			
		||||
              Column(
 | 
			
		||||
                children: [
 | 
			
		||||
                  const SizedBox(
 | 
			
		||||
                    height: 4,
 | 
			
		||||
                  ),
 | 
			
		||||
                  Text(
 | 
			
		||||
                    tr('noVersionDetection'),
 | 
			
		||||
                    style: Theme.of(context).textTheme.labelSmall,
 | 
			
		||||
                  )
 | 
			
		||||
                ],
 | 
			
		||||
              ),
 | 
			
		||||
        const SizedBox(
 | 
			
		||||
          height: 32,
 | 
			
		||||
        ),
 | 
			
		||||
        infoColumn,
 | 
			
		||||
        const SizedBox(height: 150)
 | 
			
		||||
      ],
 | 
			
		||||
    );
 | 
			
		||||
            const SizedBox(
 | 
			
		||||
              height: 32,
 | 
			
		||||
            ),
 | 
			
		||||
            Text(
 | 
			
		||||
              tr('lastUpdateCheckX', args: [
 | 
			
		||||
                app?.app.lastUpdateCheck == null
 | 
			
		||||
                    ? tr('never')
 | 
			
		||||
                    : '\n${app?.app.lastUpdateCheck?.toLocal()}'
 | 
			
		||||
              ]),
 | 
			
		||||
              textAlign: TextAlign.center,
 | 
			
		||||
              style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12),
 | 
			
		||||
            ),
 | 
			
		||||
            const SizedBox(
 | 
			
		||||
              height: 48,
 | 
			
		||||
            ),
 | 
			
		||||
            CategoryEditorSelector(
 | 
			
		||||
                alignment: WrapAlignment.center,
 | 
			
		||||
                preselected: app?.app.categories != null
 | 
			
		||||
                    ? app!.app.categories.toSet()
 | 
			
		||||
                    : {},
 | 
			
		||||
                onSelected: (categories) {
 | 
			
		||||
                  if (app != null) {
 | 
			
		||||
                    app.app.categories = categories;
 | 
			
		||||
                    appsProvider.saveApps([app.app]);
 | 
			
		||||
                  }
 | 
			
		||||
                }),
 | 
			
		||||
          ],
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
    getFullInfoColumn() => Column(
 | 
			
		||||
          mainAxisAlignment: MainAxisAlignment.center,
 | 
			
		||||
          crossAxisAlignment: CrossAxisAlignment.stretch,
 | 
			
		||||
          children: [
 | 
			
		||||
            const SizedBox(height: 125),
 | 
			
		||||
            app?.installedInfo != null
 | 
			
		||||
                ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [
 | 
			
		||||
                    Image.memory(
 | 
			
		||||
                      app!.installedInfo!.icon!,
 | 
			
		||||
                      height: 150,
 | 
			
		||||
                      gaplessPlayback: true,
 | 
			
		||||
                    )
 | 
			
		||||
                  ])
 | 
			
		||||
                : Container(),
 | 
			
		||||
            const SizedBox(
 | 
			
		||||
              height: 25,
 | 
			
		||||
            ),
 | 
			
		||||
            Text(
 | 
			
		||||
              app?.name ?? tr('app'),
 | 
			
		||||
              textAlign: TextAlign.center,
 | 
			
		||||
              style: Theme.of(context).textTheme.displayLarge,
 | 
			
		||||
            ),
 | 
			
		||||
            Text(
 | 
			
		||||
              tr('byX', args: [app?.app.author ?? tr('unknown')]),
 | 
			
		||||
              textAlign: TextAlign.center,
 | 
			
		||||
              style: Theme.of(context).textTheme.headlineMedium,
 | 
			
		||||
            ),
 | 
			
		||||
            const SizedBox(
 | 
			
		||||
              height: 8,
 | 
			
		||||
            ),
 | 
			
		||||
            Text(
 | 
			
		||||
              app?.app.id ?? '',
 | 
			
		||||
              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);
 | 
			
		||||
                    }
 | 
			
		||||
                  },
 | 
			
		||||
                ),
 | 
			
		||||
              )
 | 
			
		||||
              ..loadRequest(Uri.parse(app.app.url)))
 | 
			
		||||
        : 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;
 | 
			
		||||
              }
 | 
			
		||||
            }
 | 
			
		||||
          }
 | 
			
		||||
        } else if (originalSettings['versionDetection'] ==
 | 
			
		||||
            'releaseDateAsVersion') {
 | 
			
		||||
          app.app.installedVersion =
 | 
			
		||||
              app.installedInfo?.versionName ?? app.app.installedVersion;
 | 
			
		||||
        }
 | 
			
		||||
        appsProvider.saveApps([app.app]).then((value) {
 | 
			
		||||
          getUpdate(app.app.id);
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    getInstallOrUpdateButton() => TextButton(
 | 
			
		||||
        onPressed: (app?.app.installedVersion == null ||
 | 
			
		||||
                    app?.app.installedVersion != app?.app.latestVersion) &&
 | 
			
		||||
                !areDownloadsRunning
 | 
			
		||||
            ? () async {
 | 
			
		||||
                try {
 | 
			
		||||
                  HapticFeedback.heavyImpact();
 | 
			
		||||
                  if (app?.app.additionalSettings['trackOnly'] != true) {
 | 
			
		||||
                    await settingsProvider.getInstallPermission();
 | 
			
		||||
                  }
 | 
			
		||||
                  var res = await appsProvider.downloadAndInstallLatestApps(
 | 
			
		||||
                      [app!.app.id], globalNavigatorKey.currentContext);
 | 
			
		||||
                  if (res.isNotEmpty && mounted) {
 | 
			
		||||
                    Navigator.of(context).pop();
 | 
			
		||||
                  }
 | 
			
		||||
                } catch (e) {
 | 
			
		||||
                  showError(e, context);
 | 
			
		||||
                }
 | 
			
		||||
              }
 | 
			
		||||
            : null,
 | 
			
		||||
        child: Text(app?.app.installedVersion == null
 | 
			
		||||
            ? !trackOnly
 | 
			
		||||
                ? tr('install')
 | 
			
		||||
                : tr('markInstalled')
 | 
			
		||||
            : !trackOnly
 | 
			
		||||
                ? tr('update')
 | 
			
		||||
                : tr('markUpdated')));
 | 
			
		||||
 | 
			
		||||
    getBottomSheetMenu() => Padding(
 | 
			
		||||
        padding:
 | 
			
		||||
            EdgeInsets.fromLTRB(0, 0, 0, MediaQuery.of(context).padding.bottom),
 | 
			
		||||
        child: Column(
 | 
			
		||||
          mainAxisSize: MainAxisSize.min,
 | 
			
		||||
          children: [
 | 
			
		||||
            Padding(
 | 
			
		||||
                padding: const EdgeInsets.fromLTRB(16, 8, 16, 0),
 | 
			
		||||
                child: Row(
 | 
			
		||||
                    mainAxisAlignment: MainAxisAlignment.spaceEvenly,
 | 
			
		||||
                    children: [
 | 
			
		||||
                      if (app?.app.installedVersion != null &&
 | 
			
		||||
                          app?.app.installedVersion != app?.app.latestVersion &&
 | 
			
		||||
                          !isVersionDetectionStandard &&
 | 
			
		||||
                          !trackOnly)
 | 
			
		||||
                        IconButton(
 | 
			
		||||
                            onPressed: app?.downloadProgress != null
 | 
			
		||||
                                ? null
 | 
			
		||||
                                : showMarkUpdatedDialog,
 | 
			
		||||
                            tooltip: tr('markUpdated'),
 | 
			
		||||
                            icon: const Icon(Icons.done)),
 | 
			
		||||
                      if (source != null &&
 | 
			
		||||
                          source.combinedAppSpecificSettingFormItems.isNotEmpty)
 | 
			
		||||
                        IconButton(
 | 
			
		||||
                            onPressed: app?.downloadProgress != null
 | 
			
		||||
                                ? null
 | 
			
		||||
                                : () async {
 | 
			
		||||
                                    var values =
 | 
			
		||||
                                        await showAdditionalOptionsDialog();
 | 
			
		||||
                                    handleAdditionalOptionChanges(values);
 | 
			
		||||
                                  },
 | 
			
		||||
                            tooltip: tr('additionalOptions'),
 | 
			
		||||
                            icon: const Icon(Icons.edit)),
 | 
			
		||||
                      if (app != null && app.installedInfo != null)
 | 
			
		||||
                        IconButton(
 | 
			
		||||
                          onPressed: () {
 | 
			
		||||
                            appsProvider.openAppSettings(app.app.id);
 | 
			
		||||
                          },
 | 
			
		||||
                          icon: const Icon(Icons.settings),
 | 
			
		||||
                          tooltip: tr('settings'),
 | 
			
		||||
                        ),
 | 
			
		||||
                      if (app != null && settingsProvider.showAppWebpage)
 | 
			
		||||
                        IconButton(
 | 
			
		||||
                            onPressed: () {
 | 
			
		||||
                              showDialog(
 | 
			
		||||
                                  context: context,
 | 
			
		||||
                                  builder: (BuildContext ctx) {
 | 
			
		||||
                                    return AlertDialog(
 | 
			
		||||
                                      scrollable: true,
 | 
			
		||||
                                      content: getInfoColumn(),
 | 
			
		||||
                                      title: Text(
 | 
			
		||||
                                          '${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
 | 
			
		||||
              ? 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);
 | 
			
		||||
                              }
 | 
			
		||||
                            },
 | 
			
		||||
                          ),
 | 
			
		||||
                        )
 | 
			
		||||
                        ..loadRequest(Uri.parse(app.app.url)))
 | 
			
		||||
                  : Container()
 | 
			
		||||
              : CustomScrollView(
 | 
			
		||||
                  slivers: [
 | 
			
		||||
                    SliverToBoxAdapter(
 | 
			
		||||
                        child: Column(children: [fullInfoColumn])),
 | 
			
		||||
                  ],
 | 
			
		||||
                ),
 | 
			
		||||
          onRefresh: () async {
 | 
			
		||||
            if (app != null) {
 | 
			
		||||
              getUpdate(app.app.id);
 | 
			
		||||
            }
 | 
			
		||||
          }),
 | 
			
		||||
      bottomSheet: Padding(
 | 
			
		||||
          padding: EdgeInsets.fromLTRB(
 | 
			
		||||
              0, 0, 0, MediaQuery.of(context).padding.bottom),
 | 
			
		||||
          child: Column(
 | 
			
		||||
            mainAxisSize: MainAxisSize.min,
 | 
			
		||||
            children: [
 | 
			
		||||
              Padding(
 | 
			
		||||
                  padding: const EdgeInsets.fromLTRB(16, 8, 16, 0),
 | 
			
		||||
                  child: Row(
 | 
			
		||||
                      mainAxisAlignment: MainAxisAlignment.spaceEvenly,
 | 
			
		||||
                      children: [
 | 
			
		||||
                        if (noVersionDetection &&
 | 
			
		||||
                            !trackOnly &&
 | 
			
		||||
                            app?.app.installedVersion != null &&
 | 
			
		||||
                            app?.app.installedVersion != app?.app.latestVersion)
 | 
			
		||||
                          IconButton(
 | 
			
		||||
                              onPressed: app?.downloadProgress != null
 | 
			
		||||
                                  ? null
 | 
			
		||||
                                  : () {
 | 
			
		||||
                                      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')))
 | 
			
		||||
                                              ],
 | 
			
		||||
                                            );
 | 
			
		||||
                                          });
 | 
			
		||||
                                    },
 | 
			
		||||
                              tooltip: tr('markUpdated'),
 | 
			
		||||
                              icon: const Icon(Icons.done)),
 | 
			
		||||
                        if (source != null &&
 | 
			
		||||
                            source
 | 
			
		||||
                                .combinedAppSpecificSettingFormItems.isNotEmpty)
 | 
			
		||||
                          IconButton(
 | 
			
		||||
                              onPressed: app?.downloadProgress != null
 | 
			
		||||
                                  ? null
 | 
			
		||||
                                  : () {
 | 
			
		||||
                                      showDialog<Map<String, dynamic>?>(
 | 
			
		||||
                                          context: context,
 | 
			
		||||
                                          builder: (BuildContext ctx) {
 | 
			
		||||
                                            var items = source
 | 
			
		||||
                                                .combinedAppSpecificSettingFormItems
 | 
			
		||||
                                                .map((row) {
 | 
			
		||||
                                              row.map((e) {
 | 
			
		||||
                                                if (app?.app.additionalSettings[
 | 
			
		||||
                                                        e.key] !=
 | 
			
		||||
                                                    null) {
 | 
			
		||||
                                                  e.defaultValue = app?.app
 | 
			
		||||
                                                          .additionalSettings[
 | 
			
		||||
                                                      e.key];
 | 
			
		||||
                                                }
 | 
			
		||||
                                                return e;
 | 
			
		||||
                                              }).toList();
 | 
			
		||||
                                              return row;
 | 
			
		||||
                                            }).toList();
 | 
			
		||||
                                            return GeneratedFormModal(
 | 
			
		||||
                                              title: tr('additionalOptions'),
 | 
			
		||||
                                              items: items,
 | 
			
		||||
                                            );
 | 
			
		||||
                                          }).then((values) {
 | 
			
		||||
                                        if (app != null && values != null) {
 | 
			
		||||
                                          Map<String, dynamic>
 | 
			
		||||
                                              originalSettings =
 | 
			
		||||
                                              app.app.additionalSettings;
 | 
			
		||||
                                          app.app.additionalSettings = values;
 | 
			
		||||
                                          if (source.enforceTrackOnly) {
 | 
			
		||||
                                            app.app.additionalSettings[
 | 
			
		||||
                                                'trackOnly'] = true;
 | 
			
		||||
                                            showError(
 | 
			
		||||
                                                tr('appsFromSourceAreTrackOnly'),
 | 
			
		||||
                                                context);
 | 
			
		||||
                                          }
 | 
			
		||||
                                          if (app.app.additionalSettings[
 | 
			
		||||
                                                  'releaseDateAsVersion'] ==
 | 
			
		||||
                                              true) {
 | 
			
		||||
                                            app.app.additionalSettings[
 | 
			
		||||
                                                'noVersionDetection'] = true;
 | 
			
		||||
                                            if (originalSettings[
 | 
			
		||||
                                                    'releaseDateAsVersion'] !=
 | 
			
		||||
                                                true) {
 | 
			
		||||
                                              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;
 | 
			
		||||
                                                }
 | 
			
		||||
                                              }
 | 
			
		||||
                                            }
 | 
			
		||||
                                          } else if (originalSettings[
 | 
			
		||||
                                                  'releaseDateAsVersion'] ==
 | 
			
		||||
                                              true) {
 | 
			
		||||
                                            app.app.additionalSettings[
 | 
			
		||||
                                                'noVersionDetection'] = false;
 | 
			
		||||
                                            app.app.installedVersion = app
 | 
			
		||||
                                                    .installedInfo
 | 
			
		||||
                                                    ?.versionName ??
 | 
			
		||||
                                                app.app.installedVersion;
 | 
			
		||||
                                          }
 | 
			
		||||
                                          appsProvider.saveApps([app.app]).then(
 | 
			
		||||
                                              (value) {
 | 
			
		||||
                                            getUpdate(app.app.id);
 | 
			
		||||
                                          });
 | 
			
		||||
                                        }
 | 
			
		||||
                                      });
 | 
			
		||||
                                    },
 | 
			
		||||
                              tooltip: tr('additionalOptions'),
 | 
			
		||||
                              icon: const Icon(Icons.edit)),
 | 
			
		||||
                        if (app != null && app.installedInfo != null)
 | 
			
		||||
                          IconButton(
 | 
			
		||||
                            onPressed: () {
 | 
			
		||||
                              appsProvider.openAppSettings(app.app.id);
 | 
			
		||||
                            },
 | 
			
		||||
                            icon: const Icon(Icons.settings),
 | 
			
		||||
                            tooltip: tr('settings'),
 | 
			
		||||
                          ),
 | 
			
		||||
                        if (app != null && settingsProvider.showAppWebpage)
 | 
			
		||||
                          IconButton(
 | 
			
		||||
                              onPressed: () {
 | 
			
		||||
                                showDialog(
 | 
			
		||||
                                    context: context,
 | 
			
		||||
                                    builder: (BuildContext ctx) {
 | 
			
		||||
                                      return AlertDialog(
 | 
			
		||||
                                        scrollable: true,
 | 
			
		||||
                                        content: infoColumn,
 | 
			
		||||
                                        title: Text(
 | 
			
		||||
                                            '${app.app.name} ${tr('byX', args: [
 | 
			
		||||
                                              app.app.author
 | 
			
		||||
                                            ])}'),
 | 
			
		||||
                                        actions: [
 | 
			
		||||
                                          TextButton(
 | 
			
		||||
                                              onPressed: () {
 | 
			
		||||
                                                Navigator.of(context).pop();
 | 
			
		||||
                                              },
 | 
			
		||||
                                              child: Text(tr('continue')))
 | 
			
		||||
                                        ],
 | 
			
		||||
                                      );
 | 
			
		||||
                                    });
 | 
			
		||||
                              },
 | 
			
		||||
                              icon: const Icon(Icons.more_horiz),
 | 
			
		||||
                              tooltip: tr('more')),
 | 
			
		||||
                        const SizedBox(width: 16.0),
 | 
			
		||||
                        Expanded(
 | 
			
		||||
                            child: TextButton(
 | 
			
		||||
                                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);
 | 
			
		||||
                                          });
 | 
			
		||||
                                        }).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),
 | 
			
		||||
                        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))
 | 
			
		||||
            ],
 | 
			
		||||
          )),
 | 
			
		||||
    );
 | 
			
		||||
        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());
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1595
									
								
								lib/pages/apps.dart
									
									
									
									
									
								
							
							
						
						@@ -30,6 +30,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
 | 
			
		||||
    SourceProvider sourceProvider = SourceProvider();
 | 
			
		||||
    var appsProvider = context.read<AppsProvider>();
 | 
			
		||||
    var settingsProvider = context.read<SettingsProvider>();
 | 
			
		||||
 | 
			
		||||
    var outlineButtonStyle = ButtonStyle(
 | 
			
		||||
      shape: MaterialStateProperty.all(
 | 
			
		||||
        StadiumBorder(
 | 
			
		||||
@@ -41,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(
 | 
			
		||||
        backgroundColor: Theme.of(context).colorScheme.surface,
 | 
			
		||||
        body: CustomScrollView(slivers: <Widget>[
 | 
			
		||||
@@ -60,18 +308,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
 | 
			
		||||
                                  onPressed: appsProvider.apps.isEmpty ||
 | 
			
		||||
                                          importInProgress
 | 
			
		||||
                                      ? null
 | 
			
		||||
                                      : () {
 | 
			
		||||
                                          HapticFeedback.selectionClick();
 | 
			
		||||
                                          appsProvider
 | 
			
		||||
                                              .exportApps()
 | 
			
		||||
                                              .then((String path) {
 | 
			
		||||
                                            showError(
 | 
			
		||||
                                                tr('exportedTo', args: [path]),
 | 
			
		||||
                                                context);
 | 
			
		||||
                                          }).catchError((e) {
 | 
			
		||||
                                            showError(e, context);
 | 
			
		||||
                                          });
 | 
			
		||||
                                        },
 | 
			
		||||
                                      : runObtainiumExport,
 | 
			
		||||
                                  child: Text(tr('obtainiumExport')))),
 | 
			
		||||
                          const SizedBox(
 | 
			
		||||
                            width: 16,
 | 
			
		||||
@@ -81,59 +318,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
 | 
			
		||||
                                  style: outlineButtonStyle,
 | 
			
		||||
                                  onPressed: importInProgress
 | 
			
		||||
                                      ? null
 | 
			
		||||
                                      : () {
 | 
			
		||||
                                          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;
 | 
			
		||||
                                                    }
 | 
			
		||||
                                                  }
 | 
			
		||||
                                                });
 | 
			
		||||
                                                settingsProvider.categories =
 | 
			
		||||
                                                    cats;
 | 
			
		||||
                                                showError(
 | 
			
		||||
                                                    tr('importedX', args: [
 | 
			
		||||
                                                      plural('apps', value)
 | 
			
		||||
                                                    ]),
 | 
			
		||||
                                                    context);
 | 
			
		||||
                                              });
 | 
			
		||||
                                            } else {
 | 
			
		||||
                                              // User canceled the picker
 | 
			
		||||
                                            }
 | 
			
		||||
                                          }).catchError((e) {
 | 
			
		||||
                                            showError(e, context);
 | 
			
		||||
                                          }).whenComplete(() {
 | 
			
		||||
                                            setState(() {
 | 
			
		||||
                                              importInProgress = false;
 | 
			
		||||
                                            });
 | 
			
		||||
                                          });
 | 
			
		||||
                                        },
 | 
			
		||||
                                      : runObtainiumImport,
 | 
			
		||||
                                  child: Text(tr('obtainiumImport'))))
 | 
			
		||||
                        ],
 | 
			
		||||
                      ),
 | 
			
		||||
@@ -150,88 +335,26 @@ class _ImportExportPageState extends State<ImportExportPage> {
 | 
			
		||||
                          ],
 | 
			
		||||
                        )
 | 
			
		||||
                      else
 | 
			
		||||
                        const Divider(
 | 
			
		||||
                          height: 32,
 | 
			
		||||
                        Column(
 | 
			
		||||
                          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
 | 
			
		||||
                          .where((element) => element.canSearch)
 | 
			
		||||
                          .map((source) => Column(
 | 
			
		||||
@@ -243,104 +366,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
 | 
			
		||||
                                        onPressed: importInProgress
 | 
			
		||||
                                            ? null
 | 
			
		||||
                                            : () {
 | 
			
		||||
                                                () 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 =
 | 
			
		||||
                                                          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;
 | 
			
		||||
                                                  });
 | 
			
		||||
                                                });
 | 
			
		||||
                                                runSourceSearch(source);
 | 
			
		||||
                                              },
 | 
			
		||||
                                        child: Text(
 | 
			
		||||
                                            tr('searchX', args: [source.name])))
 | 
			
		||||
@@ -356,91 +382,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
 | 
			
		||||
                                        onPressed: importInProgress
 | 
			
		||||
                                            ? null
 | 
			
		||||
                                            : () {
 | 
			
		||||
                                                () 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 =
 | 
			
		||||
                                                        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;
 | 
			
		||||
                                                  });
 | 
			
		||||
                                                });
 | 
			
		||||
                                                runMassSourceImport(source);
 | 
			
		||||
                                              },
 | 
			
		||||
                                        child: Text(
 | 
			
		||||
                                            tr('importX', args: [source.name])))
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@ import 'package:obtainium/components/custom_app_bar.dart';
 | 
			
		||||
import 'package:obtainium/components/generated_form.dart';
 | 
			
		||||
import 'package:obtainium/custom_errors.dart';
 | 
			
		||||
import 'package:obtainium/main.dart';
 | 
			
		||||
import 'package:obtainium/providers/apps_provider.dart';
 | 
			
		||||
import 'package:obtainium/providers/logs_provider.dart';
 | 
			
		||||
import 'package:obtainium/providers/settings_provider.dart';
 | 
			
		||||
import 'package:obtainium/providers/source_provider.dart';
 | 
			
		||||
@@ -87,6 +88,7 @@ class _SettingsPageState extends State<SettingsPage> {
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
    var sortDropdown = DropdownButtonFormField(
 | 
			
		||||
        isExpanded: true,
 | 
			
		||||
        decoration: InputDecoration(labelText: tr('appSortBy')),
 | 
			
		||||
        value: settingsProvider.sortColumn,
 | 
			
		||||
        items: [
 | 
			
		||||
@@ -114,6 +116,7 @@ class _SettingsPageState extends State<SettingsPage> {
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
    var orderDropdown = DropdownButtonFormField(
 | 
			
		||||
        isExpanded: true,
 | 
			
		||||
        decoration: InputDecoration(labelText: tr('appSortOrder')),
 | 
			
		||||
        value: settingsProvider.sortOrder,
 | 
			
		||||
        items: [
 | 
			
		||||
@@ -150,7 +153,7 @@ class _SettingsPageState extends State<SettingsPage> {
 | 
			
		||||
          if (value != null) {
 | 
			
		||||
            context.setLocale(Locale(value));
 | 
			
		||||
          } else {
 | 
			
		||||
            context.resetLocale();
 | 
			
		||||
            settingsProvider.resetLocaleSafe(context);
 | 
			
		||||
          }
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
@@ -260,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(
 | 
			
		||||
                              height: 16,
 | 
			
		||||
                            ),
 | 
			
		||||
@@ -430,6 +445,7 @@ class _CategoryEditorSelectorState extends State<CategoryEditorSelector> {
 | 
			
		||||
  @override
 | 
			
		||||
  Widget build(BuildContext context) {
 | 
			
		||||
    var settingsProvider = context.watch<SettingsProvider>();
 | 
			
		||||
    var appsProvider = context.watch<AppsProvider>();
 | 
			
		||||
    storedValues = settingsProvider.categories.map((key, value) => MapEntry(
 | 
			
		||||
        key,
 | 
			
		||||
        MapEntry(value,
 | 
			
		||||
@@ -453,8 +469,9 @@ class _CategoryEditorSelectorState extends State<CategoryEditorSelector> {
 | 
			
		||||
          if (!isBuilding) {
 | 
			
		||||
            storedValues =
 | 
			
		||||
                values['categories'] as Map<String, MapEntry<int, bool>>;
 | 
			
		||||
            settingsProvider.categories =
 | 
			
		||||
                storedValues.map((key, value) => MapEntry(key, value.key));
 | 
			
		||||
            settingsProvider.setCategories(
 | 
			
		||||
                storedValues.map((key, value) => MapEntry(key, value.key)),
 | 
			
		||||
                appsProvider: appsProvider);
 | 
			
		||||
            if (widget.onSelected != null) {
 | 
			
		||||
              widget.onSelected!(storedValues.keys
 | 
			
		||||
                  .where((k) => storedValues[k]!.value)
 | 
			
		||||
 
 | 
			
		||||
@@ -34,6 +34,10 @@ class AppInMemory {
 | 
			
		||||
  AppInfo? installedInfo;
 | 
			
		||||
 | 
			
		||||
  AppInMemory(this.app, this.downloadProgress, this.installedInfo);
 | 
			
		||||
  AppInMemory deepCopy() =>
 | 
			
		||||
      AppInMemory(app.deepCopy(), downloadProgress, installedInfo);
 | 
			
		||||
 | 
			
		||||
  String get name => app.overrideName ?? installedInfo?.name ?? app.finalName;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class DownloadedApk {
 | 
			
		||||
@@ -73,6 +77,18 @@ List<String> generateStandardVersionRegExStrings() {
 | 
			
		||||
List<String> standardVersionRegExStrings =
 | 
			
		||||
    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 {
 | 
			
		||||
  // In memory App state (should always be kept in sync with local storage versions)
 | 
			
		||||
  Map<String, AppInMemory> apps = {};
 | 
			
		||||
@@ -85,6 +101,8 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
  late Stream<FGBGType>? foregroundStream;
 | 
			
		||||
  late StreamSubscription<FGBGType>? foregroundSubscription;
 | 
			
		||||
 | 
			
		||||
  Iterable<AppInMemory> getAppValues() => apps.values.map((a) => a.deepCopy());
 | 
			
		||||
 | 
			
		||||
  AppsProvider() {
 | 
			
		||||
    // Subscribe to changes in the app foreground status
 | 
			
		||||
    foregroundStream = FGBGEvents.stream.asBroadcastStream();
 | 
			
		||||
@@ -145,56 +163,67 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  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 =
 | 
			
		||||
        context?.read<NotificationsProvider>();
 | 
			
		||||
    var notif = DownloadNotification(app.name, 100);
 | 
			
		||||
    notificationsProvider?.cancel(notif.id);
 | 
			
		||||
    int? prevProg;
 | 
			
		||||
    File downloadedFile =
 | 
			
		||||
        await downloadFile(downloadUrl, fileName, (double? progress) {
 | 
			
		||||
      int? prog = progress?.ceil();
 | 
			
		||||
    var notifId = DownloadNotification(app.finalName, 0).id;
 | 
			
		||||
    if (apps[app.id] != null) {
 | 
			
		||||
      apps[app.id]!.downloadProgress = 0;
 | 
			
		||||
      notifyListeners();
 | 
			
		||||
    }
 | 
			
		||||
    try {
 | 
			
		||||
      String downloadUrl = await SourceProvider()
 | 
			
		||||
          .getSource(app.url)
 | 
			
		||||
          .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex].value);
 | 
			
		||||
      var fileName = '${app.id}-${downloadUrl.hashCode}.apk';
 | 
			
		||||
      var notif = DownloadNotification(app.finalName, 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.finalName, 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}-${downloadUrl.hashCode}.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) {
 | 
			
		||||
        apps[app.id]!.downloadProgress = progress;
 | 
			
		||||
        apps[app.id]!.downloadProgress = null;
 | 
			
		||||
        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)) {
 | 
			
		||||
        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
 | 
			
		||||
@@ -272,9 +301,10 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
    await intent.launch();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  Future<String?> confirmApkUrl(App app, BuildContext? context) async {
 | 
			
		||||
  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)
 | 
			
		||||
    String? apkUrl = app.apkUrls[app.preferredApkIndex];
 | 
			
		||||
    MapEntry<String, String>? apkUrl = app.apkUrls[app.preferredApkIndex];
 | 
			
		||||
    // get device supported architecture
 | 
			
		||||
    List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis;
 | 
			
		||||
 | 
			
		||||
@@ -297,14 +327,14 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
 | 
			
		||||
    // If the picked APK comes from an origin different from the source, get user confirmation (if context provided)
 | 
			
		||||
    if (apkUrl != null &&
 | 
			
		||||
        getHost(apkUrl) != getHost(app.url) &&
 | 
			
		||||
        getHost(apkUrl.value) != getHost(app.url) &&
 | 
			
		||||
        context != null) {
 | 
			
		||||
      // ignore: use_build_context_synchronously
 | 
			
		||||
      if (await showDialog(
 | 
			
		||||
              context: context,
 | 
			
		||||
              builder: (BuildContext ctx) {
 | 
			
		||||
                return APKOriginWarningDialog(
 | 
			
		||||
                    sourceUrl: app.url, apkUrl: apkUrl!);
 | 
			
		||||
                    sourceUrl: app.url, apkUrl: apkUrl!.value);
 | 
			
		||||
              }) !=
 | 
			
		||||
          true) {
 | 
			
		||||
        apkUrl = null;
 | 
			
		||||
@@ -329,7 +359,7 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
      if (apps[id] == null) {
 | 
			
		||||
        throw ObtainiumError(tr('appNotFound'));
 | 
			
		||||
      }
 | 
			
		||||
      String? apkUrl;
 | 
			
		||||
      MapEntry<String, String>? apkUrl;
 | 
			
		||||
      var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true;
 | 
			
		||||
      if (!trackOnly) {
 | 
			
		||||
        apkUrl = await confirmApkUrl(apps[id]!.app, context);
 | 
			
		||||
@@ -460,94 +490,117 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
    return res;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // If the App says it is installed but installedInfo is null, set it to not installed
 | 
			
		||||
  // If there is any other mismatch between installedInfo and installedVersion, try reconciling them intelligently
 | 
			
		||||
  // If that fails, just set it to the actual version string (all we can do at that point)
 | 
			
		||||
  // Don't save changes, just return the object if changes were made (else null)
 | 
			
		||||
  bool isVersionDetectionPossible(AppInMemory? app) {
 | 
			
		||||
    return app?.app.additionalSettings['trackOnly'] != true &&
 | 
			
		||||
        app?.app.additionalSettings['versionDetection'] !=
 | 
			
		||||
            '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) {
 | 
			
		||||
    var modded = false;
 | 
			
		||||
    var trackOnly = app.additionalSettings['trackOnly'] == true;
 | 
			
		||||
    var noVersionDetection =
 | 
			
		||||
        app.additionalSettings['noVersionDetection'] == true;
 | 
			
		||||
    var noVersionDetection = app.additionalSettings['versionDetection'] !=
 | 
			
		||||
        'standardVersionDetection';
 | 
			
		||||
    // FIRST, COMPARE THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE ONE IS NULL
 | 
			
		||||
    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;
 | 
			
		||||
      modded = true;
 | 
			
		||||
    } else if (installedInfo?.versionName != 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;
 | 
			
		||||
      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 &&
 | 
			
		||||
        !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!);
 | 
			
		||||
      if (correctedInstalledVersion != null) {
 | 
			
		||||
        app.installedVersion = correctedInstalledVersion;
 | 
			
		||||
      if (correctedInstalledVersion?.key == false) {
 | 
			
		||||
        app.installedVersion = correctedInstalledVersion!.value;
 | 
			
		||||
        modded = true;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // THIRD, RECONCILE THE APP'S REPORTED INSTALLED AND LATEST VERSIONS
 | 
			
		||||
    if (app.installedVersion != null &&
 | 
			
		||||
        app.installedVersion != app.latestVersion &&
 | 
			
		||||
        !noVersionDetection) {
 | 
			
		||||
      app.installedVersion = reconcileRealAndInternalVersions(
 | 
			
		||||
              app.installedVersion!, app.latestVersion,
 | 
			
		||||
              matchMode: true) ??
 | 
			
		||||
          app.installedVersion;
 | 
			
		||||
      // App's reported installed and latest versions don't match (and it uses standard version detection)
 | 
			
		||||
      // If they share a standard format, make sure the App's reported installed version uses that format
 | 
			
		||||
      var correctedInstalledVersion =
 | 
			
		||||
          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;
 | 
			
		||||
    }
 | 
			
		||||
    // 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;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  String? reconcileRealAndInternalVersions(
 | 
			
		||||
      String realVersion, String internalVersion,
 | 
			
		||||
      {bool matchMode = false}) {
 | 
			
		||||
    // 1. If one or both of these can't be converted to a "standard" format, return null (leave as is)
 | 
			
		||||
    // 2. If both have a "standard" format under which they are equal, return null (leave as is)
 | 
			
		||||
    // 3. If both have a "standard" format in common but are unequal, return realVersion (this means it was changed externally)
 | 
			
		||||
    // If in matchMode, the outcomes of rules 2 and 3 are reversed, and the "real" version is not matched strictly
 | 
			
		||||
    // Matchmode to be used when comparing internal install version and internal latest version
 | 
			
		||||
 | 
			
		||||
    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;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    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);
 | 
			
		||||
  MapEntry<bool, String>? reconcileVersionDifferences(
 | 
			
		||||
      String templateVersion, String comparisonVersion) {
 | 
			
		||||
    // Returns null if the versions don't share a common standard format
 | 
			
		||||
    // Returns <true, comparisonVersion> if they share a common format and are equal
 | 
			
		||||
    // Returns <false, templateVersion> if they share a common format but are not equal
 | 
			
		||||
    // templateVersion must fully match a standard format, while comparisonVersion can have a substring match
 | 
			
		||||
    var templateVersionFormats =
 | 
			
		||||
        findStandardFormatsForVersion(templateVersion, true);
 | 
			
		||||
    var comparisonVersionFormats =
 | 
			
		||||
        findStandardFormatsForVersion(comparisonVersion, false);
 | 
			
		||||
    var commonStandardFormats =
 | 
			
		||||
        realStandardVersionFormats.intersection(internalStandardVersionFormats);
 | 
			
		||||
        templateVersionFormats.intersection(comparisonVersionFormats);
 | 
			
		||||
    if (commonStandardFormats.isEmpty) {
 | 
			
		||||
      return null; // Incompatible; no "enhanced detection"
 | 
			
		||||
      return null;
 | 
			
		||||
    }
 | 
			
		||||
    for (String pattern in commonStandardFormats) {
 | 
			
		||||
      if (doStringsMatchUnderRegEx(pattern, internalVersion, realVersion)) {
 | 
			
		||||
        return matchMode
 | 
			
		||||
            ? internalVersion
 | 
			
		||||
            : null; // Enhanced detection says no change
 | 
			
		||||
      if (doStringsMatchUnderRegEx(
 | 
			
		||||
          pattern, comparisonVersion, templateVersion)) {
 | 
			
		||||
        return MapEntry(true, comparisonVersion);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    return matchMode
 | 
			
		||||
        ? null
 | 
			
		||||
        : realVersion; // Enhanced detection says something changed
 | 
			
		||||
    return MapEntry(false, templateVersion);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  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 {
 | 
			
		||||
@@ -559,7 +612,21 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
    List<App> newApps = (await getAppsDir())
 | 
			
		||||
        .listSync()
 | 
			
		||||
        .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();
 | 
			
		||||
    var idsToDelete = apps.values
 | 
			
		||||
        .map((e) => e.app.id)
 | 
			
		||||
@@ -576,7 +643,7 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
        sp.getSource(newApps[i].url);
 | 
			
		||||
        apps[newApps[i].id] = AppInMemory(newApps[i], null, info);
 | 
			
		||||
      } catch (e) {
 | 
			
		||||
        errors.add([newApps[i].id, newApps[i].name, e.toString()]);
 | 
			
		||||
        errors.add([newApps[i].id, newApps[i].finalName, e.toString()]);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (errors.isNotEmpty) {
 | 
			
		||||
@@ -602,10 +669,12 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  Future<void> saveApps(List<App> apps,
 | 
			
		||||
      {bool attemptToCorrectInstallStatus = true}) async {
 | 
			
		||||
      {bool attemptToCorrectInstallStatus = true,
 | 
			
		||||
      bool onlyIfExists = true}) async {
 | 
			
		||||
    attemptToCorrectInstallStatus =
 | 
			
		||||
        attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork());
 | 
			
		||||
    for (var app in apps) {
 | 
			
		||||
    for (var a in apps) {
 | 
			
		||||
      var app = a.deepCopy();
 | 
			
		||||
      AppInfo? info = await getInstalledInfo(app.id);
 | 
			
		||||
      app.name = info?.name ?? app.name;
 | 
			
		||||
      if (attemptToCorrectInstallStatus) {
 | 
			
		||||
@@ -613,9 +682,15 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
      }
 | 
			
		||||
      File('${(await getAppsDir()).path}/${app.id}.json')
 | 
			
		||||
          .writeAsStringSync(jsonEncode(app.toJson()));
 | 
			
		||||
      this.apps.update(
 | 
			
		||||
          app.id, (value) => AppInMemory(app, value.downloadProgress, info),
 | 
			
		||||
          ifAbsent: () => AppInMemory(app, null, info));
 | 
			
		||||
      try {
 | 
			
		||||
        this.apps.update(
 | 
			
		||||
            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();
 | 
			
		||||
  }
 | 
			
		||||
@@ -636,8 +711,11 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  Future<bool> removeAppsWithModal(BuildContext context, List<App> apps) async {
 | 
			
		||||
    var showUninstallOption =
 | 
			
		||||
        apps.where((a) => a.installedVersion != null).isNotEmpty;
 | 
			
		||||
    var showUninstallOption = apps
 | 
			
		||||
        .where((a) =>
 | 
			
		||||
            a.installedVersion != null &&
 | 
			
		||||
            a.additionalSettings['trackOnly'] != true)
 | 
			
		||||
        .isNotEmpty;
 | 
			
		||||
    var values = await showDialog(
 | 
			
		||||
        context: context,
 | 
			
		||||
        builder: (BuildContext ctx) {
 | 
			
		||||
@@ -686,6 +764,18 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
    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 {
 | 
			
		||||
    App? currentApp = apps[appId]!.app;
 | 
			
		||||
    SourceProvider sourceProvider = SourceProvider();
 | 
			
		||||
@@ -765,12 +855,6 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  Future<String> exportApps() async {
 | 
			
		||||
    Directory? exportDir = Directory('/storage/emulated/0/Download');
 | 
			
		||||
    String path = 'Downloads'; // TODO: See if hardcoding this can be avoided
 | 
			
		||||
    if (!exportDir.existsSync()) {
 | 
			
		||||
      exportDir = await getExternalStorageDirectory();
 | 
			
		||||
      path = exportDir!.path;
 | 
			
		||||
    }
 | 
			
		||||
    if ((await DeviceInfoPlugin().androidInfo).version.sdkInt <= 29) {
 | 
			
		||||
      if (await Permission.storage.isDenied) {
 | 
			
		||||
        await Permission.storage.request();
 | 
			
		||||
@@ -779,6 +863,18 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
        throw ObtainiumError(tr('storagePermissionDenied'));
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    Directory? exportDir = Directory('/storage/emulated/0/Download');
 | 
			
		||||
    String path = 'Downloads'; // TODO: See if hardcoding this can be avoided
 | 
			
		||||
    var downloadsAccessible = false;
 | 
			
		||||
    try {
 | 
			
		||||
      downloadsAccessible = exportDir.existsSync();
 | 
			
		||||
    } catch (e) {
 | 
			
		||||
      logs.add('Error accessing Downloads (will use fallback): $e');
 | 
			
		||||
    }
 | 
			
		||||
    if (!downloadsAccessible) {
 | 
			
		||||
      exportDir = await getExternalStorageDirectory();
 | 
			
		||||
      path = exportDir!.path;
 | 
			
		||||
    }
 | 
			
		||||
    File export = File(
 | 
			
		||||
        '${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json');
 | 
			
		||||
    export.writeAsStringSync(
 | 
			
		||||
@@ -798,7 +894,7 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
        a.installedVersion = apps[a.id]?.app.installedVersion;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    await saveApps(importedApps);
 | 
			
		||||
    await saveApps(importedApps, onlyIfExists: false);
 | 
			
		||||
    notifyListeners();
 | 
			
		||||
    return importedApps.length;
 | 
			
		||||
  }
 | 
			
		||||
@@ -818,7 +914,7 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
      if (apps.containsKey(app.id)) {
 | 
			
		||||
        errorsMap.addAll({app.id: tr('appAlreadyAdded')});
 | 
			
		||||
      } else {
 | 
			
		||||
        await saveApps([app]);
 | 
			
		||||
        await saveApps([app], onlyIfExists: false);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    List<List<String>> errors =
 | 
			
		||||
@@ -831,7 +927,7 @@ class APKPicker extends StatefulWidget {
 | 
			
		||||
  const APKPicker({super.key, required this.app, this.initVal, this.archs});
 | 
			
		||||
 | 
			
		||||
  final App app;
 | 
			
		||||
  final String? initVal;
 | 
			
		||||
  final MapEntry<String, String>? initVal;
 | 
			
		||||
  final List<String>? archs;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
@@ -839,7 +935,7 @@ class APKPicker extends StatefulWidget {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class _APKPickerState extends State<APKPicker> {
 | 
			
		||||
  String? apkUrl;
 | 
			
		||||
  MapEntry<String, String>? apkUrl;
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Widget build(BuildContext context) {
 | 
			
		||||
@@ -848,19 +944,17 @@ class _APKPickerState extends State<APKPicker> {
 | 
			
		||||
      scrollable: true,
 | 
			
		||||
      title: Text(tr('pickAnAPK')),
 | 
			
		||||
      content: Column(children: [
 | 
			
		||||
        Text(tr('appHasMoreThanOnePackage', args: [widget.app.name])),
 | 
			
		||||
        Text(tr('appHasMoreThanOnePackage', args: [widget.app.finalName])),
 | 
			
		||||
        const SizedBox(height: 16),
 | 
			
		||||
        ...widget.app.apkUrls.map(
 | 
			
		||||
          (u) => RadioListTile<String>(
 | 
			
		||||
              title: Text(Uri.parse(u)
 | 
			
		||||
                  .pathSegments
 | 
			
		||||
                  .where((element) => element.isNotEmpty)
 | 
			
		||||
                  .last),
 | 
			
		||||
              value: u,
 | 
			
		||||
              groupValue: apkUrl,
 | 
			
		||||
              title: Text(u.key),
 | 
			
		||||
              value: u.value,
 | 
			
		||||
              groupValue: apkUrl!.value,
 | 
			
		||||
              onChanged: (String? val) {
 | 
			
		||||
                setState(() {
 | 
			
		||||
                  apkUrl = val;
 | 
			
		||||
                  apkUrl =
 | 
			
		||||
                      widget.app.apkUrls.where((e) => e.value == val).first;
 | 
			
		||||
                });
 | 
			
		||||
              }),
 | 
			
		||||
        ),
 | 
			
		||||
 
 | 
			
		||||
@@ -34,9 +34,9 @@ class UpdateNotification extends ObtainiumNotification {
 | 
			
		||||
    message = updates.isEmpty
 | 
			
		||||
        ? tr('noNewUpdates')
 | 
			
		||||
        : updates.length == 1
 | 
			
		||||
            ? tr('xHasAnUpdate', args: [updates[0].name])
 | 
			
		||||
            ? tr('xHasAnUpdate', args: [updates[0].finalName])
 | 
			
		||||
            : plural('xAndNMoreUpdatesAvailable', updates.length - 1,
 | 
			
		||||
                args: [updates[0].name, (updates.length - 1).toString()]);
 | 
			
		||||
                args: [updates[0].finalName, (updates.length - 1).toString()]);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -46,9 +46,9 @@ class SilentUpdateNotification extends ObtainiumNotification {
 | 
			
		||||
            tr('appsUpdatedNotifDescription'), Importance.defaultImportance) {
 | 
			
		||||
    message = updates.length == 1
 | 
			
		||||
        ? tr('xWasUpdatedToY',
 | 
			
		||||
            args: [updates[0].name, updates[0].latestVersion])
 | 
			
		||||
            args: [updates[0].finalName, updates[0].latestVersion])
 | 
			
		||||
        : plural('xAndNMoreUpdatesInstalled', updates.length - 1,
 | 
			
		||||
            args: [updates[0].name, (updates.length - 1).toString()]);
 | 
			
		||||
            args: [updates[0].finalName, (updates.length - 1).toString()]);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -7,6 +7,8 @@ import 'package:flutter/material.dart';
 | 
			
		||||
import 'package:fluttertoast/fluttertoast.dart';
 | 
			
		||||
import 'package:obtainium/app_sources/github.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:shared_preferences/shared_preferences.dart';
 | 
			
		||||
 | 
			
		||||
@@ -139,6 +141,15 @@ class SettingsProvider with ChangeNotifier {
 | 
			
		||||
    notifyListeners();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  bool get groupByCategory {
 | 
			
		||||
    return prefs?.getBool('groupByCategory') ?? false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  set groupByCategory(bool show) {
 | 
			
		||||
    prefs?.setBool('groupByCategory', show);
 | 
			
		||||
    notifyListeners();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  String? getSettingString(String settingId) {
 | 
			
		||||
    return prefs?.getString(settingId);
 | 
			
		||||
  }
 | 
			
		||||
@@ -151,7 +162,23 @@ class SettingsProvider with ChangeNotifier {
 | 
			
		||||
  Map<String, int> get 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
 | 
			
		||||
          .getAppValues()
 | 
			
		||||
          .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));
 | 
			
		||||
    notifyListeners();
 | 
			
		||||
  }
 | 
			
		||||
@@ -178,4 +205,15 @@ class SettingsProvider with ChangeNotifier {
 | 
			
		||||
 | 
			
		||||
  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();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,7 @@
 | 
			
		||||
 | 
			
		||||
import 'dart:convert';
 | 
			
		||||
 | 
			
		||||
import 'package:device_info_plus/device_info_plus.dart';
 | 
			
		||||
import 'package:easy_localization/easy_localization.dart';
 | 
			
		||||
import 'package:html/dom.dart';
 | 
			
		||||
import 'package:http/http.dart';
 | 
			
		||||
@@ -15,9 +16,12 @@ import 'package:obtainium/app_sources/gitlab.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/neutroncode.dart';
 | 
			
		||||
import 'package:obtainium/app_sources/signal.dart';
 | 
			
		||||
import 'package:obtainium/app_sources/sourceforge.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/custom_errors.dart';
 | 
			
		||||
import 'package:obtainium/mass_app_sources/githubstars.dart';
 | 
			
		||||
@@ -31,11 +35,13 @@ class AppNames {
 | 
			
		||||
 | 
			
		||||
class APKDetails {
 | 
			
		||||
  late String version;
 | 
			
		||||
  late List<String> apkUrls;
 | 
			
		||||
  late List<MapEntry<String, String>> apkUrls;
 | 
			
		||||
  late AppNames names;
 | 
			
		||||
  late DateTime? releaseDate;
 | 
			
		||||
  late String? changeLog;
 | 
			
		||||
 | 
			
		||||
  APKDetails(this.version, this.apkUrls, this.names, {this.releaseDate});
 | 
			
		||||
  APKDetails(this.version, this.apkUrls, this.names,
 | 
			
		||||
      {this.releaseDate, this.changeLog});
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class App {
 | 
			
		||||
@@ -45,13 +51,14 @@ class App {
 | 
			
		||||
  late String name;
 | 
			
		||||
  String? installedVersion;
 | 
			
		||||
  late String latestVersion;
 | 
			
		||||
  List<String> apkUrls = [];
 | 
			
		||||
  List<MapEntry<String, String>> apkUrls = [];
 | 
			
		||||
  late int preferredApkIndex;
 | 
			
		||||
  late Map<String, dynamic> additionalSettings;
 | 
			
		||||
  late DateTime? lastUpdateCheck;
 | 
			
		||||
  bool pinned = false;
 | 
			
		||||
  List<String> categories;
 | 
			
		||||
  late DateTime? releaseDate;
 | 
			
		||||
  late String? changeLog;
 | 
			
		||||
  App(
 | 
			
		||||
      this.id,
 | 
			
		||||
      this.url,
 | 
			
		||||
@@ -65,13 +72,39 @@ class App {
 | 
			
		||||
      this.lastUpdateCheck,
 | 
			
		||||
      this.pinned,
 | 
			
		||||
      {this.categories = const [],
 | 
			
		||||
      this.releaseDate});
 | 
			
		||||
      this.releaseDate,
 | 
			
		||||
      this.changeLog});
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String toString() {
 | 
			
		||||
    return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALSETTINGS: ${additionalSettings.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned';
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  String? get overrideName =>
 | 
			
		||||
      additionalSettings['appName']?.toString().trim().isNotEmpty == true
 | 
			
		||||
          ? additionalSettings['appName']
 | 
			
		||||
          : null;
 | 
			
		||||
 | 
			
		||||
  String get finalName {
 | 
			
		||||
    return overrideName ?? name;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  App deepCopy() => App(
 | 
			
		||||
      id,
 | 
			
		||||
      url,
 | 
			
		||||
      author,
 | 
			
		||||
      name,
 | 
			
		||||
      installedVersion,
 | 
			
		||||
      latestVersion,
 | 
			
		||||
      apkUrls,
 | 
			
		||||
      preferredApkIndex,
 | 
			
		||||
      Map.from(additionalSettings),
 | 
			
		||||
      lastUpdateCheck,
 | 
			
		||||
      pinned,
 | 
			
		||||
      categories: categories,
 | 
			
		||||
      changeLog: changeLog,
 | 
			
		||||
      releaseDate: releaseDate);
 | 
			
		||||
 | 
			
		||||
  factory App.fromJson(Map<String, dynamic> json) {
 | 
			
		||||
    var source = SourceProvider().getSource(json['url']);
 | 
			
		||||
    var formItems = source.combinedAppSpecificSettingFormItems
 | 
			
		||||
@@ -100,6 +133,20 @@ class App {
 | 
			
		||||
      additionalSettings['noVersionDetection'] =
 | 
			
		||||
          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
 | 
			
		||||
    for (var item in formItems) {
 | 
			
		||||
      if (additionalSettings[item.key] != null) {
 | 
			
		||||
@@ -113,35 +160,51 @@ class App {
 | 
			
		||||
    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();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // Arch based APK filter option should be disabled if it previously did not exist
 | 
			
		||||
    if (json['additionalSettings'] != null &&
 | 
			
		||||
        jsonDecode(json['additionalSettings'])['autoApkFilterByArch'] == null) {
 | 
			
		||||
      additionalSettings['autoApkFilterByArch'] = false;
 | 
			
		||||
    }
 | 
			
		||||
    return App(
 | 
			
		||||
      json['id'] as String,
 | 
			
		||||
      json['url'] as String,
 | 
			
		||||
      json['author'] as String,
 | 
			
		||||
      json['name'] as String,
 | 
			
		||||
      json['installedVersion'] == null
 | 
			
		||||
          ? null
 | 
			
		||||
          : json['installedVersion'] as String,
 | 
			
		||||
      json['latestVersion'] as String,
 | 
			
		||||
      json['apkUrls'] == null
 | 
			
		||||
          ? []
 | 
			
		||||
          : List<String>.from(jsonDecode(json['apkUrls'])),
 | 
			
		||||
      preferredApkIndex,
 | 
			
		||||
      additionalSettings,
 | 
			
		||||
      json['lastUpdateCheck'] == null
 | 
			
		||||
          ? null
 | 
			
		||||
          : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']),
 | 
			
		||||
      json['pinned'] ?? false,
 | 
			
		||||
      categories: json['categories'] != null
 | 
			
		||||
          ? (json['categories'] as List<dynamic>)
 | 
			
		||||
              .map((e) => e.toString())
 | 
			
		||||
              .toList()
 | 
			
		||||
          : json['category'] != null
 | 
			
		||||
              ? [json['category'] as String]
 | 
			
		||||
              : [],
 | 
			
		||||
      releaseDate: json['releaseDate'] == null
 | 
			
		||||
          ? null
 | 
			
		||||
          : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']),
 | 
			
		||||
    );
 | 
			
		||||
        json['id'] as String,
 | 
			
		||||
        json['url'] as String,
 | 
			
		||||
        json['author'] as String,
 | 
			
		||||
        json['name'] as String,
 | 
			
		||||
        json['installedVersion'] == null
 | 
			
		||||
            ? null
 | 
			
		||||
            : json['installedVersion'] as String,
 | 
			
		||||
        json['latestVersion'] as String,
 | 
			
		||||
        apkUrls,
 | 
			
		||||
        preferredApkIndex,
 | 
			
		||||
        additionalSettings,
 | 
			
		||||
        json['lastUpdateCheck'] == null
 | 
			
		||||
            ? null
 | 
			
		||||
            : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']),
 | 
			
		||||
        json['pinned'] ?? false,
 | 
			
		||||
        categories: json['categories'] != null
 | 
			
		||||
            ? (json['categories'] as List<dynamic>)
 | 
			
		||||
                .map((e) => e.toString())
 | 
			
		||||
                .toList()
 | 
			
		||||
            : json['category'] != null
 | 
			
		||||
                ? [json['category'] as String]
 | 
			
		||||
                : [],
 | 
			
		||||
        releaseDate: json['releaseDate'] == null
 | 
			
		||||
            ? null
 | 
			
		||||
            : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']),
 | 
			
		||||
        changeLog:
 | 
			
		||||
            json['changeLog'] == null ? null : json['changeLog'] as String);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  Map<String, dynamic> toJson() => {
 | 
			
		||||
@@ -151,13 +214,14 @@ class App {
 | 
			
		||||
        'name': name,
 | 
			
		||||
        'installedVersion': installedVersion,
 | 
			
		||||
        'latestVersion': latestVersion,
 | 
			
		||||
        'apkUrls': jsonEncode(apkUrls),
 | 
			
		||||
        'apkUrls': jsonEncode(apkUrls.map((e) => [e.key, e.value]).toList()),
 | 
			
		||||
        'preferredApkIndex': preferredApkIndex,
 | 
			
		||||
        'additionalSettings': jsonEncode(additionalSettings),
 | 
			
		||||
        'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch,
 | 
			
		||||
        'pinned': pinned,
 | 
			
		||||
        'categories': categories,
 | 
			
		||||
        'releaseDate': releaseDate?.microsecondsSinceEpoch
 | 
			
		||||
        'releaseDate': releaseDate?.microsecondsSinceEpoch,
 | 
			
		||||
        'changeLog': changeLog
 | 
			
		||||
      };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -202,10 +266,16 @@ Map<String, dynamic> getDefaultValuesFromFormItems(
 | 
			
		||||
      .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 {
 | 
			
		||||
  String? host;
 | 
			
		||||
  late String name;
 | 
			
		||||
  bool enforceTrackOnly = false;
 | 
			
		||||
  bool changeLogIfAnyIsMarkDown = true;
 | 
			
		||||
 | 
			
		||||
  AppSource() {
 | 
			
		||||
    name = runtimeType.toString();
 | 
			
		||||
@@ -234,11 +304,16 @@ class AppSource {
 | 
			
		||||
      )
 | 
			
		||||
    ],
 | 
			
		||||
    [
 | 
			
		||||
      GeneratedFormSwitch('releaseDateAsVersion',
 | 
			
		||||
          label: tr('useReleaseDateAsVersion'))
 | 
			
		||||
    ],
 | 
			
		||||
    [
 | 
			
		||||
      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',
 | 
			
		||||
@@ -249,7 +324,12 @@ class AppSource {
 | 
			
		||||
              return regExValidator(value);
 | 
			
		||||
            }
 | 
			
		||||
          ])
 | 
			
		||||
    ]
 | 
			
		||||
    ],
 | 
			
		||||
    [
 | 
			
		||||
      GeneratedFormSwitch('autoApkFilterByArch',
 | 
			
		||||
          label: tr('autoApkFilterByArch'), defaultValue: true)
 | 
			
		||||
    ],
 | 
			
		||||
    [GeneratedFormTextField('appName', label: tr('appName'), required: false)]
 | 
			
		||||
  ];
 | 
			
		||||
 | 
			
		||||
  // Previous 2 variables combined into one at runtime for convenient usage
 | 
			
		||||
@@ -313,12 +393,16 @@ class SourceProvider {
 | 
			
		||||
    Codeberg(),
 | 
			
		||||
    FDroid(),
 | 
			
		||||
    IzzyOnDroid(),
 | 
			
		||||
    Mullvad(),
 | 
			
		||||
    Signal(),
 | 
			
		||||
    FDroidRepo(),
 | 
			
		||||
    SourceForge(),
 | 
			
		||||
    APKMirror(),
 | 
			
		||||
    FDroidRepo(),
 | 
			
		||||
    Mullvad(),
 | 
			
		||||
    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
 | 
			
		||||
  ];
 | 
			
		||||
 | 
			
		||||
@@ -329,7 +413,7 @@ class SourceProvider {
 | 
			
		||||
    url = preStandardizeUrl(url);
 | 
			
		||||
    AppSource? source;
 | 
			
		||||
    for (var s in sources.where((element) => element.host != null)) {
 | 
			
		||||
      if (url.contains('://${s.host}')) {
 | 
			
		||||
      if (RegExp('://(.+\\.)?${s.host}').hasMatch(url)) {
 | 
			
		||||
        source = s;
 | 
			
		||||
        break;
 | 
			
		||||
      }
 | 
			
		||||
@@ -367,45 +451,50 @@ class SourceProvider {
 | 
			
		||||
      (standardUrl + additionalSettings.toString()).hashCode.toString();
 | 
			
		||||
 | 
			
		||||
  bool isTempId(App app) {
 | 
			
		||||
    return app.id == generateTempID(app.url, app.additionalSettings);
 | 
			
		||||
    // return app.id == generateTempID(app.url, app.additionalSettings);
 | 
			
		||||
    return RegExp('^[0-9]+\$').hasMatch(app.id);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  Future<App> getApp(
 | 
			
		||||
      AppSource source, String url, Map<String, dynamic> additionalSettings,
 | 
			
		||||
      {App? currentApp,
 | 
			
		||||
      bool trackOnlyOverride = false,
 | 
			
		||||
      bool noVersionDetectionOverride = false,
 | 
			
		||||
      bool releaseDateAsVersionOverride = false}) async {
 | 
			
		||||
      {App? currentApp, bool trackOnlyOverride = false}) async {
 | 
			
		||||
    if (trackOnlyOverride || source.enforceTrackOnly) {
 | 
			
		||||
      additionalSettings['trackOnly'] = true;
 | 
			
		||||
    }
 | 
			
		||||
    if (releaseDateAsVersionOverride) {
 | 
			
		||||
      additionalSettings['releaseDateAsVersion'] = true;
 | 
			
		||||
      noVersionDetectionOverride =
 | 
			
		||||
          true; // Rel. date as version means no ver. det.
 | 
			
		||||
    }
 | 
			
		||||
    if (noVersionDetectionOverride) {
 | 
			
		||||
      additionalSettings['noVersionDetection'] = true;
 | 
			
		||||
    }
 | 
			
		||||
    var trackOnly = additionalSettings['trackOnly'] == true;
 | 
			
		||||
    String standardUrl = source.standardizeURL(preStandardizeUrl(url));
 | 
			
		||||
    APKDetails apk =
 | 
			
		||||
        await source.getLatestAPKDetails(standardUrl, additionalSettings);
 | 
			
		||||
    if (additionalSettings['releaseDateAsVersion'] == true &&
 | 
			
		||||
    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)).toList();
 | 
			
		||||
          apk.apkUrls.where((element) => reg.hasMatch(element.key)).toList();
 | 
			
		||||
    }
 | 
			
		||||
    if (apk.apkUrls.isEmpty && !trackOnly) {
 | 
			
		||||
      throw NoAPKError();
 | 
			
		||||
    }
 | 
			
		||||
    if (apk.apkUrls.length > 1 &&
 | 
			
		||||
        additionalSettings['autoApkFilterByArch'] == true) {
 | 
			
		||||
      var abis = (await DeviceInfoPlugin().androidInfo).supportedAbis;
 | 
			
		||||
      for (var abi in abis) {
 | 
			
		||||
        var urls2 = apk.apkUrls
 | 
			
		||||
            .where((element) => RegExp('.*$abi.*').hasMatch(element.key))
 | 
			
		||||
            .toList();
 | 
			
		||||
        if (urls2.isNotEmpty && urls2.length < apk.apkUrls.length) {
 | 
			
		||||
          apk.apkUrls = urls2;
 | 
			
		||||
          break;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    String apkVersion = apk.version.replaceAll('/', '-');
 | 
			
		||||
    var name = currentApp?.name.trim() ??
 | 
			
		||||
        apk.names.name[0].toUpperCase() + apk.names.name.substring(1);
 | 
			
		||||
    var name = currentApp != null ? currentApp.name.trim() : '';
 | 
			
		||||
    name = name.isNotEmpty
 | 
			
		||||
        ? name
 | 
			
		||||
        : apk.names.name[0].toUpperCase() + apk.names.name.substring(1);
 | 
			
		||||
    return App(
 | 
			
		||||
        currentApp?.id ??
 | 
			
		||||
            source.tryInferringAppId(standardUrl,
 | 
			
		||||
@@ -413,9 +502,7 @@ class SourceProvider {
 | 
			
		||||
            generateTempID(standardUrl, additionalSettings),
 | 
			
		||||
        standardUrl,
 | 
			
		||||
        apk.names.author[0].toUpperCase() + apk.names.author.substring(1),
 | 
			
		||||
        name.trim().isNotEmpty
 | 
			
		||||
            ? name
 | 
			
		||||
            : apk.names.name[0].toUpperCase() + apk.names.name.substring(1),
 | 
			
		||||
        name,
 | 
			
		||||
        currentApp?.installedVersion,
 | 
			
		||||
        apkVersion,
 | 
			
		||||
        apk.apkUrls,
 | 
			
		||||
@@ -424,7 +511,8 @@ class SourceProvider {
 | 
			
		||||
        DateTime.now(),
 | 
			
		||||
        currentApp?.pinned ?? false,
 | 
			
		||||
        categories: currentApp?.categories ?? const [],
 | 
			
		||||
        releaseDate: apk.releaseDate);
 | 
			
		||||
        releaseDate: apk.releaseDate,
 | 
			
		||||
        changeLog: apk.changeLog);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Returns errors in [results, errors] instead of throwing them
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										250
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						@@ -5,18 +5,18 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: android_alarm_manager_plus
 | 
			
		||||
      sha256: "8647cc5f9339f3955a2bd9ec40e0f10c3a80049f31f80b3ffdd87e07bb73fce2"
 | 
			
		||||
      sha256: f6d0347734fa2ea716349a5a3e16ffdc1800ca64e5640112896d128c6815c178
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.1"
 | 
			
		||||
    version: "2.1.2"
 | 
			
		||||
  android_intent_plus:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: android_intent_plus
 | 
			
		||||
      sha256: "54810cb33945c2c10742cd746ea994822c115e9dbe189919bc63cb436e45a6af"
 | 
			
		||||
      sha256: "6bcdcd20461ac7a0c785f6298cdda96ad275d5bcbc1ecf28829cbe03ec6690be"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.1.6"
 | 
			
		||||
    version: "3.1.7"
 | 
			
		||||
  animations:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
@@ -25,14 +25,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.7"
 | 
			
		||||
  archive:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: archive
 | 
			
		||||
      sha256: d6347d54a2d8028e0437e3c099f66fdb8ae02c4720c1e7534c9f24c10351f85d
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.3.6"
 | 
			
		||||
  args:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -65,22 +57,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "1.2.1"
 | 
			
		||||
  checked_yaml:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: checked_yaml
 | 
			
		||||
      sha256: "3d1505d91afa809d177efd4eed5bb0eb65805097a1463abdd2add076effae311"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.2"
 | 
			
		||||
  cli_util:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: cli_util
 | 
			
		||||
      sha256: "66f86e916d285c1a93d3b79587d94bd71984a66aac4ff74e524cfa7877f1395c"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.3.5"
 | 
			
		||||
  clock:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -97,14 +73,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "1.17.0"
 | 
			
		||||
  convert:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: convert
 | 
			
		||||
      sha256: "0f08b14755d163f6e2134cb58222dd25ea2a2ee8a195e53983d57c075324d592"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.1.1"
 | 
			
		||||
  cross_file:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -149,10 +117,10 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: device_info_plus
 | 
			
		||||
      sha256: "1d6e5a61674ba3a68fb048a7c7b4ff4bebfed8d7379dbe8f2b718231be9a7c95"
 | 
			
		||||
      sha256: "435383ca05f212760b0a70426b5a90354fe6bd65992b3a5e27ab6ede74c02f5c"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "8.1.0"
 | 
			
		||||
    version: "8.2.0"
 | 
			
		||||
  device_info_plus_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -165,10 +133,10 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: dynamic_color
 | 
			
		||||
      sha256: c4a508284b14ec4dda5adba2c28b2cdd34fbae1afead7e8c52cad87d51c5405b
 | 
			
		||||
      sha256: bbebb1b7ebed819e0ec83d4abdc2a8482d934f6a85289ffc1c6acf7589fa2aad
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "1.6.2"
 | 
			
		||||
    version: "1.6.3"
 | 
			
		||||
  easy_localization:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
@@ -213,10 +181,10 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: file_picker
 | 
			
		||||
      sha256: d090ae03df98b0247b82e5928f44d1b959867049d18d73635e2e0bc3f49542b9
 | 
			
		||||
      sha256: dd328189f2f4ccea042bb5b382d5e981691cc74b5a3429b9317bff2b19704489
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "5.2.5"
 | 
			
		||||
    version: "5.2.8"
 | 
			
		||||
  flutter:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description: flutter
 | 
			
		||||
@@ -230,14 +198,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.2.2"
 | 
			
		||||
  flutter_launcher_icons:
 | 
			
		||||
    dependency: "direct dev"
 | 
			
		||||
    description:
 | 
			
		||||
      name: flutter_launcher_icons
 | 
			
		||||
      sha256: ce0e501cfc258907842238e4ca605e74b7fd1cdf04b3b43e86c43f3e40a1592c
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.11.0"
 | 
			
		||||
  flutter_lints:
 | 
			
		||||
    dependency: "direct dev"
 | 
			
		||||
    description:
 | 
			
		||||
@@ -275,14 +235,22 @@ packages:
 | 
			
		||||
    description: flutter
 | 
			
		||||
    source: sdk
 | 
			
		||||
    version: "0.0.0"
 | 
			
		||||
  flutter_markdown:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: flutter_markdown
 | 
			
		||||
      sha256: "7b25c10de1fea883f3c4f9b8389506b54053cd00807beab69fd65c8653a2711f"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.6.14"
 | 
			
		||||
  flutter_plugin_android_lifecycle:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: flutter_plugin_android_lifecycle
 | 
			
		||||
      sha256: "60fc7b78455b94e6de2333d2f95196d32cf5c22f4b0b0520a628804cb463503b"
 | 
			
		||||
      sha256: c224ac897bed083dabf11f238dd11a239809b446740be0c2044608c50029ffdf
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.7"
 | 
			
		||||
    version: "2.0.9"
 | 
			
		||||
  flutter_test:
 | 
			
		||||
    dependency: "direct dev"
 | 
			
		||||
    description: flutter
 | 
			
		||||
@@ -297,18 +265,18 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: fluttertoast
 | 
			
		||||
      sha256: "774fa28b07f3a82c93596bc137be33189fec578ed3447a93a5a11c93435de394"
 | 
			
		||||
      sha256: "2f9c4d3f4836421f7067a28f8939814597b27614e021da9d63e5d3fb6e212d25"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "8.1.3"
 | 
			
		||||
    version: "8.2.1"
 | 
			
		||||
  html:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: html
 | 
			
		||||
      sha256: d9793e10dbe0e6c364f4c59bf3e01fb33a9b2a674bc7a1081693dba0614b6269
 | 
			
		||||
      sha256: "79d498e6d6761925a34ee5ea8fa6dfef38607781d2fa91e37523474282af55cb"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.15.1"
 | 
			
		||||
    version: "0.15.2"
 | 
			
		||||
  http:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
@@ -325,14 +293,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "4.0.2"
 | 
			
		||||
  image:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: image
 | 
			
		||||
      sha256: "8e9d133755c3e84c73288363e6343157c383a0c6c56fc51afcc5d4d7180306d6"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.3.0"
 | 
			
		||||
  install_plugin_v2:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
@@ -365,14 +325,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.6.5"
 | 
			
		||||
  json_annotation:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: json_annotation
 | 
			
		||||
      sha256: c33da08e136c3df0190bd5bbe51ae1df4a7d96e7954d1d7249fea2968a72d317
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "4.8.0"
 | 
			
		||||
  lints:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -381,6 +333,14 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.1"
 | 
			
		||||
  markdown:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: markdown
 | 
			
		||||
      sha256: d95a9d12954aafc97f984ca29baaa7690ed4d9ec4140a23ad40580bcdb6c87f5
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "7.0.2"
 | 
			
		||||
  matcher:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -449,50 +409,50 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider
 | 
			
		||||
      sha256: dcea5feb97d8abf90cab9e9030b497fb7c3cbf26b7a1fe9e3ef7dcb0a1ddec95
 | 
			
		||||
      sha256: c7edf82217d4b2952b2129a61d3ad60f1075b9299e629e149a8d2e39c2e6aad4
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.12"
 | 
			
		||||
    version: "2.0.14"
 | 
			
		||||
  path_provider_android:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider_android
 | 
			
		||||
      sha256: a776c088d671b27f6e3aa8881d64b87b3e80201c64e8869b811325de7a76c15e
 | 
			
		||||
      sha256: "019f18c9c10ae370b08dce1f3e3b73bc9f58e7f087bb5e921f06529438ac0ae7"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.22"
 | 
			
		||||
    version: "2.0.24"
 | 
			
		||||
  path_provider_foundation:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider_foundation
 | 
			
		||||
      sha256: "62a68e7e1c6c459f9289859e2fae58290c981ce21d1697faf54910fe1faa4c74"
 | 
			
		||||
      sha256: ad4c4d011830462633f03eb34445a45345673dfd4faf1ab0b4735fbd93b19183
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.1"
 | 
			
		||||
    version: "2.2.2"
 | 
			
		||||
  path_provider_linux:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider_linux
 | 
			
		||||
      sha256: "2e32f1640f07caef0d3cb993680f181c79e54a3827b997d5ee221490d131fbd9"
 | 
			
		||||
      sha256: "2ae08f2216225427e64ad224a24354221c2c7907e448e6e0e8b57b1eb9f10ad1"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.8"
 | 
			
		||||
    version: "2.1.10"
 | 
			
		||||
  path_provider_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider_platform_interface
 | 
			
		||||
      sha256: f0abc8ebd7253741f05488b4813d936b4d07c6bae3e86148a09e342ee4b08e76
 | 
			
		||||
      sha256: "57585299a729335f1298b43245842678cb9f43a6310351b18fb577d6e33165ec"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.5"
 | 
			
		||||
    version: "2.0.6"
 | 
			
		||||
  path_provider_windows:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider_windows
 | 
			
		||||
      sha256: bcabbe399d4042b8ee687e17548d5d3f527255253b4a639f5f8d2094a9c2b45c
 | 
			
		||||
      sha256: f53720498d5a543f9607db4b0e997c4b5438884de25b0f73098cc2671a51b130
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.3"
 | 
			
		||||
    version: "2.1.5"
 | 
			
		||||
  permission_handler:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
@@ -513,10 +473,10 @@ packages:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: permission_handler_apple
 | 
			
		||||
      sha256: "9c370ef6a18b1c4b2f7f35944d644a56aa23576f23abee654cf73968de93f163"
 | 
			
		||||
      sha256: ee96ac32f5a8e6f80756e25b25b9f8e535816c8e6665a96b6d70681f8c4f7e85
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "9.0.7"
 | 
			
		||||
    version: "9.0.8"
 | 
			
		||||
  permission_handler_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -553,18 +513,10 @@ packages:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: plugin_platform_interface
 | 
			
		||||
      sha256: dbf0f707c78beedc9200146ad3cb0ab4d5da13c246336987be6940f026500d3a
 | 
			
		||||
      sha256: "6a2128648c854906c53fa8e33986fc0247a1116122f9534dd20e3ab9e16a32bc"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.3"
 | 
			
		||||
  pointycastle:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: pointycastle
 | 
			
		||||
      sha256: db7306cf0249f838d1a24af52b5a5887c5bf7f31d8bb4e827d071dc0939ad346
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.6.2"
 | 
			
		||||
    version: "2.1.4"
 | 
			
		||||
  process:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -585,74 +537,74 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: share_plus
 | 
			
		||||
      sha256: "8c6892037b1824e2d7e8f59d54b3105932899008642e6372e5079c6939b4b625"
 | 
			
		||||
      sha256: "692261968a494e47323dcc8bc66d8d52e81bc27cb4b808e4e8d7e8079d4cc01a"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "6.3.1"
 | 
			
		||||
    version: "6.3.2"
 | 
			
		||||
  share_plus_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: share_plus_platform_interface
 | 
			
		||||
      sha256: "82ddd4ab9260c295e6e39612d4ff00390b9a7a21f1bb1da771e2f232d80ab8a1"
 | 
			
		||||
      sha256: "0c6e61471bd71b04a138b8b588fa388e66d8b005e6f2deda63371c5c505a0981"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.2.0"
 | 
			
		||||
    version: "3.2.1"
 | 
			
		||||
  shared_preferences:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences
 | 
			
		||||
      sha256: "5949029e70abe87f75cfe59d17bf5c397619c4b74a099b10116baeb34786fad9"
 | 
			
		||||
      sha256: "858aaa72d8f61637d64e776aca82e1c67e6d9ee07979123c5d17115031c1b13b"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.17"
 | 
			
		||||
    version: "2.1.0"
 | 
			
		||||
  shared_preferences_android:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_android
 | 
			
		||||
      sha256: "955e9736a12ba776bdd261cf030232b30eadfcd9c79b32a3250dd4a494e8c8f7"
 | 
			
		||||
      sha256: "8304d8a1f7d21a429f91dee552792249362b68a331ac5c3c1caf370f658873f6"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.15"
 | 
			
		||||
    version: "2.1.0"
 | 
			
		||||
  shared_preferences_foundation:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_foundation
 | 
			
		||||
      sha256: "2b55c18636a4edc529fa5cd44c03d3f3100c00513f518c5127c951978efcccd0"
 | 
			
		||||
      sha256: "0c1c16c56c9708aa9c361541a6f0e5cc6fc12a3232d866a687a7b7db30032b07"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.3"
 | 
			
		||||
    version: "2.2.1"
 | 
			
		||||
  shared_preferences_linux:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_linux
 | 
			
		||||
      sha256: f8ea038aa6da37090093974ebdcf4397010605fd2ff65c37a66f9d28394cb874
 | 
			
		||||
      sha256: "9d387433ca65717bbf1be88f4d5bb18f10508917a8fa2fb02e0fd0d7479a9afa"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.3"
 | 
			
		||||
    version: "2.2.0"
 | 
			
		||||
  shared_preferences_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_platform_interface
 | 
			
		||||
      sha256: da9431745ede5ece47bc26d5d73a9d3c6936ef6945c101a5aca46f62e52c1cf3
 | 
			
		||||
      sha256: fb5cf25c0235df2d0640ac1b1174f6466bd311f621574997ac59018a6664548d
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.0"
 | 
			
		||||
    version: "2.2.0"
 | 
			
		||||
  shared_preferences_web:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_web
 | 
			
		||||
      sha256: a4b5bc37fe1b368bbc81f953197d55e12f49d0296e7e412dfe2d2d77d6929958
 | 
			
		||||
      sha256: "74083203a8eae241e0de4a0d597dbedab3b8fef5563f33cf3c12d7e93c655ca5"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.4"
 | 
			
		||||
    version: "2.1.0"
 | 
			
		||||
  shared_preferences_windows:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_windows
 | 
			
		||||
      sha256: "5eaf05ae77658d3521d0e993ede1af962d4b326cd2153d312df716dc250f00c9"
 | 
			
		||||
      sha256: "5e588e2efef56916a3b229c3bfe81e6a525665a454519ca51dbcc4236a274173"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.3"
 | 
			
		||||
    version: "2.2.0"
 | 
			
		||||
  sky_engine:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description: flutter
 | 
			
		||||
@@ -670,18 +622,18 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: sqflite
 | 
			
		||||
      sha256: "78324387dc81df14f78df06019175a86a2ee0437624166c382e145d0a7fd9a4f"
 | 
			
		||||
      sha256: "500d6fec583d2c021f2d25a056d96654f910662c64f836cd2063167b8f1fa758"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.2.4+1"
 | 
			
		||||
    version: "2.2.6"
 | 
			
		||||
  sqflite_common:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: sqflite_common
 | 
			
		||||
      sha256: bfd6973aaeeb93475bc0d875ac9aefddf7965ef22ce09790eb963992ffc5183f
 | 
			
		||||
      sha256: "963dad8c4aa2f814ce7d2d5b1da2f36f31bd1a439d8f27e3dc189bb9d26bc684"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.4.2+2"
 | 
			
		||||
    version: "2.4.3"
 | 
			
		||||
  stack_trace:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -750,66 +702,66 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher
 | 
			
		||||
      sha256: e8f2efc804810c0f2f5b485f49e7942179f56eabcfe81dce3387fec4bb55876b
 | 
			
		||||
      sha256: "75f2846facd11168d007529d6cd8fcb2b750186bea046af9711f10b907e1587e"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "6.1.9"
 | 
			
		||||
    version: "6.1.10"
 | 
			
		||||
  url_launcher_android:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_android
 | 
			
		||||
      sha256: "3e2f6dfd2c7d9cd123296cab8ef66cfc2c1a13f5845f42c7a0f365690a8a7dd1"
 | 
			
		||||
      sha256: a52628068d282d01a07cd86e6ba99e497aa45ce8c91159015b2416907d78e411
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "6.0.23"
 | 
			
		||||
    version: "6.0.27"
 | 
			
		||||
  url_launcher_ios:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_ios
 | 
			
		||||
      sha256: "0a5af0aefdd8cf820dd739886efb1637f1f24489900204f50984634c07a54815"
 | 
			
		||||
      sha256: "9af7ea73259886b92199f9e42c116072f05ff9bea2dcb339ab935dfc957392c2"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "6.1.0"
 | 
			
		||||
    version: "6.1.4"
 | 
			
		||||
  url_launcher_linux:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_linux
 | 
			
		||||
      sha256: "318c42cba924e18180c029be69caf0a1a710191b9ec49bb42b5998fdcccee3cc"
 | 
			
		||||
      sha256: "206fb8334a700ef7754d6a9ed119e7349bc830448098f21a69bf1b4ed038cabc"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.0.2"
 | 
			
		||||
    version: "3.0.4"
 | 
			
		||||
  url_launcher_macos:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_macos
 | 
			
		||||
      sha256: "41988b55570df53b3dd2a7fc90c76756a963de6a8c5f8e113330cb35992e2094"
 | 
			
		||||
      sha256: "91ee3e75ea9dadf38036200c5d3743518f4a5eb77a8d13fda1ee5764373f185e"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.0.2"
 | 
			
		||||
    version: "3.0.5"
 | 
			
		||||
  url_launcher_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_platform_interface
 | 
			
		||||
      sha256: "4eae912628763eb48fc214522e58e942fd16ce195407dbf45638239523c759a6"
 | 
			
		||||
      sha256: "6c9ca697a5ae218ce56cece69d46128169a58aa8653c1b01d26fcd4aad8c4370"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.1"
 | 
			
		||||
    version: "2.1.2"
 | 
			
		||||
  url_launcher_web:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_web
 | 
			
		||||
      sha256: "44d79408ce9f07052095ef1f9a693c258d6373dc3944249374e30eff7219ccb0"
 | 
			
		||||
      sha256: "81fe91b6c4f84f222d186a9d23c73157dc4c8e1c71489c4d08be1ad3b228f1aa"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.14"
 | 
			
		||||
    version: "2.0.16"
 | 
			
		||||
  url_launcher_windows:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: url_launcher_windows
 | 
			
		||||
      sha256: b6217370f8eb1fd85c8890c539f5a639a01ab209a36db82c921ebeacefc7a615
 | 
			
		||||
      sha256: a83ba3607a507758669cfafb03f9de09bf6e6280c14d9b9cb18f013e406dcacd
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.0.3"
 | 
			
		||||
    version: "3.0.5"
 | 
			
		||||
  uuid:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -830,42 +782,42 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter
 | 
			
		||||
      sha256: f7ec234830f86d0ef2bd664e8460b0038b8c1a83ff076035cad74ac70273753c
 | 
			
		||||
      sha256: "47663d51a9061451aa3880a214ee9a65dcbb933b77bc44388e194279ab3ccaf6"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "4.0.2"
 | 
			
		||||
    version: "4.0.7"
 | 
			
		||||
  webview_flutter_android:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter_android
 | 
			
		||||
      sha256: da98c8cdaebea4cf89481853f37ca93ccc8d31fc386f5b3c928aea3b6e83268c
 | 
			
		||||
      sha256: "5906c9aa8c88ed372b2ad3c88c942790b4fb16f73fdd1c0647b4d747d9cf5b3f"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.3.0"
 | 
			
		||||
    version: "3.4.5"
 | 
			
		||||
  webview_flutter_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter_platform_interface
 | 
			
		||||
      sha256: "8b2262dda5d26eabc600a7282a8c16a9473a0c765526afb0ffc33eef912f7968"
 | 
			
		||||
      sha256: "6341f92977609be71391f4d4dcd64bfaa8ac657af1dfb2e231b5c1724e8c6c36"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.1"
 | 
			
		||||
    version: "2.2.0"
 | 
			
		||||
  webview_flutter_wkwebview:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter_wkwebview
 | 
			
		||||
      sha256: dcd9ad0ef0f608f399d7a54d0b289597385e59a89f04983a672b9348faddfd98
 | 
			
		||||
      sha256: "9a78d963cce191dd6a9df547301fc5c008bf3dae95a323ec281fff1284e0a037"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.1.0"
 | 
			
		||||
    version: "3.2.4"
 | 
			
		||||
  win32:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: win32
 | 
			
		||||
      sha256: c9ebe7ee4ab0c2194e65d3a07d8c54c5d00bb001b76081c4a04cdb8448b59e46
 | 
			
		||||
      sha256: a6f0236dbda0f63aa9a25ad1ff9a9d8a4eaaa5012da0dc59d21afdb1dc361ca4
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.1.3"
 | 
			
		||||
    version: "3.1.4"
 | 
			
		||||
  xdg_directories:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -882,14 +834,6 @@ packages:
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "6.2.2"
 | 
			
		||||
  yaml:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: yaml
 | 
			
		||||
      sha256: "23812a9b125b48d4007117254bca50abb6c712352927eece9e155207b1db2370"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.1.1"
 | 
			
		||||
sdks:
 | 
			
		||||
  dart: ">=2.18.2 <3.0.0"
 | 
			
		||||
  dart: ">=2.19.0 <3.0.0"
 | 
			
		||||
  flutter: ">=3.4.0-17.0.pre"
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										12
									
								
								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
 | 
			
		||||
# In Windows, build-name is used as the major, minor, and patch parts
 | 
			
		||||
# of the product and file versions while build-number is used as the build suffix.
 | 
			
		||||
version: 0.11.0+119 # When changing this, update the tag in main() accordingly
 | 
			
		||||
version: 0.11.30+152 # When changing this, update the tag in main() accordingly
 | 
			
		||||
 | 
			
		||||
environment:
 | 
			
		||||
  sdk: '>=2.18.2 <3.0.0'
 | 
			
		||||
@@ -59,12 +59,12 @@ dependencies:
 | 
			
		||||
  sqflite: ^2.2.0+3
 | 
			
		||||
  easy_localization: ^3.0.1
 | 
			
		||||
  android_intent_plus: ^3.1.5
 | 
			
		||||
  flutter_markdown: ^0.6.14
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
dev_dependencies:
 | 
			
		||||
  flutter_test:
 | 
			
		||||
    sdk: flutter
 | 
			
		||||
  flutter_launcher_icons: ^0.11.0
 | 
			
		||||
 | 
			
		||||
  # The "flutter_lints" package below contains a set of recommended lints to
 | 
			
		||||
  # encourage good coding practices. The lint set provided by the package is
 | 
			
		||||
@@ -73,12 +73,6 @@ dev_dependencies:
 | 
			
		||||
  # rules and activating additional ones.
 | 
			
		||||
  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
 | 
			
		||||
# following page: https://dart.dev/tools/pub/pubspec
 | 
			
		||||
 | 
			
		||||
@@ -97,6 +91,8 @@ flutter:
 | 
			
		||||
  
 | 
			
		||||
  assets:
 | 
			
		||||
    - assets/translations/
 | 
			
		||||
    - assets/graphics/
 | 
			
		||||
    - assets/ca/
 | 
			
		||||
 | 
			
		||||
  # An image asset can refer to one or more resolution-specific "variants", see
 | 
			
		||||
  # https://flutter.dev/assets-and-images/#resolution-aware
 | 
			
		||||
 
 | 
			
		||||