Compare commits
	
		
			119 Commits
		
	
	
		
			v0.11.6-be
			...
			v0.11.34-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 02da24aa75 | ||
|  | 3c6e66ce12 | ||
|  | 0213b542e3 | ||
|  | b0e8a4a297 | ||
|  | e72b33ebf2 | ||
|  | 283722319b | ||
|  | b406bb5c6a | ||
|  | de2b7fa7a1 | ||
|  | be61220af4 | ||
|  | 3e732a4317 | ||
|  | 9f2db4e4e7 | ||
|  | 78141998f4 | ||
|  | 934f237e34 | ||
|  | 1b2a9a39e3 | ||
|  | dc52fb6181 | ||
|  | 9e4ac397d8 | ||
|  | 0ec944eae9 | ||
|  | ad250c30e4 | ||
|  | 1090f15508 | ||
|  | 666941350e | ||
|  | eeadbce8b0 | ||
|  | ce8aeff342 | ||
|  | 0d8362a2ed | ||
|  | 3b28143a4e | ||
|  | 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 | 
| @@ -19,6 +19,9 @@ Currently supported App sources: | |||||||
| - Third Party F-Droid Repos | - Third Party F-Droid Repos | ||||||
|   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` |   - Any URLs ending with `/fdroid/<word>`, where `<word>` can be anything - most often `repo` | ||||||
| - [Steam](https://store.steampowered.com/mobile) | - [Steam](https://store.steampowered.com/mobile) | ||||||
|  | - [Telegram App](https://telegram.org) | ||||||
|  | - [VLC](https://www.videolan.org/vlc/download-android.html) | ||||||
|  | - [Neutron Code](https://neutroncode.com) | ||||||
| - "HTML" (Fallback) | - "HTML" (Fallback) | ||||||
|   - Any other URL that returns an HTML page with links to APK files (if multiple, the last file alphabetically is picked) |   - 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 | 
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "System folgen", |     "followSystem": "System folgen", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App sortieren nach", |     "appSortBy": "App sortieren nach", | ||||||
|     "authorName": "Autor/Name", |     "authorName": "Autor/Name", | ||||||
|     "nameAuthor": "Name/Autor", |     "nameAuthor": "Name/Autor", | ||||||
| @@ -207,6 +208,7 @@ | |||||||
|     "addCategory": "Kategorie hinzufügen", |     "addCategory": "Kategorie hinzufügen", | ||||||
|     "label": "Bezeichnung", |     "label": "Bezeichnung", | ||||||
|     "language": "Sprache", |     "language": "Sprache", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|     "storagePermissionDenied": "Speicherberechtigung verweigert", |     "storagePermissionDenied": "Speicherberechtigung verweigert", | ||||||
|     "selectedCategorizeWarning": "Dadurch werden alle bestehenden Kategorieeinstellungen für die ausgewählten Apps ersetzt.", |     "selectedCategorizeWarning": "Dadurch werden alle bestehenden Kategorieeinstellungen für die ausgewählten Apps ersetzt.", | ||||||
|     "filterAPKsByRegEx": "APKs nach regulärem Ausdruck filtern", |     "filterAPKsByRegEx": "APKs nach regulärem Ausdruck filtern", | ||||||
| @@ -220,9 +222,11 @@ | |||||||
|     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", |     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", | ||||||
|     "versionDetection": "Versionserkennung", |     "versionDetection": "Versionserkennung", | ||||||
|     "standardVersionDetection": "Standardversionserkennung", |     "standardVersionDetection": "Standardversionserkennung", | ||||||
|  |     "groupByCategory": "Nach Kategorie gruppieren", | ||||||
|  |     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "App entfernen?", |         "one": "App entfernen?", | ||||||
|         "other": "App entfernen?" |         "other": "Apps entfernen?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", |         "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut", | ||||||
| @@ -268,4 +272,4 @@ | |||||||
|         "one": "{} und 1 weitere Anwendung wurden aktualisiert.", |         "one": "{} und 1 weitere Anwendung wurden aktualisiert.", | ||||||
|         "other": "{} und {} weitere Anwendungen wurden aktualisiert." |         "other": "{} und {} weitere Anwendungen wurden aktualisiert." | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Follow System", |     "followSystem": "Follow System", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App Sort By", |     "appSortBy": "App Sort By", | ||||||
|     "authorName": "Author/Name", |     "authorName": "Author/Name", | ||||||
|     "nameAuthor": "Name/Author", |     "nameAuthor": "Name/Author", | ||||||
| @@ -207,6 +208,7 @@ | |||||||
|     "addCategory": "Add Category", |     "addCategory": "Add Category", | ||||||
|     "label": "Label", |     "label": "Label", | ||||||
|     "language": "Language", |     "language": "Language", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|     "storagePermissionDenied": "Storage permission denied", |     "storagePermissionDenied": "Storage permission denied", | ||||||
|     "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.", |     "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.", | ||||||
|     "filterAPKsByRegEx": "Filter APKs by Regular Expression", |     "filterAPKsByRegEx": "Filter APKs by Regular Expression", | ||||||
| @@ -220,6 +222,8 @@ | |||||||
|     "importFromURLsInFile": "Import from URLs in File (like OPML)", |     "importFromURLsInFile": "Import from URLs in File (like OPML)", | ||||||
|     "versionDetection": "Version Detection", |     "versionDetection": "Version Detection", | ||||||
|     "standardVersionDetection": "Standard version detection", |     "standardVersionDetection": "Standard version detection", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Remove App?", |         "one": "Remove App?", | ||||||
|         "other": "Remove Apps?" |         "other": "Remove Apps?" | ||||||
| @@ -268,4 +272,4 @@ | |||||||
|         "one": "{} and 1 more app were updated.", |         "one": "{} and 1 more app were updated.", | ||||||
|         "other": "{} and {} more apps were updated." |         "other": "{} and {} more apps were updated." | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "هماهنگ با سیستم", |     "followSystem": "هماهنگ با سیستم", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "مرتب سازی برنامه بر اساس", |     "appSortBy": "مرتب سازی برنامه بر اساس", | ||||||
|     "authorName": "سازنده/اسم", |     "authorName": "سازنده/اسم", | ||||||
|     "nameAuthor": "اسم/سازنده", |     "nameAuthor": "اسم/سازنده", | ||||||
| @@ -207,6 +208,7 @@ | |||||||
|     "addCategory": "اضافه کردن دسته", |     "addCategory": "اضافه کردن دسته", | ||||||
|     "label": "برچسب", |     "label": "برچسب", | ||||||
|     "language": "زبان", |     "language": "زبان", | ||||||
|  |     "copiedToClipboard": "در کلیپ بورد کپی شد", | ||||||
|     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", |     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", | ||||||
|     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", |     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", | ||||||
|     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", |     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", | ||||||
| @@ -220,6 +222,8 @@ | |||||||
|     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", |     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", | ||||||
|     "versionDetection": "تشخیص نسخه", |     "versionDetection": "تشخیص نسخه", | ||||||
|     "standardVersionDetection": "تشخیص نسخه استاندارد", |     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||||
|  |     "groupByCategory": "گروه بر اساس دسته", | ||||||
|  |     "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "برنامه حذف شود؟", |         "one": "برنامه حذف شود؟", | ||||||
|         "other": "برنامه ها حذف شوند؟" |         "other": "برنامه ها حذف شوند؟" | ||||||
|   | |||||||
							
								
								
									
										275
									
								
								assets/translations/fr.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,275 @@ | |||||||
|  | { | ||||||
|  |     "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", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|  |     "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." | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Rendszer szerint", |     "followSystem": "Rendszer szerint", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App rendezés...", |     "appSortBy": "App rendezés...", | ||||||
|     "authorName": "Szerző/Név", |     "authorName": "Szerző/Név", | ||||||
|     "nameAuthor": "Név/Szerző", |     "nameAuthor": "Név/Szerző", | ||||||
| @@ -206,6 +207,7 @@ | |||||||
|     "addCategory": "Új kategória", |     "addCategory": "Új kategória", | ||||||
|     "label": "Címke", |     "label": "Címke", | ||||||
|     "language": "Nyelv", |     "language": "Nyelv", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|     "storagePermissionDenied": "Tárhely engedély megtagadva", |     "storagePermissionDenied": "Tárhely engedély megtagadva", | ||||||
|     "selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.", |     "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", |     "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel", | ||||||
| @@ -219,6 +221,8 @@ | |||||||
|     "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)", |     "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)", | ||||||
|     "versionDetection": "Verzió érzékelés", |     "versionDetection": "Verzió érzékelés", | ||||||
|     "standardVersionDetection": "Alapért. 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": { |     "removeAppQuestion": { | ||||||
|         "one": "Eltávolítja az alkalmazást?", |         "one": "Eltávolítja az alkalmazást?", | ||||||
|         "other": "Eltávolítja az alkalmazást?" |         "other": "Eltávolítja az alkalmazást?" | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "Segui sistema", |     "followSystem": "Segui sistema", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App ordinate per", |     "appSortBy": "App ordinate per", | ||||||
|     "authorName": "Autore/Nome", |     "authorName": "Autore/Nome", | ||||||
|     "nameAuthor": "Nome/Autore", |     "nameAuthor": "Nome/Autore", | ||||||
| @@ -207,6 +208,7 @@ | |||||||
|     "addCategory": "Aggiungi categoria", |     "addCategory": "Aggiungi categoria", | ||||||
|     "label": "Etichetta", |     "label": "Etichetta", | ||||||
|     "language": "Lingua", |     "language": "Lingua", | ||||||
|  |     "copiedToClipboard": "Copiato negli appunti", | ||||||
|     "storagePermissionDenied": "Accesso ai file non autorizzato", |     "storagePermissionDenied": "Accesso ai file non autorizzato", | ||||||
|     "selectedCategorizeWarning": "Ciò sostituirà le impostazioni di categoria esistenti per le App selezionate.", |     "selectedCategorizeWarning": "Ciò sostituirà le impostazioni di categoria esistenti per le App selezionate.", | ||||||
|     "filterAPKsByRegEx": "Filtra file APK con espressioni regolari", |     "filterAPKsByRegEx": "Filtra file APK con espressioni regolari", | ||||||
| @@ -217,9 +219,11 @@ | |||||||
|     "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.", |     "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à", |     "changes": "Novità", | ||||||
|     "releaseDate": "Data di rilascio", |     "releaseDate": "Data di rilascio", | ||||||
|     "importFromURLsInFile": "Import from URLs in File (like OPML)", |     "importFromURLsInFile": "Importa da URL in file (come OPML)", | ||||||
|     "versionDetection": "Version Detection", |     "versionDetection": "Rilevamento di versione", | ||||||
|     "standardVersionDetection": "Standard version detection", |     "standardVersionDetection": "Rilevamento di versione standard", | ||||||
|  |     "groupByCategory": "Raggruppa per categoria", | ||||||
|  |     "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Rimuovere l'App?", |         "one": "Rimuovere l'App?", | ||||||
|         "other": "Rimuovere le App?" |         "other": "Rimuovere le App?" | ||||||
|   | |||||||
| @@ -122,6 +122,7 @@ | |||||||
|     "followSystem": "システムに従う", |     "followSystem": "システムに従う", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "アプリの並び方", |     "appSortBy": "アプリの並び方", | ||||||
|     "authorName": "作者名/アプリ名", |     "authorName": "作者名/アプリ名", | ||||||
|     "nameAuthor": "アプリ名/作者名", |     "nameAuthor": "アプリ名/作者名", | ||||||
| @@ -207,6 +208,7 @@ | |||||||
|     "addCategory": "カテゴリを追加", |     "addCategory": "カテゴリを追加", | ||||||
|     "label": "ラベル", |     "label": "ラベル", | ||||||
|     "language": "言語", |     "language": "言語", | ||||||
|  |     "copiedToClipboard": "クリップボードにコピーしました", | ||||||
|     "storagePermissionDenied": "ストレージ権限が拒否されました", |     "storagePermissionDenied": "ストレージ権限が拒否されました", | ||||||
|     "selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。", |     "selectedCategorizeWarning": "これにより、選択したアプリの既存のカテゴリ設定がすべて置き換えられます。", | ||||||
|     "filterAPKsByRegEx": "正規表現でAPKを絞り込む", |     "filterAPKsByRegEx": "正規表現でAPKを絞り込む", | ||||||
| @@ -220,6 +222,8 @@ | |||||||
|     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", |     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", | ||||||
|     "versionDetection": "バージョン検出", |     "versionDetection": "バージョン検出", | ||||||
|     "standardVersionDetection": "標準のバージョン検出", |     "standardVersionDetection": "標準のバージョン検出", | ||||||
|  |     "groupByCategory": "カテゴリ別にグループ化する", | ||||||
|  |     "autoApkFilterByArch": "可能であれば,CPUアーキテクチャによるAPKのフィルタリングを試みる", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "アプリを削除しますか?", |         "one": "アプリを削除しますか?", | ||||||
|         "other": "アプリを削除しますか?" |         "other": "アプリを削除しますか?" | ||||||
| @@ -268,4 +272,4 @@ | |||||||
|         "one": "{} とさらに {} 個のアプリがアップデートされました", |         "one": "{} とさらに {} 個のアプリがアップデートされました", | ||||||
|         "other": "{} とさらに {} 個のアプリがアップデートされました" |         "other": "{} とさらに {} 個のアプリがアップデートされました" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -123,6 +123,7 @@ | |||||||
|     "followSystem": "跟随系统", |     "followSystem": "跟随系统", | ||||||
|     "obtainium": "Obtainium", |     "obtainium": "Obtainium", | ||||||
|     "materialYou": "Material You", |     "materialYou": "Material You", | ||||||
|  |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "排列方式", |     "appSortBy": "排列方式", | ||||||
|     "authorName": "作者 / 名字", |     "authorName": "作者 / 名字", | ||||||
|     "nameAuthor": "名字 / 作者", |     "nameAuthor": "名字 / 作者", | ||||||
| @@ -208,6 +209,7 @@ | |||||||
|     "addCategory": "添加类别", |     "addCategory": "添加类别", | ||||||
|     "label": "标签", |     "label": "标签", | ||||||
|     "language": "语言", |     "language": "语言", | ||||||
|  |     "copiedToClipboard": "Copied to Clipboard", | ||||||
|     "storagePermissionDenied": "存储权限已被拒绝", |     "storagePermissionDenied": "存储权限已被拒绝", | ||||||
|     "selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别", |     "selectedCategorizeWarning": "这将取代所选应用程序的任何现有类别", | ||||||
|     "filterAPKsByRegEx": "Filter APKs by Regular Expression", |     "filterAPKsByRegEx": "Filter APKs by Regular Expression", | ||||||
| @@ -220,6 +222,8 @@ | |||||||
|     "importFromURLsInFile": "Import from URLs in File (like OPML)", |     "importFromURLsInFile": "Import from URLs in File (like OPML)", | ||||||
|     "versionDetection": "Version Detection", |     "versionDetection": "Version Detection", | ||||||
|     "standardVersionDetection": "Standard version detection", |     "standardVersionDetection": "Standard version detection", | ||||||
|  |     "groupByCategory": "Group by Category", | ||||||
|  |     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "删除应用?", |         "one": "删除应用?", | ||||||
|         "other": "删除应用?" |         "other": "删除应用?" | ||||||
| @@ -268,4 +272,4 @@ | |||||||
|         "one": "{} 和 {} 更多应用已被安装", |         "one": "{} 和 {} 更多应用已被安装", | ||||||
|         "other": "{} 和 {} 更多应用已被安装" |         "other": "{} 和 {} 更多应用已被安装" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| @@ -35,6 +36,8 @@ class Codeberg extends AppSource { | |||||||
|     canSearch = true; |     canSearch = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   var gh = GitHub(); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String standardizeURL(String url) { |   String standardizeURL(String url) { | ||||||
|     RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+'); |     RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+'); | ||||||
| @@ -54,76 +57,10 @@ class Codeberg extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases'] == true; |     return gh.getLatestAPKDetailsCommon( | ||||||
|     bool fallbackToOlderReleases = |         'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100', | ||||||
|         additionalSettings['fallbackToOlderReleases'] == true; |         standardUrl, | ||||||
|     String? regexFilter = |         additionalSettings); | ||||||
|         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) |  | ||||||
|                     ?.isNotEmpty == |  | ||||||
|                 true |  | ||||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] |  | ||||||
|             : null; |  | ||||||
|     Response res = await get(Uri.parse( |  | ||||||
|         'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100')); |  | ||||||
|     if (res.statusCode == 200) { |  | ||||||
|       var releases = jsonDecode(res.body) as List<dynamic>; |  | ||||||
|  |  | ||||||
|       List<String> getReleaseAPKUrls(dynamic release) => |  | ||||||
|           (release['assets'] as List<dynamic>?) |  | ||||||
|               ?.map((e) { |  | ||||||
|                 return e['name'] != null && e['browser_download_url'] != null |  | ||||||
|                     ? MapEntry(e['name'] as String, |  | ||||||
|                         e['browser_download_url'] as String) |  | ||||||
|                     : const MapEntry('', ''); |  | ||||||
|               }) |  | ||||||
|               .where((element) => element.key.toLowerCase().endsWith('.apk')) |  | ||||||
|               .map((e) => e.value) |  | ||||||
|               .toList() ?? |  | ||||||
|           []; |  | ||||||
|  |  | ||||||
|       dynamic targetRelease; |  | ||||||
|  |  | ||||||
|       for (int i = 0; i < releases.length; i++) { |  | ||||||
|         if (!fallbackToOlderReleases && i > 0) break; |  | ||||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|         if (releases[i]['draft'] == true) { |  | ||||||
|           // Draft releases not supported |  | ||||||
|         } |  | ||||||
|         var nameToFilter = releases[i]['name'] as String?; |  | ||||||
|         if (nameToFilter == null || nameToFilter.trim().isEmpty) { |  | ||||||
|           // Some leave titles empty so tag is used |  | ||||||
|           nameToFilter = releases[i]['tag_name'] as String; |  | ||||||
|         } |  | ||||||
|         if (regexFilter != null && |  | ||||||
|             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|         var apkUrls = getReleaseAPKUrls(releases[i]); |  | ||||||
|         if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) { |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|         targetRelease = releases[i]; |  | ||||||
|         targetRelease['apkUrls'] = apkUrls; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       if (targetRelease == null) { |  | ||||||
|         throw NoReleasesError(); |  | ||||||
|       } |  | ||||||
|       String? version = targetRelease['tag_name']; |  | ||||||
|       DateTime? releaseDate = targetRelease['published_at'] != null |  | ||||||
|           ? DateTime.parse(targetRelease['published_at']) |  | ||||||
|           : null; |  | ||||||
|       if (version == null) { |  | ||||||
|         throw NoVersionError(); |  | ||||||
|       } |  | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |  | ||||||
|           getAppNames(standardUrl), |  | ||||||
|           releaseDate: releaseDate); |  | ||||||
|     } else { |  | ||||||
|       throw getObtainiumHttpError(res); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   AppNames getAppNames(String standardUrl) { |   AppNames getAppNames(String standardUrl) { | ||||||
| @@ -134,20 +71,9 @@ class Codeberg extends AppSource { | |||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<Map<String, String>> search(String query) async { |   Future<Map<String, String>> search(String query) async { | ||||||
|     Response res = await get(Uri.parse( |     return gh.searchCommon( | ||||||
|         'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100')); |         query, | ||||||
|     if (res.statusCode == 200) { |         'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100', | ||||||
|       Map<String, String> urlsWithDescriptions = {}; |         'data'); | ||||||
|       for (var e in (jsonDecode(res.body)['data'] as List<dynamic>)) { |  | ||||||
|         urlsWithDescriptions.addAll({ |  | ||||||
|           e['html_url'] as String: e['description'] != null |  | ||||||
|               ? e['description'] as String |  | ||||||
|               : tr('noDescription') |  | ||||||
|         }); |  | ||||||
|       } |  | ||||||
|       return urlsWithDescriptions; |  | ||||||
|     } else { |  | ||||||
|       throw getObtainiumHttpError(res); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -14,12 +14,14 @@ class FDroid extends AppSource { | |||||||
|   @override |   @override | ||||||
|   String standardizeURL(String url) { |   String standardizeURL(String url) { | ||||||
|     RegExp standardUrlRegExB = |     RegExp standardUrlRegExB = | ||||||
|         RegExp('^https?://$host/+[^/]+/+packages/+[^/]+'); |         RegExp('^https?://(cloudflare\\.)?$host/+[^/]+/+packages/+[^/]+'); | ||||||
|     RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase()); |     RegExpMatch? match = standardUrlRegExB.firstMatch(url.toLowerCase()); | ||||||
|     if (match != null) { |     if (match != null) { | ||||||
|       url = 'https://$host/packages/${Uri.parse(url).pathSegments.last}'; |       url = | ||||||
|  |           'https://${Uri.parse(url.substring(0, match.end)).host}/packages/${Uri.parse(url).pathSegments.last}'; | ||||||
|     } |     } | ||||||
|     RegExp standardUrlRegExA = RegExp('^https?://$host/+packages/+[^/]+'); |     RegExp standardUrlRegExA = | ||||||
|  |         RegExp('^https?://(cloudflare\\.)?$host/+packages/+[^/]+'); | ||||||
|     match = standardUrlRegExA.firstMatch(url.toLowerCase()); |     match = standardUrlRegExA.firstMatch(url.toLowerCase()); | ||||||
|     if (match == null) { |     if (match == null) { | ||||||
|       throw InvalidURLError(name); |       throw InvalidURLError(name); | ||||||
| @@ -27,9 +29,6 @@ class FDroid extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String? tryInferringAppId(String standardUrl, |   String? tryInferringAppId(String standardUrl, | ||||||
|       {Map<String, dynamic> additionalSettings = const {}}) { |       {Map<String, dynamic> additionalSettings = const {}}) { | ||||||
| @@ -51,7 +50,7 @@ class FDroid extends AppSource { | |||||||
|           .where((element) => element['versionName'] == latestVersion) |           .where((element) => element['versionName'] == latestVersion) | ||||||
|           .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk') |           .map((e) => '${apkUrlPrefix}_${e['versionCode']}.apk') | ||||||
|           .toList(); |           .toList(); | ||||||
|       return APKDetails(latestVersion, apkUrls, |       return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls), | ||||||
|           AppNames(name, Uri.parse(standardUrl).pathSegments.last)); |           AppNames(name, Uri.parse(standardUrl).pathSegments.last)); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
| @@ -64,9 +63,10 @@ class FDroid extends AppSource { | |||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     String? appId = tryInferringAppId(standardUrl); |     String? appId = tryInferringAppId(standardUrl); | ||||||
|  |     String host = Uri.parse(standardUrl).host; | ||||||
|     return getAPKUrlsFromFDroidPackagesAPIResponse( |     return getAPKUrlsFromFDroidPackagesAPIResponse( | ||||||
|         await get(Uri.parse('https://f-droid.org/api/v1/packages/$appId')), |         await get(Uri.parse('https://$host/api/v1/packages/$appId')), | ||||||
|         'https://f-droid.org/repo/$appId', |         'https://$host/repo/$appId', | ||||||
|         standardUrl); |         standardUrl); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -80,7 +80,8 @@ class FDroidRepo extends AppSource { | |||||||
|               element.querySelector('apkname') != null) |               element.querySelector('apkname') != null) | ||||||
|           .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}') |           .map((e) => '$standardUrl/${e.querySelector('apkname')!.innerHtml}') | ||||||
|           .toList(); |           .toList(); | ||||||
|       return APKDetails(latestVersion, apkUrls, AppNames(authorName, appName), |       return APKDetails(latestVersion, getApkUrlsFromUrls(apkUrls), | ||||||
|  |           AppNames(authorName, appName), | ||||||
|           releaseDate: releaseDate); |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|   | |||||||
| @@ -96,11 +96,9 @@ class GitHub extends AppSource { | |||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => |   String? changeLogPageFromStandardUrl(String standardUrl) => | ||||||
|       '$standardUrl/releases'; |       '$standardUrl/releases'; | ||||||
|  |  | ||||||
|   @override |   Future<APKDetails> getLatestAPKDetailsCommon(String requestUrl, | ||||||
|   Future<APKDetails> getLatestAPKDetails( |       String standardUrl, Map<String, dynamic> additionalSettings, | ||||||
|     String standardUrl, |       {Function(Response)? onHttpErrorCode}) async { | ||||||
|     Map<String, dynamic> additionalSettings, |  | ||||||
|   ) async { |  | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases'] == true; |     bool includePrereleases = additionalSettings['includePrereleases'] == true; | ||||||
|     bool fallbackToOlderReleases = |     bool fallbackToOlderReleases = | ||||||
|         additionalSettings['fallbackToOlderReleases'] == true; |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
| @@ -110,27 +108,50 @@ class GitHub extends AppSource { | |||||||
|                 true |                 true | ||||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|             : null; |             : null; | ||||||
|     Response res = await get(Uri.parse( |     Response res = await get(Uri.parse(requestUrl)); | ||||||
|         'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100')); |  | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var releases = jsonDecode(res.body) as List<dynamic>; |       var releases = jsonDecode(res.body) as List<dynamic>; | ||||||
|  |  | ||||||
|       List<String> getReleaseAPKUrls(dynamic release) => |       List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) => | ||||||
|           (release['assets'] as List<dynamic>?) |           (release['assets'] as List<dynamic>?) | ||||||
|               ?.map((e) { |               ?.map((e) { | ||||||
|                 return e['browser_download_url'] != null |                 return e['name'] != null && e['browser_download_url'] != null | ||||||
|                     ? e['browser_download_url'] as String |                     ? MapEntry(e['name'] as String, | ||||||
|                     : ''; |                         e['browser_download_url'] as String) | ||||||
|  |                     : const MapEntry('', ''); | ||||||
|               }) |               }) | ||||||
|               .where((element) => element.toLowerCase().endsWith('.apk')) |               .where((element) => element.key.toLowerCase().endsWith('.apk')) | ||||||
|               .toList() ?? |               .toList() ?? | ||||||
|           []; |           []; | ||||||
|  |  | ||||||
|  |       DateTime? getReleaseDateFromRelease(dynamic rel) => | ||||||
|  |           rel?['published_at'] != null | ||||||
|  |               ? DateTime.parse(rel['published_at']) | ||||||
|  |               : null; | ||||||
|  |       releases.sort((a, b) { | ||||||
|  |         // See #478 | ||||||
|  |         if (a == b) { | ||||||
|  |           return 0; | ||||||
|  |         } else if (a == null) { | ||||||
|  |           return -1; | ||||||
|  |         } else if (b == null) { | ||||||
|  |           return 1; | ||||||
|  |         } else { | ||||||
|  |           return getReleaseDateFromRelease(a)! | ||||||
|  |               .compareTo(getReleaseDateFromRelease(b)!); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |       releases = releases.reversed.toList(); | ||||||
|       dynamic targetRelease; |       dynamic targetRelease; | ||||||
|  |       var prerrelsSkipped = 0; | ||||||
|       for (int i = 0; i < releases.length; i++) { |       for (int i = 0; i < releases.length; i++) { | ||||||
|         if (!fallbackToOlderReleases && i > 0) break; |         if (!fallbackToOlderReleases && i > prerrelsSkipped) break; | ||||||
|         if (!includePrereleases && releases[i]['prerelease'] == true) { |         if (!includePrereleases && releases[i]['prerelease'] == true) { | ||||||
|  |           prerrelsSkipped++; | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         if (releases[i]['draft'] == true) { | ||||||
|  |           // Draft releases not supported | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|         var nameToFilter = releases[i]['name'] as String?; |         var nameToFilter = releases[i]['name'] as String?; | ||||||
| @@ -154,47 +175,78 @@ class GitHub extends AppSource { | |||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String? version = targetRelease['tag_name']; |       String? version = targetRelease['tag_name']; | ||||||
|       DateTime? releaseDate = targetRelease['published_at'] != null |       DateTime? releaseDate = getReleaseDateFromRelease(targetRelease); | ||||||
|           ? DateTime.parse(targetRelease['published_at']) |  | ||||||
|           : null; |  | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |       var changeLog = targetRelease['body'].toString(); | ||||||
|  |       return APKDetails( | ||||||
|  |           version, | ||||||
|  |           targetRelease['apkUrls'] as List<MapEntry<String, String>>, | ||||||
|           getAppNames(standardUrl), |           getAppNames(standardUrl), | ||||||
|           releaseDate: releaseDate); |           releaseDate: releaseDate, | ||||||
|  |           changeLog: changeLog.isEmpty ? null : changeLog); | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       if (onHttpErrorCode != null) { | ||||||
|  |         onHttpErrorCode(res); | ||||||
|  |       } | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     return getLatestAPKDetailsCommon( | ||||||
|  |         'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100', | ||||||
|  |         standardUrl, | ||||||
|  |         additionalSettings, onHttpErrorCode: (Response res) { | ||||||
|  |       rateLimitErrorCheck(res); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   AppNames getAppNames(String standardUrl) { |   AppNames getAppNames(String standardUrl) { | ||||||
|     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); |     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); | ||||||
|     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); |     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); | ||||||
|     return AppNames(names[0], names[1]); |     return AppNames(names[0], names[1]); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   Future<Map<String, String>> searchCommon( | ||||||
|   Future<Map<String, String>> search(String query) async { |       String query, String requestUrl, String rootProp, | ||||||
|     Response res = await get(Uri.parse( |       {Function(Response)? onHttpErrorCode}) async { | ||||||
|         'https://${await getCredentialPrefixIfAny()}api.$host/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100')); |     Response res = await get(Uri.parse(requestUrl)); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       Map<String, String> urlsWithDescriptions = {}; |       Map<String, String> urlsWithDescriptions = {}; | ||||||
|       for (var e in (jsonDecode(res.body)['items'] as List<dynamic>)) { |       for (var e in (jsonDecode(res.body)[rootProp] as List<dynamic>)) { | ||||||
|         urlsWithDescriptions.addAll({ |         urlsWithDescriptions.addAll({ | ||||||
|           e['html_url'] as String: e['description'] != null |           e['html_url'] as String: | ||||||
|               ? e['description'] as String |               ((e['archived'] == true ? '[ARCHIVED] ' : '') + | ||||||
|               : tr('noDescription') |                   (e['description'] != null | ||||||
|  |                       ? e['description'] as String | ||||||
|  |                       : tr('noDescription'))) | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|       return urlsWithDescriptions; |       return urlsWithDescriptions; | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       if (onHttpErrorCode != null) { | ||||||
|  |         onHttpErrorCode(res); | ||||||
|  |       } | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<Map<String, String>> search(String query) async { | ||||||
|  |     return searchCommon( | ||||||
|  |         query, | ||||||
|  |         'https://${await getCredentialPrefixIfAny()}api.$host/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100', | ||||||
|  |         'items', onHttpErrorCode: (Response res) { | ||||||
|  |       rateLimitErrorCheck(res); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   rateLimitErrorCheck(Response res) { |   rateLimitErrorCheck(Response res) { | ||||||
|     if (res.headers['x-ratelimit-remaining'] == '0') { |     if (res.headers['x-ratelimit-remaining'] == '0') { | ||||||
|       throw RateLimitError( |       throw RateLimitError( | ||||||
|   | |||||||
| @@ -3,10 +3,19 @@ import 'package:http/http.dart'; | |||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  |  | ||||||
| class GitLab extends AppSource { | class GitLab extends AppSource { | ||||||
|   GitLab() { |   GitLab() { | ||||||
|     host = 'gitlab.com'; |     host = 'gitlab.com'; | ||||||
|  |  | ||||||
|  |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('fallbackToOlderReleases', | ||||||
|  |             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||||
|  |       ] | ||||||
|  |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -28,40 +37,58 @@ class GitLab extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|  |     bool fallbackToOlderReleases = | ||||||
|  |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|     Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom')); |     Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var standardUri = Uri.parse(standardUrl); |       var standardUri = Uri.parse(standardUrl); | ||||||
|       var parsedHtml = parse(res.body); |       var parsedHtml = parse(res.body); | ||||||
|       var entry = parsedHtml.querySelector('entry'); |       var apkDetailsList = parsedHtml.querySelectorAll('entry').map((entry) { | ||||||
|       var entryContent = |         var entryContent = parse( | ||||||
|           parse(parseFragment(entry?.querySelector('content')!.innerHtml).text); |             parseFragment(entry.querySelector('content')!.innerHtml).text); | ||||||
|       var apkUrls = [ |         var apkUrls = [ | ||||||
|         ...getLinksFromParsedHTML( |           ...getLinksFromParsedHTML( | ||||||
|             entryContent, |               entryContent, | ||||||
|             RegExp( |               RegExp( | ||||||
|                 '^${standardUri.path.replaceAllMapped(RegExp(r'[.*+?^${}()|[\]\\]'), (x) { |                   '^${standardUri.path.replaceAllMapped(RegExp(r'[.*+?^${}()|[\]\\]'), (x) { | ||||||
|                   return '\\${x[0]}'; |                     return '\\${x[0]}'; | ||||||
|                 })}/uploads/[^/]+/[^/]+\\.apk\$', |                   })}/uploads/[^/]+/[^/]+\\.apk\$', | ||||||
|                 caseSensitive: false), |                   caseSensitive: false), | ||||||
|             standardUri.origin), |               standardUri.origin), | ||||||
|         // GitLab releases may contain links to externally hosted APKs |           // GitLab releases may contain links to externally hosted APKs | ||||||
|         ...getLinksFromParsedHTML(entryContent, |           ...getLinksFromParsedHTML(entryContent, | ||||||
|                 RegExp('/[^/]+\\.apk\$', caseSensitive: false), '') |                   RegExp('/[^/]+\\.apk\$', caseSensitive: false), '') | ||||||
|             .where((element) => Uri.parse(element).host != '') |               .where((element) => Uri.parse(element).host != '') | ||||||
|             .toList() |               .toList() | ||||||
|       ]; |         ]; | ||||||
|  |  | ||||||
|       var entryId = entry?.querySelector('id')?.innerHtml; |         var entryId = entry.querySelector('id')?.innerHtml; | ||||||
|       var version = |         var version = | ||||||
|           entryId == null ? null : Uri.parse(entryId).pathSegments.last; |             entryId == null ? null : Uri.parse(entryId).pathSegments.last; | ||||||
|       var releaseDateString = entry?.querySelector('updated')?.innerHtml; |         var releaseDateString = entry.querySelector('updated')?.innerHtml; | ||||||
|       DateTime? releaseDate = |         DateTime? releaseDate = releaseDateString != null | ||||||
|           releaseDateString != null ? DateTime.parse(releaseDateString) : null; |             ? DateTime.parse(releaseDateString) | ||||||
|       if (version == null) { |             : null; | ||||||
|         throw NoVersionError(); |         if (version == null) { | ||||||
|  |           throw NoVersionError(); | ||||||
|  |         } | ||||||
|  |         return APKDetails(version, getApkUrlsFromUrls(apkUrls), | ||||||
|  |             GitHub().getAppNames(standardUrl), | ||||||
|  |             releaseDate: releaseDate); | ||||||
|  |       }); | ||||||
|  |       if (apkDetailsList.isEmpty) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl), |       if (fallbackToOlderReleases) { | ||||||
|           releaseDate: releaseDate); |         if (additionalSettings['trackOnly'] != true) { | ||||||
|  |           apkDetailsList = | ||||||
|  |               apkDetailsList.where((e) => e.apkUrls.isNotEmpty).toList(); | ||||||
|  |         } | ||||||
|  |         if (apkDetailsList.isEmpty) { | ||||||
|  |           throw NoReleasesError(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       return apkDetailsList.first; | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -10,9 +10,6 @@ class HTML extends AppSource { | |||||||
|     return url; |     return url; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -37,15 +34,22 @@ class HTML extends AppSource { | |||||||
|       var rel = links.last; |       var rel = links.last; | ||||||
|       var apkName = rel.split('/').last; |       var apkName = rel.split('/').last; | ||||||
|       var version = apkName.substring(0, apkName.length - 4); |       var version = apkName.substring(0, apkName.length - 4); | ||||||
|       List<String> apkUrls = [rel] |       List<String> apkUrls = [rel].map((e) { | ||||||
|           .map((e) => e.toLowerCase().startsWith('http://') || |         try { | ||||||
|                   e.toLowerCase().startsWith('https://') |           Uri.parse(e).origin; | ||||||
|               ? e |           return e; | ||||||
|               : e.startsWith('/') |         } catch (err) { | ||||||
|                   ? '${uri.origin}/$e' |           // is relative | ||||||
|                   : '${uri.origin}/${uri.path}/$e') |         } | ||||||
|           .toList(); |         var currPathSegments = uri.path.split('/'); | ||||||
|       return APKDetails(version, apkUrls, AppNames(uri.host, tr('app'))); |         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 { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -18,9 +18,6 @@ class IzzyOnDroid extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String? tryInferringAppId(String standardUrl, |   String? tryInferringAppId(String standardUrl, | ||||||
|       {Map<String, dynamic> additionalSettings = const {}}) { |       {Map<String, dynamic> additionalSettings = const {}}) { | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| import 'package:html/parser.dart'; | import 'package:html/parser.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| @@ -29,19 +30,37 @@ class Mullvad extends AppSource { | |||||||
|   ) async { |   ) async { | ||||||
|     Response res = await get(Uri.parse('$standardUrl/en/download/android')); |     Response res = await get(Uri.parse('$standardUrl/en/download/android')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var version = parse(res.body) |       var versions = parse(res.body) | ||||||
|           .querySelector('p.subtitle.is-6') |           .querySelectorAll('p') | ||||||
|           ?.querySelector('a') |           .map((e) => e.innerHtml) | ||||||
|           ?.attributes['href'] |           .where((p) => p.contains('Latest version: ')) | ||||||
|           ?.split('/') |           .map((e) { | ||||||
|           .last; |             var match = RegExp('[0-9]+(\\.[0-9]+)*').firstMatch(e); | ||||||
|       if (version == null) { |             if (match == null) { | ||||||
|  |               return ''; | ||||||
|  |             } else { | ||||||
|  |               return e.substring(match.start, match.end); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .where((element) => element.isNotEmpty) | ||||||
|  |           .toList(); | ||||||
|  |       if (versions.isEmpty) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|  |       String? changeLog; | ||||||
|  |       try { | ||||||
|  |         changeLog = (await GitHub().getLatestAPKDetails( | ||||||
|  |                 'https://github.com/mullvad/mullvadvpn-app', | ||||||
|  |                 {'fallbackToOlderReleases': true})) | ||||||
|  |             .changeLog; | ||||||
|  |       } catch (e) { | ||||||
|  |         // Ignore | ||||||
|  |       } | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, |           versions[0], | ||||||
|           ['https://mullvad.net/download/app/apk/latest'], |           getApkUrlsFromUrls(['https://mullvad.net/download/app/apk/latest']), | ||||||
|           AppNames(name, 'Mullvad-VPN')); |           AppNames(name, 'Mullvad-VPN'), | ||||||
|  |           changeLog: changeLog); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
							
								
								
									
										111
									
								
								lib/app_sources/neutroncode.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,111 @@ | |||||||
|  | import 'package:html/parser.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class NeutronCode extends AppSource { | ||||||
|  |   NeutronCode() { | ||||||
|  |     host = 'neutroncode.com'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String standardizeURL(String url) { | ||||||
|  |     RegExp standardUrlRegEx = RegExp('^https?://$host/downloads/file/[^/]+'); | ||||||
|  |     RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase()); | ||||||
|  |     if (match == null) { | ||||||
|  |       throw InvalidURLError(name); | ||||||
|  |     } | ||||||
|  |     return url.substring(0, match.end); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String? changeLogPageFromStandardUrl(String standardUrl) => standardUrl; | ||||||
|  |  | ||||||
|  |   String monthNameToNumberString(String s) { | ||||||
|  |     switch (s.toLowerCase()) { | ||||||
|  |       case 'january': | ||||||
|  |         return '01'; | ||||||
|  |       case 'february': | ||||||
|  |         return '02'; | ||||||
|  |       case 'march': | ||||||
|  |         return '03'; | ||||||
|  |       case 'april': | ||||||
|  |         return '04'; | ||||||
|  |       case 'may': | ||||||
|  |         return '05'; | ||||||
|  |       case 'june': | ||||||
|  |         return '06'; | ||||||
|  |       case 'july': | ||||||
|  |         return '07'; | ||||||
|  |       case 'august': | ||||||
|  |         return '08'; | ||||||
|  |       case 'september': | ||||||
|  |         return '09'; | ||||||
|  |       case 'october': | ||||||
|  |         return '10'; | ||||||
|  |       case 'november': | ||||||
|  |         return '11'; | ||||||
|  |       case 'december': | ||||||
|  |         return '12'; | ||||||
|  |       default: | ||||||
|  |         throw ArgumentError('Invalid month name: $s'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   customDateParse(String dateString) { | ||||||
|  |     List<String> parts = dateString.split(' '); | ||||||
|  |     if (parts.length != 3) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |     String result = ''; | ||||||
|  |     for (var s in parts.reversed) { | ||||||
|  |       try { | ||||||
|  |         try { | ||||||
|  |           int.parse(s); | ||||||
|  |           result += '$s-'; | ||||||
|  |         } catch (e) { | ||||||
|  |           result += '${monthNameToNumberString(s)}-'; | ||||||
|  |         } | ||||||
|  |       } catch (e) { | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return result.substring(0, result.length - 1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     Response res = await get(Uri.parse(standardUrl)); | ||||||
|  |     if (res.statusCode == 200) { | ||||||
|  |       var http = parse(res.body); | ||||||
|  |       var name = http.querySelector('.pd-title')?.innerHtml; | ||||||
|  |       var filename = http.querySelector('.pd-filename .pd-float')?.innerHtml; | ||||||
|  |       if (filename == null) { | ||||||
|  |         throw NoReleasesError(); | ||||||
|  |       } | ||||||
|  |       var version = | ||||||
|  |           http.querySelector('.pd-version-txt')?.nextElementSibling?.innerHtml; | ||||||
|  |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |       String? apkUrl = 'https://$host/download/$filename'; | ||||||
|  |       var dateStringOriginal = | ||||||
|  |           http.querySelector('.pd-date-txt')?.nextElementSibling?.innerHtml; | ||||||
|  |       var dateString = dateStringOriginal != null | ||||||
|  |           ? (customDateParse(dateStringOriginal)) | ||||||
|  |           : null; | ||||||
|  |       var changeLogElements = http.querySelectorAll('.pd-fdesc p'); | ||||||
|  |       return APKDetails(version, getApkUrlsFromUrls([apkUrl]), | ||||||
|  |           AppNames(runtimeType.toString(), name ?? standardUrl.split('/').last), | ||||||
|  |           releaseDate: dateString != null ? DateTime.parse(dateString) : null, | ||||||
|  |           changeLog: changeLogElements.isNotEmpty | ||||||
|  |               ? changeLogElements.last.innerHtml | ||||||
|  |               : null); | ||||||
|  |     } else { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -13,9 +13,6 @@ class Signal extends AppSource { | |||||||
|     return 'https://$host'; |     return 'https://$host'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -31,7 +28,8 @@ class Signal extends AppSource { | |||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, apkUrls, AppNames(name, 'Signal')); |       return APKDetails( | ||||||
|  |           version, getApkUrlsFromUrls(apkUrls), AppNames(name, 'Signal')); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -18,9 +18,6 @@ class SourceForge extends AppSource { | |||||||
|     return url.substring(0, match.end); |     return url.substring(0, match.end); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -34,7 +31,8 @@ class SourceForge extends AppSource { | |||||||
|       getVersion(String url) { |       getVersion(String url) { | ||||||
|         try { |         try { | ||||||
|           var tokens = url.split('/'); |           var tokens = url.split('/'); | ||||||
|           return tokens[tokens.length - 3]; |           var fi = tokens.indexOf('files'); | ||||||
|  |           return tokens[tokens[fi + 2] == 'download' ? fi - 1 : fi + 1]; | ||||||
|         } catch (e) { |         } catch (e) { | ||||||
|           return null; |           return null; | ||||||
|         } |         } | ||||||
| @@ -53,7 +51,7 @@ class SourceForge extends AppSource { | |||||||
|               .toList(); |               .toList(); | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, |           version, | ||||||
|           apkUrlList, |           getApkUrlsFromUrls(apkUrlList), | ||||||
|           AppNames( |           AppNames( | ||||||
|               name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1))); |               name, standardUrl.substring(standardUrl.lastIndexOf('/') + 1))); | ||||||
|     } else { |     } else { | ||||||
|   | |||||||
| @@ -24,9 +24,6 @@ class SteamMobile extends AppSource { | |||||||
|     return 'https://$host'; |     return 'https://$host'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => null; |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -56,7 +53,8 @@ class SteamMobile extends AppSource { | |||||||
|       var version = links[0].substring( |       var version = links[0].substring( | ||||||
|           versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4); |           versionMatch.start + apkNamePrefix.length + 2, versionMatch.end - 4); | ||||||
|       var apkUrls = [links[0]]; |       var apkUrls = [links[0]]; | ||||||
|       return APKDetails(version, apkUrls, AppNames(name, apks[apkNamePrefix]!)); |       return APKDetails(version, getApkUrlsFromUrls(apkUrls), | ||||||
|  |           AppNames(name, apks[apkNamePrefix]!)); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       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 { | class GeneratedFormDropdown extends GeneratedFormItem { | ||||||
|   late List<MapEntry<String, String>>? opts; |   late List<MapEntry<String, String>>? opts; | ||||||
|  |   List<String>? disabledOptKeys; | ||||||
|  |  | ||||||
|   GeneratedFormDropdown( |   GeneratedFormDropdown( | ||||||
|     String key, |     String key, | ||||||
| @@ -55,6 +56,7 @@ class GeneratedFormDropdown extends GeneratedFormItem { | |||||||
|     String label = 'Input', |     String label = 'Input', | ||||||
|     List<Widget> belowWidgets = const [], |     List<Widget> belowWidgets = const [], | ||||||
|     String defaultValue = '', |     String defaultValue = '', | ||||||
|  |     this.disabledOptKeys, | ||||||
|     List<String? Function(String? value)> additionalValidators = const [], |     List<String? Function(String? value)> additionalValidators = const [], | ||||||
|   }) : super(key, |   }) : super(key, | ||||||
|             label: label, |             label: label, | ||||||
| @@ -225,10 +227,15 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|           return DropdownButtonFormField( |           return DropdownButtonFormField( | ||||||
|               decoration: InputDecoration(labelText: formItem.label), |               decoration: InputDecoration(labelText: formItem.label), | ||||||
|               value: values[formItem.key], |               value: values[formItem.key], | ||||||
|               items: formItem.opts! |               items: formItem.opts!.map((e2) { | ||||||
|                   .map((e2) => |                 var enabled = | ||||||
|                       DropdownMenuItem(value: e2.key, child: Text(e2.value))) |                     formItem.disabledOptKeys?.contains(e2.key) != true; | ||||||
|                   .toList(), |                 return DropdownMenuItem( | ||||||
|  |                     value: e2.key, | ||||||
|  |                     enabled: enabled, | ||||||
|  |                     child: Opacity( | ||||||
|  |                         opacity: enabled ? 1 : 0.5, child: Text(e2.value))); | ||||||
|  |               }).toList(), | ||||||
|               onChanged: (value) { |               onChanged: (value) { | ||||||
|                 setState(() { |                 setState(() { | ||||||
|                   values[formItem.key] = value ?? formItem.opts!.first.key; |                   values[formItem.key] = value ?? formItem.opts!.first.key; | ||||||
| @@ -260,7 +267,10 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|           formInputs[r][e] = Row( |           formInputs[r][e] = Row( | ||||||
|             mainAxisAlignment: MainAxisAlignment.spaceBetween, |             mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|             children: [ |             children: [ | ||||||
|               Text(widget.items[r][e].label), |               Flexible(child: Text(widget.items[r][e].label)), | ||||||
|  |               const SizedBox( | ||||||
|  |                 width: 8, | ||||||
|  |               ), | ||||||
|               Switch( |               Switch( | ||||||
|                   value: values[widget.items[r][e].key], |                   value: values[widget.items[r][e].key], | ||||||
|                   onChanged: (value) { |                   onChanged: (value) { | ||||||
| @@ -460,10 +470,9 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|       if (rowInputs.key > 0) { |       if (rowInputs.key > 0) { | ||||||
|         rows.add([ |         rows.add([ | ||||||
|           SizedBox( |           SizedBox( | ||||||
|             height: widget.items[rowInputs.key][0] is GeneratedFormSwitch && |             height: widget.items[rowInputs.key - 1][0] is GeneratedFormSwitch | ||||||
|                     widget.items[rowInputs.key - 1][0] is! GeneratedFormSwitch |                 ? 8 | ||||||
|                 ? 25 |                 : 25, | ||||||
|                 : 8, |  | ||||||
|           ) |           ) | ||||||
|         ]); |         ]); | ||||||
|       } |       } | ||||||
| @@ -477,6 +486,7 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|         rowItems.add(Expanded( |         rowItems.add(Expanded( | ||||||
|             child: Column( |             child: Column( | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.stretch, |                 crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |                 mainAxisSize: MainAxisSize.min, | ||||||
|                 children: [ |                 children: [ | ||||||
|               rowInput.value, |               rowInput.value, | ||||||
|               ...widget.items[rowInputs.key][rowInput.key].belowWidgets |               ...widget.items[rowInputs.key][rowInput.key].belowWidgets | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart'; | |||||||
| // ignore: implementation_imports | // ignore: implementation_imports | ||||||
| import 'package:easy_localization/src/localization.dart'; | import 'package:easy_localization/src/localization.dart'; | ||||||
|  |  | ||||||
| const String currentVersion = '0.11.6'; | const String currentVersion = '0.11.34'; | ||||||
| const String currentReleaseTag = | const String currentReleaseTag = | ||||||
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES |     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES | ||||||
|  |  | ||||||
| @@ -34,7 +34,8 @@ const supportedLocales = [ | |||||||
|   Locale('ja'), |   Locale('ja'), | ||||||
|   Locale('hu'), |   Locale('hu'), | ||||||
|   Locale('de'), |   Locale('de'), | ||||||
|   Locale('fa') |   Locale('fa'), | ||||||
|  |   Locale('fr') | ||||||
| ]; | ]; | ||||||
| const fallbackLocale = Locale('en'); | const fallbackLocale = Locale('en'); | ||||||
| const localeDir = 'assets/translations'; | const localeDir = 'assets/translations'; | ||||||
| @@ -146,6 +147,14 @@ Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async { | |||||||
|  |  | ||||||
| void main() async { | void main() async { | ||||||
|   WidgetsFlutterBinding.ensureInitialized(); |   WidgetsFlutterBinding.ensureInitialized(); | ||||||
|  |   try { | ||||||
|  |     ByteData data = | ||||||
|  |         await PlatformAssetBundle().load('assets/ca/lets-encrypt-r3.pem'); | ||||||
|  |     SecurityContext.defaultContext | ||||||
|  |         .setTrustedCertificatesBytes(data.buffer.asUint8List()); | ||||||
|  |   } catch (e) { | ||||||
|  |     // Already added, do nothing (see #375) | ||||||
|  |   } | ||||||
|   await EasyLocalization.ensureInitialized(); |   await EasyLocalization.ensureInitialized(); | ||||||
|   if ((await DeviceInfoPlugin().androidInfo).version.sdkInt >= 29) { |   if ((await DeviceInfoPlugin().androidInfo).version.sdkInt >= 29) { | ||||||
|     SystemChrome.setSystemUIOverlayStyle( |     SystemChrome.setSystemUIOverlayStyle( | ||||||
| @@ -209,7 +218,7 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|               {'includePrereleases': true}, |               {'includePrereleases': true}, | ||||||
|               null, |               null, | ||||||
|               false) |               false) | ||||||
|         ]); |         ], onlyIfExists: false); | ||||||
|       } |       } | ||||||
|       if (!supportedLocales |       if (!supportedLocales | ||||||
|               .map((e) => e.languageCode) |               .map((e) => e.languageCode) | ||||||
| @@ -254,6 +263,14 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|         darkColorScheme = ColorScheme.fromSeed( |         darkColorScheme = ColorScheme.fromSeed( | ||||||
|             seedColor: defaultThemeColour, brightness: Brightness.dark); |             seedColor: defaultThemeColour, brightness: Brightness.dark); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       // set the background and surface colors to pure black in the amoled theme | ||||||
|  |       if (settingsProvider.useBlackTheme) { | ||||||
|  |         darkColorScheme = darkColorScheme | ||||||
|  |             .copyWith(background: Colors.black, surface: Colors.black) | ||||||
|  |             .harmonized(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|       return MaterialApp( |       return MaterialApp( | ||||||
|           title: 'Obtainium', |           title: 'Obtainium', | ||||||
|           localizationsDelegates: context.localizationDelegates, |           localizationsDelegates: context.localizationDelegates, | ||||||
|   | |||||||
| @@ -33,10 +33,10 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|   bool additionalSettingsValid = true; |   bool additionalSettingsValid = true; | ||||||
|   List<String> pickedCategories = []; |   List<String> pickedCategories = []; | ||||||
|   int searchnum = 0; |   int searchnum = 0; | ||||||
|  |   SourceProvider sourceProvider = SourceProvider(); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |  | ||||||
|     AppsProvider appsProvider = context.read<AppsProvider>(); |     AppsProvider appsProvider = context.read<AppsProvider>(); | ||||||
|  |  | ||||||
|     bool doingSomething = gettingAppInfo || searching; |     bool doingSomething = gettingAppInfo || searching; | ||||||
| @@ -64,65 +64,56 @@ 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 { |     addApp({bool resetUserInputAfter = false}) async { | ||||||
|       setState(() { |       setState(() { | ||||||
|         gettingAppInfo = true; |         gettingAppInfo = true; | ||||||
|       }); |       }); | ||||||
|       var settingsProvider = context.read<SettingsProvider>(); |       try { | ||||||
|       () async { |         var settingsProvider = context.read<SettingsProvider>(); | ||||||
|         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; |         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; | ||||||
|         var cont = true; |         App? app; | ||||||
|         if ((userPickedTrackOnly || pickedSource!.enforceTrackOnly) && |         if ((await getTrackOnlyConfirmationIfNeeded(userPickedTrackOnly)) && | ||||||
|             // ignore: use_build_context_synchronously |             (await getReleaseDateAsVersionConfirmationIfNeeded( | ||||||
|             await showDialog( |                 userPickedTrackOnly))) { | ||||||
|                     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 (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) { |  | ||||||
|           cont = false; |  | ||||||
|         } |  | ||||||
|         if (additionalSettings['versionDetection'] == 'noVersionDetection' && |  | ||||||
|             // 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(); |  | ||||||
|           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; |           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; | ||||||
|           App app = await sourceProvider.getApp( |           app = await sourceProvider.getApp( | ||||||
|               pickedSource!, userInput, additionalSettings, |               pickedSource!, userInput, additionalSettings, | ||||||
|               trackOnlyOverride: trackOnly); |               trackOnlyOverride: trackOnly); | ||||||
|           if (!trackOnly) { |           if (!trackOnly) { | ||||||
| @@ -136,7 +127,8 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|             if (apkUrl == null) { |             if (apkUrl == null) { | ||||||
|               throw ObtainiumError(tr('cancelled')); |               throw ObtainiumError(tr('cancelled')); | ||||||
|             } |             } | ||||||
|             app.preferredApkIndex = app.apkUrls.indexOf(apkUrl); |             app.preferredApkIndex = | ||||||
|  |                 app.apkUrls.map((e) => e.value).toList().indexOf(apkUrl.value); | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
|             var downloadedApk = await appsProvider.downloadApp( |             var downloadedApk = await appsProvider.downloadApp( | ||||||
|                 app, globalNavigatorKey.currentContext); |                 app, globalNavigatorKey.currentContext); | ||||||
| @@ -149,262 +141,253 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|             app.installedVersion = app.latestVersion; |             app.installedVersion = app.latestVersion; | ||||||
|           } |           } | ||||||
|           app.categories = pickedCategories; |           app.categories = pickedCategories; | ||||||
|           await appsProvider.saveApps([app]); |           await appsProvider.saveApps([app], onlyIfExists: false); | ||||||
|  |  | ||||||
|           return app; |  | ||||||
|         } |         } | ||||||
|       }() |  | ||||||
|           .then((app) { |  | ||||||
|         if (app != null) { |         if (app != null) { | ||||||
|           Navigator.push(context, |           Navigator.push(globalNavigatorKey.currentContext ?? context, | ||||||
|               MaterialPageRoute(builder: (context) => AppPage(appId: app.id))); |               MaterialPageRoute(builder: (context) => AppPage(appId: app!.id))); | ||||||
|         } |         } | ||||||
|       }).catchError((e) { |       } catch (e) { | ||||||
|         showError(e, context); |         showError(e, context); | ||||||
|       }).whenComplete(() { |       } finally { | ||||||
|         setState(() { |         setState(() { | ||||||
|           gettingAppInfo = false; |           gettingAppInfo = false; | ||||||
|           if (resetUserInputAfter) { |           if (resetUserInputAfter) { | ||||||
|             changeUserInput('', false, true); |             changeUserInput('', false, true); | ||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|       }); |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     Widget getUrlInputRow() => Row( | ||||||
|  |           children: [ | ||||||
|  |             Expanded( | ||||||
|  |                 child: GeneratedForm( | ||||||
|  |                     key: Key(searchnum.toString()), | ||||||
|  |                     items: [ | ||||||
|  |                       [ | ||||||
|  |                         GeneratedFormTextField('appSourceURL', | ||||||
|  |                             label: tr('appSourceURL'), | ||||||
|  |                             defaultValue: userInput, | ||||||
|  |                             additionalValidators: [ | ||||||
|  |                               (value) { | ||||||
|  |                                 try { | ||||||
|  |                                   sourceProvider | ||||||
|  |                                       .getSource(value ?? '') | ||||||
|  |                                       .standardizeURL( | ||||||
|  |                                           preStandardizeUrl(value ?? '')); | ||||||
|  |                                 } catch (e) { | ||||||
|  |                                   return e is String | ||||||
|  |                                       ? e | ||||||
|  |                                       : e is ObtainiumError | ||||||
|  |                                           ? e.toString() | ||||||
|  |                                           : tr('error'); | ||||||
|  |                                 } | ||||||
|  |                                 return null; | ||||||
|  |                               } | ||||||
|  |                             ]) | ||||||
|  |                       ] | ||||||
|  |                     ], | ||||||
|  |                     onValueChanges: (values, valid, isBuilding) { | ||||||
|  |                       changeUserInput( | ||||||
|  |                           values['appSourceURL']!, valid, isBuilding); | ||||||
|  |                     })), | ||||||
|  |             const SizedBox( | ||||||
|  |               width: 16, | ||||||
|  |             ), | ||||||
|  |             gettingAppInfo | ||||||
|  |                 ? const CircularProgressIndicator() | ||||||
|  |                 : ElevatedButton( | ||||||
|  |                     onPressed: doingSomething || | ||||||
|  |                             pickedSource == null || | ||||||
|  |                             (pickedSource!.combinedAppSpecificSettingFormItems | ||||||
|  |                                     .isNotEmpty && | ||||||
|  |                                 !additionalSettingsValid) | ||||||
|  |                         ? null | ||||||
|  |                         : () { | ||||||
|  |                             HapticFeedback.selectionClick(); | ||||||
|  |                             addApp(); | ||||||
|  |                           }, | ||||||
|  |                     child: Text(tr('add'))) | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     runSearch() async { | ||||||
|  |       setState(() { | ||||||
|  |         searching = true; | ||||||
|  |       }); | ||||||
|  |       try { | ||||||
|  |         var results = await Future.wait(sourceProvider.sources | ||||||
|  |             .where((e) => e.canSearch) | ||||||
|  |             .map((e) => e.search(searchQuery))); | ||||||
|  |  | ||||||
|  |         // .then((results) async { | ||||||
|  |         // Interleave results instead of simple reduce | ||||||
|  |         Map<String, String> res = {}; | ||||||
|  |         var si = 0; | ||||||
|  |         var done = false; | ||||||
|  |         while (!done) { | ||||||
|  |           done = true; | ||||||
|  |           for (var r in results) { | ||||||
|  |             if (r.length > si) { | ||||||
|  |               done = false; | ||||||
|  |               res.addEntries([r.entries.elementAt(si)]); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |           si++; | ||||||
|  |         } | ||||||
|  |         List<String>? selectedUrls = res.isEmpty | ||||||
|  |             ? [] | ||||||
|  |             // ignore: use_build_context_synchronously | ||||||
|  |             : await showDialog<List<String>?>( | ||||||
|  |                 context: context, | ||||||
|  |                 builder: (BuildContext ctx) { | ||||||
|  |                   return UrlSelectionModal( | ||||||
|  |                     urlsWithDescriptions: res, | ||||||
|  |                     selectedByDefault: false, | ||||||
|  |                     onlyOneSelectionAllowed: true, | ||||||
|  |                   ); | ||||||
|  |                 }); | ||||||
|  |         if (selectedUrls != null && selectedUrls.isNotEmpty) { | ||||||
|  |           changeUserInput(selectedUrls[0], true, false, isSearch: true); | ||||||
|  |         } | ||||||
|  |       } catch (e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       } finally { | ||||||
|  |         setState(() { | ||||||
|  |           searching = false; | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     bool shouldShowSearchBar() => | ||||||
|  |         sourceProvider.sources.where((e) => e.canSearch).isNotEmpty && | ||||||
|  |         pickedSource == null && | ||||||
|  |         userInput.isEmpty; | ||||||
|  |  | ||||||
|  |     Widget getSearchBarRow() => Row( | ||||||
|  |           children: [ | ||||||
|  |             Expanded( | ||||||
|  |               child: GeneratedForm( | ||||||
|  |                   items: [ | ||||||
|  |                     [ | ||||||
|  |                       GeneratedFormTextField('searchSomeSources', | ||||||
|  |                           label: tr('searchSomeSourcesLabel'), required: false), | ||||||
|  |                     ] | ||||||
|  |                   ], | ||||||
|  |                   onValueChanges: (values, valid, isBuilding) { | ||||||
|  |                     if (values.isNotEmpty && valid && !isBuilding) { | ||||||
|  |                       setState(() { | ||||||
|  |                         searchQuery = values['searchSomeSources']!.trim(); | ||||||
|  |                       }); | ||||||
|  |                     } | ||||||
|  |                   }), | ||||||
|  |             ), | ||||||
|  |             const SizedBox( | ||||||
|  |               width: 16, | ||||||
|  |             ), | ||||||
|  |             ElevatedButton( | ||||||
|  |                 onPressed: searchQuery.isEmpty || doingSomething | ||||||
|  |                     ? null | ||||||
|  |                     : () { | ||||||
|  |                         runSearch(); | ||||||
|  |                       }, | ||||||
|  |                 child: Text(tr('search'))) | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     Widget getAdditionalOptsCol() => Column( | ||||||
|  |           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |           children: [ | ||||||
|  |             const Divider( | ||||||
|  |               height: 64, | ||||||
|  |             ), | ||||||
|  |             Text( | ||||||
|  |                 tr('additionalOptsFor', | ||||||
|  |                     args: [pickedSource?.name ?? tr('source')]), | ||||||
|  |                 style: TextStyle(color: Theme.of(context).colorScheme.primary)), | ||||||
|  |             const SizedBox( | ||||||
|  |               height: 16, | ||||||
|  |             ), | ||||||
|  |             GeneratedForm( | ||||||
|  |                 key: Key(pickedSource.runtimeType.toString()), | ||||||
|  |                 items: pickedSource!.combinedAppSpecificSettingFormItems, | ||||||
|  |                 onValueChanges: (values, valid, isBuilding) { | ||||||
|  |                   if (!isBuilding) { | ||||||
|  |                     setState(() { | ||||||
|  |                       additionalSettings = values; | ||||||
|  |                       additionalSettingsValid = valid; | ||||||
|  |                     }); | ||||||
|  |                   } | ||||||
|  |                 }), | ||||||
|  |             Column( | ||||||
|  |               children: [ | ||||||
|  |                 const SizedBox( | ||||||
|  |                   height: 16, | ||||||
|  |                 ), | ||||||
|  |                 CategoryEditorSelector( | ||||||
|  |                     alignment: WrapAlignment.start, | ||||||
|  |                     onSelected: (categories) { | ||||||
|  |                       pickedCategories = categories; | ||||||
|  |                     }), | ||||||
|  |               ], | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     Widget getSourcesListWidget() => Column( | ||||||
|  |             crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|  |             mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |             children: [ | ||||||
|  |               const SizedBox( | ||||||
|  |                 height: 48, | ||||||
|  |               ), | ||||||
|  |               Text( | ||||||
|  |                 tr('supportedSourcesBelow'), | ||||||
|  |               ), | ||||||
|  |               const SizedBox( | ||||||
|  |                 height: 8, | ||||||
|  |               ), | ||||||
|  |               ...sourceProvider.sources | ||||||
|  |                   .map((e) => GestureDetector( | ||||||
|  |                       onTap: e.host != null | ||||||
|  |                           ? () { | ||||||
|  |                               launchUrlString('https://${e.host}', | ||||||
|  |                                   mode: LaunchMode.externalApplication); | ||||||
|  |                             } | ||||||
|  |                           : null, | ||||||
|  |                       child: Text( | ||||||
|  |                         '${e.name}${e.enforceTrackOnly ? ' ${tr('trackOnlyInBrackets')}' : ''}${e.canSearch ? ' ${tr('searchableInBrackets')}' : ''}', | ||||||
|  |                         style: TextStyle( | ||||||
|  |                             decoration: e.host != null | ||||||
|  |                                 ? TextDecoration.underline | ||||||
|  |                                 : TextDecoration.none, | ||||||
|  |                             fontStyle: FontStyle.italic), | ||||||
|  |                       ))) | ||||||
|  |                   .toList() | ||||||
|  |             ]); | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|         backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|         body: CustomScrollView(slivers: <Widget>[ |         body: CustomScrollView(shrinkWrap: true, slivers: <Widget>[ | ||||||
|           CustomAppBar(title: tr('addApp')), |           CustomAppBar(title: tr('addApp')), | ||||||
|           SliverFillRemaining( |           SliverToBoxAdapter( | ||||||
|             child: Padding( |             child: Padding( | ||||||
|                 padding: const EdgeInsets.all(16), |                 padding: const EdgeInsets.all(16), | ||||||
|                 child: Column( |                 child: Column( | ||||||
|  |                     mainAxisSize: MainAxisSize.min, | ||||||
|                     crossAxisAlignment: CrossAxisAlignment.stretch, |                     crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|                     children: [ |                     children: [ | ||||||
|                       Row( |                       getUrlInputRow(), | ||||||
|                         children: [ |                       if (shouldShowSearchBar()) | ||||||
|                           Expanded( |  | ||||||
|                               child: GeneratedForm( |  | ||||||
|                                   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) |  | ||||||
|                         const SizedBox( |                         const SizedBox( | ||||||
|                           height: 16, |                           height: 16, | ||||||
|                         ), |                         ), | ||||||
|                       if (sourceProvider.sources |                       if (shouldShowSearchBar()) getSearchBarRow(), | ||||||
|                               .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 (pickedSource != null) |                       if (pickedSource != null) | ||||||
|                         Column( |                         getAdditionalOptsCol() | ||||||
|                           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; |  | ||||||
|                                     }), |  | ||||||
|                               ], |  | ||||||
|                             ), |  | ||||||
|                           ], |  | ||||||
|                         ) |  | ||||||
|                       else |                       else | ||||||
|                         Expanded( |                         getSourcesListWidget(), | ||||||
|                             child: Column( |  | ||||||
|                                 crossAxisAlignment: CrossAxisAlignment.center, |  | ||||||
|                                 mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|                                 children: [ |  | ||||||
|                               const SizedBox( |  | ||||||
|                                 height: 48, |  | ||||||
|                               ), |  | ||||||
|                               Text( |  | ||||||
|                                 tr('supportedSourcesBelow'), |  | ||||||
|                               ), |  | ||||||
|                               const SizedBox( |  | ||||||
|                                 height: 8, |  | ||||||
|                               ), |  | ||||||
|                               ...sourceProvider.sources |  | ||||||
|                                   .map((e) => GestureDetector( |  | ||||||
|                                       onTap: e.host != null |  | ||||||
|                                           ? () { |  | ||||||
|                                               launchUrlString( |  | ||||||
|                                                   'https://${e.host}', |  | ||||||
|                                                   mode: LaunchMode |  | ||||||
|                                                       .externalApplication); |  | ||||||
|                                             } |  | ||||||
|                                           : null, |  | ||||||
|                                       child: Text( |  | ||||||
|                                         '${e.name}${e.enforceTrackOnly ? ' ${tr('trackOnlyInBrackets')}' : ''}${e.canSearch ? ' ${tr('searchableInBrackets')}' : ''}', |  | ||||||
|                                         style: TextStyle( |  | ||||||
|                                             decoration: e.host != null |  | ||||||
|                                                 ? TextDecoration.underline |  | ||||||
|                                                 : TextDecoration.none, |  | ||||||
|                                             fontStyle: FontStyle.italic), |  | ||||||
|                                       ))) |  | ||||||
|                                   .toList() |  | ||||||
|                             ])), |  | ||||||
|                       const SizedBox( |                       const SizedBox( | ||||||
|                         height: 8, |                         height: 8, | ||||||
|                       ), |                       ), | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
| @@ -34,406 +35,433 @@ class _AppPageState extends State<AppPage> { | |||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     bool areDownloadsRunning = appsProvider.areDownloadsRunning(); | ||||||
|  |  | ||||||
|     var sourceProvider = SourceProvider(); |     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; |     var source = app != null ? sourceProvider.getSource(app.app.url) : null; | ||||||
|     if (!appsProvider.areDownloadsRunning() && prevApp == null && app != null) { |     if (!areDownloadsRunning && prevApp == null && app != null) { | ||||||
|       prevApp = app; |       prevApp = app; | ||||||
|       getUpdate(app.app.id); |       getUpdate(app.app.id); | ||||||
|     } |     } | ||||||
|     var trackOnly = app?.app.additionalSettings['trackOnly'] == true; |     var trackOnly = app?.app.additionalSettings['trackOnly'] == true; | ||||||
|  |  | ||||||
|     var infoColumn = Column( |     bool isVersionDetectionStandard = | ||||||
|       mainAxisAlignment: MainAxisAlignment.center, |         app?.app.additionalSettings['versionDetection'] == | ||||||
|       crossAxisAlignment: CrossAxisAlignment.stretch, |             'standardVersionDetection'; | ||||||
|       children: [ |  | ||||||
|         GestureDetector( |  | ||||||
|             onTap: () { |  | ||||||
|               if (app?.app.url != null) { |  | ||||||
|                 launchUrlString(app?.app.url ?? '', |  | ||||||
|                     mode: LaunchMode.externalApplication); |  | ||||||
|               } |  | ||||||
|             }, |  | ||||||
|             child: Text( |  | ||||||
|               app?.app.url ?? '', |  | ||||||
|               textAlign: TextAlign.center, |  | ||||||
|               style: const TextStyle( |  | ||||||
|                   decoration: TextDecoration.underline, |  | ||||||
|                   fontStyle: FontStyle.italic, |  | ||||||
|                   fontSize: 12), |  | ||||||
|             )), |  | ||||||
|         const SizedBox( |  | ||||||
|           height: 32, |  | ||||||
|         ), |  | ||||||
|         Text( |  | ||||||
|           tr('latestVersionX', args: [app?.app.latestVersion ?? tr('unknown')]), |  | ||||||
|           textAlign: TextAlign.center, |  | ||||||
|           style: Theme.of(context).textTheme.bodyLarge, |  | ||||||
|         ), |  | ||||||
|         Text( |  | ||||||
|           '${tr('installedVersionX', args: [ |  | ||||||
|                 app?.app.installedVersion ?? tr('none') |  | ||||||
|               ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [ |  | ||||||
|                   tr('app') |  | ||||||
|                 ])}' : ''}', |  | ||||||
|           textAlign: TextAlign.center, |  | ||||||
|           style: Theme.of(context).textTheme.bodyLarge, |  | ||||||
|         ), |  | ||||||
|         const SizedBox( |  | ||||||
|           height: 32, |  | ||||||
|         ), |  | ||||||
|         Text( |  | ||||||
|           tr('lastUpdateCheckX', args: [ |  | ||||||
|             app?.app.lastUpdateCheck == null |  | ||||||
|                 ? tr('never') |  | ||||||
|                 : '\n${app?.app.lastUpdateCheck?.toLocal()}' |  | ||||||
|           ]), |  | ||||||
|           textAlign: TextAlign.center, |  | ||||||
|           style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12), |  | ||||||
|         ), |  | ||||||
|         const SizedBox( |  | ||||||
|           height: 48, |  | ||||||
|         ), |  | ||||||
|         CategoryEditorSelector( |  | ||||||
|             alignment: WrapAlignment.center, |  | ||||||
|             preselected: |  | ||||||
|                 app?.app.categories != null ? app!.app.categories.toSet() : {}, |  | ||||||
|             onSelected: (categories) { |  | ||||||
|               if (app != null) { |  | ||||||
|                 app.app.categories = categories; |  | ||||||
|                 appsProvider.saveApps([app.app]); |  | ||||||
|               } |  | ||||||
|             }), |  | ||||||
|       ], |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     var fullInfoColumn = Column( |     getInfoColumn() => Column( | ||||||
|       mainAxisAlignment: MainAxisAlignment.center, |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|       crossAxisAlignment: CrossAxisAlignment.stretch, |           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|       children: [ |           children: [ | ||||||
|         const SizedBox(height: 125), |             GestureDetector( | ||||||
|         app?.installedInfo != null |                 onTap: () { | ||||||
|             ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [ |                   if (app?.app.url != null) { | ||||||
|                 Image.memory( |                     launchUrlString(app?.app.url ?? '', | ||||||
|                   app!.installedInfo!.icon!, |                         mode: LaunchMode.externalApplication); | ||||||
|                   height: 150, |                   } | ||||||
|                   gaplessPlayback: true, |                 }, | ||||||
|                 ) |                 onLongPress: () { | ||||||
|               ]) |                   Clipboard.setData(ClipboardData(text: app?.app.url ?? '')); | ||||||
|             : Container(), |                   ScaffoldMessenger.of(context).showSnackBar(SnackBar( | ||||||
|         const SizedBox( |                     content: Text(tr('copiedToClipboard')), | ||||||
|           height: 25, |                   )); | ||||||
|         ), |                 }, | ||||||
|         Text( |                 child: Text( | ||||||
|           app?.installedInfo?.name ?? app?.app.name ?? tr('app'), |                   app?.app.url ?? '', | ||||||
|           textAlign: TextAlign.center, |                   textAlign: TextAlign.center, | ||||||
|           style: Theme.of(context).textTheme.displayLarge, |                   style: const TextStyle( | ||||||
|         ), |                       decoration: TextDecoration.underline, | ||||||
|         Text( |                       fontStyle: FontStyle.italic, | ||||||
|           tr('byX', args: [app?.app.author ?? tr('unknown')]), |                       fontSize: 12), | ||||||
|           textAlign: TextAlign.center, |                 )), | ||||||
|           style: Theme.of(context).textTheme.headlineMedium, |             const SizedBox( | ||||||
|         ), |               height: 32, | ||||||
|         const SizedBox( |             ), | ||||||
|           height: 8, |             Text( | ||||||
|         ), |               tr('latestVersionX', | ||||||
|         Text( |                   args: [app?.app.latestVersion ?? tr('unknown')]), | ||||||
|           app?.app.id ?? '', |               textAlign: TextAlign.center, | ||||||
|           textAlign: TextAlign.center, |               style: Theme.of(context).textTheme.bodyLarge, | ||||||
|           style: Theme.of(context).textTheme.labelSmall, |             ), | ||||||
|         ), |             Text( | ||||||
|         app?.app.releaseDate == null |               '${tr('installedVersionX', args: [ | ||||||
|             ? const SizedBox.shrink() |                     app?.app.installedVersion ?? tr('none') | ||||||
|             : Text( |                   ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [ | ||||||
|                 app!.app.releaseDate.toString(), |                       tr('app') | ||||||
|                 textAlign: TextAlign.center, |                     ])}' : ''}', | ||||||
|                 style: Theme.of(context).textTheme.labelSmall, |               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( |             const SizedBox( | ||||||
|           height: 32, |               height: 32, | ||||||
|         ), |             ), | ||||||
|         infoColumn, |             Text( | ||||||
|         const SizedBox(height: 150) |               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); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     getResetInstallStatusButton() => TextButton( | ||||||
|  |         onPressed: app?.app == null | ||||||
|  |             ? null | ||||||
|  |             : () { | ||||||
|  |                 app!.app.installedVersion = null; | ||||||
|  |                 appsProvider.saveApps([app.app]); | ||||||
|  |               }, | ||||||
|  |         child: Text(tr('resetInstallStatus'))); | ||||||
|  |  | ||||||
|  |     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: !isVersionDetectionStandard && | ||||||
|  |                                   app?.app.installedVersion != null && | ||||||
|  |                                   app?.app.installedVersion == | ||||||
|  |                                       app?.app.latestVersion | ||||||
|  |                               ? getResetInstallStatusButton() | ||||||
|  |                               : 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( |     return Scaffold( | ||||||
|       appBar: settingsProvider.showAppWebpage ? AppBar() : null, |         appBar: settingsProvider.showAppWebpage ? AppBar() : null, | ||||||
|       backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|       body: RefreshIndicator( |         body: RefreshIndicator( | ||||||
|           child: settingsProvider.showAppWebpage |             child: settingsProvider.showAppWebpage | ||||||
|               ? app != null |                 ? getAppWebView() | ||||||
|                   ? WebViewWidget( |                 : CustomScrollView( | ||||||
|                       controller: WebViewController() |                     slivers: [ | ||||||
|                         ..setJavaScriptMode(JavaScriptMode.unrestricted) |                       SliverToBoxAdapter( | ||||||
|                         ..setBackgroundColor( |                           child: Column(children: [getFullInfoColumn()])), | ||||||
|                             Theme.of(context).colorScheme.background) |                     ], | ||||||
|                         ..setJavaScriptMode(JavaScriptMode.unrestricted) |                   ), | ||||||
|                         ..setNavigationDelegate( |             onRefresh: () async { | ||||||
|                           NavigationDelegate( |               if (app != null) { | ||||||
|                             onWebResourceError: (WebResourceError error) { |                 getUpdate(app.app.id); | ||||||
|                               if (error.isForMainFrame == true) { |               } | ||||||
|                                 showError( |             }), | ||||||
|                                     ObtainiumError(error.description, |         bottomSheet: getBottomSheetMenu()); | ||||||
|                                         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 (app?.app.additionalSettings['versionDetection'] != |  | ||||||
|                                 'standardVersionDetection' && |  | ||||||
|                             !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[ |  | ||||||
|                                                   '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); |  | ||||||
|                                           }); |  | ||||||
|                                         } |  | ||||||
|                                       }); |  | ||||||
|                                     }, |  | ||||||
|                               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)) |  | ||||||
|             ], |  | ||||||
|           )), |  | ||||||
|     ); |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1609
									
								
								lib/pages/apps.dart
									
									
									
									
									
								
							
							
						
						| @@ -30,6 +30,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|     SourceProvider sourceProvider = SourceProvider(); |     SourceProvider sourceProvider = SourceProvider(); | ||||||
|     var appsProvider = context.read<AppsProvider>(); |     var appsProvider = context.read<AppsProvider>(); | ||||||
|     var settingsProvider = context.read<SettingsProvider>(); |     var settingsProvider = context.read<SettingsProvider>(); | ||||||
|  |  | ||||||
|     var outlineButtonStyle = ButtonStyle( |     var outlineButtonStyle = ButtonStyle( | ||||||
|       shape: MaterialStateProperty.all( |       shape: MaterialStateProperty.all( | ||||||
|         StadiumBorder( |         StadiumBorder( | ||||||
| @@ -101,6 +102,193 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     runObtainiumExport() { | ||||||
|  |       HapticFeedback.selectionClick(); | ||||||
|  |       appsProvider.exportApps().then((String path) { | ||||||
|  |         showError(tr('exportedTo', args: [path]), context); | ||||||
|  |       }).catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runObtainiumImport() { | ||||||
|  |       HapticFeedback.selectionClick(); | ||||||
|  |       FilePicker.platform.pickFiles().then((result) { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = true; | ||||||
|  |         }); | ||||||
|  |         if (result != null) { | ||||||
|  |           String data = File(result.files.single.path!).readAsStringSync(); | ||||||
|  |           try { | ||||||
|  |             jsonDecode(data); | ||||||
|  |           } catch (e) { | ||||||
|  |             throw ObtainiumError(tr('invalidInput')); | ||||||
|  |           } | ||||||
|  |           appsProvider.importApps(data).then((value) { | ||||||
|  |             var cats = settingsProvider.categories; | ||||||
|  |             appsProvider.apps.forEach((key, value) { | ||||||
|  |               for (var c in value.app.categories) { | ||||||
|  |                 if (!cats.containsKey(c)) { | ||||||
|  |                   cats[c] = generateRandomLightColor().value; | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |             appsProvider.addMissingCategories(settingsProvider); | ||||||
|  |             showError(tr('importedX', args: [plural('apps', value)]), context); | ||||||
|  |           }); | ||||||
|  |         } else { | ||||||
|  |           // User canceled the picker | ||||||
|  |         } | ||||||
|  |       }).catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }).whenComplete(() { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = false; | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runUrlImport() { | ||||||
|  |       FilePicker.platform.pickFiles().then((result) { | ||||||
|  |         if (result != null) { | ||||||
|  |           urlListImport( | ||||||
|  |               overrideInitValid: true, | ||||||
|  |               initValue: RegExp('https?://[^"]+') | ||||||
|  |                   .allMatches( | ||||||
|  |                       File(result.files.single.path!).readAsStringSync()) | ||||||
|  |                   .map((e) => e.input.substring(e.start, e.end)) | ||||||
|  |                   .toSet() | ||||||
|  |                   .toList() | ||||||
|  |                   .where((url) { | ||||||
|  |                 try { | ||||||
|  |                   sourceProvider.getSource(url); | ||||||
|  |                   return true; | ||||||
|  |                 } catch (e) { | ||||||
|  |                   return false; | ||||||
|  |                 } | ||||||
|  |               }).join('\n')); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runSourceSearch(AppSource source) { | ||||||
|  |       () async { | ||||||
|  |         var values = await showDialog<Map<String, dynamic>?>( | ||||||
|  |             context: context, | ||||||
|  |             builder: (BuildContext ctx) { | ||||||
|  |               return GeneratedFormModal( | ||||||
|  |                 title: tr('searchX', args: [source.name]), | ||||||
|  |                 items: [ | ||||||
|  |                   [ | ||||||
|  |                     GeneratedFormTextField('searchQuery', | ||||||
|  |                         label: tr('searchQuery')) | ||||||
|  |                   ] | ||||||
|  |                 ], | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |         if (values != null && | ||||||
|  |             (values['searchQuery'] as String?)?.isNotEmpty == true) { | ||||||
|  |           setState(() { | ||||||
|  |             importInProgress = true; | ||||||
|  |           }); | ||||||
|  |           var urlsWithDescriptions = | ||||||
|  |               await source.search(values['searchQuery'] as String); | ||||||
|  |           if (urlsWithDescriptions.isNotEmpty) { | ||||||
|  |             var selectedUrls = | ||||||
|  |                 // ignore: use_build_context_synchronously | ||||||
|  |                 await showDialog<List<String>?>( | ||||||
|  |                     context: context, | ||||||
|  |                     builder: (BuildContext ctx) { | ||||||
|  |                       return UrlSelectionModal( | ||||||
|  |                         urlsWithDescriptions: urlsWithDescriptions, | ||||||
|  |                         selectedByDefault: false, | ||||||
|  |                       ); | ||||||
|  |                     }); | ||||||
|  |             if (selectedUrls != null && selectedUrls.isNotEmpty) { | ||||||
|  |               var errors = await appsProvider.addAppsByURL(selectedUrls); | ||||||
|  |               if (errors.isEmpty) { | ||||||
|  |                 // ignore: use_build_context_synchronously | ||||||
|  |                 showError( | ||||||
|  |                     tr('importedX', args: [plural('app', selectedUrls.length)]), | ||||||
|  |                     context); | ||||||
|  |               } else { | ||||||
|  |                 // ignore: use_build_context_synchronously | ||||||
|  |                 showDialog( | ||||||
|  |                     context: context, | ||||||
|  |                     builder: (BuildContext ctx) { | ||||||
|  |                       return ImportErrorDialog( | ||||||
|  |                           urlsLength: selectedUrls.length, errors: errors); | ||||||
|  |                     }); | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } else { | ||||||
|  |             throw ObtainiumError(tr('noResults')); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }() | ||||||
|  |           .catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }).whenComplete(() { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = false; | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     runMassSourceImport(MassAppUrlSource source) { | ||||||
|  |       () async { | ||||||
|  |         var values = await showDialog<Map<String, dynamic>?>( | ||||||
|  |             context: context, | ||||||
|  |             builder: (BuildContext ctx) { | ||||||
|  |               return GeneratedFormModal( | ||||||
|  |                 title: tr('importX', args: [source.name]), | ||||||
|  |                 items: source.requiredArgs | ||||||
|  |                     .map((e) => [GeneratedFormTextField(e, label: e)]) | ||||||
|  |                     .toList(), | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |         if (values != null) { | ||||||
|  |           setState(() { | ||||||
|  |             importInProgress = true; | ||||||
|  |           }); | ||||||
|  |           var urlsWithDescriptions = await source.getUrlsWithDescriptions( | ||||||
|  |               values.values.map((e) => e.toString()).toList()); | ||||||
|  |           var selectedUrls = | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               await showDialog<List<String>?>( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return UrlSelectionModal( | ||||||
|  |                         urlsWithDescriptions: urlsWithDescriptions); | ||||||
|  |                   }); | ||||||
|  |           if (selectedUrls != null) { | ||||||
|  |             var errors = await appsProvider.addAppsByURL(selectedUrls); | ||||||
|  |             if (errors.isEmpty) { | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               showError( | ||||||
|  |                   tr('importedX', args: [plural('app', selectedUrls.length)]), | ||||||
|  |                   context); | ||||||
|  |             } else { | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               showDialog( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return ImportErrorDialog( | ||||||
|  |                         urlsLength: selectedUrls.length, errors: errors); | ||||||
|  |                   }); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }() | ||||||
|  |           .catchError((e) { | ||||||
|  |         showError(e, context); | ||||||
|  |       }).whenComplete(() { | ||||||
|  |         setState(() { | ||||||
|  |           importInProgress = false; | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|         backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|         body: CustomScrollView(slivers: <Widget>[ |         body: CustomScrollView(slivers: <Widget>[ | ||||||
| @@ -120,18 +308,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                   onPressed: appsProvider.apps.isEmpty || |                                   onPressed: appsProvider.apps.isEmpty || | ||||||
|                                           importInProgress |                                           importInProgress | ||||||
|                                       ? null |                                       ? null | ||||||
|                                       : () { |                                       : runObtainiumExport, | ||||||
|                                           HapticFeedback.selectionClick(); |  | ||||||
|                                           appsProvider |  | ||||||
|                                               .exportApps() |  | ||||||
|                                               .then((String path) { |  | ||||||
|                                             showError( |  | ||||||
|                                                 tr('exportedTo', args: [path]), |  | ||||||
|                                                 context); |  | ||||||
|                                           }).catchError((e) { |  | ||||||
|                                             showError(e, context); |  | ||||||
|                                           }); |  | ||||||
|                                         }, |  | ||||||
|                                   child: Text(tr('obtainiumExport')))), |                                   child: Text(tr('obtainiumExport')))), | ||||||
|                           const SizedBox( |                           const SizedBox( | ||||||
|                             width: 16, |                             width: 16, | ||||||
| @@ -141,59 +318,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                   style: outlineButtonStyle, |                                   style: outlineButtonStyle, | ||||||
|                                   onPressed: importInProgress |                                   onPressed: importInProgress | ||||||
|                                       ? null |                                       ? null | ||||||
|                                       : () { |                                       : runObtainiumImport, | ||||||
|                                           HapticFeedback.selectionClick(); |  | ||||||
|                                           FilePicker.platform |  | ||||||
|                                               .pickFiles() |  | ||||||
|                                               .then((result) { |  | ||||||
|                                             setState(() { |  | ||||||
|                                               importInProgress = true; |  | ||||||
|                                             }); |  | ||||||
|                                             if (result != null) { |  | ||||||
|                                               String data = File( |  | ||||||
|                                                       result.files.single.path!) |  | ||||||
|                                                   .readAsStringSync(); |  | ||||||
|                                               try { |  | ||||||
|                                                 jsonDecode(data); |  | ||||||
|                                               } catch (e) { |  | ||||||
|                                                 throw ObtainiumError( |  | ||||||
|                                                     tr('invalidInput')); |  | ||||||
|                                               } |  | ||||||
|                                               appsProvider |  | ||||||
|                                                   .importApps(data) |  | ||||||
|                                                   .then((value) { |  | ||||||
|                                                 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; |  | ||||||
|                                             }); |  | ||||||
|                                           }); |  | ||||||
|                                         }, |  | ||||||
|                                   child: Text(tr('obtainiumImport')))) |                                   child: Text(tr('obtainiumImport')))) | ||||||
|                         ], |                         ], | ||||||
|                       ), |                       ), | ||||||
| @@ -216,49 +341,15 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                               height: 32, |                               height: 32, | ||||||
|                             ), |                             ), | ||||||
|                             TextButton( |                             TextButton( | ||||||
|                                 onPressed: importInProgress |                                 onPressed: | ||||||
|                                     ? null |                                     importInProgress ? null : urlListImport, | ||||||
|                                     : () { |  | ||||||
|                                         urlListImport(); |  | ||||||
|                                       }, |  | ||||||
|                                 child: Text( |                                 child: Text( | ||||||
|                                   tr('importFromURLList'), |                                   tr('importFromURLList'), | ||||||
|                                 )), |                                 )), | ||||||
|                             const SizedBox(height: 8), |                             const SizedBox(height: 8), | ||||||
|                             TextButton( |                             TextButton( | ||||||
|                                 onPressed: importInProgress |                                 onPressed: | ||||||
|                                     ? null |                                     importInProgress ? null : 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')); |  | ||||||
|                                           } |  | ||||||
|                                         }); |  | ||||||
|                                       }, |  | ||||||
|                                 child: Text( |                                 child: Text( | ||||||
|                                   tr('importFromURLsInFile'), |                                   tr('importFromURLsInFile'), | ||||||
|                                 )), |                                 )), | ||||||
| @@ -275,106 +366,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                         onPressed: importInProgress |                                         onPressed: importInProgress | ||||||
|                                             ? null |                                             ? null | ||||||
|                                             : () { |                                             : () { | ||||||
|                                                 () async { |                                                 runSourceSearch(source); | ||||||
|                                                   var values = await showDialog< |  | ||||||
|                                                           Map<String, |  | ||||||
|                                                               dynamic>?>( |  | ||||||
|                                                       context: context, |  | ||||||
|                                                       builder: |  | ||||||
|                                                           (BuildContext ctx) { |  | ||||||
|                                                         return GeneratedFormModal( |  | ||||||
|                                                           title: tr('searchX', |  | ||||||
|                                                               args: [ |  | ||||||
|                                                                 source.name |  | ||||||
|                                                               ]), |  | ||||||
|                                                           items: [ |  | ||||||
|                                                             [ |  | ||||||
|                                                               GeneratedFormTextField( |  | ||||||
|                                                                   'searchQuery', |  | ||||||
|                                                                   label: tr( |  | ||||||
|                                                                       'searchQuery')) |  | ||||||
|                                                             ] |  | ||||||
|                                                           ], |  | ||||||
|                                                         ); |  | ||||||
|                                                       }); |  | ||||||
|                                                   if (values != null && |  | ||||||
|                                                       (values['searchQuery'] |  | ||||||
|                                                                   as String?) |  | ||||||
|                                                               ?.isNotEmpty == |  | ||||||
|                                                           true) { |  | ||||||
|                                                     setState(() { |  | ||||||
|                                                       importInProgress = true; |  | ||||||
|                                                     }); |  | ||||||
|                                                     var urlsWithDescriptions = |  | ||||||
|                                                         await source.search( |  | ||||||
|                                                             values['searchQuery'] |  | ||||||
|                                                                 as String); |  | ||||||
|                                                     if (urlsWithDescriptions |  | ||||||
|                                                         .isNotEmpty) { |  | ||||||
|                                                       var selectedUrls = |  | ||||||
|                                                           // 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; |  | ||||||
|                                                   }); |  | ||||||
|                                                 }); |  | ||||||
|                                               }, |                                               }, | ||||||
|                                         child: Text( |                                         child: Text( | ||||||
|                                             tr('searchX', args: [source.name]))) |                                             tr('searchX', args: [source.name]))) | ||||||
| @@ -390,93 +382,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                         onPressed: importInProgress |                                         onPressed: importInProgress | ||||||
|                                             ? null |                                             ? null | ||||||
|                                             : () { |                                             : () { | ||||||
|                                                 () async { |                                                 runMassSourceImport(source); | ||||||
|                                                   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; |  | ||||||
|                                                   }); |  | ||||||
|                                                 }); |  | ||||||
|                                               }, |                                               }, | ||||||
|                                         child: Text( |                                         child: Text( | ||||||
|                                             tr('importX', args: [source.name]))) |                                             tr('importX', args: [source.name]))) | ||||||
| @@ -600,7 +506,7 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|           widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')), |           widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')), | ||||||
|       content: Column(children: [ |       content: Column(children: [ | ||||||
|         ...urlWithDescriptionSelections.keys.map((urlWithD) { |         ...urlWithDescriptionSelections.keys.map((urlWithD) { | ||||||
|           select(bool? value) { |           selectThis(bool? value) { | ||||||
|             setState(() { |             setState(() { | ||||||
|               value ??= false; |               value ??= false; | ||||||
|               if (value! && widget.onlyOneSelectionAllowed) { |               if (value! && widget.onlyOneSelectionAllowed) { | ||||||
| @@ -611,11 +517,56 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|             }); |             }); | ||||||
|           } |           } | ||||||
|  |  | ||||||
|           return Row(children: [ |           var urlLink = GestureDetector( | ||||||
|  |               onTap: () { | ||||||
|  |                 launchUrlString(urlWithD.key, | ||||||
|  |                     mode: LaunchMode.externalApplication); | ||||||
|  |               }, | ||||||
|  |               child: Text( | ||||||
|  |                 Uri.parse(urlWithD.key).path.substring(1), | ||||||
|  |                 style: const TextStyle(decoration: TextDecoration.underline), | ||||||
|  |                 textAlign: TextAlign.start, | ||||||
|  |               )); | ||||||
|  |  | ||||||
|  |           var descriptionText = Text( | ||||||
|  |             urlWithD.value.length > 128 | ||||||
|  |                 ? '${urlWithD.value.substring(0, 128)}...' | ||||||
|  |                 : urlWithD.value, | ||||||
|  |             style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12), | ||||||
|  |           ); | ||||||
|  |  | ||||||
|  |           var selectedUrlsWithDs = urlWithDescriptionSelections.entries | ||||||
|  |               .where((e) => e.value) | ||||||
|  |               .toList(); | ||||||
|  |  | ||||||
|  |           var singleSelectTile = ListTile( | ||||||
|  |             title: urlLink, | ||||||
|  |             subtitle: GestureDetector( | ||||||
|  |               onTap: () { | ||||||
|  |                 setState(() { | ||||||
|  |                   selectOnlyOne(urlWithD.key); | ||||||
|  |                 }); | ||||||
|  |               }, | ||||||
|  |               child: descriptionText, | ||||||
|  |             ), | ||||||
|  |             leading: Radio<String>( | ||||||
|  |               value: urlWithD.key, | ||||||
|  |               groupValue: selectedUrlsWithDs.isEmpty | ||||||
|  |                   ? null | ||||||
|  |                   : selectedUrlsWithDs.first.key.key, | ||||||
|  |               onChanged: (value) { | ||||||
|  |                 setState(() { | ||||||
|  |                   selectOnlyOne(urlWithD.key); | ||||||
|  |                 }); | ||||||
|  |               }, | ||||||
|  |             ), | ||||||
|  |           ); | ||||||
|  |  | ||||||
|  |           var multiSelectTile = Row(children: [ | ||||||
|             Checkbox( |             Checkbox( | ||||||
|                 value: urlWithDescriptionSelections[urlWithD], |                 value: urlWithDescriptionSelections[urlWithD], | ||||||
|                 onChanged: (value) { |                 onChanged: (value) { | ||||||
|                   select(value); |                   selectThis(value); | ||||||
|                 }), |                 }), | ||||||
|             const SizedBox( |             const SizedBox( | ||||||
|               width: 8, |               width: 8, | ||||||
| @@ -628,28 +579,13 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
|                   height: 8, |                   height: 8, | ||||||
|                 ), |                 ), | ||||||
|                 GestureDetector( |                 urlLink, | ||||||
|                     onTap: () { |  | ||||||
|                       launchUrlString(urlWithD.key, |  | ||||||
|                           mode: LaunchMode.externalApplication); |  | ||||||
|                     }, |  | ||||||
|                     child: Text( |  | ||||||
|                       Uri.parse(urlWithD.key).path.substring(1), |  | ||||||
|                       style: |  | ||||||
|                           const TextStyle(decoration: TextDecoration.underline), |  | ||||||
|                       textAlign: TextAlign.start, |  | ||||||
|                     )), |  | ||||||
|                 GestureDetector( |                 GestureDetector( | ||||||
|                   onTap: () { |                   onTap: () { | ||||||
|                     select(!(urlWithDescriptionSelections[urlWithD] ?? false)); |                     selectThis( | ||||||
|  |                         !(urlWithDescriptionSelections[urlWithD] ?? false)); | ||||||
|                   }, |                   }, | ||||||
|                   child: Text( |                   child: descriptionText, | ||||||
|                     urlWithD.value.length > 128 |  | ||||||
|                         ? '${urlWithD.value.substring(0, 128)}...' |  | ||||||
|                         : urlWithD.value, |  | ||||||
|                     style: const TextStyle( |  | ||||||
|                         fontStyle: FontStyle.italic, fontSize: 12), |  | ||||||
|                   ), |  | ||||||
|                 ), |                 ), | ||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
|                   height: 8, |                   height: 8, | ||||||
| @@ -657,6 +593,10 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> { | |||||||
|               ], |               ], | ||||||
|             )) |             )) | ||||||
|           ]); |           ]); | ||||||
|  |  | ||||||
|  |           return widget.onlyOneSelectionAllowed | ||||||
|  |               ? singleSelectTile | ||||||
|  |               : multiSelectTile; | ||||||
|         }) |         }) | ||||||
|       ]), |       ]), | ||||||
|       actions: [ |       actions: [ | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import 'package:obtainium/components/custom_app_bar.dart'; | |||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
|  | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| import 'package:obtainium/providers/logs_provider.dart'; | import 'package:obtainium/providers/logs_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| @@ -223,6 +224,17 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                             ), |                             ), | ||||||
|                             themeDropdown, |                             themeDropdown, | ||||||
|                             height16, |                             height16, | ||||||
|  |                             Row( | ||||||
|  |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|  |                               children: [ | ||||||
|  |                                 Text(tr('useBlackTheme')), | ||||||
|  |                                 Switch( | ||||||
|  |                                     value: settingsProvider.useBlackTheme, | ||||||
|  |                                     onChanged: (value) { | ||||||
|  |                                       settingsProvider.useBlackTheme = value; | ||||||
|  |                                     }) | ||||||
|  |                               ], | ||||||
|  |                             ), | ||||||
|                             colourDropdown, |                             colourDropdown, | ||||||
|                             height16, |                             height16, | ||||||
|                             Row( |                             Row( | ||||||
| @@ -262,6 +274,18 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                     }) |                                     }) | ||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|  |                             height16, | ||||||
|  |                             Row( | ||||||
|  |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|  |                               children: [ | ||||||
|  |                                 Text(tr('groupByCategory')), | ||||||
|  |                                 Switch( | ||||||
|  |                                     value: settingsProvider.groupByCategory, | ||||||
|  |                                     onChanged: (value) { | ||||||
|  |                                       settingsProvider.groupByCategory = value; | ||||||
|  |                                     }) | ||||||
|  |                               ], | ||||||
|  |                             ), | ||||||
|                             const Divider( |                             const Divider( | ||||||
|                               height: 16, |                               height: 16, | ||||||
|                             ), |                             ), | ||||||
| @@ -432,6 +456,7 @@ class _CategoryEditorSelectorState extends State<CategoryEditorSelector> { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     var settingsProvider = context.watch<SettingsProvider>(); |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|  |     var appsProvider = context.watch<AppsProvider>(); | ||||||
|     storedValues = settingsProvider.categories.map((key, value) => MapEntry( |     storedValues = settingsProvider.categories.map((key, value) => MapEntry( | ||||||
|         key, |         key, | ||||||
|         MapEntry(value, |         MapEntry(value, | ||||||
| @@ -455,8 +480,9 @@ class _CategoryEditorSelectorState extends State<CategoryEditorSelector> { | |||||||
|           if (!isBuilding) { |           if (!isBuilding) { | ||||||
|             storedValues = |             storedValues = | ||||||
|                 values['categories'] as Map<String, MapEntry<int, bool>>; |                 values['categories'] as Map<String, MapEntry<int, bool>>; | ||||||
|             settingsProvider.categories = |             settingsProvider.setCategories( | ||||||
|                 storedValues.map((key, value) => MapEntry(key, value.key)); |                 storedValues.map((key, value) => MapEntry(key, value.key)), | ||||||
|  |                 appsProvider: appsProvider); | ||||||
|             if (widget.onSelected != null) { |             if (widget.onSelected != null) { | ||||||
|               widget.onSelected!(storedValues.keys |               widget.onSelected!(storedValues.keys | ||||||
|                   .where((k) => storedValues[k]!.value) |                   .where((k) => storedValues[k]!.value) | ||||||
|   | |||||||
| @@ -34,6 +34,10 @@ class AppInMemory { | |||||||
|   AppInfo? installedInfo; |   AppInfo? installedInfo; | ||||||
|  |  | ||||||
|   AppInMemory(this.app, this.downloadProgress, this.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 { | class DownloadedApk { | ||||||
| @@ -73,6 +77,18 @@ List<String> generateStandardVersionRegExStrings() { | |||||||
| List<String> standardVersionRegExStrings = | List<String> standardVersionRegExStrings = | ||||||
|     generateStandardVersionRegExStrings(); |     generateStandardVersionRegExStrings(); | ||||||
|  |  | ||||||
|  | Set<String> findStandardFormatsForVersion(String version, bool strict) { | ||||||
|  |   // If !strict, even a substring match is valid | ||||||
|  |   Set<String> results = {}; | ||||||
|  |   for (var pattern in standardVersionRegExStrings) { | ||||||
|  |     if (RegExp('${strict ? '^' : ''}$pattern${strict ? '\$' : ''}') | ||||||
|  |         .hasMatch(version)) { | ||||||
|  |       results.add(pattern); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return results; | ||||||
|  | } | ||||||
|  |  | ||||||
| class AppsProvider with ChangeNotifier { | class AppsProvider with ChangeNotifier { | ||||||
|   // In memory App state (should always be kept in sync with local storage versions) |   // In memory App state (should always be kept in sync with local storage versions) | ||||||
|   Map<String, AppInMemory> apps = {}; |   Map<String, AppInMemory> apps = {}; | ||||||
| @@ -85,6 +101,8 @@ class AppsProvider with ChangeNotifier { | |||||||
|   late Stream<FGBGType>? foregroundStream; |   late Stream<FGBGType>? foregroundStream; | ||||||
|   late StreamSubscription<FGBGType>? foregroundSubscription; |   late StreamSubscription<FGBGType>? foregroundSubscription; | ||||||
|  |  | ||||||
|  |   Iterable<AppInMemory> getAppValues() => apps.values.map((a) => a.deepCopy()); | ||||||
|  |  | ||||||
|   AppsProvider() { |   AppsProvider() { | ||||||
|     // Subscribe to changes in the app foreground status |     // Subscribe to changes in the app foreground status | ||||||
|     foregroundStream = FGBGEvents.stream.asBroadcastStream(); |     foregroundStream = FGBGEvents.stream.asBroadcastStream(); | ||||||
| @@ -145,56 +163,68 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { |   Future<DownloadedApk> downloadApp(App app, BuildContext? context) async { | ||||||
|     var fileName = |  | ||||||
|         '${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'; |  | ||||||
|     String downloadUrl = await SourceProvider() |  | ||||||
|         .getSource(app.url) |  | ||||||
|         .apkUrlPrefetchModifier(app.apkUrls[app.preferredApkIndex]); |  | ||||||
|     NotificationsProvider? notificationsProvider = |     NotificationsProvider? notificationsProvider = | ||||||
|         context?.read<NotificationsProvider>(); |         context?.read<NotificationsProvider>(); | ||||||
|     var notif = DownloadNotification(app.name, 100); |     var notifId = DownloadNotification(app.finalName, 0).id; | ||||||
|     notificationsProvider?.cancel(notif.id); |     if (apps[app.id] != null) { | ||||||
|     int? prevProg; |       apps[app.id]!.downloadProgress = 0; | ||||||
|     File downloadedFile = |       notifyListeners(); | ||||||
|         await downloadFile(downloadUrl, fileName, (double? progress) { |     } | ||||||
|       int? prog = progress?.ceil(); |     try { | ||||||
|  |       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) { | ||||||
|  |         var isTempId = SourceProvider().isTempId(app); | ||||||
|  |         if (apps[app.id] != null && !isTempId) { | ||||||
|  |           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], onlyIfExists: !isTempId); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       return DownloadedApk(app.id, downloadedFile); | ||||||
|  |     } finally { | ||||||
|  |       notificationsProvider?.cancel(notifId); | ||||||
|       if (apps[app.id] != null) { |       if (apps[app.id] != null) { | ||||||
|         apps[app.id]!.downloadProgress = progress; |         apps[app.id]!.downloadProgress = null; | ||||||
|         notifyListeners(); |         notifyListeners(); | ||||||
|       } |       } | ||||||
|       notif = DownloadNotification(app.name, prog ?? 100); |  | ||||||
|       if (prog != null && prevProg != prog) { |  | ||||||
|         notificationsProvider?.notify(notif); |  | ||||||
|       } |  | ||||||
|       prevProg = prog; |  | ||||||
|     }); |  | ||||||
|     notificationsProvider?.cancel(notif.id); |  | ||||||
|     // Delete older versions of the APK if any |  | ||||||
|     for (var file in downloadedFile.parent.listSync()) { |  | ||||||
|       var fn = file.path.split('/').last; |  | ||||||
|       if (fn.startsWith('${app.id}-') && |  | ||||||
|           fn.endsWith('.apk') && |  | ||||||
|           fn != fileName) { |  | ||||||
|         file.delete(); |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|     // If the APK package ID is different from the App ID, it is either new (using a placeholder ID) or the ID has changed |  | ||||||
|     // The former case should be handled (give the App its real ID), the latter is a security issue |  | ||||||
|     var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path); |  | ||||||
|     if (app.id != newInfo.packageName) { |  | ||||||
|       if (apps[app.id] != null && !SourceProvider().isTempId(app)) { |  | ||||||
|         throw IDChangedError(); |  | ||||||
|       } |  | ||||||
|       var originalAppId = app.id; |  | ||||||
|       app.id = newInfo.packageName; |  | ||||||
|       downloadedFile = downloadedFile.renameSync( |  | ||||||
|           '${downloadedFile.parent.path}/${app.id}-${app.latestVersion}-${app.preferredApkIndex}.apk'); |  | ||||||
|       if (apps[originalAppId] != null) { |  | ||||||
|         await removeApps([originalAppId]); |  | ||||||
|         await saveApps([app]); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     return DownloadedApk(app.id, downloadedFile); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool areDownloadsRunning() => apps.values |   bool areDownloadsRunning() => apps.values | ||||||
| @@ -272,9 +302,11 @@ class AppsProvider with ChangeNotifier { | |||||||
|     await intent.launch(); |     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) |     // 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 >= 0 ? app.preferredApkIndex : 0]; | ||||||
|     // get device supported architecture |     // get device supported architecture | ||||||
|     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; |     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||||
|  |  | ||||||
| @@ -297,14 +329,14 @@ class AppsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|     // If the picked APK comes from an origin different from the source, get user confirmation (if context provided) |     // If the picked APK comes from an origin different from the source, get user confirmation (if context provided) | ||||||
|     if (apkUrl != null && |     if (apkUrl != null && | ||||||
|         getHost(apkUrl) != getHost(app.url) && |         getHost(apkUrl.value) != getHost(app.url) && | ||||||
|         context != null) { |         context != null) { | ||||||
|       // ignore: use_build_context_synchronously |       // ignore: use_build_context_synchronously | ||||||
|       if (await showDialog( |       if (await showDialog( | ||||||
|               context: context, |               context: context, | ||||||
|               builder: (BuildContext ctx) { |               builder: (BuildContext ctx) { | ||||||
|                 return APKOriginWarningDialog( |                 return APKOriginWarningDialog( | ||||||
|                     sourceUrl: app.url, apkUrl: apkUrl!); |                     sourceUrl: app.url, apkUrl: apkUrl!.value); | ||||||
|               }) != |               }) != | ||||||
|           true) { |           true) { | ||||||
|         apkUrl = null; |         apkUrl = null; | ||||||
| @@ -329,14 +361,19 @@ class AppsProvider with ChangeNotifier { | |||||||
|       if (apps[id] == null) { |       if (apps[id] == null) { | ||||||
|         throw ObtainiumError(tr('appNotFound')); |         throw ObtainiumError(tr('appNotFound')); | ||||||
|       } |       } | ||||||
|       String? apkUrl; |       MapEntry<String, String>? apkUrl; | ||||||
|       var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true; |       var trackOnly = apps[id]!.app.additionalSettings['trackOnly'] == true; | ||||||
|       if (!trackOnly) { |       if (!trackOnly) { | ||||||
|         apkUrl = await confirmApkUrl(apps[id]!.app, context); |         apkUrl = await confirmApkUrl(apps[id]!.app, context); | ||||||
|       } |       } | ||||||
|       if (apkUrl != null) { |       if (apkUrl != null) { | ||||||
|         int urlInd = apps[id]!.app.apkUrls.indexOf(apkUrl); |         int urlInd = apps[id]! | ||||||
|         if (urlInd != apps[id]!.app.preferredApkIndex) { |             .app | ||||||
|  |             .apkUrls | ||||||
|  |             .map((e) => e.value) | ||||||
|  |             .toList() | ||||||
|  |             .indexOf(apkUrl.value); | ||||||
|  |         if (urlInd >= 0 && urlInd != apps[id]!.app.preferredApkIndex) { | ||||||
|           apps[id]!.app.preferredApkIndex = urlInd; |           apps[id]!.app.preferredApkIndex = urlInd; | ||||||
|           await saveApps([apps[id]!.app]); |           await saveApps([apps[id]!.app]); | ||||||
|         } |         } | ||||||
| @@ -460,94 +497,117 @@ class AppsProvider with ChangeNotifier { | |||||||
|     return res; |     return res; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // If the App says it is installed but installedInfo is null, set it to not installed |   bool isVersionDetectionPossible(AppInMemory? app) { | ||||||
|   // If there is any other mismatch between installedInfo and installedVersion, try reconciling them intelligently |     return app?.app.additionalSettings['trackOnly'] != true && | ||||||
|   // If that fails, just set it to the actual version string (all we can do at that point) |         app?.app.additionalSettings['versionDetection'] != | ||||||
|   // Don't save changes, just return the object if changes were made (else null) |             'releaseDateAsVersion' && | ||||||
|  |         app?.installedInfo?.versionName != null && | ||||||
|  |         app?.app.installedVersion != null && | ||||||
|  |         reconcileVersionDifferences( | ||||||
|  |                 app!.installedInfo!.versionName!, app.app.installedVersion!) != | ||||||
|  |             null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Given an App and it's on-device info... | ||||||
|  |   // Reconcile unexpected differences between its reported installed version, real installed version, and reported latest version | ||||||
|   App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) { |   App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) { | ||||||
|     var modded = false; |     var modded = false; | ||||||
|     var trackOnly = app.additionalSettings['trackOnly'] == true; |     var trackOnly = app.additionalSettings['trackOnly'] == true; | ||||||
|     var noVersionDetection = app.additionalSettings['versionDetection'] != |     var noVersionDetection = app.additionalSettings['versionDetection'] != | ||||||
|         'standardVersionDetection'; |         'standardVersionDetection'; | ||||||
|  |     // FIRST, COMPARE THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE ONE IS NULL | ||||||
|     if (installedInfo == null && app.installedVersion != null && !trackOnly) { |     if (installedInfo == null && app.installedVersion != null && !trackOnly) { | ||||||
|  |       // App says it's installed but isn't really (and isn't track only) - set to not installed | ||||||
|       app.installedVersion = null; |       app.installedVersion = null; | ||||||
|       modded = true; |       modded = true; | ||||||
|     } else if (installedInfo?.versionName != null && |     } else if (installedInfo?.versionName != null && | ||||||
|         app.installedVersion == null) { |         app.installedVersion == null) { | ||||||
|  |       // App says it's not installed but really is - set to installed and use real package versionName | ||||||
|       app.installedVersion = installedInfo!.versionName; |       app.installedVersion = installedInfo!.versionName; | ||||||
|       modded = true; |       modded = true; | ||||||
|     } else if (installedInfo?.versionName != null && |     } | ||||||
|  |     // SECOND, RECONCILE DIFFERENCES BETWEEN THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE NEITHER IS NULL | ||||||
|  |     if (installedInfo?.versionName != null && | ||||||
|         installedInfo!.versionName != app.installedVersion && |         installedInfo!.versionName != app.installedVersion && | ||||||
|         !noVersionDetection) { |         !noVersionDetection) { | ||||||
|       String? correctedInstalledVersion = reconcileRealAndInternalVersions( |       // App's reported version and real version don't match (and it uses standard version detection) | ||||||
|  |       // If they share a standard format (and are still different under it), update the reported version accordingly | ||||||
|  |       var correctedInstalledVersion = reconcileVersionDifferences( | ||||||
|           installedInfo.versionName!, app.installedVersion!); |           installedInfo.versionName!, app.installedVersion!); | ||||||
|       if (correctedInstalledVersion != null) { |       if (correctedInstalledVersion?.key == false) { | ||||||
|         app.installedVersion = correctedInstalledVersion; |         app.installedVersion = correctedInstalledVersion!.value; | ||||||
|         modded = true; |         modded = true; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |     // THIRD, RECONCILE THE APP'S REPORTED INSTALLED AND LATEST VERSIONS | ||||||
|     if (app.installedVersion != null && |     if (app.installedVersion != null && | ||||||
|         app.installedVersion != app.latestVersion && |         app.installedVersion != app.latestVersion && | ||||||
|         !noVersionDetection) { |         !noVersionDetection) { | ||||||
|       app.installedVersion = reconcileRealAndInternalVersions( |       // App's reported installed and latest versions don't match (and it uses standard version detection) | ||||||
|               app.installedVersion!, app.latestVersion, |       // If they share a standard format, make sure the App's reported installed version uses that format | ||||||
|               matchMode: true) ?? |       var correctedInstalledVersion = | ||||||
|           app.installedVersion; |           reconcileVersionDifferences(app.installedVersion!, app.latestVersion); | ||||||
|  |       if (correctedInstalledVersion?.key == true) { | ||||||
|  |         app.installedVersion = correctedInstalledVersion!.value; | ||||||
|  |         modded = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     // FOURTH, DISABLE VERSION DETECTION IF ENABLED AND THE REPORTED/REAL INSTALLED VERSIONS ARE NOT STANDARDIZED | ||||||
|  |     if (installedInfo != null && | ||||||
|  |         app.additionalSettings['versionDetection'] == | ||||||
|  |             'standardVersionDetection' && | ||||||
|  |         !isVersionDetectionPossible(AppInMemory(app, null, installedInfo))) { | ||||||
|  |       app.additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|  |       logs.add('Could not reconcile version formats for: ${app.id}'); | ||||||
|       modded = true; |       modded = true; | ||||||
|     } |     } | ||||||
|  |     // if (app.installedVersion != null && | ||||||
|  |     //     app.additionalSettings['versionDetection'] == | ||||||
|  |     //         'standardVersionDetection') { | ||||||
|  |     //   var correctedInstalledVersion = | ||||||
|  |     //       reconcileVersionDifferences(app.installedVersion!, app.latestVersion); | ||||||
|  |     //   if (correctedInstalledVersion == null) { | ||||||
|  |     //     app.additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|  |     //     logs.add('Could not reconcile version formats for: ${app.id}'); | ||||||
|  |     //     modded = true; | ||||||
|  |     //   } | ||||||
|  |     // } | ||||||
|  |  | ||||||
|     return modded ? app : null; |     return modded ? app : null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   String? reconcileRealAndInternalVersions( |   MapEntry<bool, String>? reconcileVersionDifferences( | ||||||
|       String realVersion, String internalVersion, |       String templateVersion, String comparisonVersion) { | ||||||
|       {bool matchMode = false}) { |     // Returns null if the versions don't share a common standard format | ||||||
|     // 1. If one or both of these can't be converted to a "standard" format, return null (leave as is) |     // Returns <true, comparisonVersion> if they share a common format and are equal | ||||||
|     // 2. If both have a "standard" format under which they are equal, return null (leave as is) |     // Returns <false, templateVersion> if they share a common format but are not equal | ||||||
|     // 3. If both have a "standard" format in common but are unequal, return realVersion (this means it was changed externally) |     // templateVersion must fully match a standard format, while comparisonVersion can have a substring match | ||||||
|     // If in matchMode, the outcomes of rules 2 and 3 are reversed, and the "real" version is not matched strictly |     var templateVersionFormats = | ||||||
|     // Matchmode to be used when comparing internal install version and internal latest version |         findStandardFormatsForVersion(templateVersion, true); | ||||||
|  |     var comparisonVersionFormats = | ||||||
|     bool doStringsMatchUnderRegEx( |         findStandardFormatsForVersion(comparisonVersion, false); | ||||||
|         String pattern, String value1, String value2) { |  | ||||||
|       var r = RegExp(pattern); |  | ||||||
|       var m1 = r.firstMatch(value1); |  | ||||||
|       var m2 = r.firstMatch(value2); |  | ||||||
|       return m1 != null && m2 != null |  | ||||||
|           ? value1.substring(m1.start, m1.end) == |  | ||||||
|               value2.substring(m2.start, m2.end) |  | ||||||
|           : false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     Set<String> findStandardFormatsForVersion(String version, bool strict) { |  | ||||||
|       Set<String> results = {}; |  | ||||||
|       for (var pattern in standardVersionRegExStrings) { |  | ||||||
|         if (RegExp('${strict ? '^' : ''}$pattern${strict ? '\$' : ''}') |  | ||||||
|             .hasMatch(version)) { |  | ||||||
|           results.add(pattern); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|       return results; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     var realStandardVersionFormats = |  | ||||||
|         findStandardFormatsForVersion(realVersion, true); |  | ||||||
|     var internalStandardVersionFormats = |  | ||||||
|         findStandardFormatsForVersion(internalVersion, false); |  | ||||||
|     var commonStandardFormats = |     var commonStandardFormats = | ||||||
|         realStandardVersionFormats.intersection(internalStandardVersionFormats); |         templateVersionFormats.intersection(comparisonVersionFormats); | ||||||
|     if (commonStandardFormats.isEmpty) { |     if (commonStandardFormats.isEmpty) { | ||||||
|       return null; // Incompatible; no "enhanced detection" |       return null; | ||||||
|     } |     } | ||||||
|     for (String pattern in commonStandardFormats) { |     for (String pattern in commonStandardFormats) { | ||||||
|       if (doStringsMatchUnderRegEx(pattern, internalVersion, realVersion)) { |       if (doStringsMatchUnderRegEx( | ||||||
|         return matchMode |           pattern, comparisonVersion, templateVersion)) { | ||||||
|             ? internalVersion |         return MapEntry(true, comparisonVersion); | ||||||
|             : null; // Enhanced detection says no change |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return matchMode |     return MapEntry(false, templateVersion); | ||||||
|         ? null |   } | ||||||
|         : realVersion; // Enhanced detection says something changed |  | ||||||
|  |   bool doStringsMatchUnderRegEx(String pattern, String value1, String value2) { | ||||||
|  |     var r = RegExp(pattern); | ||||||
|  |     var m1 = r.firstMatch(value1); | ||||||
|  |     var m2 = r.firstMatch(value2); | ||||||
|  |     return m1 != null && m2 != null | ||||||
|  |         ? value1.substring(m1.start, m1.end) == | ||||||
|  |             value2.substring(m2.start, m2.end) | ||||||
|  |         : false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> loadApps() async { |   Future<void> loadApps() async { | ||||||
| @@ -559,7 +619,21 @@ class AppsProvider with ChangeNotifier { | |||||||
|     List<App> newApps = (await getAppsDir()) |     List<App> newApps = (await getAppsDir()) | ||||||
|         .listSync() |         .listSync() | ||||||
|         .where((item) => item.path.toLowerCase().endsWith('.json')) |         .where((item) => item.path.toLowerCase().endsWith('.json')) | ||||||
|         .map((e) => App.fromJson(jsonDecode(File(e.path).readAsStringSync()))) |         .map((e) { | ||||||
|  |           try { | ||||||
|  |             return App.fromJson(jsonDecode(File(e.path).readAsStringSync())); | ||||||
|  |           } catch (err) { | ||||||
|  |             if (err is FormatException) { | ||||||
|  |               logs.add('Corrupt JSON when loading App (will be ignored): $e'); | ||||||
|  |               e.renameSync('${e.path}.corrupt'); | ||||||
|  |               return App( | ||||||
|  |                   '', '', '', '', '', '', [], 0, {}, DateTime.now(), false); | ||||||
|  |             } else { | ||||||
|  |               rethrow; | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .where((element) => element.id.isNotEmpty) | ||||||
|         .toList(); |         .toList(); | ||||||
|     var idsToDelete = apps.values |     var idsToDelete = apps.values | ||||||
|         .map((e) => e.app.id) |         .map((e) => e.app.id) | ||||||
| @@ -576,7 +650,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|         sp.getSource(newApps[i].url); |         sp.getSource(newApps[i].url); | ||||||
|         apps[newApps[i].id] = AppInMemory(newApps[i], null, info); |         apps[newApps[i].id] = AppInMemory(newApps[i], null, info); | ||||||
|       } catch (e) { |       } 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) { |     if (errors.isNotEmpty) { | ||||||
| @@ -602,10 +676,12 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> saveApps(List<App> apps, |   Future<void> saveApps(List<App> apps, | ||||||
|       {bool attemptToCorrectInstallStatus = true}) async { |       {bool attemptToCorrectInstallStatus = true, | ||||||
|  |       bool onlyIfExists = true}) async { | ||||||
|     attemptToCorrectInstallStatus = |     attemptToCorrectInstallStatus = | ||||||
|         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); |         attemptToCorrectInstallStatus && (await doesInstalledAppsPluginWork()); | ||||||
|     for (var app in apps) { |     for (var a in apps) { | ||||||
|  |       var app = a.deepCopy(); | ||||||
|       AppInfo? info = await getInstalledInfo(app.id); |       AppInfo? info = await getInstalledInfo(app.id); | ||||||
|       app.name = info?.name ?? app.name; |       app.name = info?.name ?? app.name; | ||||||
|       if (attemptToCorrectInstallStatus) { |       if (attemptToCorrectInstallStatus) { | ||||||
| @@ -613,9 +689,15 @@ class AppsProvider with ChangeNotifier { | |||||||
|       } |       } | ||||||
|       File('${(await getAppsDir()).path}/${app.id}.json') |       File('${(await getAppsDir()).path}/${app.id}.json') | ||||||
|           .writeAsStringSync(jsonEncode(app.toJson())); |           .writeAsStringSync(jsonEncode(app.toJson())); | ||||||
|       this.apps.update( |       try { | ||||||
|           app.id, (value) => AppInMemory(app, value.downloadProgress, info), |         this.apps.update( | ||||||
|           ifAbsent: () => AppInMemory(app, null, info)); |             app.id, (value) => AppInMemory(app, value.downloadProgress, info), | ||||||
|  |             ifAbsent: onlyIfExists ? null : () => AppInMemory(app, null, info)); | ||||||
|  |       } catch (e) { | ||||||
|  |         if (e is! ArgumentError || e.name != 'key') { | ||||||
|  |           rethrow; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
| @@ -636,8 +718,11 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<bool> removeAppsWithModal(BuildContext context, List<App> apps) async { |   Future<bool> removeAppsWithModal(BuildContext context, List<App> apps) async { | ||||||
|     var showUninstallOption = |     var showUninstallOption = apps | ||||||
|         apps.where((a) => a.installedVersion != null).isNotEmpty; |         .where((a) => | ||||||
|  |             a.installedVersion != null && | ||||||
|  |             a.additionalSettings['trackOnly'] != true) | ||||||
|  |         .isNotEmpty; | ||||||
|     var values = await showDialog( |     var values = await showDialog( | ||||||
|         context: context, |         context: context, | ||||||
|         builder: (BuildContext ctx) { |         builder: (BuildContext ctx) { | ||||||
| @@ -686,6 +771,18 @@ class AppsProvider with ChangeNotifier { | |||||||
|     await intent.launch(); |     await intent.launch(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   addMissingCategories(SettingsProvider settingsProvider) { | ||||||
|  |     var cats = settingsProvider.categories; | ||||||
|  |     apps.forEach((key, value) { | ||||||
|  |       for (var c in value.app.categories) { | ||||||
|  |         if (!cats.containsKey(c)) { | ||||||
|  |           cats[c] = generateRandomLightColor().value; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     settingsProvider.setCategories(cats, appsProvider: this); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   Future<App?> checkUpdate(String appId) async { |   Future<App?> checkUpdate(String appId) async { | ||||||
|     App? currentApp = apps[appId]!.app; |     App? currentApp = apps[appId]!.app; | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |     SourceProvider sourceProvider = SourceProvider(); | ||||||
| @@ -765,12 +862,6 @@ class AppsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<String> exportApps() async { |   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 DeviceInfoPlugin().androidInfo).version.sdkInt <= 29) { | ||||||
|       if (await Permission.storage.isDenied) { |       if (await Permission.storage.isDenied) { | ||||||
|         await Permission.storage.request(); |         await Permission.storage.request(); | ||||||
| @@ -779,6 +870,18 @@ class AppsProvider with ChangeNotifier { | |||||||
|         throw ObtainiumError(tr('storagePermissionDenied')); |         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( |     File export = File( | ||||||
|         '${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json'); |         '${exportDir.path}/${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().millisecondsSinceEpoch}.json'); | ||||||
|     export.writeAsStringSync( |     export.writeAsStringSync( | ||||||
| @@ -798,7 +901,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|         a.installedVersion = apps[a.id]?.app.installedVersion; |         a.installedVersion = apps[a.id]?.app.installedVersion; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     await saveApps(importedApps); |     await saveApps(importedApps, onlyIfExists: false); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|     return importedApps.length; |     return importedApps.length; | ||||||
|   } |   } | ||||||
| @@ -811,14 +914,14 @@ class AppsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|   Future<List<List<String>>> addAppsByURL(List<String> urls) async { |   Future<List<List<String>>> addAppsByURL(List<String> urls) async { | ||||||
|     List<dynamic> results = await SourceProvider().getAppsByURLNaive(urls, |     List<dynamic> results = await SourceProvider().getAppsByURLNaive(urls, | ||||||
|         ignoreUrls: apps.values.map((e) => e.app.url).toList()); |         alreadyAddedUrls: apps.values.map((e) => e.app.url).toList()); | ||||||
|     List<App> pps = results[0]; |     List<App> pps = results[0]; | ||||||
|     Map<String, dynamic> errorsMap = results[1]; |     Map<String, dynamic> errorsMap = results[1]; | ||||||
|     for (var app in pps) { |     for (var app in pps) { | ||||||
|       if (apps.containsKey(app.id)) { |       if (apps.containsKey(app.id)) { | ||||||
|         errorsMap.addAll({app.id: tr('appAlreadyAdded')}); |         errorsMap.addAll({app.id: tr('appAlreadyAdded')}); | ||||||
|       } else { |       } else { | ||||||
|         await saveApps([app]); |         await saveApps([app], onlyIfExists: false); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     List<List<String>> errors = |     List<List<String>> errors = | ||||||
| @@ -831,7 +934,7 @@ class APKPicker extends StatefulWidget { | |||||||
|   const APKPicker({super.key, required this.app, this.initVal, this.archs}); |   const APKPicker({super.key, required this.app, this.initVal, this.archs}); | ||||||
|  |  | ||||||
|   final App app; |   final App app; | ||||||
|   final String? initVal; |   final MapEntry<String, String>? initVal; | ||||||
|   final List<String>? archs; |   final List<String>? archs; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -839,7 +942,7 @@ class APKPicker extends StatefulWidget { | |||||||
| } | } | ||||||
|  |  | ||||||
| class _APKPickerState extends State<APKPicker> { | class _APKPickerState extends State<APKPicker> { | ||||||
|   String? apkUrl; |   MapEntry<String, String>? apkUrl; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
| @@ -848,19 +951,17 @@ class _APKPickerState extends State<APKPicker> { | |||||||
|       scrollable: true, |       scrollable: true, | ||||||
|       title: Text(tr('pickAnAPK')), |       title: Text(tr('pickAnAPK')), | ||||||
|       content: Column(children: [ |       content: Column(children: [ | ||||||
|         Text(tr('appHasMoreThanOnePackage', args: [widget.app.name])), |         Text(tr('appHasMoreThanOnePackage', args: [widget.app.finalName])), | ||||||
|         const SizedBox(height: 16), |         const SizedBox(height: 16), | ||||||
|         ...widget.app.apkUrls.map( |         ...widget.app.apkUrls.map( | ||||||
|           (u) => RadioListTile<String>( |           (u) => RadioListTile<String>( | ||||||
|               title: Text(Uri.parse(u) |               title: Text(u.key), | ||||||
|                   .pathSegments |               value: u.value, | ||||||
|                   .where((element) => element.isNotEmpty) |               groupValue: apkUrl!.value, | ||||||
|                   .last), |  | ||||||
|               value: u, |  | ||||||
|               groupValue: apkUrl, |  | ||||||
|               onChanged: (String? val) { |               onChanged: (String? val) { | ||||||
|                 setState(() { |                 setState(() { | ||||||
|                   apkUrl = val; |                   apkUrl = | ||||||
|  |                       widget.app.apkUrls.where((e) => e.value == val).first; | ||||||
|                 }); |                 }); | ||||||
|               }), |               }), | ||||||
|         ), |         ), | ||||||
|   | |||||||
| @@ -34,9 +34,9 @@ class UpdateNotification extends ObtainiumNotification { | |||||||
|     message = updates.isEmpty |     message = updates.isEmpty | ||||||
|         ? tr('noNewUpdates') |         ? tr('noNewUpdates') | ||||||
|         : updates.length == 1 |         : updates.length == 1 | ||||||
|             ? tr('xHasAnUpdate', args: [updates[0].name]) |             ? tr('xHasAnUpdate', args: [updates[0].finalName]) | ||||||
|             : plural('xAndNMoreUpdatesAvailable', updates.length - 1, |             : 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) { |             tr('appsUpdatedNotifDescription'), Importance.defaultImportance) { | ||||||
|     message = updates.length == 1 |     message = updates.length == 1 | ||||||
|         ? tr('xWasUpdatedToY', |         ? tr('xWasUpdatedToY', | ||||||
|             args: [updates[0].name, updates[0].latestVersion]) |             args: [updates[0].finalName, updates[0].latestVersion]) | ||||||
|         : plural('xAndNMoreUpdatesInstalled', updates.length - 1, |         : 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:fluttertoast/fluttertoast.dart'; | ||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
|  | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:permission_handler/permission_handler.dart'; | import 'package:permission_handler/permission_handler.dart'; | ||||||
| import 'package:shared_preferences/shared_preferences.dart'; | import 'package:shared_preferences/shared_preferences.dart'; | ||||||
|  |  | ||||||
| @@ -62,6 +64,15 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool get useBlackTheme { | ||||||
|  |     return prefs?.getBool('useBlackTheme') ?? false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set useBlackTheme(bool useBlackTheme) { | ||||||
|  |     prefs?.setBool('useBlackTheme', useBlackTheme); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   int get updateInterval { |   int get updateInterval { | ||||||
|     var min = prefs?.getInt('updateInterval') ?? 360; |     var min = prefs?.getInt('updateInterval') ?? 360; | ||||||
|     if (!updateIntervals.contains(min)) { |     if (!updateIntervals.contains(min)) { | ||||||
| @@ -139,6 +150,15 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool get groupByCategory { | ||||||
|  |     return prefs?.getBool('groupByCategory') ?? false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set groupByCategory(bool show) { | ||||||
|  |     prefs?.setBool('groupByCategory', show); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   String? getSettingString(String settingId) { |   String? getSettingString(String settingId) { | ||||||
|     return prefs?.getString(settingId); |     return prefs?.getString(settingId); | ||||||
|   } |   } | ||||||
| @@ -151,7 +171,23 @@ class SettingsProvider with ChangeNotifier { | |||||||
|   Map<String, int> get categories => |   Map<String, int> get categories => | ||||||
|       Map<String, int>.from(jsonDecode(prefs?.getString('categories') ?? '{}')); |       Map<String, int>.from(jsonDecode(prefs?.getString('categories') ?? '{}')); | ||||||
|  |  | ||||||
|   set categories(Map<String, int> cats) { |   void setCategories(Map<String, int> cats, {AppsProvider? appsProvider}) { | ||||||
|  |     if (appsProvider != null) { | ||||||
|  |       List<App> changedApps = appsProvider | ||||||
|  |           .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)); |     prefs?.setString('categories', jsonEncode(cats)); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
|  |  | ||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
|  |  | ||||||
|  | import 'package:device_info_plus/device_info_plus.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:html/dom.dart'; | import 'package:html/dom.dart'; | ||||||
| import 'package:http/http.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/izzyondroid.dart'; | ||||||
| import 'package:obtainium/app_sources/html.dart'; | import 'package:obtainium/app_sources/html.dart'; | ||||||
| import 'package:obtainium/app_sources/mullvad.dart'; | import 'package:obtainium/app_sources/mullvad.dart'; | ||||||
|  | import 'package:obtainium/app_sources/neutroncode.dart'; | ||||||
| import 'package:obtainium/app_sources/signal.dart'; | import 'package:obtainium/app_sources/signal.dart'; | ||||||
| import 'package:obtainium/app_sources/sourceforge.dart'; | import 'package:obtainium/app_sources/sourceforge.dart'; | ||||||
| import 'package:obtainium/app_sources/steammobile.dart'; | import 'package:obtainium/app_sources/steammobile.dart'; | ||||||
|  | import 'package:obtainium/app_sources/telegramapp.dart'; | ||||||
|  | import 'package:obtainium/app_sources/vlc.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/mass_app_sources/githubstars.dart'; | import 'package:obtainium/mass_app_sources/githubstars.dart'; | ||||||
| @@ -31,11 +35,13 @@ class AppNames { | |||||||
|  |  | ||||||
| class APKDetails { | class APKDetails { | ||||||
|   late String version; |   late String version; | ||||||
|   late List<String> apkUrls; |   late List<MapEntry<String, String>> apkUrls; | ||||||
|   late AppNames names; |   late AppNames names; | ||||||
|   late DateTime? releaseDate; |   late 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 { | class App { | ||||||
| @@ -45,13 +51,14 @@ class App { | |||||||
|   late String name; |   late String name; | ||||||
|   String? installedVersion; |   String? installedVersion; | ||||||
|   late String latestVersion; |   late String latestVersion; | ||||||
|   List<String> apkUrls = []; |   List<MapEntry<String, String>> apkUrls = []; | ||||||
|   late int preferredApkIndex; |   late int preferredApkIndex; | ||||||
|   late Map<String, dynamic> additionalSettings; |   late Map<String, dynamic> additionalSettings; | ||||||
|   late DateTime? lastUpdateCheck; |   late DateTime? lastUpdateCheck; | ||||||
|   bool pinned = false; |   bool pinned = false; | ||||||
|   List<String> categories; |   List<String> categories; | ||||||
|   late DateTime? releaseDate; |   late DateTime? releaseDate; | ||||||
|  |   late String? changeLog; | ||||||
|   App( |   App( | ||||||
|       this.id, |       this.id, | ||||||
|       this.url, |       this.url, | ||||||
| @@ -65,13 +72,39 @@ class App { | |||||||
|       this.lastUpdateCheck, |       this.lastUpdateCheck, | ||||||
|       this.pinned, |       this.pinned, | ||||||
|       {this.categories = const [], |       {this.categories = const [], | ||||||
|       this.releaseDate}); |       this.releaseDate, | ||||||
|  |       this.changeLog}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
|     return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALSETTINGS: ${additionalSettings.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned'; |     return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls PREFERREDAPK: $preferredApkIndex ADDITIONALSETTINGS: ${additionalSettings.toString()} LASTCHECK: ${lastUpdateCheck.toString()} PINNED $pinned'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   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) { |   factory App.fromJson(Map<String, dynamic> json) { | ||||||
|     var source = SourceProvider().getSource(json['url']); |     var source = SourceProvider().getSource(json['url']); | ||||||
|     var formItems = source.combinedAppSpecificSettingFormItems |     var formItems = source.combinedAppSpecificSettingFormItems | ||||||
| @@ -103,16 +136,16 @@ class App { | |||||||
|     // Convert bool style version detection options to dropdown style |     // Convert bool style version detection options to dropdown style | ||||||
|     if (additionalSettings['noVersionDetection'] == true) { |     if (additionalSettings['noVersionDetection'] == true) { | ||||||
|       additionalSettings['versionDetection'] = 'noVersionDetection'; |       additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|     } |       if (additionalSettings['releaseDateAsVersion'] == true) { | ||||||
|     if (additionalSettings['releaseDateAsVersion'] == true) { |         additionalSettings['versionDetection'] = 'releaseDateAsVersion'; | ||||||
|       additionalSettings['versionDetection'] = 'releaseDateAsVersion'; |         additionalSettings.remove('releaseDateAsVersion'); | ||||||
|       additionalSettings.remove('releaseDateAsVersion'); |       } | ||||||
|     } |       if (additionalSettings['noVersionDetection'] != null) { | ||||||
|     if (additionalSettings['noVersionDetection'] != null) { |         additionalSettings.remove('noVersionDetection'); | ||||||
|       additionalSettings.remove('noVersionDetection'); |       } | ||||||
|     } |       if (additionalSettings['releaseDateAsVersion'] != null) { | ||||||
|     if (additionalSettings['releaseDateAsVersion'] != null) { |         additionalSettings.remove('releaseDateAsVersion'); | ||||||
|       additionalSettings.remove('releaseDateAsVersion'); |       } | ||||||
|     } |     } | ||||||
|     // Ensure additionalSettings are correctly typed |     // Ensure additionalSettings are correctly typed | ||||||
|     for (var item in formItems) { |     for (var item in formItems) { | ||||||
| @@ -127,35 +160,51 @@ class App { | |||||||
|     if (preferredApkIndex < 0) { |     if (preferredApkIndex < 0) { | ||||||
|       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( |     return App( | ||||||
|       json['id'] as String, |         json['id'] as String, | ||||||
|       json['url'] as String, |         json['url'] as String, | ||||||
|       json['author'] as String, |         json['author'] as String, | ||||||
|       json['name'] as String, |         json['name'] as String, | ||||||
|       json['installedVersion'] == null |         json['installedVersion'] == null | ||||||
|           ? null |             ? null | ||||||
|           : json['installedVersion'] as String, |             : json['installedVersion'] as String, | ||||||
|       json['latestVersion'] as String, |         json['latestVersion'] as String, | ||||||
|       json['apkUrls'] == null |         apkUrls, | ||||||
|           ? [] |         preferredApkIndex, | ||||||
|           : List<String>.from(jsonDecode(json['apkUrls'])), |         additionalSettings, | ||||||
|       preferredApkIndex, |         json['lastUpdateCheck'] == null | ||||||
|       additionalSettings, |             ? null | ||||||
|       json['lastUpdateCheck'] == null |             : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), | ||||||
|           ? null |         json['pinned'] ?? false, | ||||||
|           : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), |         categories: json['categories'] != null | ||||||
|       json['pinned'] ?? false, |             ? (json['categories'] as List<dynamic>) | ||||||
|       categories: json['categories'] != null |                 .map((e) => e.toString()) | ||||||
|           ? (json['categories'] as List<dynamic>) |                 .toList() | ||||||
|               .map((e) => e.toString()) |             : json['category'] != null | ||||||
|               .toList() |                 ? [json['category'] as String] | ||||||
|           : json['category'] != null |                 : [], | ||||||
|               ? [json['category'] as String] |         releaseDate: json['releaseDate'] == null | ||||||
|               : [], |             ? null | ||||||
|       releaseDate: json['releaseDate'] == null |             : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']), | ||||||
|           ? null |         changeLog: | ||||||
|           : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']), |             json['changeLog'] == null ? null : json['changeLog'] as String); | ||||||
|     ); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> toJson() => { |   Map<String, dynamic> toJson() => { | ||||||
| @@ -165,13 +214,14 @@ class App { | |||||||
|         'name': name, |         'name': name, | ||||||
|         'installedVersion': installedVersion, |         'installedVersion': installedVersion, | ||||||
|         'latestVersion': latestVersion, |         'latestVersion': latestVersion, | ||||||
|         'apkUrls': jsonEncode(apkUrls), |         'apkUrls': jsonEncode(apkUrls.map((e) => [e.key, e.value]).toList()), | ||||||
|         'preferredApkIndex': preferredApkIndex, |         'preferredApkIndex': preferredApkIndex, | ||||||
|         'additionalSettings': jsonEncode(additionalSettings), |         'additionalSettings': jsonEncode(additionalSettings), | ||||||
|         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, |         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, | ||||||
|         'pinned': pinned, |         'pinned': pinned, | ||||||
|         'categories': categories, |         'categories': categories, | ||||||
|         'releaseDate': releaseDate?.microsecondsSinceEpoch |         'releaseDate': releaseDate?.microsecondsSinceEpoch, | ||||||
|  |         'changeLog': changeLog | ||||||
|       }; |       }; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -216,10 +266,18 @@ Map<String, dynamic> getDefaultValuesFromFormItems( | |||||||
|       .reduce((value, element) => [...value, ...element])); |       .reduce((value, element) => [...value, ...element])); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | List<MapEntry<String, String>> getApkUrlsFromUrls(List<String> urls) => | ||||||
|  |     urls.map((e) { | ||||||
|  |       var segments = e.split('/').where((el) => el.trim().isNotEmpty); | ||||||
|  |       var apkSegs = segments.where((s) => s.toLowerCase().endsWith('.apk')); | ||||||
|  |       return MapEntry(apkSegs.isNotEmpty ? apkSegs.last : segments.last, e); | ||||||
|  |     }).toList(); | ||||||
|  |  | ||||||
| class AppSource { | class AppSource { | ||||||
|   String? host; |   String? host; | ||||||
|   late String name; |   late String name; | ||||||
|   bool enforceTrackOnly = false; |   bool enforceTrackOnly = false; | ||||||
|  |   bool changeLogIfAnyIsMarkDown = true; | ||||||
|  |  | ||||||
|   AppSource() { |   AppSource() { | ||||||
|     name = runtimeType.toString(); |     name = runtimeType.toString(); | ||||||
| @@ -268,7 +326,12 @@ class AppSource { | |||||||
|               return regExValidator(value); |               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 |   // Previous 2 variables combined into one at runtime for convenient usage | ||||||
| @@ -332,12 +395,16 @@ class SourceProvider { | |||||||
|     Codeberg(), |     Codeberg(), | ||||||
|     FDroid(), |     FDroid(), | ||||||
|     IzzyOnDroid(), |     IzzyOnDroid(), | ||||||
|     Mullvad(), |     FDroidRepo(), | ||||||
|     Signal(), |  | ||||||
|     SourceForge(), |     SourceForge(), | ||||||
|     APKMirror(), |     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(), |     SteamMobile(), | ||||||
|  |     NeutronCode(), | ||||||
|     HTML() // This should ALWAYS be the last option as they are tried in order |     HTML() // This should ALWAYS be the last option as they are tried in order | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
| @@ -348,7 +415,7 @@ class SourceProvider { | |||||||
|     url = preStandardizeUrl(url); |     url = preStandardizeUrl(url); | ||||||
|     AppSource? source; |     AppSource? source; | ||||||
|     for (var s in sources.where((element) => element.host != null)) { |     for (var s in sources.where((element) => element.host != null)) { | ||||||
|       if (url.contains('://${s.host}')) { |       if (RegExp('://(.+\\.)?${s.host}').hasMatch(url)) { | ||||||
|         source = s; |         source = s; | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
| @@ -407,14 +474,29 @@ class SourceProvider { | |||||||
|     if (additionalSettings['apkFilterRegEx'] != null) { |     if (additionalSettings['apkFilterRegEx'] != null) { | ||||||
|       var reg = RegExp(additionalSettings['apkFilterRegEx']); |       var reg = RegExp(additionalSettings['apkFilterRegEx']); | ||||||
|       apk.apkUrls = |       apk.apkUrls = | ||||||
|           apk.apkUrls.where((element) => reg.hasMatch(element)).toList(); |           apk.apkUrls.where((element) => reg.hasMatch(element.key)).toList(); | ||||||
|     } |     } | ||||||
|     if (apk.apkUrls.isEmpty && !trackOnly) { |     if (apk.apkUrls.isEmpty && !trackOnly) { | ||||||
|       throw NoAPKError(); |       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('/', '-'); |     String apkVersion = apk.version.replaceAll('/', '-'); | ||||||
|     var name = currentApp?.name.trim() ?? |     var name = currentApp != null ? currentApp.name.trim() : ''; | ||||||
|         apk.names.name[0].toUpperCase() + apk.names.name.substring(1); |     name = name.isNotEmpty | ||||||
|  |         ? name | ||||||
|  |         : apk.names.name[0].toUpperCase() + apk.names.name.substring(1); | ||||||
|     return App( |     return App( | ||||||
|         currentApp?.id ?? |         currentApp?.id ?? | ||||||
|             source.tryInferringAppId(standardUrl, |             source.tryInferringAppId(standardUrl, | ||||||
| @@ -422,9 +504,7 @@ class SourceProvider { | |||||||
|             generateTempID(standardUrl, additionalSettings), |             generateTempID(standardUrl, additionalSettings), | ||||||
|         standardUrl, |         standardUrl, | ||||||
|         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), |         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), | ||||||
|         name.trim().isNotEmpty |         name, | ||||||
|             ? name |  | ||||||
|             : apk.names.name[0].toUpperCase() + apk.names.name.substring(1), |  | ||||||
|         currentApp?.installedVersion, |         currentApp?.installedVersion, | ||||||
|         apkVersion, |         apkVersion, | ||||||
|         apk.apkUrls, |         apk.apkUrls, | ||||||
| @@ -433,16 +513,20 @@ class SourceProvider { | |||||||
|         DateTime.now(), |         DateTime.now(), | ||||||
|         currentApp?.pinned ?? false, |         currentApp?.pinned ?? false, | ||||||
|         categories: currentApp?.categories ?? const [], |         categories: currentApp?.categories ?? const [], | ||||||
|         releaseDate: apk.releaseDate); |         releaseDate: apk.releaseDate, | ||||||
|  |         changeLog: apk.changeLog); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Returns errors in [results, errors] instead of throwing them |   // Returns errors in [results, errors] instead of throwing them | ||||||
|   Future<List<dynamic>> getAppsByURLNaive(List<String> urls, |   Future<List<dynamic>> getAppsByURLNaive(List<String> urls, | ||||||
|       {List<String> ignoreUrls = const []}) async { |       {List<String> alreadyAddedUrls = const []}) async { | ||||||
|     List<App> apps = []; |     List<App> apps = []; | ||||||
|     Map<String, dynamic> errors = {}; |     Map<String, dynamic> errors = {}; | ||||||
|     for (var url in urls.where((element) => !ignoreUrls.contains(element))) { |     for (var url in urls) { | ||||||
|       try { |       try { | ||||||
|  |         if (alreadyAddedUrls.contains(url)) { | ||||||
|  |           throw ObtainiumError(tr('appAlreadyAdded')); | ||||||
|  |         } | ||||||
|         var source = getSource(url); |         var source = getSource(url); | ||||||
|         apps.add(await getApp( |         apps.add(await getApp( | ||||||
|             source, |             source, | ||||||
|   | |||||||
							
								
								
									
										238
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						| @@ -5,18 +5,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: android_alarm_manager_plus |       name: android_alarm_manager_plus | ||||||
|       sha256: "8647cc5f9339f3955a2bd9ec40e0f10c3a80049f31f80b3ffdd87e07bb73fce2" |       sha256: f6d0347734fa2ea716349a5a3e16ffdc1800ca64e5640112896d128c6815c178 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.1" |     version: "2.1.2" | ||||||
|   android_intent_plus: |   android_intent_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: android_intent_plus |       name: android_intent_plus | ||||||
|       sha256: "54810cb33945c2c10742cd746ea994822c115e9dbe189919bc63cb436e45a6af" |       sha256: "6bcdcd20461ac7a0c785f6298cdda96ad275d5bcbc1ecf28829cbe03ec6690be" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.1.6" |     version: "3.1.7" | ||||||
|   animations: |   animations: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -25,14 +25,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.7" |     version: "2.0.7" | ||||||
|   archive: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: archive |  | ||||||
|       sha256: d6347d54a2d8028e0437e3c099f66fdb8ae02c4720c1e7534c9f24c10351f85d |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "3.3.6" |  | ||||||
|   args: |   args: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -65,22 +57,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.2.1" |     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: |   clock: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -97,14 +73,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.17.0" |     version: "1.17.0" | ||||||
|   convert: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: convert |  | ||||||
|       sha256: "0f08b14755d163f6e2134cb58222dd25ea2a2ee8a195e53983d57c075324d592" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "3.1.1" |  | ||||||
|   cross_file: |   cross_file: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -149,10 +117,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: device_info_plus |       name: device_info_plus | ||||||
|       sha256: "1d6e5a61674ba3a68fb048a7c7b4ff4bebfed8d7379dbe8f2b718231be9a7c95" |       sha256: "435383ca05f212760b0a70426b5a90354fe6bd65992b3a5e27ab6ede74c02f5c" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "8.1.0" |     version: "8.2.0" | ||||||
|   device_info_plus_platform_interface: |   device_info_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -165,10 +133,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: dynamic_color |       name: dynamic_color | ||||||
|       sha256: c4a508284b14ec4dda5adba2c28b2cdd34fbae1afead7e8c52cad87d51c5405b |       sha256: bbebb1b7ebed819e0ec83d4abdc2a8482d934f6a85289ffc1c6acf7589fa2aad | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.6.2" |     version: "1.6.3" | ||||||
|   easy_localization: |   easy_localization: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -213,10 +181,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: file_picker |       name: file_picker | ||||||
|       sha256: d090ae03df98b0247b82e5928f44d1b959867049d18d73635e2e0bc3f49542b9 |       sha256: b85eb92b175767fdaa0c543bf3b0d1f610fe966412ea72845fe5ba7801e763ff | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.2.5" |     version: "5.2.10" | ||||||
|   flutter: |   flutter: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -230,14 +198,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.2.2" |     version: "0.2.2" | ||||||
|   flutter_launcher_icons: |  | ||||||
|     dependency: "direct dev" |  | ||||||
|     description: |  | ||||||
|       name: flutter_launcher_icons |  | ||||||
|       sha256: "02dcaf49d405f652b7160e882bacfc02cb497041bb2eab2a49b1c393cf9aac12" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "0.12.0" |  | ||||||
|   flutter_lints: |   flutter_lints: | ||||||
|     dependency: "direct dev" |     dependency: "direct dev" | ||||||
|     description: |     description: | ||||||
| @@ -275,14 +235,22 @@ packages: | |||||||
|     description: flutter |     description: flutter | ||||||
|     source: sdk |     source: sdk | ||||||
|     version: "0.0.0" |     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: |   flutter_plugin_android_lifecycle: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: flutter_plugin_android_lifecycle |       name: flutter_plugin_android_lifecycle | ||||||
|       sha256: "4bef634684b2c7f3468c77c766c831229af829a0cd2d4ee6c1b99558bd14e5d2" |       sha256: c224ac897bed083dabf11f238dd11a239809b446740be0c2044608c50029ffdf | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.8" |     version: "2.0.9" | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     dependency: "direct dev" |     dependency: "direct dev" | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -305,10 +273,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: html |       name: html | ||||||
|       sha256: d9793e10dbe0e6c364f4c59bf3e01fb33a9b2a674bc7a1081693dba0614b6269 |       sha256: "79d498e6d6761925a34ee5ea8fa6dfef38607781d2fa91e37523474282af55cb" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.15.1" |     version: "0.15.2" | ||||||
|   http: |   http: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -325,14 +293,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.0.2" |     version: "4.0.2" | ||||||
|   image: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: image |  | ||||||
|       sha256: "483a389d6ccb292b570c31b3a193779b1b0178e7eb571986d9a49904b6861227" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "4.0.15" |  | ||||||
|   install_plugin_v2: |   install_plugin_v2: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -365,14 +325,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.6.5" |     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: |   lints: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -381,6 +333,14 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.1" |     version: "2.0.1" | ||||||
|  |   markdown: | ||||||
|  |     dependency: transitive | ||||||
|  |     description: | ||||||
|  |       name: markdown | ||||||
|  |       sha256: d95a9d12954aafc97f984ca29baaa7690ed4d9ec4140a23ad40580bcdb6c87f5 | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "7.0.2" | ||||||
|   matcher: |   matcher: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -449,34 +409,34 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: path_provider |       name: path_provider | ||||||
|       sha256: "04890b994ee89bfa80bf3080bfec40d5a92c5c7a785ebb02c13084a099d2b6f9" |       sha256: c7edf82217d4b2952b2129a61d3ad60f1075b9299e629e149a8d2e39c2e6aad4 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.13" |     version: "2.0.14" | ||||||
|   path_provider_android: |   path_provider_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_android |       name: path_provider_android | ||||||
|       sha256: "7623b7d4be0f0f7d9a8b5ee6879fc13e4522d4c875ab86801dee4af32b54b83e" |       sha256: da97262be945a72270513700a92b39dd2f4a54dad55d061687e2e37a6390366a | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.23" |     version: "2.0.25" | ||||||
|   path_provider_foundation: |   path_provider_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_foundation |       name: path_provider_foundation | ||||||
|       sha256: eec003594f19fe2456ea965ae36b3fc967bc5005f508890aafe31fa75e41d972 |       sha256: ad4c4d011830462633f03eb34445a45345673dfd4faf1ab0b4735fbd93b19183 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.2" |     version: "2.2.2" | ||||||
|   path_provider_linux: |   path_provider_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_linux |       name: path_provider_linux | ||||||
|       sha256: "525ad5e07622d19447ad740b1ed5070031f7a5437f44355ae915ff56e986429a" |       sha256: "2ae08f2216225427e64ad224a24354221c2c7907e448e6e0e8b57b1eb9f10ad1" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.9" |     version: "2.1.10" | ||||||
|   path_provider_platform_interface: |   path_provider_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -489,10 +449,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_windows |       name: path_provider_windows | ||||||
|       sha256: "642ddf65fde5404f83267e8459ddb4556316d3ee6d511ed193357e25caa3632d" |       sha256: f53720498d5a543f9607db4b0e997c4b5438884de25b0f73098cc2671a51b130 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.4" |     version: "2.1.5" | ||||||
|   permission_handler: |   permission_handler: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -513,10 +473,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: permission_handler_apple |       name: permission_handler_apple | ||||||
|       sha256: "9c370ef6a18b1c4b2f7f35944d644a56aa23576f23abee654cf73968de93f163" |       sha256: ee96ac32f5a8e6f80756e25b25b9f8e535816c8e6665a96b6d70681f8c4f7e85 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "9.0.7" |     version: "9.0.8" | ||||||
|   permission_handler_platform_interface: |   permission_handler_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -557,14 +517,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.4" |     version: "2.1.4" | ||||||
|   pointycastle: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: pointycastle |  | ||||||
|       sha256: db7306cf0249f838d1a24af52b5a5887c5bf7f31d8bb4e827d071dc0939ad346 |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "3.6.2" |  | ||||||
|   process: |   process: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -585,74 +537,74 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: share_plus |       name: share_plus | ||||||
|       sha256: "8c6892037b1824e2d7e8f59d54b3105932899008642e6372e5079c6939b4b625" |       sha256: "692261968a494e47323dcc8bc66d8d52e81bc27cb4b808e4e8d7e8079d4cc01a" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.1" |     version: "6.3.2" | ||||||
|   share_plus_platform_interface: |   share_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: share_plus_platform_interface |       name: share_plus_platform_interface | ||||||
|       sha256: "82ddd4ab9260c295e6e39612d4ff00390b9a7a21f1bb1da771e2f232d80ab8a1" |       sha256: "0c6e61471bd71b04a138b8b588fa388e66d8b005e6f2deda63371c5c505a0981" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.2.0" |     version: "3.2.1" | ||||||
|   shared_preferences: |   shared_preferences: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences |       name: shared_preferences | ||||||
|       sha256: ee6257848f822b8481691f20c3e6d2bfee2e9eccb2a3d249907fcfb198c55b41 |       sha256: "858aaa72d8f61637d64e776aca82e1c67e6d9ee07979123c5d17115031c1b13b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.18" |     version: "2.1.0" | ||||||
|   shared_preferences_android: |   shared_preferences_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_android |       name: shared_preferences_android | ||||||
|       sha256: a51a4f9375097f94df1c6e0a49c0374440d31ab026b59d58a7e7660675879db4 |       sha256: "7fa90471a6875d26ad78c7e4a675874b2043874586891128dc5899662c97db46" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.16" |     version: "2.1.2" | ||||||
|   shared_preferences_foundation: |   shared_preferences_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_foundation |       name: shared_preferences_foundation | ||||||
|       sha256: "6b84fdf06b32bb336f972d373cd38b63734f3461ba56ac2ba01b56d052796259" |       sha256: "0c1c16c56c9708aa9c361541a6f0e5cc6fc12a3232d866a687a7b7db30032b07" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.4" |     version: "2.2.1" | ||||||
|   shared_preferences_linux: |   shared_preferences_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_linux |       name: shared_preferences_linux | ||||||
|       sha256: d7fb71e6e20cd3dfffcc823a28da3539b392e53ed5fc5c2b90b55fdaa8a7e8fa |       sha256: "9d387433ca65717bbf1be88f4d5bb18f10508917a8fa2fb02e0fd0d7479a9afa" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.4" |     version: "2.2.0" | ||||||
|   shared_preferences_platform_interface: |   shared_preferences_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_platform_interface |       name: shared_preferences_platform_interface | ||||||
|       sha256: "824bfd02713e37603b2bdade0842e47d56e7db32b1dcdd1cae533fb88e2913fc" |       sha256: fb5cf25c0235df2d0640ac1b1174f6466bd311f621574997ac59018a6664548d | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.1" |     version: "2.2.0" | ||||||
|   shared_preferences_web: |   shared_preferences_web: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_web |       name: shared_preferences_web | ||||||
|       sha256: "6737b757e49ba93de2a233df229d0b6a87728cea1684da828cbc718b65dcf9d7" |       sha256: "74083203a8eae241e0de4a0d597dbedab3b8fef5563f33cf3c12d7e93c655ca5" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.5" |     version: "2.1.0" | ||||||
|   shared_preferences_windows: |   shared_preferences_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_windows |       name: shared_preferences_windows | ||||||
|       sha256: bd014168e8484837c39ef21065b78f305810ceabc1d4f90be6e3b392ce81b46d |       sha256: "5e588e2efef56916a3b229c3bfe81e6a525665a454519ca51dbcc4236a274173" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.4" |     version: "2.2.0" | ||||||
|   sky_engine: |   sky_engine: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -670,18 +622,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: sqflite |       name: sqflite | ||||||
|       sha256: "851d5040552cf911f4cabda08d003eca76b27da3ed0002978272e27c8fbf8ecc" |       sha256: e7dfb6482d5d02b661d0b2399efa72b98909e5aa7b8336e1fb37e226264ade00 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.2.5" |     version: "2.2.7" | ||||||
|   sqflite_common: |   sqflite_common: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: sqflite_common |       name: sqflite_common | ||||||
|       sha256: bfd6973aaeeb93475bc0d875ac9aefddf7965ef22ce09790eb963992ffc5183f |       sha256: "220831bf0bd5333ff2445eee35ec131553b804e6b5d47a4a37ca6f5eb66e282c" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.2+2" |     version: "2.4.4" | ||||||
|   stack_trace: |   stack_trace: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -710,10 +662,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: synchronized |       name: synchronized | ||||||
|       sha256: "33b31b6beb98100bf9add464a36a8dd03eb10c7a8cf15aeec535e9b054aaf04b" |       sha256: "5fcbd27688af6082f5abd611af56ee575342c30e87541d0245f7ff99faa02c60" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.1" |     version: "3.1.0" | ||||||
|   term_glyph: |   term_glyph: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -734,10 +686,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: timezone |       name: timezone | ||||||
|       sha256: "24c8fcdd49a805d95777a39064862133ff816ebfffe0ceff110fb5960e557964" |       sha256: "1cfd8ddc2d1cfd836bc93e67b9be88c3adaeca6f40a00ca999104c30693cdca0" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.9.1" |     version: "0.9.2" | ||||||
|   typed_data: |   typed_data: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -758,34 +710,34 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_android |       name: url_launcher_android | ||||||
|       sha256: "1f4d9ebe86f333c15d318f81dcdc08b01d45da44af74552608455ebdc08d9732" |       sha256: a52628068d282d01a07cd86e6ba99e497aa45ce8c91159015b2416907d78e411 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.0.24" |     version: "6.0.27" | ||||||
|   url_launcher_ios: |   url_launcher_ios: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_ios |       name: url_launcher_ios | ||||||
|       sha256: c9cd648d2f7ab56968e049d4e9116f96a85517f1dd806b96a86ea1018a3a82e5 |       sha256: "9af7ea73259886b92199f9e42c116072f05ff9bea2dcb339ab935dfc957392c2" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.1.1" |     version: "6.1.4" | ||||||
|   url_launcher_linux: |   url_launcher_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_linux |       name: url_launcher_linux | ||||||
|       sha256: e29039160ab3730e42f3d811dc2a6d5f2864b90a70fb765ea60144b03307f682 |       sha256: "206fb8334a700ef7754d6a9ed119e7349bc830448098f21a69bf1b4ed038cabc" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.3" |     version: "3.0.4" | ||||||
|   url_launcher_macos: |   url_launcher_macos: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_macos |       name: url_launcher_macos | ||||||
|       sha256: "2dddb3291a57b074dade66b5e07e64401dd2487caefd4e9e2f467138d8c7eb06" |       sha256: "91ee3e75ea9dadf38036200c5d3743518f4a5eb77a8d13fda1ee5764373f185e" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.3" |     version: "3.0.5" | ||||||
|   url_launcher_platform_interface: |   url_launcher_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -798,18 +750,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_web |       name: url_launcher_web | ||||||
|       sha256: "574cfbe2390666003c3a1d129bdc4574aaa6728f0c00a4829a81c316de69dd9b" |       sha256: "81fe91b6c4f84f222d186a9d23c73157dc4c8e1c71489c4d08be1ad3b228f1aa" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.15" |     version: "2.0.16" | ||||||
|   url_launcher_windows: |   url_launcher_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_windows |       name: url_launcher_windows | ||||||
|       sha256: "97c9067950a0d09cbd93e2e3f0383d1403989362b97102fbf446473a48079a4b" |       sha256: a83ba3607a507758669cfafb03f9de09bf6e6280c14d9b9cb18f013e406dcacd | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.4" |     version: "3.0.5" | ||||||
|   uuid: |   uuid: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -830,42 +782,42 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter |       name: webview_flutter | ||||||
|       sha256: "9ba213434f13e760ea0f175fbc4d6bb6aeafd7dfc6c7d973f15d3e47a5d6686e" |       sha256: "1a37bdbaaf5fbe09ad8579ab09ecfd473284ce482f900b5aea27cf834386a567" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.0.5" |     version: "4.2.0" | ||||||
|   webview_flutter_android: |   webview_flutter_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_android |       name: webview_flutter_android | ||||||
|       sha256: "48c8cfb023168473c0a3a4c21ffea6c23a32cc7156701c39f618b303c6a3c96e" |       sha256: "134ed5d36127b6f5865e86a82174886eae0b983dacd8df14b0448371debde755" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.3.1" |     version: "3.6.0" | ||||||
|   webview_flutter_platform_interface: |   webview_flutter_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_platform_interface |       name: webview_flutter_platform_interface | ||||||
|       sha256: df6472164b3f4eaf3280422227f361dc8424b106726b7f21d79a8656ba53f71f |       sha256: "78715dc442b7849dbde74e92bb67de1cecf5addf95531c5fb474e72f5fe9a507" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.2" |     version: "2.3.0" | ||||||
|   webview_flutter_wkwebview: |   webview_flutter_wkwebview: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_wkwebview |       name: webview_flutter_wkwebview | ||||||
|       sha256: "283a38c2a2544768033864c698e0133aa9eee0f2c800f494b538a3d1044f7ecb" |       sha256: c94d242d8cbe1012c06ba7ac790c46d6e6b68723b7d34f8c74ed19f68d166f49 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.1.1" |     version: "3.4.0" | ||||||
|   win32: |   win32: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: win32 |       name: win32 | ||||||
|       sha256: c9ebe7ee4ab0c2194e65d3a07d8c54c5d00bb001b76081c4a04cdb8448b59e46 |       sha256: a6f0236dbda0f63aa9a25ad1ff9a9d8a4eaaa5012da0dc59d21afdb1dc361ca4 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.1.3" |     version: "3.1.4" | ||||||
|   xdg_directories: |   xdg_directories: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -882,14 +834,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.2.2" |     version: "6.2.2" | ||||||
|   yaml: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: yaml |  | ||||||
|       sha256: "23812a9b125b48d4007117254bca50abb6c712352927eece9e155207b1db2370" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "3.1.1" |  | ||||||
| sdks: | sdks: | ||||||
|   dart: ">=2.18.2 <3.0.0" |   dart: ">=2.19.0 <3.0.0" | ||||||
|   flutter: ">=3.4.0-17.0.pre" |   flutter: ">=3.4.0-17.0.pre" | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								pubspec.yaml
									
									
									
									
									
								
							
							
						
						| @@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev | |||||||
| # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | ||||||
| # In Windows, build-name is used as the major, minor, and patch parts | # In Windows, build-name is used as the major, minor, and patch parts | ||||||
| # of the product and file versions while build-number is used as the build suffix. | # of the product and file versions while build-number is used as the build suffix. | ||||||
| version: 0.11.6+125 # When changing this, update the tag in main() accordingly | version: 0.11.34+156 # When changing this, update the tag in main() accordingly | ||||||
|  |  | ||||||
| environment: | environment: | ||||||
|   sdk: '>=2.18.2 <3.0.0' |   sdk: '>=2.18.2 <3.0.0' | ||||||
| @@ -49,7 +49,7 @@ dependencies: | |||||||
|   permission_handler: ^10.0.0 |   permission_handler: ^10.0.0 | ||||||
|   fluttertoast: ^8.0.9 |   fluttertoast: ^8.0.9 | ||||||
|   device_info_plus: ^8.0.0 |   device_info_plus: ^8.0.0 | ||||||
|   file_picker: ^5.1.0 |   file_picker: ^5.2.10 | ||||||
|   animations: ^2.0.4 |   animations: ^2.0.4 | ||||||
|   install_plugin_v2: ^1.0.0 |   install_plugin_v2: ^1.0.0 | ||||||
|   share_plus: ^6.0.1 |   share_plus: ^6.0.1 | ||||||
| @@ -59,12 +59,12 @@ dependencies: | |||||||
|   sqflite: ^2.2.0+3 |   sqflite: ^2.2.0+3 | ||||||
|   easy_localization: ^3.0.1 |   easy_localization: ^3.0.1 | ||||||
|   android_intent_plus: ^3.1.5 |   android_intent_plus: ^3.1.5 | ||||||
|  |   flutter_markdown: ^0.6.14 | ||||||
|  |  | ||||||
|  |  | ||||||
| dev_dependencies: | dev_dependencies: | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     sdk: flutter |     sdk: flutter | ||||||
|   flutter_launcher_icons: ^0.12.0 |  | ||||||
|  |  | ||||||
|   # The "flutter_lints" package below contains a set of recommended lints to |   # The "flutter_lints" package below contains a set of recommended lints to | ||||||
|   # encourage good coding practices. The lint set provided by the package is |   # encourage good coding practices. The lint set provided by the package is | ||||||
| @@ -73,12 +73,6 @@ dev_dependencies: | |||||||
|   # rules and activating additional ones. |   # rules and activating additional ones. | ||||||
|   flutter_lints: ^2.0.1 |   flutter_lints: ^2.0.1 | ||||||
|  |  | ||||||
| flutter_icons: |  | ||||||
|   android: true |  | ||||||
|   image_path: "assets/graphics/icon.png" |  | ||||||
|   adaptive_icon_background: "#FFFFFF" |  | ||||||
|   adaptive_icon_foreground: "assets/graphics/icon.png" |  | ||||||
|  |  | ||||||
| # For information on the generic Dart part of this file, see the | # For information on the generic Dart part of this file, see the | ||||||
| # following page: https://dart.dev/tools/pub/pubspec | # following page: https://dart.dev/tools/pub/pubspec | ||||||
|  |  | ||||||
| @@ -98,6 +92,7 @@ flutter: | |||||||
|   assets: |   assets: | ||||||
|     - assets/translations/ |     - assets/translations/ | ||||||
|     - assets/graphics/ |     - assets/graphics/ | ||||||
|  |     - assets/ca/ | ||||||
|  |  | ||||||
|   # An image asset can refer to one or more resolution-specific "variants", see |   # An image asset can refer to one or more resolution-specific "variants", see | ||||||
|   # https://flutter.dev/assets-and-images/#resolution-aware |   # https://flutter.dev/assets-and-images/#resolution-aware | ||||||
|   | |||||||