Compare commits
	
		
			78 Commits
		
	
	
		
			v0.10.9-be
			...
			v0.11.8-be
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | a0199f0ceb | ||
|  | 0528936e5a | ||
|  | 4de98b2f36 | ||
|  | dfb5f5596c | ||
|  | 2e706aac47 | ||
|  | 24a600e595 | ||
|  | 1596a44ec5 | ||
|  | 9ee2be76ca | ||
|  | 83b770294d | ||
|  | 2679d5a1aa | ||
|  | e49c09c0ff | ||
|  | c9318ef2b5 | ||
|  | 2e88c8eede | ||
|  | 8648c1bea7 | ||
|  | b22e2bab0c | ||
|  | 57f7bf44c2 | ||
|  | ce526d8d26 | ||
|  | 5f3eeb9971 | ||
|  | e67a6b8627 | ||
|  | f8e99bb0cb | ||
|  | 09b5dd41d3 | ||
|  | b1bd36408c | ||
|  | 54d8dff32f | ||
|  | 7b1416e28e | ||
|  | 926e7b89ce | ||
|  | 43d4f89d61 | ||
|  | 2190da162d | ||
|  | f10bb5ac91 | ||
|  | 8e52f9666d | ||
|  | a8a47bb153 | ||
|  | 728dafcc28 | ||
|  | d53b21906c | ||
|  | d6dcac0f97 | ||
|  | dae5a67652 | ||
|  | 508fcccec9 | ||
|  | cc8a4c3760 | ||
|  | 814e2b7306 | ||
|  | 2e159c9886 | ||
|  | b82d28f2a7 | ||
|  | 3c61735706 | ||
|  | a2879f5bfa | ||
|  | b57f023739 | ||
|  | c376a7abec | ||
|  | 31c6cc3f6f | ||
|  | 8de8438aeb | ||
|  | 2b0225dd5b | ||
|  | f6af3a7998 | ||
|  | bd29d7bc10 | ||
|  | ffb3516a4b | ||
|  | 6a5e7942ee | ||
|  | 859158e84a | ||
|  | 435116e10b | ||
|  | a788d9d7cd | ||
|  | 4be3478b97 | ||
|  | fe0126095a | ||
|  | d5fdf28a98 | ||
|  | f06d245e20 | ||
|  | 2b4f94b407 | ||
|  | 5f7e342e6b | ||
|  | 191776d0d5 | ||
|  | ea81b0e66e | ||
|  | 86131ae3ce | ||
|  | 64ded1d720 | ||
|  | a11c2f1d37 | ||
|  | 890787f87f | ||
|  | c5ff1de950 | ||
|  | 56658abd60 | ||
|  | b60622e2cb | ||
|  | e149f0b225 | ||
|  | d9729f08c0 | ||
|  | eda5c1bac6 | ||
|  | 5574ea870b | ||
|  | 9f03234ac1 | ||
|  | b2503dd43d | ||
|  | e01ca704bc | ||
|  | 6aa4ace8e2 | ||
|  | d762467a31 | ||
|  | b07cce8ecd | 
| 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 | 
| Before Width: | Height: | Size: 109 KiB | 
							
								
								
									
										78
									
								
								assets/graphics/icon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,78 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||||
|  | <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||||
|  |  | ||||||
|  | <svg | ||||||
|  |    width="142.12897mm" | ||||||
|  |    height="142.12897mm" | ||||||
|  |    viewBox="0 0 142.12897 142.12897" | ||||||
|  |    version="1.1" | ||||||
|  |    id="svg5" | ||||||
|  |    xml:space="preserve" | ||||||
|  |    inkscape:version="1.2.2 (b0a8486541, 2022-12-01)" | ||||||
|  |    sodipodi:docname="icon.svg" | ||||||
|  |    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||||
|  |    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||||
|  |    xmlns:xlink="http://www.w3.org/1999/xlink" | ||||||
|  |    xmlns="http://www.w3.org/2000/svg" | ||||||
|  |    xmlns:svg="http://www.w3.org/2000/svg"><sodipodi:namedview | ||||||
|  |      id="namedview7" | ||||||
|  |      pagecolor="#ffffff" | ||||||
|  |      bordercolor="#000000" | ||||||
|  |      borderopacity="0.25" | ||||||
|  |      inkscape:showpageshadow="2" | ||||||
|  |      inkscape:pageopacity="0.0" | ||||||
|  |      inkscape:pagecheckerboard="0" | ||||||
|  |      inkscape:deskcolor="#d1d1d1" | ||||||
|  |      inkscape:document-units="mm" | ||||||
|  |      showgrid="false" | ||||||
|  |      inkscape:zoom="2.4175295" | ||||||
|  |      inkscape:cx="371.03994" | ||||||
|  |      inkscape:cy="273.62644" | ||||||
|  |      inkscape:window-width="2256" | ||||||
|  |      inkscape:window-height="1427" | ||||||
|  |      inkscape:window-x="0" | ||||||
|  |      inkscape:window-y="0" | ||||||
|  |      inkscape:window-maximized="1" | ||||||
|  |      inkscape:current-layer="layer1" /><defs | ||||||
|  |      id="defs2"><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        id="linearGradient3657"><stop | ||||||
|  |          style="stop-color:#9b58dc;stop-opacity:1;" | ||||||
|  |          offset="0" | ||||||
|  |          id="stop3653" /><stop | ||||||
|  |          style="stop-color:#321c92;stop-opacity:1;" | ||||||
|  |          offset="1" | ||||||
|  |          id="stop3655" /></linearGradient><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        id="linearGradient945"><stop | ||||||
|  |          style="stop-color:#9b58dc;stop-opacity:1;" | ||||||
|  |          offset="0" | ||||||
|  |          id="stop941" /><stop | ||||||
|  |          style="stop-color:#321c92;stop-opacity:1;" | ||||||
|  |          offset="1" | ||||||
|  |          id="stop943" /></linearGradient><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        xlink:href="#linearGradient945" | ||||||
|  |        id="linearGradient947" | ||||||
|  |        x1="76.787094" | ||||||
|  |        y1="113.40435" | ||||||
|  |        x2="110.68458" | ||||||
|  |        y2="152.48038" | ||||||
|  |        gradientUnits="userSpaceOnUse" | ||||||
|  |        gradientTransform="translate(-0.04012535,0.02025786)" /><linearGradient | ||||||
|  |        inkscape:collect="always" | ||||||
|  |        xlink:href="#linearGradient3657" | ||||||
|  |        id="linearGradient3659" | ||||||
|  |        x1="76.787094" | ||||||
|  |        y1="113.40435" | ||||||
|  |        x2="110.68458" | ||||||
|  |        y2="152.48038" | ||||||
|  |        gradientUnits="userSpaceOnUse" | ||||||
|  |        gradientTransform="translate(-0.04012535,0.02025786)" /></defs><g | ||||||
|  |      inkscape:label="Layer 1" | ||||||
|  |      inkscape:groupmode="layer" | ||||||
|  |      id="layer1" | ||||||
|  |      transform="translate(-30.394373,-54.680428)"><path | ||||||
|  |        style="fill:url(#linearGradient3659);fill-opacity:1;stroke:url(#linearGradient947);stroke-width:0.139;stroke-dasharray:none" | ||||||
|  |        d="m 109.8808,153.22596 c -0.73146,-0.38777 -5.00657,-2.75679 -25.032416,-13.87149 -5.57273,-3.09297 -10.93823,-6.06723 -11.92332,-6.60948 -2.23728,-1.23152 -2.58105,-1.53456 -2.58105,-2.27528 0,-0.3879 0.89293,-2.87231 2.98561,-8.30689 1.64209,-4.2644 3.09426,-8.0014 3.22705,-8.30444 0.3024,-0.69008 0.78972,-1.27621 1.26573,-1.52236 0.44558,-0.23042 11.58052,-4.29685 12.14814,-4.43644 0.61355,-0.1509 1.1428,0.13977 1.45487,0.79901 0.14976,0.31638 0.77213,1.94934 1.38303,3.6288 0.6109,1.67945 1.52036,4.16275 2.02104,5.51844 1.14709,3.10604 1.18992,3.54589 0.3912,4.01771 -0.2117,0.12505 -1.58874,0.66539 -3.06009,1.20075 -1.47136,0.53536 -2.87533,1.08982 -3.11993,1.23213 -0.56422,0.32826 -0.64913,0.83523 -0.20815,1.24273 0.17523,0.16193 3.00434,1.77571 6.28691,3.58618 9.174936,5.06035 8.665596,4.83136 9.277626,4.17097 0.29987,-0.32356 5.78141,-14.266 6.09596,-15.50521 0.1344,-0.5295 0.11969,-0.60308 -0.16695,-0.83519 -0.39165,-0.31714 -0.335,-0.33071 -3.93797,0.9431 -3.56937,1.26192 -3.90926,1.28864 -4.38744,0.34488 -0.25108,-0.49556 -4.095796,-11.05481 -4.334456,-11.90432 -0.15438,-0.5495 0.0344,-1.0717 0.49701,-1.37482 0.19228,-0.12598 2.990116,-1.19935 6.217406,-2.38526 4.78924,-1.75986 6.0081,-2.15842 6.63117,-2.16837 0.8037,-0.0128 0.90917,0.0424 15.64514,8.19599 1.02104,0.56495 1.56579,1.15961 1.56579,1.70925 0,0.21814 -3.6538,9.91011 -8.11957,21.53771 -6.2982,16.39877 -8.19916,21.21114 -8.4744,21.45338 -0.46789,0.41179 -0.8512,0.39392 -1.74794,-0.0815 z" | ||||||
|  |        id="path239" /></g></svg> | ||||||
| After Width: | Height: | Size: 4.1 KiB | 
| @@ -207,12 +207,19 @@ | |||||||
|     "addCategory": "Kategorie hinzufügen", |     "addCategory": "Kategorie hinzufügen", | ||||||
|     "label": "Bezeichnung", |     "label": "Bezeichnung", | ||||||
|     "language": "Sprache", |     "language": "Sprache", | ||||||
|     "storagePermissionDenied": "Storage permission denied", |     "storagePermissionDenied": "Speicherberechtigung verweigert", | ||||||
|     "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.", |     "selectedCategorizeWarning": "Dadurch werden alle bestehenden Kategorieeinstellungen für die ausgewählten Apps ersetzt.", | ||||||
|     "filterAPKsByRegEx": "Filter APKs by Regular Expression", |     "filterAPKsByRegEx": "APKs nach regulärem Ausdruck filtern", | ||||||
|     "removeFromObtainium": "Remove from Obtainium", |     "removeFromObtainium": "Aus Obtainium entfernen", | ||||||
|     "uninstallFromDevice": "Uninstall from Device", |     "uninstallFromDevice": "Vom Gerät deinstallieren", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.", |     "onlyWorksWithNonVersionDetectApps": "Funktioniert nur bei Apps mit deaktivierter Versionserkennung.", | ||||||
|  |     "releaseDateAsVersion": "Veröffentlichungsdatum als Version verwenden", | ||||||
|  |     "releaseDateAsVersionExplanation": "Diese Option sollte nur für Apps verwendet werden, bei denen die Versionserkennung nicht korrekt funktioniert, aber ein Veröffentlichungsdatum verfügbar ist.", | ||||||
|  |     "changes": "Änderungen", | ||||||
|  |     "releaseDate": "Veröffentlichungsdatum", | ||||||
|  |     "importFromURLsInFile": "Importieren von URLs aus Datei ( z.B. OPML)", | ||||||
|  |     "versionDetection": "Versionserkennung", | ||||||
|  |     "standardVersionDetection": "Standardversionserkennung", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "App entfernen?", |         "one": "App entfernen?", | ||||||
|         "other": "App entfernen?" |         "other": "App entfernen?" | ||||||
|   | |||||||
| @@ -213,6 +213,13 @@ | |||||||
|     "removeFromObtainium": "Remove from Obtainium", |     "removeFromObtainium": "Remove from Obtainium", | ||||||
|     "uninstallFromDevice": "Uninstall from Device", |     "uninstallFromDevice": "Uninstall from Device", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.", |     "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.", | ||||||
|  |     "releaseDateAsVersion": "Use Release Date as Version", | ||||||
|  |     "releaseDateAsVersionExplanation": "This option should only be used for Apps where version detection does not work correctly, but a release date is available.", | ||||||
|  |     "changes": "Changes", | ||||||
|  |     "releaseDate": "Release Date", | ||||||
|  |     "importFromURLsInFile": "Import from URLs in File (like OPML)", | ||||||
|  |     "versionDetection": "Version Detection", | ||||||
|  |     "standardVersionDetection": "Standard version detection", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Remove App?", |         "one": "Remove App?", | ||||||
|         "other": "Remove Apps?" |         "other": "Remove Apps?" | ||||||
|   | |||||||
							
								
								
									
										271
									
								
								assets/translations/fa.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,271 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "آدرس اینترنتی برنامه {} معتبر نیست", | ||||||
|  |     "noReleaseFound": "نسخه مناسبی پیدا نشد", | ||||||
|  |     "noVersionFound": "نمی توان نسخه منتشر شده را تعیین کرد", | ||||||
|  |     "urlMatchesNoSource": "آدرس اینترنتی با منبع شناخته شده مطابقت ندارد", | ||||||
|  |     "cantInstallOlderVersion": "نمی توان نسخه قدیمی یک برنامه را نصب کرد", | ||||||
|  |     "appIdMismatch": "شناسه بسته دانلود شده با شناسه برنامه موجود مطابقت ندارد", | ||||||
|  |     "functionNotImplemented": "این کلاس این تابع را پیاده سازی نکرده است", | ||||||
|  |     "placeholder": "نگهدارنده مکان", | ||||||
|  |     "someErrors": "برخی از خطاها رخ داده است", | ||||||
|  |     "unexpectedError": "خطای غیرمنتظره", | ||||||
|  |     "ok": "باشه", | ||||||
|  |     "and": "و", | ||||||
|  |     "startedBgUpdateTask": "شروع بررسی بروزرسانی BG", | ||||||
|  |     "bgUpdateIgnoreAfterIs": "نادیده گرفتن بروزرسانی BG بعد از {} است", | ||||||
|  |     "startedActualBGUpdateCheck": "بررسی بهروزرسانی واقعی BG آغاز شد", | ||||||
|  |     "bgUpdateTaskFinished": "کار بررسی بهروزرسانی BG تمام شد", | ||||||
|  |     "firstRun": "این اولین اجرای Obtainium است", | ||||||
|  |     "settingUpdateCheckIntervalTo": "تنظیم فاصله بهروزرسانی روی {}", | ||||||
|  |     "githubPATLabel": "توکن دسترسی شخصی گیت هاب(محدودیت نرخ را افزایش میدهد)", | ||||||
|  |     "githubPATHint": "PAT باید در این قالب باشد: username:token", | ||||||
|  |     "githubPATFormat": "username:token", | ||||||
|  |     "githubPATLinkText": "درباره گیتهاب PATs", | ||||||
|  |     "includePrereleases": "شامل نسخه های اولیه", | ||||||
|  |     "fallbackToOlderReleases": "بازگشت به نسخه های قدیمی تر", | ||||||
|  |     "filterReleaseTitlesByRegEx": "عناوین انتشار را با بیان منظم فیلتر کنید", | ||||||
|  |     "invalidRegEx": "عبارت منظم نامعتبر است", | ||||||
|  |     "noDescription": "بدون توضیحات", | ||||||
|  |     "cancel": "لغو", | ||||||
|  |     "continue": "ادامه دهید", | ||||||
|  |     "requiredInBrackets": "(ضروری)", | ||||||
|  |     "dropdownNoOptsError": "خطا: کشویی باید حداقل یک گزینه داشته باشد", | ||||||
|  |     "colour": "رنگ", | ||||||
|  |     "githubStarredRepos": "مخازن ستاره دار گیتهاب", | ||||||
|  |     "uname": "نام کاربری", | ||||||
|  |     "wrongArgNum": "تعداد آرگومان های ارائه شده اشتباه است", | ||||||
|  |     "xIsTrackOnly": "{} فقط ردیابی", | ||||||
|  |     "source": "منبع", | ||||||
|  |     "app": "برنامه", | ||||||
|  |     "appsFromSourceAreTrackOnly": "برنامههای این منبع «فقط ردیابی» هستند", | ||||||
|  |     "youPickedTrackOnly": "شما گزینه ی «فقط ردیابی» را انتخاب کرده اید", | ||||||
|  |     "trackOnlyAppDescription": "برنامه برای به روز رسانی ها ردیابی می شود، اما Obtainium قادر به دانلود یا نصب آن نخواهد بود.", | ||||||
|  |     "cancelled": "لغو شد", | ||||||
|  |     "appAlreadyAdded": "برنامه قبلاً اضافه شده است", | ||||||
|  |     "alreadyUpToDateQuestion": "برنامه از قبل به روز شده است؟", | ||||||
|  |     "addApp": "افزودن برنامه", | ||||||
|  |     "appSourceURL": "آدرس اینترنتی منبع برنامه", | ||||||
|  |     "error": "خطا", | ||||||
|  |     "add": "اضافه کردن", | ||||||
|  |     "searchSomeSourcesLabel": "جستجو (فقط برخی منابع)", | ||||||
|  |     "search": "جستجو کردن", | ||||||
|  |     "additionalOptsFor": "گزینه های اضافی برای {}", | ||||||
|  |     "supportedSourcesBelow": "منابع پشتیبانی شده:", | ||||||
|  |     "trackOnlyInBrackets": "«فقط ردیابی»", | ||||||
|  |     "searchableInBrackets": "(قابل جستجو)", | ||||||
|  |     "appsString": "برنامه ها", | ||||||
|  |     "noApps": "برنامه ای وجود ندارد", | ||||||
|  |     "noAppsForFilter": "برنامه ای برای فیلتر کردن وجود ندارد", | ||||||
|  |     "byX": "توسط {}", | ||||||
|  |     "percentProgress": "پیش رفتن: {}%", | ||||||
|  |     "pleaseWait": "لطفا صبر کنید", | ||||||
|  |     "updateAvailable": "بروزرسانی در دسترس", | ||||||
|  |     "estimateInBracketsShort": "(تخمین)", | ||||||
|  |     "notInstalled": "نصب نشده", | ||||||
|  |     "estimateInBrackets": "(تخمین زدن)", | ||||||
|  |     "selectAll": "انتخاب همه", | ||||||
|  |     "deselectN": "لغو انتخاب {}", | ||||||
|  |     "xWillBeRemovedButRemainInstalled": "{} از Obtainium حذف میشود اما روی دستگاه نصب میماند.", | ||||||
|  |     "removeSelectedAppsQuestion": "برنامه های انتخابی حذف شود؟", | ||||||
|  |     "removeSelectedApps": "حذف برنامه های انتخاب شده", | ||||||
|  |     "updateX": "به روز رسانی {}", | ||||||
|  |     "installX": "نصب {}", | ||||||
|  |     "markXTrackOnlyAsUpdated": "علامت {}\n(فقط ردیابی)\nبروز شده", | ||||||
|  |     "changeX": "تغییر دادن {}", | ||||||
|  |     "installUpdateApps": "نصب/بهروزرسانی برنامهها", | ||||||
|  |     "installUpdateSelectedApps": "برنامههای انتخابی را نصب/بهروزرسانی کنید", | ||||||
|  |     "markXSelectedAppsAsUpdated": "{} برنامه های انتخابی را به عنوان به روز علامت گذاری کنید؟", | ||||||
|  |     "no": "خیر", | ||||||
|  |     "yes": "بله", | ||||||
|  |     "markSelectedAppsUpdated": "برنامه های انتخاب شده را به عنوان به روز علامت گذاری کنید", | ||||||
|  |     "pinToTop": "پین به بالا", | ||||||
|  |     "unpinFromTop": "برداشتن پین از بالا", | ||||||
|  |     "resetInstallStatusForSelectedAppsQuestion": "وضعیت نصب برنامههای انتخابی بازنشانی شود؟", | ||||||
|  |     "installStatusOfXWillBeResetExplanation": "وضعیت نصب برنامههای انتخابشده بازنشانی میشود.\n\nاگر نسخه برنامه نشاندادهشده در Obtainium به دلیل بهروزرسانیهای ناموفق یا مشکلات دیگر نادرست باشد، میتواند کمک کند.", | ||||||
|  |     "shareSelectedAppURLs": "اشتراک گذاری آدرس اینترنتی برنامه های انتخاب شده", | ||||||
|  |     "resetInstallStatus": "بازنشانی وضعیت نصب", | ||||||
|  |     "more": "بیشتر", | ||||||
|  |     "removeOutdatedFilter": "فیلتر برنامه قدیمی را حذف کنید", | ||||||
|  |     "showOutdatedOnly": "فقط برنامه های قدیمی را نشان دهید", | ||||||
|  |     "filter": "فیلتر", | ||||||
|  |     "filterActive": "فیلتر *", | ||||||
|  |     "filterApps": "فیلتر کردن برنامه ها", | ||||||
|  |     "appName": "نام برنامه", | ||||||
|  |     "author": "سازنده", | ||||||
|  |     "upToDateApps": "برنامه های به روز", | ||||||
|  |     "nonInstalledApps": "برنامه های نصب نشده", | ||||||
|  |     "importExport": "وادر کردن/صادر کردن", | ||||||
|  |     "settings": "تنظیمات", | ||||||
|  |     "exportedTo": "صادر کردن به{}", | ||||||
|  |     "obtainiumExport": "صادرکردن Obtainium", | ||||||
|  |     "invalidInput": "ورودی نامعتبر", | ||||||
|  |     "importedX": "وارد شده {}", | ||||||
|  |     "obtainiumImport": "واردکردن Obtainium", | ||||||
|  |     "importFromURLList": "وارد کردن از فهرست آدرس اینترنتی", | ||||||
|  |     "searchQuery": "جستجوی سوال", | ||||||
|  |     "appURLList": "فهرست آدرس اینترنتی برنامه", | ||||||
|  |     "line": "خط", | ||||||
|  |     "searchX": "جستجو {}", | ||||||
|  |     "noResults": "نتیجه ای پیدا نشد", | ||||||
|  |     "importX": "وارد کردن {}", | ||||||
|  |     "importedAppsIdDisclaimer": "ممکن است برنامههای وارد شده به اشتباه بهعنوان \"نصب نشده\" نشان داده شوند.\nبرای رفع این مشکل، آنها را دوباره از طریق Obtainium نصب کنید.\nاین نباید روی دادههای برنامه تأثیر بگذارد.\n\nفقط بر روی آدرس اینترنتی و روشهای وارد کردن شخص ثالث تأثیر میگذارد.", | ||||||
|  |     "importErrors": "خطاهای وارد کردن", | ||||||
|  |     "importedXOfYApps": "{} از {} برنامه وارد شد.", | ||||||
|  |     "followingURLsHadErrors": "آدرس های اینترنتی زیر دارای خطا بودند:", | ||||||
|  |     "okay": "باشه", | ||||||
|  |     "selectURL": "آدرس اینترنتی انتخاب شده", | ||||||
|  |     "selectURLs": "آدرس های اینترنتی انتخاب شده", | ||||||
|  |     "pick": "انتخاب", | ||||||
|  |     "theme": "تم", | ||||||
|  |     "dark": "تاریک", | ||||||
|  |     "light": "روشن", | ||||||
|  |     "followSystem": "هماهنگ با سیستم", | ||||||
|  |     "obtainium": "Obtainium", | ||||||
|  |     "materialYou": "Material You", | ||||||
|  |     "appSortBy": "مرتب سازی برنامه بر اساس", | ||||||
|  |     "authorName": "سازنده/اسم", | ||||||
|  |     "nameAuthor": "اسم/سازنده", | ||||||
|  |     "asAdded": "همانطور که اضافه شد", | ||||||
|  |     "appSortOrder": "ترتیب مرتب سازی برنامه", | ||||||
|  |     "ascending": "صعودی", | ||||||
|  |     "descending": "نزولی", | ||||||
|  |     "bgUpdateCheckInterval": "فاصله بررسی بهروزرسانی در پسزمینه", | ||||||
|  |     "neverManualOnly": "هرگز - فقط دستی", | ||||||
|  |     "appearance": "ظاهر", | ||||||
|  |     "showWebInAppView": "نمایش صفحه وب منبع در نمای برنامه", | ||||||
|  |     "pinUpdates": "بهروزرسانیها را به نمای بالای برنامهها پین کنید", | ||||||
|  |     "updates": "به روز رسانی ها", | ||||||
|  |     "sourceSpecific": "منبع خاص", | ||||||
|  |     "appSource": "منبع برنامه", | ||||||
|  |     "noLogs": "بدون گزارش", | ||||||
|  |     "appLogs": "گزارش های برنامه", | ||||||
|  |     "close": "بستن", | ||||||
|  |     "share": "اشتراک گذاری", | ||||||
|  |     "appNotFound": "برنامه پیدا نشد", | ||||||
|  |     "obtainiumExportHyphenatedLowercase": "صادر کردن-obtainium", | ||||||
|  |     "pickAnAPK": "یک APK انتخاب کنید", | ||||||
|  |     "appHasMoreThanOnePackage": "{} بیش از یک بسته دارد:", | ||||||
|  |     "deviceSupportsXArch": "دستگاه شما از معماری پردازنده {} پشتیبانی میکند", | ||||||
|  |     "deviceSupportsFollowingArchs": "دستگاه شما از معماری های پردازنده زیر پشتیبانی می کند:", | ||||||
|  |     "warning": "اخطار", | ||||||
|  |     "sourceIsXButPackageFromYPrompt": "منبع برنامه \"{}\" است اما بسته انتشار از \"{}\" آمده است. ادامه هید؟", | ||||||
|  |     "updatesAvailable": "بروزرسانی در دسترس ", | ||||||
|  |     "updatesAvailableNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی برای یک یا چند برنامه ردیابی شده توسط Obtainium در دسترس است", | ||||||
|  |     "noNewUpdates": "به روز رسانی جدیدی وجود ندارد.", | ||||||
|  |     "xHasAnUpdate": "{} یک به روز رسانی دارد.", | ||||||
|  |     "appsUpdated": "برنامه ها به روز شدند", | ||||||
|  |     "appsUpdatedNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی یک یا چند برنامه در پس زمینه اعمال شده است", | ||||||
|  |     "xWasUpdatedToY": "{} به {} به روز شد.", | ||||||
|  |     "errorCheckingUpdates": "خطا در بررسی بهروزرسانیها", | ||||||
|  |     "errorCheckingUpdatesNotifDescription": "اعلانی که وقتی بررسی بهروزرسانی پسزمینه ناموفق است نشان میدهد", | ||||||
|  |     "appsRemoved": "برنامه ها حذف شدند", | ||||||
|  |     "appsRemovedNotifDescription": "به کاربر اطلاع می دهد که یک یا چند برنامه به دلیل خطا در هنگام بارگیری حذف شده است", | ||||||
|  |     "xWasRemovedDueToErrorY": "{} به دلیل این خطا حذف شد: {}", | ||||||
|  |     "completeAppInstallation": "نصب کامل برنامه", | ||||||
|  |     "obtainiumMustBeOpenToInstallApps": "Obtainium باید برای نصب برنامه ها باز باشد", | ||||||
|  |     "completeAppInstallationNotifDescription": "از کاربر میخواهد برای پایان نصب برنامه به Obtainium برگردد", | ||||||
|  |     "checkingForUpdates": "بررسی بهروزرسانیها", | ||||||
|  |     "checkingForUpdatesNotifDescription": "اعلان گذرا که هنگام بررسی به روز رسانی ظاهر می شود", | ||||||
|  |     "pleaseAllowInstallPerm": "لطفاً به Obtainium اجازه دهید برنامهها را نصب کند", | ||||||
|  |     "trackOnly": "فقط ردیابی", | ||||||
|  |     "errorWithHttpStatusCode": "خطا {}", | ||||||
|  |     "versionCorrectionDisabled": "تصحیح نسخه غیرفعال شد (به نظر می رسد افزونه کار نمی کند)", | ||||||
|  |     "unknown": "ناشناخته", | ||||||
|  |     "none": "هیچ", | ||||||
|  |     "never": "هرگز", | ||||||
|  |     "latestVersionX": "آخرین نسخه: {}", | ||||||
|  |     "installedVersionX": "نسخه نصب شده: {}", | ||||||
|  |     "lastUpdateCheckX": "بررسی آخرین بهروزرسانی: {}", | ||||||
|  |     "remove": "حذف", | ||||||
|  |     "yesMarkUpdated": "بله، علامت گذاری به عنوان به روز شده", | ||||||
|  |     "fdroid": "F-Droid", | ||||||
|  |     "appIdOrName": "شناسه یا نام برنامه", | ||||||
|  |     "appWithIdOrNameNotFound": "هیچ برنامه ای با آن شناسه یا نام یافت نشد", | ||||||
|  |     "reposHaveMultipleApps": "مخازن ممکن است شامل چندین برنامه باشد", | ||||||
|  |     "fdroidThirdPartyRepo": "مخازن شخص ثالث F-Droid", | ||||||
|  |     "steam": "Steam", | ||||||
|  |     "steamMobile": "Steam Mobile", | ||||||
|  |     "steamChat": "Steam Chat", | ||||||
|  |     "install": "نصب", | ||||||
|  |     "markInstalled": "علامت گذاری به عنوان نصب شده", | ||||||
|  |     "update": "به روز رسانی", | ||||||
|  |     "markUpdated": "علامت گذاری به روز شد", | ||||||
|  |     "additionalOptions": "گزینه های اضافی", | ||||||
|  |     "disableVersionDetection": "غیرفعال کردن تشخیص نسخه", | ||||||
|  |     "noVersionDetectionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند.", | ||||||
|  |     "downloadingX": "در حال دانلود {}", | ||||||
|  |     "downloadNotifDescription": "کاربر را از پیشرفت دانلود یک برنامه مطلع می کند", | ||||||
|  |     "noAPKFound": "APK پیدا نشد فایل", | ||||||
|  |     "noVersionDetection": "بدون تشخیص نسخه", | ||||||
|  |     "categorize": "دسته بندی کردن", | ||||||
|  |     "categories": "دسته بندی ها", | ||||||
|  |     "category": "دسته بندی", | ||||||
|  |     "noCategory": "بدون دسته بندی", | ||||||
|  |     "noCategories": "بدون دسته بندی ها", | ||||||
|  |     "deleteCategoriesQuestion": "دسته بندی ها حذف شوند؟", | ||||||
|  |     "categoryDeleteWarning": "همه برنامهها در دستههای حذف شده روی دستهبندی نشده تنظیم میشوند.", | ||||||
|  |     "addCategory": "اضافه کردن دسته", | ||||||
|  |     "label": "برچسب", | ||||||
|  |     "language": "زبان", | ||||||
|  |     "storagePermissionDenied": "مجوز ذخیره سازی رد شد", | ||||||
|  |     "selectedCategorizeWarning": "این جایگزین تنظیمات دسته بندی موجود برای برنامه های انتخابی می شود.", | ||||||
|  |     "filterAPKsByRegEx": "فایلهای APK را با نظم فیلتر کنید", | ||||||
|  |     "removeFromObtainium": "از Obtainium حذف کنید", | ||||||
|  |     "uninstallFromDevice": "حذف نصب از دستگاه", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "فقط برای برنامههایی کار میکند که تشخیص نسخه غیرفعال است.", | ||||||
|  |     "releaseDateAsVersion": "از تاریخ انتشار به عنوان نسخه استفاده کنید", | ||||||
|  |     "releaseDateAsVersionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند، اما تاریخ انتشار در دسترس است.", | ||||||
|  |     "changes": "تغییرات", | ||||||
|  |     "releaseDate": "تاریخ انتشار", | ||||||
|  |     "importFromURLsInFile": "وارد کردن از آدرس های اینترنتی موجود در فایل (مانند OPML)", | ||||||
|  |     "versionDetection": "تشخیص نسخه", | ||||||
|  |     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "برنامه حذف شود؟", | ||||||
|  |         "other": "برنامه ها حذف شوند؟" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "درخواستهای بسیار زیاد (نرخ محدود) - {} دقیقه دیگر دوباره امتحان کنید", | ||||||
|  |         "other": "درخواست های بسیار زیاد (نرخ محدود) - بعد از {} دقیقه دوباره امتحان کنید" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "بررسی بهروزرسانی BG با یک {} مواجه شد، یک بررسی مجدد را در {} دقیقه برنامهریزی میکند", | ||||||
|  |         "other": "بررسی بهروزرسانی BG با {} مواجه شد، یک بررسی مجدد را در {} دقیقه برنامهریزی میکند" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "بررسی بهروزرسانی BG پیدا شد {} بهروزرسانی - در صورت نیاز به کاربر اطلاع میدهد", | ||||||
|  |         "other": "بررسی بهروزرسانی BG {} بهروزرسانیهای یافت شده - در صورت نیاز به کاربر اطلاع میدهد" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "برنامه {}", | ||||||
|  |         "other": "{} برنامه ها" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} آدرس اینترنتی", | ||||||
|  |         "other": "{} آدرس های اینترنتی" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} دقیقه", | ||||||
|  |         "other": "{} دقیقه" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} ساعت", | ||||||
|  |         "other": "{} ساعت" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} روز", | ||||||
|  |         "other": "{} روز" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "گزارش {n} پاک شد (قبل از = {پیش از}، بعد = {بعد})", | ||||||
|  |         "other": "{n} گزارش پاک شد (قبل از = {پیش از}، بعد = {بعد})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} و 1 برنامه دیگر بهروزرسانی دارند.", | ||||||
|  |         "other": "{} و {} برنامه دیگر به روز رسانی دارند." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} و 1 برنامه دیگر به روز شدند.", | ||||||
|  |         "other": "{} و {} برنامه دیگر به روز شدند." | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										271
									
								
								assets/translations/fr.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,271 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "URL d'application {} invalide", | ||||||
|  |     "noReleaseFound": "Impossible de trouver une version appropriée", | ||||||
|  |     "noVersionFound": "Impossible de déterminer la version de la version", | ||||||
|  |     "urlMatchesNoSource": "L'URL ne correspond pas à une source connue", | ||||||
|  |     "cantInstallOlderVersion": "Impossible d'installer une ancienne version d'une application", | ||||||
|  |     "appIdMismatch": "L'ID de paquet téléchargé ne correspond pas à l'ID de l'application existante", | ||||||
|  |     "functionNotImplemented": "Cette classe n'a pas implémenté cette fonction", | ||||||
|  |     "placeholder": "Espace réservé", | ||||||
|  |     "someErrors": "Des erreurs se sont produites", | ||||||
|  |     "unexpectedError": "Erreur inattendue", | ||||||
|  |     "ok": "Okay", | ||||||
|  |     "and": "et", | ||||||
|  |     "startedBgUpdateTask": "Démarrage de la tâche de vérification de mise à jour en arrière-plan", | ||||||
|  |     "bgUpdateIgnoreAfterIs": "Mise à jour en arrière-plan est ignoré après  {}", | ||||||
|  |     "startedActualBGUpdateCheck": "Démarrage de la vérification de la mise à jour en arrière-plan", | ||||||
|  |     "bgUpdateTaskFinished": "Tâche de vérification de la mise à jour en arrière-plan terminée", | ||||||
|  |     "firstRun": "Il s'agit de la toute première exécution d'Obtainium", | ||||||
|  |     "settingUpdateCheckIntervalTo": "Définition de l'intervalle de mise à jour sur {}", | ||||||
|  |     "githubPATLabel": "Jeton d'Accès Personnel GitHub (Augmente la limite de débit)", | ||||||
|  |     "githubPATHint": "Le JAP doit être dans ce format : username:token", | ||||||
|  |     "githubPATFormat": "username:token", | ||||||
|  |     "githubPATLinkText": "À propos des JAP GitHub", | ||||||
|  |     "includePrereleases": "Inclure les avant-premières", | ||||||
|  |     "fallbackToOlderReleases": "Retour aux anciennes versions", | ||||||
|  |     "filterReleaseTitlesByRegEx": "Filtrer les titres de version par expression régulière", | ||||||
|  |     "invalidRegEx": "Expression régulière invalide", | ||||||
|  |     "noDescription": "Pas de description", | ||||||
|  |     "cancel": "Annuler", | ||||||
|  |     "continue": "Continuer", | ||||||
|  |     "requiredInBrackets": "(Requis)", | ||||||
|  |     "dropdownNoOptsError": "ERREUR : LE DÉROULEMENT DOIT AVOIR AU MOINS UNE OPT", | ||||||
|  |     "colour": "Couleur", | ||||||
|  |     "githubStarredRepos": "Dépôts étoilés GitHub", | ||||||
|  |     "uname": "Nom d'utilisateur", | ||||||
|  |     "wrongArgNum": "Mauvais nombre d'arguments fournis", | ||||||
|  |     "xIsTrackOnly": "{} est en 'Suivi uniquement'", | ||||||
|  |     "source": "Source", | ||||||
|  |     "app": "Application", | ||||||
|  |     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", | ||||||
|  |     "youPickedTrackOnly": "Vous avez sélectionné l'option 'Suivi uniquement'.", | ||||||
|  |     "trackOnlyAppDescription": "L'application sera suivie pour les mises à jour, mais Obtainium ne pourra pas la télécharger ou l'installer.", | ||||||
|  |     "cancelled": "Annulé", | ||||||
|  |     "appAlreadyAdded": "Application déjà ajoutée", | ||||||
|  |     "alreadyUpToDateQuestion": "Application déjà à jour ?", | ||||||
|  |     "addApp": "Ajouter une application", | ||||||
|  |     "appSourceURL": "URL de la source de l'application", | ||||||
|  |     "error": "Erreur", | ||||||
|  |     "add": "Ajoutée", | ||||||
|  |     "searchSomeSourcesLabel": "Rechercher (certaines sources uniquement)", | ||||||
|  |     "search": "Rechercher", | ||||||
|  |     "additionalOptsFor": "Options supplémentaires pour {}", | ||||||
|  |     "supportedSourcesBelow": "Sources prises en charge :", | ||||||
|  |     "trackOnlyInBrackets": "(Suivi uniquement)", | ||||||
|  |     "searchableInBrackets": "(Recherchable)", | ||||||
|  |     "appsString": "Applications", | ||||||
|  |     "noApps": "Aucune application", | ||||||
|  |     "noAppsForFilter": "Aucune application pour le filtre", | ||||||
|  |     "byX": "Par {}", | ||||||
|  |     "percentProgress": "Progrès: {}%", | ||||||
|  |     "pleaseWait": "Veuillez patienter", | ||||||
|  |     "updateAvailable": "Mise à jour disponible", | ||||||
|  |     "estimateInBracketsShort": "(Est.)", | ||||||
|  |     "notInstalled": "Pas installé", | ||||||
|  |     "estimateInBrackets": "(Estimation)", | ||||||
|  |     "selectAll": "Tout sélectionner", | ||||||
|  |     "deselectN": "Déselectionner {}", | ||||||
|  |     "xWillBeRemovedButRemainInstalled": "{} sera supprimé d'Obtainium mais restera installé sur l'appareil.", | ||||||
|  |     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", | ||||||
|  |     "removeSelectedApps": "Supprimer les applications sélectionnées", | ||||||
|  |     "updateX": "Mise à jour {}", | ||||||
|  |     "installX": "Installer {}", | ||||||
|  |     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\nas mis à jour", | ||||||
|  |     "changeX": "Changer {}", | ||||||
|  |     "installUpdateApps": "Installer/Mettre à jour les applications", | ||||||
|  |     "installUpdateSelectedApps": "Installer/Mettre à jour les applications sélectionnées", | ||||||
|  |     "markXSelectedAppsAsUpdated": "Marquer {} les applications sélectionnées comme mises à jour ?", | ||||||
|  |     "no": "Non", | ||||||
|  |     "yes": "Oui", | ||||||
|  |     "markSelectedAppsUpdated": "Marquer les applications sélectionnées comme mises à jour", | ||||||
|  |     "pinToTop": "Épingler en haut", | ||||||
|  |     "unpinFromTop": "Détacher du haut", | ||||||
|  |     "resetInstallStatusForSelectedAppsQuestion": "Réinitialiser l'état d'installation des applications sélectionnées ?", | ||||||
|  |     "installStatusOfXWillBeResetExplanation": "L'état d'installation de toutes les applications sélectionnées sera réinitialisé.\n\nCela peut aider lorsque la version de l'application affichée dans Obtainium est incorrecte en raison d'échecs de mises à jour ou d'autres problèmes.", | ||||||
|  |     "shareSelectedAppURLs": "Partager les URL d'application sélectionnées", | ||||||
|  |     "resetInstallStatus": "Réinitialiser le statut d'installation", | ||||||
|  |     "more": "Plus", | ||||||
|  |     "removeOutdatedFilter": "Supprimer le filtre d'application obsolète", | ||||||
|  |     "showOutdatedOnly": "Afficher uniquement les applications obsolètes", | ||||||
|  |     "filter": "Filtre", | ||||||
|  |     "filterActive": "Filtre *", | ||||||
|  |     "filterApps": "Filtrer les applications", | ||||||
|  |     "appName": "Nom de l'application", | ||||||
|  |     "author": "Auteur", | ||||||
|  |     "upToDateApps": "Applications à jour", | ||||||
|  |     "nonInstalledApps": "Applications non installées", | ||||||
|  |     "importExport": "Importer/Exporter", | ||||||
|  |     "settings": "Paramètres", | ||||||
|  |     "exportedTo": "Exporté vers {}", | ||||||
|  |     "obtainiumExport": "Exportation d'Obtainium", | ||||||
|  |     "invalidInput": "Entrée invalide", | ||||||
|  |     "importedX": "Importé {}", | ||||||
|  |     "obtainiumImport": "Importation d'Obtainium", | ||||||
|  |     "importFromURLList": "Importer à partir de la liste d'URL", | ||||||
|  |     "searchQuery": "Requête de recherche", | ||||||
|  |     "appURLList": "Liste d'URL d'application", | ||||||
|  |     "line": "Queue", | ||||||
|  |     "searchX": "Rechercher {}", | ||||||
|  |     "noResults": "Aucun résultat trouvé", | ||||||
|  |     "importX": "Importer {}", | ||||||
|  |     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher à tort comme \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela ne devrait pas affecter les données de l'application.\n\nN'affecte que les URL et les méthodes d'importation tierces.", | ||||||
|  |     "importErrors": "Erreurs d'importation", | ||||||
|  |     "importedXOfYApps": "{} sur {} applications importées.", | ||||||
|  |     "followingURLsHadErrors": "Les URL suivantes comportaient des erreurs :", | ||||||
|  |     "okay": "Okay", | ||||||
|  |     "selectURL": "Sélectionnez l'URL", | ||||||
|  |     "selectURLs": "Sélectionnez les URLs", | ||||||
|  |     "pick": "Prendre", | ||||||
|  |     "theme": "Thème", | ||||||
|  |     "dark": "Sombre", | ||||||
|  |     "light": "Clair", | ||||||
|  |     "followSystem": "Suivre le système", | ||||||
|  |     "obtainium": "Obtainium", | ||||||
|  |     "materialYou": "Material You", | ||||||
|  |     "appSortBy": "Applications triées par", | ||||||
|  |     "authorName": "Auteur/Nom", | ||||||
|  |     "nameAuthor": "Nom/Auteur", | ||||||
|  |     "asAdded": "Comme ajouté", | ||||||
|  |     "appSortOrder": "Ordre de tri des applications", | ||||||
|  |     "ascending": "Ascendant", | ||||||
|  |     "descending": "Descendanr", | ||||||
|  |     "bgUpdateCheckInterval": "Intervalle de vérification des mises à jour en arrière-plan", | ||||||
|  |     "neverManualOnly": "Jamais - Manuel uniquement", | ||||||
|  |     "appearance": "Apparence", | ||||||
|  |     "showWebInAppView": "Afficher la page Web source dans la vue de l'application", | ||||||
|  |     "pinUpdates": "Épingler les mises à jour dans la vue Top des applications", | ||||||
|  |     "updates": "Mises à jour", | ||||||
|  |     "sourceSpecific": "Spécifique à la source", | ||||||
|  |     "appSource": "Source de l'application", | ||||||
|  |     "noLogs": "Aucun journal", | ||||||
|  |     "appLogs": "Journaux d'application", | ||||||
|  |     "close": "Fermer", | ||||||
|  |     "share": "Partager", | ||||||
|  |     "appNotFound": "Application introuvable", | ||||||
|  |     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||||
|  |     "pickAnAPK": "Choisissez un APK", | ||||||
|  |     "appHasMoreThanOnePackage": "{} a plus d'un paquet :", | ||||||
|  |     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture de processeur {}.", | ||||||
|  |     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivantes :", | ||||||
|  |     "warning": "Avertissement", | ||||||
|  |     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de version provient de '{}'. Continuer?", | ||||||
|  |     "updatesAvailable": "Mises à jour disponibles", | ||||||
|  |     "updatesAvailableNotifDescription": "Avertit l'utilisateur que des mises à jour sont disponibles pour une ou plusieurs applications suivies par Obtainium", | ||||||
|  |     "noNewUpdates": "Aucune nouvelle mise à jour.", | ||||||
|  |     "xHasAnUpdate": "{} a une mise à jour.", | ||||||
|  |     "appsUpdated": "Applications mises à jour", | ||||||
|  |     "appsUpdatedNotifDescription": "Avertit l'utilisateur que les mises à jour d'une ou plusieurs applications ont été appliquées en arrière-plan", | ||||||
|  |     "xWasUpdatedToY": "{} a été mis à jour pour {}.", | ||||||
|  |     "errorCheckingUpdates": "Erreur lors de la vérification des mises à jour", | ||||||
|  |     "errorCheckingUpdatesNotifDescription": "Une notification qui s'affiche lorsque la vérification de la mise à jour en arrière-plan échoue", | ||||||
|  |     "appsRemoved": "Applications supprimées", | ||||||
|  |     "appsRemovedNotifDescription": "Avertit l'utilisateur qu'une ou plusieurs applications ont été supprimées en raison d'erreurs lors de leur chargement", | ||||||
|  |     "xWasRemovedDueToErrorY": "{} a été supprimé en raison de cette erreur : {}", | ||||||
|  |     "completeAppInstallation": "Installation complète de l'application", | ||||||
|  |     "obtainiumMustBeOpenToInstallApps": "Obtainium doit être ouvert pour installer des applications", | ||||||
|  |     "completeAppInstallationNotifDescription": "Demande à l'utilisateur de retourner sur Obtainium pour terminer l'installation d'une application", | ||||||
|  |     "checkingForUpdates": "Vérification des mises à jour", | ||||||
|  |     "checkingForUpdatesNotifDescription": "Notification transitoire qui apparaît lors de la recherche de mises à jour", | ||||||
|  |     "pleaseAllowInstallPerm": "Veuillez autoriser Obtainium à installer des applications", | ||||||
|  |     "trackOnly": "Suivi uniquement", | ||||||
|  |     "errorWithHttpStatusCode": "Erreur {}", | ||||||
|  |     "versionCorrectionDisabled": "Correction de version désactivée (le plugin ne semble pas fonctionner)", | ||||||
|  |     "unknown": "Inconnu", | ||||||
|  |     "none": "Aucun", | ||||||
|  |     "never": "Jamais", | ||||||
|  |     "latestVersionX": "Dernière version: {}", | ||||||
|  |     "installedVersionX": "Version installée : {}", | ||||||
|  |     "lastUpdateCheckX": "Vérification de la dernière mise à jour : {}", | ||||||
|  |     "remove": "Retirer", | ||||||
|  |     "yesMarkUpdated": "Oui, marquer comme mis à jour", | ||||||
|  |     "fdroid": "F-Droid", | ||||||
|  |     "appIdOrName": "ID ou nom de l'application", | ||||||
|  |     "appWithIdOrNameNotFound": "Aucune application n'a été trouvée avec cet identifiant ou ce nom", | ||||||
|  |     "reposHaveMultipleApps": "Les dépôts peuvent contenir plusieurs applications", | ||||||
|  |     "fdroidThirdPartyRepo": "Dépôt tiers F-Droid", | ||||||
|  |     "steam": "Steam", | ||||||
|  |     "steamMobile": "Steam Mobile", | ||||||
|  |     "steamChat": "Steam Chat", | ||||||
|  |     "install": "Installer", | ||||||
|  |     "markInstalled": "Marquer installée", | ||||||
|  |     "update": "Mettre à jour", | ||||||
|  |     "markUpdated": "Marquer à jour", | ||||||
|  |     "additionalOptions": "Options additionelles", | ||||||
|  |     "disableVersionDetection": "Désactiver la détection de version", | ||||||
|  |     "noVersionDetectionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de version ne fonctionne pas correctement.", | ||||||
|  |     "downloadingX": "Téléchargement {}", | ||||||
|  |     "downloadNotifDescription": "Avertit l'utilisateur de la progression du téléchargement d'une application", | ||||||
|  |     "noAPKFound": "Aucun APK trouvé", | ||||||
|  |     "noVersionDetection": "Pas de détection de version", | ||||||
|  |     "categorize": "Catégoriser", | ||||||
|  |     "categories": "Catégories", | ||||||
|  |     "category": "Catégorie", | ||||||
|  |     "noCategory": "No Category", | ||||||
|  |     "noCategories": "Aucune catégorie", | ||||||
|  |     "deleteCategoriesQuestion": "Supprimer les catégories ?", | ||||||
|  |     "categoryDeleteWarning": "Toutes les applications dans les catégories supprimées seront définies sur non catégorisées.", | ||||||
|  |     "addCategory": "Ajouter une catégorie", | ||||||
|  |     "label": "Étiquette", | ||||||
|  |     "language": "Langue", | ||||||
|  |     "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", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "Supprimer l'application ?", | ||||||
|  |         "other": "Supprimer les applications ?" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "Trop de demandes (taux limité) - réessayez dans {} minute", | ||||||
|  |         "other": "Trop de demandes (taux limité) - réessayez dans {} minutes" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "La vérification de la mise à jour en arrière-plan a rencontré un {}, planifiera une nouvelle tentative de vérification dans {} minute", | ||||||
|  |         "other": "La vérification de la mise à jour en arrière-plan a rencontré un {}, planifiera une nouvelle tentative de vérification dans {} minutes" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "La vérification des mises à jour en arrière-plan trouvée {} mise à jour - avertira l'utilisateur si nécessaire", | ||||||
|  |         "other": "La vérification des mises à jour en arrière-plan a trouvé {} mises à jour - avertira l'utilisateur si nécessaire" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "{} Application", | ||||||
|  |         "other": "{} Applications" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} URL", | ||||||
|  |         "other": "{} URLs" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} Minute", | ||||||
|  |         "other": "{} Minutes" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} Heure", | ||||||
|  |         "other": "{} Heures" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} Jour", | ||||||
|  |         "other": "{} Jours" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "{n} journal effacé (avant = {before}, après = {after})", | ||||||
|  |         "other": "{n} journaux effacés (avant = {before}, après = {after})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} et 1 autre application ont des mises à jour.", | ||||||
|  |         "other": "{} et {} autres applications ont des mises à jour." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} et 1 autre application ont été mises à jour.", | ||||||
|  |         "other": "{} et {} autres applications ont été mises à jour." | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -34,7 +34,7 @@ | |||||||
|     "githubStarredRepos": "GitHub Csillagos Repo-k", |     "githubStarredRepos": "GitHub Csillagos Repo-k", | ||||||
|     "uname": "Felh.név", |     "uname": "Felh.név", | ||||||
|     "wrongArgNum": "Rossz számú argumentumot adott meg", |     "wrongArgNum": "Rossz számú argumentumot adott meg", | ||||||
|     "xIsTrackOnly": "A(z) {} csak nyomkövethető", |     "xIsTrackOnly": "A(z) {} csak nyomonkövethető", | ||||||
|     "source": "Forrás", |     "source": "Forrás", | ||||||
|     "app": "App", |     "app": "App", | ||||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", |     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", | ||||||
| @@ -56,7 +56,7 @@ | |||||||
|     "appsString": "Appok", |     "appsString": "Appok", | ||||||
|     "noApps": "Nincs App", |     "noApps": "Nincs App", | ||||||
|     "noAppsForFilter": "Nincsenek appok a szűrőhöz", |     "noAppsForFilter": "Nincsenek appok a szűrőhöz", | ||||||
|     "byX": "{} által", |     "byX": "Fejlesztő: {}", | ||||||
|     "percentProgress": "Folyamat: {}%", |     "percentProgress": "Folyamat: {}%", | ||||||
|     "pleaseWait": "Kis türelmet", |     "pleaseWait": "Kis türelmet", | ||||||
|     "updateAvailable": "Frissítés érhető el", |     "updateAvailable": "Frissítés érhető el", | ||||||
| @@ -78,7 +78,7 @@ | |||||||
|     "no": "Nem", |     "no": "Nem", | ||||||
|     "yes": "Igen", |     "yes": "Igen", | ||||||
|     "markSelectedAppsUpdated": "Jelölje meg a kiválasztott appokat frissítettként", |     "markSelectedAppsUpdated": "Jelölje meg a kiválasztott appokat frissítettként", | ||||||
|     "pinToTop": "Rögzítés a felülre", |     "pinToTop": "Rögzítés felülre", | ||||||
|     "unpinFromTop": "Eltávolít felülről", |     "unpinFromTop": "Eltávolít felülről", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott appok telepítési állapotát?", |     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott appok telepítési állapotát?", | ||||||
|     "installStatusOfXWillBeResetExplanation": "A kiválasztott appok telepítési állapota visszaáll.\n\nEz akkor segíthet, ha az Obtainiumban megjelenített app verzió hibás, frissítések vagy egyéb problémák miatt.", |     "installStatusOfXWillBeResetExplanation": "A kiválasztott appok telepítési állapota visszaáll.\n\nEz akkor segíthet, ha az Obtainiumban megjelenített app verzió hibás, frissítések vagy egyéb problémák miatt.", | ||||||
| @@ -191,7 +191,7 @@ | |||||||
|     "update": "Frissít", |     "update": "Frissít", | ||||||
|     "markUpdated": "Frissítettnek jelöl", |     "markUpdated": "Frissítettnek jelöl", | ||||||
|     "additionalOptions": "További lehetőségek", |     "additionalOptions": "További lehetőségek", | ||||||
|     "disableVersionDetection": "Verzióérzékelés letiltása", |     "disableVersionDetection": "Verzió érzékelés letiltása", | ||||||
|     "noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően.", |     "noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően.", | ||||||
|     "downloadingX": "{} letöltés", |     "downloadingX": "{} letöltés", | ||||||
|     "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról", |     "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról", | ||||||
| @@ -212,6 +212,13 @@ | |||||||
|     "removeFromObtainium": "Eltávolítás az Obtainiumból", |     "removeFromObtainium": "Eltávolítás az Obtainiumból", | ||||||
|     "uninstallFromDevice": "Eltávolítás a készülékről", |     "uninstallFromDevice": "Eltávolítás a készülékről", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzióérzékelés le van tiltva.", |     "onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzióérzékelés le van tiltva.", | ||||||
|  |     "releaseDateAsVersion": "Használja a Kiadás dátumát, mint verziót", | ||||||
|  |     "releaseDateAsVersionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzió érzékelése nem működik megfelelően, de elérhető a kiadás dátuma.", | ||||||
|  |     "changes": "Változtatások", | ||||||
|  |     "releaseDate": "Kiadás dátuma", | ||||||
|  |     "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)", | ||||||
|  |     "versionDetection": "Verzió érzékelés", | ||||||
|  |     "standardVersionDetection": "Alapért. verzió érzékelés", | ||||||
|     "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?" | ||||||
|   | |||||||
| @@ -56,9 +56,9 @@ | |||||||
|     "appsString": "App", |     "appsString": "App", | ||||||
|     "noApps": "Nessuna App", |     "noApps": "Nessuna App", | ||||||
|     "noAppsForFilter": "Nessuna App per i filtri selezionati", |     "noAppsForFilter": "Nessuna App per i filtri selezionati", | ||||||
|     "byX": "Da {}", |     "byX": "Di {}", | ||||||
|     "percentProgress": "Progresso: {}%", |     "percentProgress": "Progresso: {}%", | ||||||
|     "pleaseWait": "Attendere prego", |     "pleaseWait": "In attesa", | ||||||
|     "updateAvailable": "Aggiornamento disponibile", |     "updateAvailable": "Aggiornamento disponibile", | ||||||
|     "estimateInBracketsShort": "(prev.)", |     "estimateInBracketsShort": "(prev.)", | ||||||
|     "notInstalled": "Non installato", |     "notInstalled": "Non installato", | ||||||
| @@ -94,7 +94,7 @@ | |||||||
|     "author": "Autore", |     "author": "Autore", | ||||||
|     "upToDateApps": "App aggiornate", |     "upToDateApps": "App aggiornate", | ||||||
|     "nonInstalledApps": "App non installate", |     "nonInstalledApps": "App non installate", | ||||||
|     "importExport": "Importa - Esporta", |     "importExport": "Importa/Esporta", | ||||||
|     "settings": "Impostazioni", |     "settings": "Impostazioni", | ||||||
|     "exportedTo": "Esportato in {}", |     "exportedTo": "Esportato in {}", | ||||||
|     "obtainiumExport": "Esporta da Obtainium", |     "obtainiumExport": "Esporta da Obtainium", | ||||||
| @@ -213,6 +213,13 @@ | |||||||
|     "removeFromObtainium": "Rimuovi da Obtainium", |     "removeFromObtainium": "Rimuovi da Obtainium", | ||||||
|     "uninstallFromDevice": "Disinstalla dal dispositivo", |     "uninstallFromDevice": "Disinstalla dal dispositivo", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "Funziona solo per le App con il rilevamento della versione disattivato.", |     "onlyWorksWithNonVersionDetectApps": "Funziona solo per le App con il rilevamento della versione disattivato.", | ||||||
|  |     "releaseDateAsVersion": "Usa data di rilascio come versione", | ||||||
|  |     "releaseDateAsVersionExplanation": "Questa opzione dovrebbe essere usata solo per le App in cui il rilevamento della versione non funziona correttamente, ma è disponibile una data di rilascio.", | ||||||
|  |     "changes": "Novità", | ||||||
|  |     "releaseDate": "Data di rilascio", | ||||||
|  |     "importFromURLsInFile": "Import from URLs in File (like OPML)", | ||||||
|  |     "versionDetection": "Version Detection", | ||||||
|  |     "standardVersionDetection": "Standard version detection", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Rimuovere l'App?", |         "one": "Rimuovere l'App?", | ||||||
|         "other": "Rimuovere le App?" |         "other": "Rimuovere le App?" | ||||||
|   | |||||||
| @@ -107,7 +107,7 @@ | |||||||
|     "line": "行", |     "line": "行", | ||||||
|     "searchX": "{}で検索", |     "searchX": "{}で検索", | ||||||
|     "noResults": "結果は見つかりませんでした", |     "noResults": "結果は見つかりませんでした", | ||||||
|     "importX": "{}をインポートする", |     "importX": "{}をインポート", | ||||||
|     "importedAppsIdDisclaimer": "インポートしたアプリが「未インストール」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響しません。\n\nURLとサードパーティのインポートメソッドにのみ影響します。", |     "importedAppsIdDisclaimer": "インポートしたアプリが「未インストール」と表示されることがあります。\nこれを解決するには、Obtainiumから再インストールしてください。\nアプリのデータには影響しません。\n\nURLとサードパーティのインポートメソッドにのみ影響します。", | ||||||
|     "importErrors": "インポートエラー", |     "importErrors": "インポートエラー", | ||||||
|     "importedXOfYApps": "{} / {} アプリをインポートしました", |     "importedXOfYApps": "{} / {} アプリをインポートしました", | ||||||
| @@ -213,6 +213,13 @@ | |||||||
|     "removeFromObtainium": "Obtainiumから削除する", |     "removeFromObtainium": "Obtainiumから削除する", | ||||||
|     "uninstallFromDevice": "デバイスからアンインストールする", |     "uninstallFromDevice": "デバイスからアンインストールする", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。", |     "onlyWorksWithNonVersionDetectApps": "バージョン検出を無効にしているアプリにのみ動作します。", | ||||||
|  |     "releaseDateAsVersion": "リリース日をバージョンとして使用する", | ||||||
|  |     "releaseDateAsVersionExplanation": "このオプションは、バージョン検出が正しく機能しないアプリで、リリース日が利用可能な場合にのみ使用する必要があります。", | ||||||
|  |     "changes": "変更点", | ||||||
|  |     "releaseDate": "リリース日", | ||||||
|  |     "importFromURLsInFile": "ファイル(OPMLなど)内のURLからインポート", | ||||||
|  |     "versionDetection": "バージョン検出", | ||||||
|  |     "standardVersionDetection": "標準のバージョン検出", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "アプリを削除しますか?", |         "one": "アプリを削除しますか?", | ||||||
|         "other": "アプリを削除しますか?" |         "other": "アプリを削除しますか?" | ||||||
|   | |||||||
| @@ -213,6 +213,13 @@ | |||||||
|     "filterAPKsByRegEx": "Filter APKs by Regular Expression", |     "filterAPKsByRegEx": "Filter APKs by Regular Expression", | ||||||
|     "removeFromObtainium": "Remove from Obtainium", |     "removeFromObtainium": "Remove from Obtainium", | ||||||
|     "uninstallFromDevice": "Uninstall from Device", |     "uninstallFromDevice": "Uninstall from Device", | ||||||
|  |     "releaseDateAsVersion": "Use Release Date as Version", | ||||||
|  |     "releaseDateAsVersionExplanation": "This option should only be used for Apps where version detection does not work correctly, but a release date is available.", | ||||||
|  |     "changes": "Changes", | ||||||
|  |     "releaseDate": "Release Date", | ||||||
|  |     "importFromURLsInFile": "Import from URLs in File (like OPML)", | ||||||
|  |     "versionDetection": "Version Detection", | ||||||
|  |     "standardVersionDetection": "Standard version detection", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "删除应用?", |         "one": "删除应用?", | ||||||
|         "other": "删除应用?" |         "other": "删除应用?" | ||||||
|   | |||||||
| @@ -1,5 +1,9 @@ | |||||||
|  | import 'dart:io'; | ||||||
|  |  | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:html/parser.dart'; | import 'package:html/parser.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| @@ -7,6 +11,23 @@ class APKMirror extends AppSource { | |||||||
|   APKMirror() { |   APKMirror() { | ||||||
|     host = 'apkmirror.com'; |     host = 'apkmirror.com'; | ||||||
|     enforceTrackOnly = true; |     enforceTrackOnly = true; | ||||||
|  |  | ||||||
|  |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('fallbackToOlderReleases', | ||||||
|  |             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||||
|  |       ], | ||||||
|  |       [ | ||||||
|  |         GeneratedFormTextField('filterReleaseTitlesByRegEx', | ||||||
|  |             label: tr('filterReleaseTitlesByRegEx'), | ||||||
|  |             required: false, | ||||||
|  |             additionalValidators: [ | ||||||
|  |               (value) { | ||||||
|  |                 return regExValidator(value); | ||||||
|  |               } | ||||||
|  |             ]) | ||||||
|  |       ] | ||||||
|  |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -28,12 +49,38 @@ class APKMirror extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|  |     bool fallbackToOlderReleases = | ||||||
|  |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|  |     String? regexFilter = | ||||||
|  |         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) | ||||||
|  |                     ?.isNotEmpty == | ||||||
|  |                 true | ||||||
|  |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|  |             : null; | ||||||
|     Response res = await get(Uri.parse('$standardUrl/feed')); |     Response res = await get(Uri.parse('$standardUrl/feed')); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       String? titleString = parse(res.body) |       var items = parse(res.body).querySelectorAll('item'); | ||||||
|           .querySelector('item') |       dynamic targetRelease; | ||||||
|           ?.querySelector('title') |       for (int i = 0; i < items.length; i++) { | ||||||
|           ?.innerHtml; |         if (!fallbackToOlderReleases && i > 0) break; | ||||||
|  |         String? nameToFilter = items[i].querySelector('title')?.innerHtml; | ||||||
|  |         if (regexFilter != null && | ||||||
|  |             nameToFilter != null && | ||||||
|  |             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         targetRelease = items[i]; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       String? titleString = targetRelease?.querySelector('title')?.innerHtml; | ||||||
|  |       String? dateString = targetRelease | ||||||
|  |           ?.querySelector('pubDate') | ||||||
|  |           ?.innerHtml | ||||||
|  |           .split(' ') | ||||||
|  |           .sublist(0, 5) | ||||||
|  |           .join(' '); | ||||||
|  |       DateTime? releaseDate = | ||||||
|  |           dateString != null ? HttpDate.parse('$dateString GMT') : null; | ||||||
|       String? version = titleString |       String? version = titleString | ||||||
|           ?.substring(RegExp('[0-9]').firstMatch(titleString)?.start ?? 0, |           ?.substring(RegExp('[0-9]').firstMatch(titleString)?.start ?? 0, | ||||||
|               RegExp(' by ').firstMatch(titleString)?.start ?? 0) |               RegExp(' by ').firstMatch(titleString)?.start ?? 0) | ||||||
| @@ -44,7 +91,8 @@ class APKMirror extends AppSource { | |||||||
|       if (version == null || version.isEmpty) { |       if (version == null || version.isEmpty) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, [], getAppNames(standardUrl)); |       return APKDetails(version, [], getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -54,9 +54,9 @@ class Codeberg extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases']; |     bool includePrereleases = additionalSettings['includePrereleases'] == true; | ||||||
|     bool fallbackToOlderReleases = |     bool fallbackToOlderReleases = | ||||||
|         additionalSettings['fallbackToOlderReleases']; |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|     String? regexFilter = |     String? regexFilter = | ||||||
|         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) |         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) | ||||||
|                     ?.isNotEmpty == |                     ?.isNotEmpty == | ||||||
| @@ -112,11 +112,15 @@ class Codeberg extends AppSource { | |||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String? version = targetRelease['tag_name']; |       String? version = targetRelease['tag_name']; | ||||||
|  |       DateTime? releaseDate = targetRelease['published_at'] != null | ||||||
|  |           ? DateTime.parse(targetRelease['published_at']) | ||||||
|  |           : null; | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |       return APKDetails(version, targetRelease['apkUrls'] as List<String>, | ||||||
|           getAppNames(standardUrl)); |           getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -69,6 +69,8 @@ class FDroidRepo extends AppSource { | |||||||
|           foundApps[0].querySelector('name')?.innerHtml ?? appIdOrName; |           foundApps[0].querySelector('name')?.innerHtml ?? appIdOrName; | ||||||
|       var releases = foundApps[0].querySelectorAll('package'); |       var releases = foundApps[0].querySelectorAll('package'); | ||||||
|       String? latestVersion = releases[0].querySelector('version')?.innerHtml; |       String? latestVersion = releases[0].querySelector('version')?.innerHtml; | ||||||
|  |       String? added = releases[0].querySelector('added')?.innerHtml; | ||||||
|  |       DateTime? releaseDate = added != null ? DateTime.parse(added) : null; | ||||||
|       if (latestVersion == null) { |       if (latestVersion == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
| @@ -78,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, apkUrls, AppNames(authorName, appName), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -101,9 +101,9 @@ class GitHub extends AppSource { | |||||||
|     String standardUrl, |     String standardUrl, | ||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     bool includePrereleases = additionalSettings['includePrereleases']; |     bool includePrereleases = additionalSettings['includePrereleases'] == true; | ||||||
|     bool fallbackToOlderReleases = |     bool fallbackToOlderReleases = | ||||||
|         additionalSettings['fallbackToOlderReleases']; |         additionalSettings['fallbackToOlderReleases'] == true; | ||||||
|     String? regexFilter = |     String? regexFilter = | ||||||
|         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) |         (additionalSettings['filterReleaseTitlesByRegEx'] as String?) | ||||||
|                     ?.isNotEmpty == |                     ?.isNotEmpty == | ||||||
| @@ -154,11 +154,15 @@ class GitHub extends AppSource { | |||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String? version = targetRelease['tag_name']; |       String? version = targetRelease['tag_name']; | ||||||
|  |       DateTime? releaseDate = targetRelease['published_at'] != null | ||||||
|  |           ? DateTime.parse(targetRelease['published_at']) | ||||||
|  |           : null; | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, targetRelease['apkUrls'] as List<String>, |       return APKDetails(version, targetRelease['apkUrls'] as List<String>, | ||||||
|           getAppNames(standardUrl)); |           getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       rateLimitErrorCheck(res); |       rateLimitErrorCheck(res); | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|   | |||||||
| @@ -54,10 +54,14 @@ class GitLab extends AppSource { | |||||||
|       var entryId = entry?.querySelector('id')?.innerHtml; |       var entryId = entry?.querySelector('id')?.innerHtml; | ||||||
|       var version = |       var version = | ||||||
|           entryId == null ? null : Uri.parse(entryId).pathSegments.last; |           entryId == null ? null : Uri.parse(entryId).pathSegments.last; | ||||||
|  |       var releaseDateString = entry?.querySelector('updated')?.innerHtml; | ||||||
|  |       DateTime? releaseDate = | ||||||
|  |           releaseDateString != null ? DateTime.parse(releaseDateString) : null; | ||||||
|       if (version == null) { |       if (version == null) { | ||||||
|         throw NoVersionError(); |         throw NoVersionError(); | ||||||
|       } |       } | ||||||
|       return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl)); |       return APKDetails(version, apkUrls, GitHub().getAppNames(standardUrl), | ||||||
|  |           releaseDate: releaseDate); | ||||||
|     } else { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -27,6 +27,10 @@ class HTML extends AppSource { | |||||||
|           .where((element) => element.toLowerCase().endsWith('.apk')) |           .where((element) => element.toLowerCase().endsWith('.apk')) | ||||||
|           .toList(); |           .toList(); | ||||||
|       links.sort((a, b) => a.split('/').last.compareTo(b.split('/').last)); |       links.sort((a, b) => a.split('/').last.compareTo(b.split('/').last)); | ||||||
|  |       if (additionalSettings['apkFilterRegEx'] != null) { | ||||||
|  |         var reg = RegExp(additionalSettings['apkFilterRegEx']); | ||||||
|  |         links = links.where((element) => reg.hasMatch(element)).toList(); | ||||||
|  |       } | ||||||
|       if (links.isEmpty) { |       if (links.isEmpty) { | ||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
| @@ -37,7 +41,9 @@ class HTML extends AppSource { | |||||||
|           .map((e) => e.toLowerCase().startsWith('http://') || |           .map((e) => e.toLowerCase().startsWith('http://') || | ||||||
|                   e.toLowerCase().startsWith('https://') |                   e.toLowerCase().startsWith('https://') | ||||||
|               ? e |               ? e | ||||||
|               : '${uri.origin}/$e') |               : e.startsWith('/') | ||||||
|  |                   ? '${uri.origin}/$e' | ||||||
|  |                   : '${uri.origin}/${uri.path}/$e') | ||||||
|           .toList(); |           .toList(); | ||||||
|       return APKDetails(version, apkUrls, AppNames(uri.host, tr('app'))); |       return APKDetails(version, apkUrls, AppNames(uri.host, tr('app'))); | ||||||
|     } else { |     } else { | ||||||
|   | |||||||
| @@ -10,7 +10,10 @@ class SteamMobile extends AppSource { | |||||||
|     host = 'store.steampowered.com'; |     host = 'store.steampowered.com'; | ||||||
|     name = tr('steam'); |     name = tr('steam'); | ||||||
|     additionalSourceAppSpecificSettingFormItems = [ |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|       [GeneratedFormDropdown('app', apks.entries.toList(), label: tr('app'))] |       [ | ||||||
|  |         GeneratedFormDropdown('app', apks.entries.toList(), | ||||||
|  |             label: tr('app'), defaultValue: apks.entries.toList()[0].key) | ||||||
|  |       ] | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -35,7 +38,8 @@ class SteamMobile extends AppSource { | |||||||
|       if (apkNamePrefix == null) { |       if (apkNamePrefix == null) { | ||||||
|         throw NoReleasesError(); |         throw NoReleasesError(); | ||||||
|       } |       } | ||||||
|       String apkInURLRegexPattern = '/$apkNamePrefix-[^/]+\\.apk\$'; |       String apkInURLRegexPattern = | ||||||
|  |           '/$apkNamePrefix-([0-9]+\\.)*[0-9]+\\.apk\$'; | ||||||
|       var links = parse(res.body) |       var links = parse(res.body) | ||||||
|           .querySelectorAll('a') |           .querySelectorAll('a') | ||||||
|           .map((e) => e.attributes['href'] ?? '') |           .map((e) => e.attributes['href'] ?? '') | ||||||
|   | |||||||
| @@ -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.10.9'; | const String currentVersion = '0.11.8'; | ||||||
| const String currentReleaseTag = | const String currentReleaseTag = | ||||||
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES |     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES | ||||||
|  |  | ||||||
| @@ -33,7 +33,9 @@ const supportedLocales = [ | |||||||
|   Locale('it'), |   Locale('it'), | ||||||
|   Locale('ja'), |   Locale('ja'), | ||||||
|   Locale('hu'), |   Locale('hu'), | ||||||
|   Locale('de') |   Locale('de'), | ||||||
|  |   Locale('fa'), | ||||||
|  |   Locale('fr') | ||||||
| ]; | ]; | ||||||
| const fallbackLocale = Locale('en'); | const fallbackLocale = Locale('en'); | ||||||
| const localeDir = 'assets/translations'; | const localeDir = 'assets/translations'; | ||||||
| @@ -210,6 +212,14 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|               false) |               false) | ||||||
|         ]); |         ]); | ||||||
|       } |       } | ||||||
|  |       if (!supportedLocales | ||||||
|  |               .map((e) => e.languageCode) | ||||||
|  |               .contains(context.locale.languageCode) || | ||||||
|  |           settingsProvider.forcedLocale == null && | ||||||
|  |               context.deviceLocale.languageCode != | ||||||
|  |                   context.locale.languageCode) { | ||||||
|  |         settingsProvider.resetLocaleSafe(context); | ||||||
|  |       } | ||||||
|       // Register the background update task according to the user's setting |       // Register the background update task according to the user's setting | ||||||
|       if (existingUpdateInterval != settingsProvider.updateInterval) { |       if (existingUpdateInterval != settingsProvider.updateInterval) { | ||||||
|         if (existingUpdateInterval != -1) { |         if (existingUpdateInterval != -1) { | ||||||
|   | |||||||
| @@ -71,8 +71,6 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|       var settingsProvider = context.read<SettingsProvider>(); |       var settingsProvider = context.read<SettingsProvider>(); | ||||||
|       () async { |       () async { | ||||||
|         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; |         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; | ||||||
|         var userPickedNoVersionDetection = |  | ||||||
|             additionalSettings['noVersionDetection'] == true; |  | ||||||
|         var cont = true; |         var cont = true; | ||||||
|         if ((userPickedTrackOnly || pickedSource!.enforceTrackOnly) && |         if ((userPickedTrackOnly || pickedSource!.enforceTrackOnly) && | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
| @@ -93,7 +91,21 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|                 null) { |                 null) { | ||||||
|           cont = false; |           cont = false; | ||||||
|         } |         } | ||||||
|         if (userPickedNoVersionDetection && |         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 |             // ignore: use_build_context_synchronously | ||||||
|             await showDialog( |             await showDialog( | ||||||
|                     context: context, |                     context: context, | ||||||
| @@ -112,13 +124,12 @@ class _AddAppPageState extends State<AddAppPage> { | |||||||
|           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; |           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; | ||||||
|           App app = await sourceProvider.getApp( |           App app = await sourceProvider.getApp( | ||||||
|               pickedSource!, userInput, additionalSettings, |               pickedSource!, userInput, additionalSettings, | ||||||
|               trackOnlyOverride: trackOnly, |               trackOnlyOverride: trackOnly); | ||||||
|               noVersionDetectionOverride: userPickedNoVersionDetection); |  | ||||||
|           if (!trackOnly) { |           if (!trackOnly) { | ||||||
|             await settingsProvider.getInstallPermission(); |             await settingsProvider.getInstallPermission(); | ||||||
|           } |           } | ||||||
|           // Only download the APK here if you need to for the package ID |           // Only download the APK here if you need to for the package ID | ||||||
|           if (sourceProvider.isTempId(app.id) && |           if (sourceProvider.isTempId(app) && | ||||||
|               app.additionalSettings['trackOnly'] != true) { |               app.additionalSettings['trackOnly'] != true) { | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
|             var apkUrl = await appsProvider.confirmApkUrl(app, context); |             var apkUrl = await appsProvider.confirmApkUrl(app, context); | ||||||
|   | |||||||
| @@ -42,8 +42,6 @@ class _AppPageState extends State<AppPage> { | |||||||
|       getUpdate(app.app.id); |       getUpdate(app.app.id); | ||||||
|     } |     } | ||||||
|     var trackOnly = app?.app.additionalSettings['trackOnly'] == true; |     var trackOnly = app?.app.additionalSettings['trackOnly'] == true; | ||||||
|     var noVersionDetection = |  | ||||||
|         app?.app.additionalSettings['noVersionDetection'] == true; |  | ||||||
|  |  | ||||||
|     var infoColumn = Column( |     var infoColumn = Column( | ||||||
|       mainAxisAlignment: MainAxisAlignment.center, |       mainAxisAlignment: MainAxisAlignment.center, | ||||||
| @@ -113,7 +111,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|       mainAxisAlignment: MainAxisAlignment.center, |       mainAxisAlignment: MainAxisAlignment.center, | ||||||
|       crossAxisAlignment: CrossAxisAlignment.stretch, |       crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|       children: [ |       children: [ | ||||||
|         const SizedBox(height: 150), |         const SizedBox(height: 125), | ||||||
|         app?.installedInfo != null |         app?.installedInfo != null | ||||||
|             ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [ |             ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [ | ||||||
|                 Image.memory( |                 Image.memory( | ||||||
| @@ -136,6 +134,21 @@ class _AppPageState extends State<AppPage> { | |||||||
|           textAlign: TextAlign.center, |           textAlign: TextAlign.center, | ||||||
|           style: Theme.of(context).textTheme.headlineMedium, |           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( |         const SizedBox( | ||||||
|           height: 32, |           height: 32, | ||||||
|         ), |         ), | ||||||
| @@ -192,7 +205,8 @@ class _AppPageState extends State<AppPage> { | |||||||
|                   child: Row( |                   child: Row( | ||||||
|                       mainAxisAlignment: MainAxisAlignment.spaceEvenly, |                       mainAxisAlignment: MainAxisAlignment.spaceEvenly, | ||||||
|                       children: [ |                       children: [ | ||||||
|                         if (noVersionDetection && |                         if (app?.app.additionalSettings['versionDetection'] != | ||||||
|  |                                 'standardVersionDetection' && | ||||||
|                             !trackOnly && |                             !trackOnly && | ||||||
|                             app?.app.installedVersion != null && |                             app?.app.installedVersion != null && | ||||||
|                             app?.app.installedVersion != app?.app.latestVersion) |                             app?.app.installedVersion != app?.app.latestVersion) | ||||||
| @@ -268,19 +282,49 @@ class _AppPageState extends State<AppPage> { | |||||||
|                                             ); |                                             ); | ||||||
|                                           }).then((values) { |                                           }).then((values) { | ||||||
|                                         if (app != null && values != null) { |                                         if (app != null && values != null) { | ||||||
|                                           var changedApp = app.app; |                                           Map<String, dynamic> | ||||||
|                                           changedApp.additionalSettings = |                                               originalSettings = | ||||||
|                                               values; |                                               app.app.additionalSettings; | ||||||
|  |                                           app.app.additionalSettings = values; | ||||||
|                                           if (source.enforceTrackOnly) { |                                           if (source.enforceTrackOnly) { | ||||||
|                                             changedApp.additionalSettings[ |                                             app.app.additionalSettings[ | ||||||
|                                                 'trackOnly'] = true; |                                                 'trackOnly'] = true; | ||||||
|                                             showError( |                                             showError( | ||||||
|                                                 tr('appsFromSourceAreTrackOnly'), |                                                 tr('appsFromSourceAreTrackOnly'), | ||||||
|                                                 context); |                                                 context); | ||||||
|                                           } |                                           } | ||||||
|                                           appsProvider.saveApps( |                                           if (app.app.additionalSettings[ | ||||||
|                                               [changedApp]).then((value) { |                                                   'versionDetection'] == | ||||||
|                                             getUpdate(changedApp.id); |                                               '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); | ||||||
|                                           }); |                                           }); | ||||||
|                                         } |                                         } | ||||||
|                                       }); |                                       }); | ||||||
|   | |||||||
| @@ -54,12 +54,12 @@ class AppsPageState extends State<AppsPage> { | |||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     var appsProvider = context.watch<AppsProvider>(); |     var appsProvider = context.watch<AppsProvider>(); | ||||||
|     var settingsProvider = context.watch<SettingsProvider>(); |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|     var sortedApps = appsProvider.apps.values.toList(); |     var listedApps = appsProvider.apps.values.toList(); | ||||||
|     var currentFilterIsUpdatesOnly = |     var currentFilterIsUpdatesOnly = | ||||||
|         filter.isIdenticalTo(updatesOnlyFilter, settingsProvider); |         filter.isIdenticalTo(updatesOnlyFilter, settingsProvider); | ||||||
|  |  | ||||||
|     selectedApps = selectedApps |     selectedApps = selectedApps | ||||||
|         .where((element) => sortedApps.map((e) => e.app).contains(element)) |         .where((element) => listedApps.map((e) => e.app).contains(element)) | ||||||
|         .toSet(); |         .toSet(); | ||||||
|  |  | ||||||
|     toggleAppSelected(App app) { |     toggleAppSelected(App app) { | ||||||
| @@ -72,7 +72,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     sortedApps = sortedApps.where((app) { |     listedApps = listedApps.where((app) { | ||||||
|       if (app.app.installedVersion == app.app.latestVersion && |       if (app.app.installedVersion == app.app.latestVersion && | ||||||
|           !(filter.includeUptodate)) { |           !(filter.includeUptodate)) { | ||||||
|         return false; |         return false; | ||||||
| @@ -111,7 +111,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|       return true; |       return true; | ||||||
|     }).toList(); |     }).toList(); | ||||||
|  |  | ||||||
|     sortedApps.sort((a, b) { |     listedApps.sort((a, b) { | ||||||
|       var nameA = a.installedInfo?.name ?? a.app.name; |       var nameA = a.installedInfo?.name ?? a.app.name; | ||||||
|       var nameB = b.installedInfo?.name ?? b.app.name; |       var nameB = b.installedInfo?.name ?? b.app.name; | ||||||
|       int result = 0; |       int result = 0; | ||||||
| @@ -119,25 +119,30 @@ class AppsPageState extends State<AppsPage> { | |||||||
|         result = (a.app.author + nameA).compareTo(b.app.author + nameB); |         result = (a.app.author + nameA).compareTo(b.app.author + nameB); | ||||||
|       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { |       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { | ||||||
|         result = (nameA + a.app.author).compareTo(nameB + b.app.author); |         result = (nameA + a.app.author).compareTo(nameB + b.app.author); | ||||||
|  |       } else if (settingsProvider.sortColumn == | ||||||
|  |           SortColumnSettings.releaseDate) { | ||||||
|  |         result = (a.app.releaseDate)?.compareTo( | ||||||
|  |                 b.app.releaseDate ?? DateTime.fromMicrosecondsSinceEpoch(0)) ?? | ||||||
|  |             0; | ||||||
|       } |       } | ||||||
|       return result; |       return result; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     if (settingsProvider.sortOrder == SortOrderSettings.descending) { |     if (settingsProvider.sortOrder == SortOrderSettings.descending) { | ||||||
|       sortedApps = sortedApps.reversed.toList(); |       listedApps = listedApps.reversed.toList(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     var existingUpdates = appsProvider.findExistingUpdates(installedOnly: true); |     var existingUpdates = appsProvider.findExistingUpdates(installedOnly: true); | ||||||
|  |  | ||||||
|     var existingUpdateIdsAllOrSelected = existingUpdates |     var existingUpdateIdsAllOrSelected = existingUpdates | ||||||
|         .where((element) => selectedApps.isEmpty |         .where((element) => selectedApps.isEmpty | ||||||
|             ? sortedApps.where((a) => a.app.id == element).isNotEmpty |             ? listedApps.where((a) => a.app.id == element).isNotEmpty | ||||||
|             : selectedApps.map((e) => e.id).contains(element)) |             : selectedApps.map((e) => e.id).contains(element)) | ||||||
|         .toList(); |         .toList(); | ||||||
|     var newInstallIdsAllOrSelected = appsProvider |     var newInstallIdsAllOrSelected = appsProvider | ||||||
|         .findExistingUpdates(nonInstalledOnly: true) |         .findExistingUpdates(nonInstalledOnly: true) | ||||||
|         .where((element) => selectedApps.isEmpty |         .where((element) => selectedApps.isEmpty | ||||||
|             ? sortedApps.where((a) => a.app.id == element).isNotEmpty |             ? listedApps.where((a) => a.app.id == element).isNotEmpty | ||||||
|             : selectedApps.map((e) => e.id).contains(element)) |             : selectedApps.map((e) => e.id).contains(element)) | ||||||
|         .toList(); |         .toList(); | ||||||
|  |  | ||||||
| @@ -159,26 +164,26 @@ class AppsPageState extends State<AppsPage> { | |||||||
|  |  | ||||||
|     if (settingsProvider.pinUpdates) { |     if (settingsProvider.pinUpdates) { | ||||||
|       var temp = []; |       var temp = []; | ||||||
|       sortedApps = sortedApps.where((sa) { |       listedApps = listedApps.where((sa) { | ||||||
|         if (existingUpdates.contains(sa.app.id)) { |         if (existingUpdates.contains(sa.app.id)) { | ||||||
|           temp.add(sa); |           temp.add(sa); | ||||||
|           return false; |           return false; | ||||||
|         } |         } | ||||||
|         return true; |         return true; | ||||||
|       }).toList(); |       }).toList(); | ||||||
|       sortedApps = [...temp, ...sortedApps]; |       listedApps = [...temp, ...listedApps]; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     var tempPinned = []; |     var tempPinned = []; | ||||||
|     var tempNotPinned = []; |     var tempNotPinned = []; | ||||||
|     for (var a in sortedApps) { |     for (var a in listedApps) { | ||||||
|       if (a.app.pinned) { |       if (a.app.pinned) { | ||||||
|         tempPinned.add(a); |         tempPinned.add(a); | ||||||
|       } else { |       } else { | ||||||
|         tempNotPinned.add(a); |         tempNotPinned.add(a); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     sortedApps = [...tempPinned, ...tempNotPinned]; |     listedApps = [...tempPinned, ...tempNotPinned]; | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|       backgroundColor: Theme.of(context).colorScheme.surface, |       backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
| @@ -198,7 +203,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           }, |           }, | ||||||
|           child: CustomScrollView(slivers: <Widget>[ |           child: CustomScrollView(slivers: <Widget>[ | ||||||
|             CustomAppBar(title: tr('appsString')), |             CustomAppBar(title: tr('appsString')), | ||||||
|             if (appsProvider.loadingApps || sortedApps.isEmpty) |             if (appsProvider.loadingApps || listedApps.isEmpty) | ||||||
|               SliverFillRemaining( |               SliverFillRemaining( | ||||||
|                   child: Center( |                   child: Center( | ||||||
|                       child: appsProvider.loadingApps |                       child: appsProvider.loadingApps | ||||||
| @@ -225,86 +230,142 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                 delegate: SliverChildBuilderDelegate( |                 delegate: SliverChildBuilderDelegate( | ||||||
|                     (BuildContext context, int index) { |                     (BuildContext context, int index) { | ||||||
|               String? changesUrl = SourceProvider() |               String? changesUrl = SourceProvider() | ||||||
|                   .getSource(sortedApps[index].app.url) |                   .getSource(listedApps[index].app.url) | ||||||
|                   .changeLogPageFromStandardUrl(sortedApps[index].app.url); |                   .changeLogPageFromStandardUrl(listedApps[index].app.url); | ||||||
|               var transparent = const Color.fromARGB(0, 0, 0, 0).value; |               var transparent = const Color.fromARGB(0, 0, 0, 0).value; | ||||||
|  |               var hasUpdate = listedApps[index].app.installedVersion != null && | ||||||
|  |                   listedApps[index].app.installedVersion != | ||||||
|  |                       listedApps[index].app.latestVersion; | ||||||
|  |               var updateButton = IconButton( | ||||||
|  |                   visualDensity: VisualDensity.compact, | ||||||
|  |                   color: Theme.of(context).colorScheme.primary, | ||||||
|  |                   tooltip: | ||||||
|  |                       listedApps[index].app.additionalSettings['trackOnly'] == | ||||||
|  |                               true | ||||||
|  |                           ? tr('markUpdated') | ||||||
|  |                           : tr('update'), | ||||||
|  |                   onPressed: appsProvider.areDownloadsRunning() | ||||||
|  |                       ? null | ||||||
|  |                       : () { | ||||||
|  |                           appsProvider.downloadAndInstallLatestApps([ | ||||||
|  |                             listedApps[index].app.id | ||||||
|  |                           ], globalNavigatorKey.currentContext).catchError((e) { | ||||||
|  |                             showError(e, context); | ||||||
|  |                           }); | ||||||
|  |                         }, | ||||||
|  |                   icon: Icon( | ||||||
|  |                       listedApps[index].app.additionalSettings['trackOnly'] == | ||||||
|  |                               true | ||||||
|  |                           ? Icons.check_circle_outline | ||||||
|  |                           : Icons.install_mobile)); | ||||||
|               return Container( |               return Container( | ||||||
|                   decoration: BoxDecoration( |                   decoration: BoxDecoration( | ||||||
|                       border: Border.symmetric( |                       border: Border.symmetric( | ||||||
|                           vertical: BorderSide( |                           vertical: BorderSide( | ||||||
|                               width: 4, |                               width: 4, | ||||||
|                               color: Color( |                               color: Color( | ||||||
|                                   sortedApps[index].app.categories.isNotEmpty |                                   listedApps[index].app.categories.isNotEmpty | ||||||
|                                       ? settingsProvider.categories[ |                                       ? settingsProvider.categories[ | ||||||
|                                               sortedApps[index] |                                               listedApps[index] | ||||||
|                                                   .app |                                                   .app | ||||||
|                                                   .categories |                                                   .categories | ||||||
|                                                   .first] ?? |                                                   .first] ?? | ||||||
|                                           transparent |                                           transparent | ||||||
|                                       : transparent)))), |                                       : transparent)))), | ||||||
|                   child: ListTile( |                   child: ListTile( | ||||||
|                     tileColor: sortedApps[index].app.pinned |                     tileColor: listedApps[index].app.pinned | ||||||
|                         ? Colors.grey.withOpacity(0.1) |                         ? Colors.grey.withOpacity(0.1) | ||||||
|                         : Colors.transparent, |                         : Colors.transparent, | ||||||
|                     selectedTileColor: Theme.of(context) |                     selectedTileColor: Theme.of(context) | ||||||
|                         .colorScheme |                         .colorScheme | ||||||
|                         .primary |                         .primary | ||||||
|                         .withOpacity(sortedApps[index].app.pinned ? 0.2 : 0.1), |                         .withOpacity(listedApps[index].app.pinned ? 0.2 : 0.1), | ||||||
|                     selected: selectedApps.contains(sortedApps[index].app), |                     selected: selectedApps.contains(listedApps[index].app), | ||||||
|                     onLongPress: () { |                     onLongPress: () { | ||||||
|                       toggleAppSelected(sortedApps[index].app); |                       toggleAppSelected(listedApps[index].app); | ||||||
|                     }, |                     }, | ||||||
|                     leading: sortedApps[index].installedInfo != null |                     leading: listedApps[index].installedInfo != null | ||||||
|                         ? Image.memory( |                         ? Image.memory( | ||||||
|                             sortedApps[index].installedInfo!.icon!, |                             listedApps[index].installedInfo!.icon!, | ||||||
|                             gaplessPlayback: true, |                             gaplessPlayback: true, | ||||||
|                           ) |                           ) | ||||||
|                         : null, |                         : Row( | ||||||
|  |                             mainAxisSize: MainAxisSize.min, | ||||||
|  |                             mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                             children: [ | ||||||
|  |                                 Transform( | ||||||
|  |                                     alignment: Alignment.center, | ||||||
|  |                                     transform: Matrix4.rotationZ(0.31), | ||||||
|  |                                     child: Padding( | ||||||
|  |                                       padding: const EdgeInsets.all(15), | ||||||
|  |                                       child: Image( | ||||||
|  |                                         image: const AssetImage( | ||||||
|  |                                             'assets/graphics/icon_small.png'), | ||||||
|  |                                         color: Colors.white.withOpacity(0.1), | ||||||
|  |                                         colorBlendMode: BlendMode.modulate, | ||||||
|  |                                         gaplessPlayback: true, | ||||||
|  |                                       ), | ||||||
|  |                                     )), | ||||||
|  |                               ]), | ||||||
|                     title: Text( |                     title: Text( | ||||||
|                       sortedApps[index].installedInfo?.name ?? |                       maxLines: 1, | ||||||
|                           sortedApps[index].app.name, |                       listedApps[index].installedInfo?.name ?? | ||||||
|  |                           listedApps[index].app.name, | ||||||
|                       style: TextStyle( |                       style: TextStyle( | ||||||
|                         fontWeight: sortedApps[index].app.pinned |                         overflow: TextOverflow.ellipsis, | ||||||
|  |                         fontWeight: listedApps[index].app.pinned | ||||||
|                             ? FontWeight.bold |                             ? FontWeight.bold | ||||||
|                             : FontWeight.normal, |                             : FontWeight.normal, | ||||||
|                       ), |                       ), | ||||||
|                     ), |                     ), | ||||||
|                     subtitle: Text( |                     subtitle: Text( | ||||||
|                         tr('byX', args: [sortedApps[index].app.author]), |                         tr('byX', args: [listedApps[index].app.author]), | ||||||
|  |                         maxLines: 1, | ||||||
|                         style: TextStyle( |                         style: TextStyle( | ||||||
|                             fontWeight: sortedApps[index].app.pinned |                             overflow: TextOverflow.ellipsis, | ||||||
|  |                             fontWeight: listedApps[index].app.pinned | ||||||
|                                 ? FontWeight.bold |                                 ? FontWeight.bold | ||||||
|                                 : FontWeight.normal)), |                                 : FontWeight.normal)), | ||||||
|                     trailing: SingleChildScrollView( |                     trailing: listedApps[index].downloadProgress != null | ||||||
|                         reverse: true, |                         ? Text(tr('percentProgress', args: [ | ||||||
|                         child: sortedApps[index].downloadProgress != null |                             listedApps[index] | ||||||
|                             ? Text(tr('percentProgress', args: [ |                                     .downloadProgress | ||||||
|                                 sortedApps[index] |                                     ?.toInt() | ||||||
|                                         .downloadProgress |                                     .toString() ?? | ||||||
|                                         ?.toInt() |                                 '100' | ||||||
|                                         .toString() ?? |                           ])) | ||||||
|                                     '100' |                         : (Row( | ||||||
|                               ])) |                             mainAxisSize: MainAxisSize.min, | ||||||
|                             : (Column( |                             crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |                             children: [ | ||||||
|  |                               hasUpdate | ||||||
|  |                                   ? updateButton | ||||||
|  |                                   : const SizedBox.shrink(), | ||||||
|  |                               hasUpdate | ||||||
|  |                                   ? const SizedBox( | ||||||
|  |                                       width: 10, | ||||||
|  |                                     ) | ||||||
|  |                                   : const SizedBox.shrink(), | ||||||
|  |                               Column( | ||||||
|                                 mainAxisAlignment: MainAxisAlignment.center, |                                 mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                                 crossAxisAlignment: CrossAxisAlignment.end, |                                 crossAxisAlignment: CrossAxisAlignment.end, | ||||||
|                                 children: [ |                                 children: [ | ||||||
|                                   SizedBox( |                                   Row( | ||||||
|                                       width: 100, |                                       mainAxisSize: MainAxisSize.min, | ||||||
|                                       child: Text( |                                       children: [ | ||||||
|                                         '${sortedApps[index].app.installedVersion ?? tr('notInstalled')}${sortedApps[index].app.additionalSettings['trackOnly'] == true ? ' ${tr('estimateInBrackets')}' : ''}', |                                         Container( | ||||||
|                                         overflow: TextOverflow.fade, |                                             constraints: const BoxConstraints( | ||||||
|                                         textAlign: TextAlign.end, |                                                 maxWidth: 150), | ||||||
|                                       )), |                                             child: Text( | ||||||
|                                   sortedApps[index].app.installedVersion != |                                               '${listedApps[index].app.installedVersion ?? tr('notInstalled')}${listedApps[index].app.additionalSettings['trackOnly'] == true ? ' ${tr('estimateInBrackets')}' : ''}', | ||||||
|                                               null && |                                               overflow: TextOverflow.ellipsis, | ||||||
|                                           sortedApps[index] |                                               textAlign: TextAlign.end, | ||||||
|                                                   .app |                                             )), | ||||||
|                                                   .installedVersion != |                                       ]), | ||||||
|                                               sortedApps[index] |                                   Row( | ||||||
|                                                   .app |                                     mainAxisSize: MainAxisSize.min, | ||||||
|                                                   .latestVersion |                                     children: [ | ||||||
|                                       ? GestureDetector( |                                       GestureDetector( | ||||||
|                                           onTap: changesUrl == null |                                           onTap: changesUrl == null | ||||||
|                                               ? null |                                               ? null | ||||||
|                                               : () { |                                               : () { | ||||||
| @@ -312,37 +373,39 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                                                       mode: LaunchMode |                                                       mode: LaunchMode | ||||||
|                                                           .externalApplication); |                                                           .externalApplication); | ||||||
|                                                 }, |                                                 }, | ||||||
|                                           child: appsProvider |                                           child: Text( | ||||||
|                                                   .areDownloadsRunning() |                                             listedApps[index].app.releaseDate == | ||||||
|                                               ? Text(tr('pleaseWait')) |                                                     null | ||||||
|                                               : Text( |                                                 ? tr('changes') | ||||||
|                                                   '${tr('updateAvailable')}${sortedApps[index].app.additionalSettings['trackOnly'] == true ? ' ${tr('estimateInBracketsShort')}' : ''}', |                                                 : DateFormat('yyyy-MM-dd') | ||||||
|                                                   style: TextStyle( |                                                     .format(listedApps[index] | ||||||
|                                                       fontStyle: |                                                         .app | ||||||
|                                                           FontStyle.italic, |                                                         .releaseDate!), | ||||||
|                                                       decoration: changesUrl == |                                             style: const TextStyle( | ||||||
|                                                               null |                                                 fontStyle: FontStyle.italic, | ||||||
|                                                           ? TextDecoration.none |                                                 decoration: | ||||||
|                                                           : TextDecoration |                                                     TextDecoration.underline), | ||||||
|                                                               .underline), |                                           )) | ||||||
|                                                 )) |                                     ], | ||||||
|                                       : const SizedBox(), |                                   ), | ||||||
|                                 ], |                                 ], | ||||||
|                               ))), |                               ) | ||||||
|  |                             ], | ||||||
|  |                           )), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
|                       if (selectedApps.isNotEmpty) { |                       if (selectedApps.isNotEmpty) { | ||||||
|                         toggleAppSelected(sortedApps[index].app); |                         toggleAppSelected(listedApps[index].app); | ||||||
|                       } else { |                       } else { | ||||||
|                         Navigator.push( |                         Navigator.push( | ||||||
|                           context, |                           context, | ||||||
|                           MaterialPageRoute( |                           MaterialPageRoute( | ||||||
|                               builder: (context) => |                               builder: (context) => | ||||||
|                                   AppPage(appId: sortedApps[index].app.id)), |                                   AppPage(appId: listedApps[index].app.id)), | ||||||
|                         ); |                         ); | ||||||
|                       } |                       } | ||||||
|                     }, |                     }, | ||||||
|                   )); |                   )); | ||||||
|             }, childCount: sortedApps.length)) |             }, childCount: listedApps.length)) | ||||||
|           ])), |           ])), | ||||||
|       persistentFooterButtons: appsProvider.apps.isEmpty |       persistentFooterButtons: appsProvider.apps.isEmpty | ||||||
|           ? null |           ? null | ||||||
| @@ -354,20 +417,20 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                           style: const ButtonStyle( |                           style: const ButtonStyle( | ||||||
|                               visualDensity: VisualDensity.compact), |                               visualDensity: VisualDensity.compact), | ||||||
|                           onPressed: () { |                           onPressed: () { | ||||||
|                             selectThese(sortedApps.map((e) => e.app).toList()); |                             selectThese(listedApps.map((e) => e.app).toList()); | ||||||
|                           }, |                           }, | ||||||
|                           icon: Icon( |                           icon: Icon( | ||||||
|                             Icons.select_all_outlined, |                             Icons.select_all_outlined, | ||||||
|                             color: Theme.of(context).colorScheme.primary, |                             color: Theme.of(context).colorScheme.primary, | ||||||
|                           ), |                           ), | ||||||
|                           label: Text(sortedApps.length.toString())) |                           label: Text(listedApps.length.toString())) | ||||||
|                       : TextButton.icon( |                       : TextButton.icon( | ||||||
|                           style: const ButtonStyle( |                           style: const ButtonStyle( | ||||||
|                               visualDensity: VisualDensity.compact), |                               visualDensity: VisualDensity.compact), | ||||||
|                           onPressed: () { |                           onPressed: () { | ||||||
|                             selectedApps.isEmpty |                             selectedApps.isEmpty | ||||||
|                                 ? selectThese( |                                 ? selectThese( | ||||||
|                                     sortedApps.map((e) => e.app).toList()) |                                     listedApps.map((e) => e.app).toList()) | ||||||
|                                 : clearSelected(); |                                 : clearSelected(); | ||||||
|                           }, |                           }, | ||||||
|                           icon: Icon( |                           icon: Icon( | ||||||
| @@ -653,7 +716,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                                                                                   onPressed: () { |                                                                                   onPressed: () { | ||||||
|                                                                                     HapticFeedback.selectionClick(); |                                                                                     HapticFeedback.selectionClick(); | ||||||
|                                                                                     appsProvider.saveApps(selectedApps.map((a) { |                                                                                     appsProvider.saveApps(selectedApps.map((a) { | ||||||
|                                                                                       if (a.installedVersion != null && a.additionalSettings['noVersionDetection'] == true) { |                                                                                       if (a.installedVersion != null && a.additionalSettings['versionDetection'] != 'standardVersionDetection') { | ||||||
|                                                                                         a.installedVersion = a.latestVersion; |                                                                                         a.installedVersion = a.latestVersion; | ||||||
|                                                                                       } |                                                                                       } | ||||||
|                                                                                       return a; |                                                                                       return a; | ||||||
|   | |||||||
| @@ -41,6 +41,66 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     urlListImport({String? initValue, bool overrideInitValid = false}) { | ||||||
|  |       showDialog<Map<String, dynamic>?>( | ||||||
|  |           context: context, | ||||||
|  |           builder: (BuildContext ctx) { | ||||||
|  |             return GeneratedFormModal( | ||||||
|  |               initValid: overrideInitValid, | ||||||
|  |               title: tr('importFromURLList'), | ||||||
|  |               items: [ | ||||||
|  |                 [ | ||||||
|  |                   GeneratedFormTextField('appURLList', | ||||||
|  |                       defaultValue: initValue ?? '', | ||||||
|  |                       label: tr('appURLList'), | ||||||
|  |                       max: 7, | ||||||
|  |                       additionalValidators: [ | ||||||
|  |                         (dynamic value) { | ||||||
|  |                           if (value != null && value.isNotEmpty) { | ||||||
|  |                             var lines = value.trim().split('\n'); | ||||||
|  |                             for (int i = 0; i < lines.length; i++) { | ||||||
|  |                               try { | ||||||
|  |                                 sourceProvider.getSource(lines[i]); | ||||||
|  |                               } catch (e) { | ||||||
|  |                                 return '${tr('line')} ${i + 1}: $e'; | ||||||
|  |                               } | ||||||
|  |                             } | ||||||
|  |                           } | ||||||
|  |                           return null; | ||||||
|  |                         } | ||||||
|  |                       ]) | ||||||
|  |                 ] | ||||||
|  |               ], | ||||||
|  |             ); | ||||||
|  |           }).then((values) { | ||||||
|  |         if (values != null) { | ||||||
|  |           var urls = (values['appURLList'] as String).split('\n'); | ||||||
|  |           setState(() { | ||||||
|  |             importInProgress = true; | ||||||
|  |           }); | ||||||
|  |           appsProvider.addAppsByURL(urls).then((errors) { | ||||||
|  |             if (errors.isEmpty) { | ||||||
|  |               showError(tr('importedX', args: [plural('apps', urls.length)]), | ||||||
|  |                   context); | ||||||
|  |             } else { | ||||||
|  |               showDialog( | ||||||
|  |                   context: context, | ||||||
|  |                   builder: (BuildContext ctx) { | ||||||
|  |                     return ImportErrorDialog( | ||||||
|  |                         urlsLength: urls.length, errors: errors); | ||||||
|  |                   }); | ||||||
|  |             } | ||||||
|  |           }).catchError((e) { | ||||||
|  |             showError(e, context); | ||||||
|  |           }).whenComplete(() { | ||||||
|  |             setState(() { | ||||||
|  |               importInProgress = false; | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return Scaffold( |     return Scaffold( | ||||||
|         backgroundColor: Theme.of(context).colorScheme.surface, |         backgroundColor: Theme.of(context).colorScheme.surface, | ||||||
|         body: CustomScrollView(slivers: <Widget>[ |         body: CustomScrollView(slivers: <Widget>[ | ||||||
| @@ -150,88 +210,60 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                           ], |                           ], | ||||||
|                         ) |                         ) | ||||||
|                       else |                       else | ||||||
|                         const Divider( |                         Column( | ||||||
|                           height: 32, |                           children: [ | ||||||
|                         ), |                             const Divider( | ||||||
|                       TextButton( |                               height: 32, | ||||||
|                           onPressed: importInProgress |                             ), | ||||||
|                               ? null |                             TextButton( | ||||||
|                               : () { |                                 onPressed: importInProgress | ||||||
|                                   showDialog<Map<String, dynamic>?>( |                                     ? null | ||||||
|                                       context: context, |                                     : () { | ||||||
|                                       builder: (BuildContext ctx) { |                                         urlListImport(); | ||||||
|                                         return GeneratedFormModal( |                                       }, | ||||||
|                                           title: tr('importFromURLList'), |                                 child: Text( | ||||||
|                                           items: [ |                                   tr('importFromURLList'), | ||||||
|                                             [ |                                 )), | ||||||
|                                               GeneratedFormTextField( |                             const SizedBox(height: 8), | ||||||
|                                                   'appURLList', |                             TextButton( | ||||||
|                                                   label: tr('appURLList'), |                                 onPressed: importInProgress | ||||||
|                                                   max: 7, |                                     ? null | ||||||
|                                                   additionalValidators: [ |                                     : () { | ||||||
|                                                     (dynamic value) { |                                         FilePicker.platform | ||||||
|                                                       if (value != null && |                                             .pickFiles() | ||||||
|                                                           value.isNotEmpty) { |                                             .then((result) { | ||||||
|                                                         var lines = value |                                           if (result != null) { | ||||||
|                                                             .trim() |                                             urlListImport( | ||||||
|                                                             .split('\n'); |                                                 overrideInitValid: true, | ||||||
|                                                         for (int i = 0; |                                                 initValue: | ||||||
|                                                             i < lines.length; |                                                     RegExp('https?://[^"]+') | ||||||
|                                                             i++) { |                                                         .allMatches(File(result | ||||||
|                                                           try { |                                                                 .files | ||||||
|                                                             sourceProvider |                                                                 .single | ||||||
|                                                                 .getSource( |                                                                 .path!) | ||||||
|                                                                     lines[i]); |                                                             .readAsStringSync()) | ||||||
|                                                           } catch (e) { |                                                         .map((e) => | ||||||
|                                                             return '${tr('line')} ${i + 1}: $e'; |                                                             e.input.substring( | ||||||
|                                                           } |                                                                 e.start, e.end)) | ||||||
|                                                         } |                                                         .toSet() | ||||||
|                                                       } |                                                         .toList() | ||||||
|                                                       return null; |                                                         .where((url) { | ||||||
|                                                     } |                                                   try { | ||||||
|                                                   ]) |                                                     sourceProvider | ||||||
|                                             ] |                                                         .getSource(url); | ||||||
|                                           ], |                                                     return true; | ||||||
|                                         ); |                                                   } catch (e) { | ||||||
|                                       }).then((values) { |                                                     return false; | ||||||
|                                     if (values != null) { |                                                   } | ||||||
|                                       var urls = |                                                 }).join('\n')); | ||||||
|                                           (values['appURLList'] as String) |                                           } | ||||||
|                                               .split('\n'); |  | ||||||
|                                       setState(() { |  | ||||||
|                                         importInProgress = true; |  | ||||||
|                                       }); |  | ||||||
|                                       appsProvider |  | ||||||
|                                           .addAppsByURL(urls) |  | ||||||
|                                           .then((errors) { |  | ||||||
|                                         if (errors.isEmpty) { |  | ||||||
|                                           showError( |  | ||||||
|                                               tr('importedX', args: [ |  | ||||||
|                                                 plural('apps', urls.length) |  | ||||||
|                                               ]), |  | ||||||
|                                               context); |  | ||||||
|                                         } else { |  | ||||||
|                                           showDialog( |  | ||||||
|                                               context: context, |  | ||||||
|                                               builder: (BuildContext ctx) { |  | ||||||
|                                                 return ImportErrorDialog( |  | ||||||
|                                                     urlsLength: urls.length, |  | ||||||
|                                                     errors: errors); |  | ||||||
|                                               }); |  | ||||||
|                                         } |  | ||||||
|                                       }).catchError((e) { |  | ||||||
|                                         showError(e, context); |  | ||||||
|                                       }).whenComplete(() { |  | ||||||
|                                         setState(() { |  | ||||||
|                                           importInProgress = false; |  | ||||||
|                                         }); |                                         }); | ||||||
|                                       }); |                                       }, | ||||||
|                                     } |                                 child: Text( | ||||||
|                                   }); |                                   tr('importFromURLsInFile'), | ||||||
|                                 }, |                                 )), | ||||||
|                           child: Text( |                           ], | ||||||
|                             tr('importFromURLList'), |                         ), | ||||||
|                           )), |  | ||||||
|                       ...sourceProvider.sources |                       ...sourceProvider.sources | ||||||
|                           .where((element) => element.canSearch) |                           .where((element) => element.canSearch) | ||||||
|                           .map((source) => Column( |                           .map((source) => Column( | ||||||
| @@ -280,6 +312,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                                     if (urlsWithDescriptions |                                                     if (urlsWithDescriptions | ||||||
|                                                         .isNotEmpty) { |                                                         .isNotEmpty) { | ||||||
|                                                       var selectedUrls = |                                                       var selectedUrls = | ||||||
|  |                                                           // ignore: use_build_context_synchronously | ||||||
|                                                           await showDialog< |                                                           await showDialog< | ||||||
|                                                                   List< |                                                                   List< | ||||||
|                                                                       String>?>( |                                                                       String>?>( | ||||||
| @@ -314,6 +347,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                                                   ]), |                                                                   ]), | ||||||
|                                                               context); |                                                               context); | ||||||
|                                                         } else { |                                                         } else { | ||||||
|  |                                                           // ignore: use_build_context_synchronously | ||||||
|                                                           showDialog( |                                                           showDialog( | ||||||
|                                                               context: context, |                                                               context: context, | ||||||
|                                                               builder: |                                                               builder: | ||||||
| @@ -391,6 +425,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                                                         e.toString()) |                                                                         e.toString()) | ||||||
|                                                                     .toList()); |                                                                     .toList()); | ||||||
|                                                     var selectedUrls = |                                                     var selectedUrls = | ||||||
|  |                                                         // ignore: use_build_context_synchronously | ||||||
|                                                         await showDialog< |                                                         await showDialog< | ||||||
|                                                                 List<String>?>( |                                                                 List<String>?>( | ||||||
|                                                             context: context, |                                                             context: context, | ||||||
| @@ -418,6 +453,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                                                 ]), |                                                                 ]), | ||||||
|                                                             context); |                                                             context); | ||||||
|                                                       } else { |                                                       } else { | ||||||
|  |                                                         // ignore: use_build_context_synchronously | ||||||
|                                                         showDialog( |                                                         showDialog( | ||||||
|                                                             context: context, |                                                             context: context, | ||||||
|                                                             builder: |                                                             builder: | ||||||
|   | |||||||
| @@ -87,6 +87,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|         }); |         }); | ||||||
|  |  | ||||||
|     var sortDropdown = DropdownButtonFormField( |     var sortDropdown = DropdownButtonFormField( | ||||||
|  |         isExpanded: true, | ||||||
|         decoration: InputDecoration(labelText: tr('appSortBy')), |         decoration: InputDecoration(labelText: tr('appSortBy')), | ||||||
|         value: settingsProvider.sortColumn, |         value: settingsProvider.sortColumn, | ||||||
|         items: [ |         items: [ | ||||||
| @@ -101,6 +102,10 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|           DropdownMenuItem( |           DropdownMenuItem( | ||||||
|             value: SortColumnSettings.added, |             value: SortColumnSettings.added, | ||||||
|             child: Text(tr('asAdded')), |             child: Text(tr('asAdded')), | ||||||
|  |           ), | ||||||
|  |           DropdownMenuItem( | ||||||
|  |             value: SortColumnSettings.releaseDate, | ||||||
|  |             child: Text(tr('releaseDate')), | ||||||
|           ) |           ) | ||||||
|         ], |         ], | ||||||
|         onChanged: (value) { |         onChanged: (value) { | ||||||
| @@ -110,6 +115,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|         }); |         }); | ||||||
|  |  | ||||||
|     var orderDropdown = DropdownButtonFormField( |     var orderDropdown = DropdownButtonFormField( | ||||||
|  |         isExpanded: true, | ||||||
|         decoration: InputDecoration(labelText: tr('appSortOrder')), |         decoration: InputDecoration(labelText: tr('appSortOrder')), | ||||||
|         value: settingsProvider.sortOrder, |         value: settingsProvider.sortOrder, | ||||||
|         items: [ |         items: [ | ||||||
| @@ -146,7 +152,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|           if (value != null) { |           if (value != null) { | ||||||
|             context.setLocale(Locale(value)); |             context.setLocale(Locale(value)); | ||||||
|           } else { |           } else { | ||||||
|             context.resetLocale(); |             settingsProvider.resetLocaleSafe(context); | ||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -182,7 +182,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|     // The former case should be handled (give the App its real ID), the latter is a security issue |     // 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); |     var newInfo = await PackageArchiveInfo.fromPath(downloadedFile.path); | ||||||
|     if (app.id != newInfo.packageName) { |     if (app.id != newInfo.packageName) { | ||||||
|       if (apps[app.id] != null && !SourceProvider().isTempId(app.id)) { |       if (apps[app.id] != null && !SourceProvider().isTempId(app)) { | ||||||
|         throw IDChangedError(); |         throw IDChangedError(); | ||||||
|       } |       } | ||||||
|       var originalAppId = app.id; |       var originalAppId = app.id; | ||||||
| @@ -467,8 +467,8 @@ class AppsProvider with ChangeNotifier { | |||||||
|   App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) { |   App? getCorrectedInstallStatusAppIfPossible(App app, AppInfo? installedInfo) { | ||||||
|     var modded = false; |     var modded = false; | ||||||
|     var trackOnly = app.additionalSettings['trackOnly'] == true; |     var trackOnly = app.additionalSettings['trackOnly'] == true; | ||||||
|     var noVersionDetection = |     var noVersionDetection = app.additionalSettings['versionDetection'] != | ||||||
|         app.additionalSettings['noVersionDetection'] == true; |         'standardVersionDetection'; | ||||||
|     if (installedInfo == null && app.installedVersion != null && !trackOnly) { |     if (installedInfo == null && app.installedVersion != null && !trackOnly) { | ||||||
|       app.installedVersion = null; |       app.installedVersion = null; | ||||||
|       modded = true; |       modded = true; | ||||||
|   | |||||||
| @@ -6,7 +6,6 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:fluttertoast/fluttertoast.dart'; | import 'package:fluttertoast/fluttertoast.dart'; | ||||||
| import 'package:obtainium/app_sources/github.dart'; | import 'package:obtainium/app_sources/github.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; |  | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
| import 'package:permission_handler/permission_handler.dart'; | import 'package:permission_handler/permission_handler.dart'; | ||||||
| import 'package:shared_preferences/shared_preferences.dart'; | import 'package:shared_preferences/shared_preferences.dart'; | ||||||
| @@ -18,7 +17,7 @@ enum ThemeSettings { system, light, dark } | |||||||
|  |  | ||||||
| enum ColourSettings { basic, materialYou } | enum ColourSettings { basic, materialYou } | ||||||
|  |  | ||||||
| enum SortColumnSettings { added, nameAuthor, authorName } | enum SortColumnSettings { added, nameAuthor, authorName, releaseDate } | ||||||
|  |  | ||||||
| enum SortOrderSettings { ascending, descending } | enum SortOrderSettings { ascending, descending } | ||||||
|  |  | ||||||
| @@ -179,4 +178,15 @@ class SettingsProvider with ChangeNotifier { | |||||||
|  |  | ||||||
|   bool setEqual(Set<String> a, Set<String> b) => |   bool setEqual(Set<String> a, Set<String> b) => | ||||||
|       a.length == b.length && a.union(b).length == a.length; |       a.length == b.length && a.union(b).length == a.length; | ||||||
|  |  | ||||||
|  |   void resetLocaleSafe(BuildContext context) { | ||||||
|  |     if (context.supportedLocales | ||||||
|  |         .map((e) => e.languageCode) | ||||||
|  |         .contains(context.deviceLocale.languageCode)) { | ||||||
|  |       context.resetLocale(); | ||||||
|  |     } else { | ||||||
|  |       context.setLocale(context.fallbackLocale!); | ||||||
|  |       context.deleteSaveLocale(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -33,8 +33,9 @@ class APKDetails { | |||||||
|   late String version; |   late String version; | ||||||
|   late List<String> apkUrls; |   late List<String> apkUrls; | ||||||
|   late AppNames names; |   late AppNames names; | ||||||
|  |   late DateTime? releaseDate; | ||||||
|  |  | ||||||
|   APKDetails(this.version, this.apkUrls, this.names); |   APKDetails(this.version, this.apkUrls, this.names, {this.releaseDate}); | ||||||
| } | } | ||||||
|  |  | ||||||
| class App { | class App { | ||||||
| @@ -50,6 +51,7 @@ class App { | |||||||
|   late DateTime? lastUpdateCheck; |   late DateTime? lastUpdateCheck; | ||||||
|   bool pinned = false; |   bool pinned = false; | ||||||
|   List<String> categories; |   List<String> categories; | ||||||
|  |   late DateTime? releaseDate; | ||||||
|   App( |   App( | ||||||
|       this.id, |       this.id, | ||||||
|       this.url, |       this.url, | ||||||
| @@ -62,7 +64,8 @@ class App { | |||||||
|       this.additionalSettings, |       this.additionalSettings, | ||||||
|       this.lastUpdateCheck, |       this.lastUpdateCheck, | ||||||
|       this.pinned, |       this.pinned, | ||||||
|       {this.categories = const []}); |       {this.categories = const [], | ||||||
|  |       this.releaseDate}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
| @@ -97,6 +100,20 @@ class App { | |||||||
|       additionalSettings['noVersionDetection'] = |       additionalSettings['noVersionDetection'] = | ||||||
|           json['noVersionDetection'] == 'true' || json['trackOnly'] == true; |           json['noVersionDetection'] == 'true' || json['trackOnly'] == true; | ||||||
|     } |     } | ||||||
|  |     // Convert bool style version detection options to dropdown style | ||||||
|  |     if (additionalSettings['noVersionDetection'] == true) { | ||||||
|  |       additionalSettings['versionDetection'] = 'noVersionDetection'; | ||||||
|  |     } | ||||||
|  |     if (additionalSettings['releaseDateAsVersion'] == true) { | ||||||
|  |       additionalSettings['versionDetection'] = 'releaseDateAsVersion'; | ||||||
|  |       additionalSettings.remove('releaseDateAsVersion'); | ||||||
|  |     } | ||||||
|  |     if (additionalSettings['noVersionDetection'] != null) { | ||||||
|  |       additionalSettings.remove('noVersionDetection'); | ||||||
|  |     } | ||||||
|  |     if (additionalSettings['releaseDateAsVersion'] != null) { | ||||||
|  |       additionalSettings.remove('releaseDateAsVersion'); | ||||||
|  |     } | ||||||
|     // Ensure additionalSettings are correctly typed |     // Ensure additionalSettings are correctly typed | ||||||
|     for (var item in formItems) { |     for (var item in formItems) { | ||||||
|       if (additionalSettings[item.key] != null) { |       if (additionalSettings[item.key] != null) { | ||||||
| @@ -111,30 +128,34 @@ class App { | |||||||
|       preferredApkIndex = 0; |       preferredApkIndex = 0; | ||||||
|     } |     } | ||||||
|     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 |       json['apkUrls'] == null | ||||||
|             ? [] |           ? [] | ||||||
|             : List<String>.from(jsonDecode(json['apkUrls'])), |           : List<String>.from(jsonDecode(json['apkUrls'])), | ||||||
|         preferredApkIndex, |       preferredApkIndex, | ||||||
|         additionalSettings, |       additionalSettings, | ||||||
|         json['lastUpdateCheck'] == null |       json['lastUpdateCheck'] == null | ||||||
|             ? null |           ? null | ||||||
|             : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), |           : DateTime.fromMicrosecondsSinceEpoch(json['lastUpdateCheck']), | ||||||
|         json['pinned'] ?? false, |       json['pinned'] ?? false, | ||||||
|         categories: json['categories'] != null |       categories: json['categories'] != null | ||||||
|             ? (json['categories'] as List<dynamic>) |           ? (json['categories'] as List<dynamic>) | ||||||
|                 .map((e) => e.toString()) |               .map((e) => e.toString()) | ||||||
|                 .toList() |               .toList() | ||||||
|             : json['category'] != null |           : json['category'] != null | ||||||
|                 ? [json['category'] as String] |               ? [json['category'] as String] | ||||||
|                 : []); |               : [], | ||||||
|  |       releaseDate: json['releaseDate'] == null | ||||||
|  |           ? null | ||||||
|  |           : DateTime.fromMicrosecondsSinceEpoch(json['releaseDate']), | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> toJson() => { |   Map<String, dynamic> toJson() => { | ||||||
| @@ -149,7 +170,8 @@ class App { | |||||||
|         'additionalSettings': jsonEncode(additionalSettings), |         'additionalSettings': jsonEncode(additionalSettings), | ||||||
|         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, |         'lastUpdateCheck': lastUpdateCheck?.microsecondsSinceEpoch, | ||||||
|         'pinned': pinned, |         'pinned': pinned, | ||||||
|         'categories': categories |         'categories': categories, | ||||||
|  |         'releaseDate': releaseDate?.microsecondsSinceEpoch | ||||||
|       }; |       }; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -226,7 +248,16 @@ class AppSource { | |||||||
|       ) |       ) | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormSwitch('noVersionDetection', label: tr('noVersionDetection')) |       GeneratedFormDropdown( | ||||||
|  |           'versionDetection', | ||||||
|  |           [ | ||||||
|  |             MapEntry( | ||||||
|  |                 'standardVersionDetection', tr('standardVersionDetection')), | ||||||
|  |             MapEntry('releaseDateAsVersion', tr('releaseDateAsVersion')), | ||||||
|  |             MapEntry('noVersionDetection', tr('noVersionDetection')) | ||||||
|  |           ], | ||||||
|  |           label: tr('versionDetection'), | ||||||
|  |           defaultValue: 'standardVersionDetection') | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormTextField('apkFilterRegEx', |       GeneratedFormTextField('apkFilterRegEx', | ||||||
| @@ -350,41 +381,29 @@ class SourceProvider { | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   String generateTempID(AppNames names, AppSource source) => |   String generateTempID( | ||||||
|       '${names.author.toLowerCase()}_${names.name.toLowerCase()}_${source.host}'; |           String standardUrl, Map<String, dynamic> additionalSettings) => | ||||||
|  |       (standardUrl + additionalSettings.toString()).hashCode.toString(); | ||||||
|  |  | ||||||
|   bool isTempId(String id) { |   bool isTempId(App app) { | ||||||
|     List<String> parts = id.split('_'); |     // return app.id == generateTempID(app.url, app.additionalSettings); | ||||||
|     if (parts.length < 3) { |     return RegExp('^[0-9]+\$').hasMatch(app.id); | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|     for (int i = 0; i < parts.length - 1; i++) { |  | ||||||
|       if (RegExp('.*[A-Z].*').hasMatch(parts[i])) { |  | ||||||
|         // TODO: Look into RegEx for non-Latin characters |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     return true; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<App> getApp( |   Future<App> getApp( | ||||||
|     AppSource source, |       AppSource source, String url, Map<String, dynamic> additionalSettings, | ||||||
|     String url, |       {App? currentApp, bool trackOnlyOverride = false}) async { | ||||||
|     Map<String, dynamic> additionalSettings, { |  | ||||||
|     App? currentApp, |  | ||||||
|     bool trackOnlyOverride = false, |  | ||||||
|     noVersionDetectionOverride = false, |  | ||||||
|   }) async { |  | ||||||
|     if (trackOnlyOverride || source.enforceTrackOnly) { |     if (trackOnlyOverride || source.enforceTrackOnly) { | ||||||
|       additionalSettings['trackOnly'] = true; |       additionalSettings['trackOnly'] = true; | ||||||
|     } |     } | ||||||
|     if (noVersionDetectionOverride) { |  | ||||||
|       additionalSettings['noVersionDetection'] = true; |  | ||||||
|     } |  | ||||||
|     var trackOnly = additionalSettings['trackOnly'] == true; |     var trackOnly = additionalSettings['trackOnly'] == true; | ||||||
|     String standardUrl = source.standardizeURL(preStandardizeUrl(url)); |     String standardUrl = source.standardizeURL(preStandardizeUrl(url)); | ||||||
|     APKDetails apk = |     APKDetails apk = | ||||||
|         await source.getLatestAPKDetails(standardUrl, additionalSettings); |         await source.getLatestAPKDetails(standardUrl, additionalSettings); | ||||||
|  |     if (additionalSettings['versionDetection'] == 'releaseDateAsVersion' && | ||||||
|  |         apk.releaseDate != null) { | ||||||
|  |       apk.version = apk.releaseDate!.microsecondsSinceEpoch.toString(); | ||||||
|  |     } | ||||||
|     if (additionalSettings['apkFilterRegEx'] != null) { |     if (additionalSettings['apkFilterRegEx'] != null) { | ||||||
|       var reg = RegExp(additionalSettings['apkFilterRegEx']); |       var reg = RegExp(additionalSettings['apkFilterRegEx']); | ||||||
|       apk.apkUrls = |       apk.apkUrls = | ||||||
| @@ -400,7 +419,7 @@ class SourceProvider { | |||||||
|         currentApp?.id ?? |         currentApp?.id ?? | ||||||
|             source.tryInferringAppId(standardUrl, |             source.tryInferringAppId(standardUrl, | ||||||
|                 additionalSettings: additionalSettings) ?? |                 additionalSettings: additionalSettings) ?? | ||||||
|             generateTempID(apk.names, source), |             generateTempID(standardUrl, additionalSettings), | ||||||
|         standardUrl, |         standardUrl, | ||||||
|         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), |         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), | ||||||
|         name.trim().isNotEmpty |         name.trim().isNotEmpty | ||||||
| @@ -413,7 +432,8 @@ class SourceProvider { | |||||||
|         additionalSettings, |         additionalSettings, | ||||||
|         DateTime.now(), |         DateTime.now(), | ||||||
|         currentApp?.pinned ?? false, |         currentApp?.pinned ?? false, | ||||||
|         categories: currentApp?.categories ?? const []); |         categories: currentApp?.categories ?? const [], | ||||||
|  |         releaseDate: apk.releaseDate); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Returns errors in [results, errors] instead of throwing them |   // Returns errors in [results, errors] instead of throwing them | ||||||
|   | |||||||
							
								
								
									
										216
									
								
								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: "71e796198588e0038dd125bf8c91683b3237b938ffad037413245c689b87ae28" |       sha256: "8647cc5f9339f3955a2bd9ec40e0f10c3a80049f31f80b3ffdd87e07bb73fce2" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.0" |     version: "2.1.1" | ||||||
|   android_intent_plus: |   android_intent_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: android_intent_plus |       name: android_intent_plus | ||||||
|       sha256: ebd110b60723334bdc6eeb373116d6c52e9bed8feb9dcbd9f034531f56636e31 |       sha256: "54810cb33945c2c10742cd746ea994822c115e9dbe189919bc63cb436e45a6af" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.1.5" |     version: "3.1.6" | ||||||
|   animations: |   animations: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -25,22 +25,14 @@ 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: | ||||||
|       name: args |       name: args | ||||||
|       sha256: "139d809800a412ebb26a3892da228b2d0ba36f0ef5d9a82166e5e52ec8d61611" |       sha256: "4cab82a83ffef80b262ddedf47a0a8e56ee6fbf7fe21e6e768b02792034dd440" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.3.2" |     version: "2.4.0" | ||||||
|   async: |   async: | ||||||
|     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: "7ff671ed0a6356fa8f2e1ae7d3558d3fb7b6a41e24455e4f8df75b811fb8e4ab" |       sha256: "1d6e5a61674ba3a68fb048a7c7b4ff4bebfed8d7379dbe8f2b718231be9a7c95" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "8.0.0" |     version: "8.1.0" | ||||||
|   device_info_plus_platform_interface: |   device_info_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -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: ce0e501cfc258907842238e4ca605e74b7fd1cdf04b3b43e86c43f3e40a1592c |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "0.11.0" |  | ||||||
|   flutter_lints: |   flutter_lints: | ||||||
|     dependency: "direct dev" |     dependency: "direct dev" | ||||||
|     description: |     description: | ||||||
| @@ -258,10 +218,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: flutter_local_notifications_linux |       name: flutter_local_notifications_linux | ||||||
|       sha256: "8f6c1611e0c4a88a382691a97bb3c3feb24cc0c0b54152b8b5fb7ffb837f7fbf" |       sha256: ccb08b93703aeedb58856e5637450bf3ffec899adb66dc325630b68994734b89 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.0" |     version: "3.0.0+1" | ||||||
|   flutter_local_notifications_platform_interface: |   flutter_local_notifications_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -279,10 +239,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: flutter_plugin_android_lifecycle |       name: flutter_plugin_android_lifecycle | ||||||
|       sha256: "60fc7b78455b94e6de2333d2f95196d32cf5c22f4b0b0520a628804cb463503b" |       sha256: c224ac897bed083dabf11f238dd11a239809b446740be0c2044608c50029ffdf | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.7" |     version: "2.0.9" | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     dependency: "direct dev" |     dependency: "direct dev" | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -297,18 +257,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: fluttertoast |       name: fluttertoast | ||||||
|       sha256: "7cc92eabe01e3f1babe1571c5560b135dfc762a34e41e9056881e2196b178ec1" |       sha256: "2f9c4d3f4836421f7067a28f8939814597b27614e021da9d63e5d3fb6e212d25" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "8.1.2" |     version: "8.2.1" | ||||||
|   html: |   html: | ||||||
|     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 +285,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: "8e9d133755c3e84c73288363e6343157c383a0c6c56fc51afcc5d4d7180306d6" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "3.3.0" |  | ||||||
|   install_plugin_v2: |   install_plugin_v2: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -365,14 +317,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: | ||||||
| @@ -449,50 +393,50 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: path_provider |       name: path_provider | ||||||
|       sha256: dcea5feb97d8abf90cab9e9030b497fb7c3cbf26b7a1fe9e3ef7dcb0a1ddec95 |       sha256: "04890b994ee89bfa80bf3080bfec40d5a92c5c7a785ebb02c13084a099d2b6f9" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.12" |     version: "2.0.13" | ||||||
|   path_provider_android: |   path_provider_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_android |       name: path_provider_android | ||||||
|       sha256: a776c088d671b27f6e3aa8881d64b87b3e80201c64e8869b811325de7a76c15e |       sha256: "7623b7d4be0f0f7d9a8b5ee6879fc13e4522d4c875ab86801dee4af32b54b83e" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.22" |     version: "2.0.23" | ||||||
|   path_provider_foundation: |   path_provider_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_foundation |       name: path_provider_foundation | ||||||
|       sha256: "62a68e7e1c6c459f9289859e2fae58290c981ce21d1697faf54910fe1faa4c74" |       sha256: eec003594f19fe2456ea965ae36b3fc967bc5005f508890aafe31fa75e41d972 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.1" |     version: "2.1.2" | ||||||
|   path_provider_linux: |   path_provider_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_linux |       name: path_provider_linux | ||||||
|       sha256: ab0987bf95bc591da42dffb38c77398fc43309f0b9b894dcc5d6f40c4b26c379 |       sha256: "525ad5e07622d19447ad740b1ed5070031f7a5437f44355ae915ff56e986429a" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.7" |     version: "2.1.9" | ||||||
|   path_provider_platform_interface: |   path_provider_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_platform_interface |       name: path_provider_platform_interface | ||||||
|       sha256: f0abc8ebd7253741f05488b4813d936b4d07c6bae3e86148a09e342ee4b08e76 |       sha256: "57585299a729335f1298b43245842678cb9f43a6310351b18fb577d6e33165ec" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.5" |     version: "2.0.6" | ||||||
|   path_provider_windows: |   path_provider_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_windows |       name: path_provider_windows | ||||||
|       sha256: bcabbe399d4042b8ee687e17548d5d3f527255253b4a639f5f8d2094a9c2b45c |       sha256: "642ddf65fde5404f83267e8459ddb4556316d3ee6d511ed193357e25caa3632d" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.3" |     version: "2.1.4" | ||||||
|   permission_handler: |   permission_handler: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -553,18 +497,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: plugin_platform_interface |       name: plugin_platform_interface | ||||||
|       sha256: dbf0f707c78beedc9200146ad3cb0ab4d5da13c246336987be6940f026500d3a |       sha256: "6a2128648c854906c53fa8e33986fc0247a1116122f9534dd20e3ab9e16a32bc" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.3" |     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,10 +521,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: share_plus |       name: share_plus | ||||||
|       sha256: e387077716f80609bb979cd199331033326033ecd1c8f200a90c5f57b1c9f55e |       sha256: "8c6892037b1824e2d7e8f59d54b3105932899008642e6372e5079c6939b4b625" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.0" |     version: "6.3.1" | ||||||
|   share_plus_platform_interface: |   share_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -601,58 +537,58 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences |       name: shared_preferences | ||||||
|       sha256: "5949029e70abe87f75cfe59d17bf5c397619c4b74a099b10116baeb34786fad9" |       sha256: ee6257848f822b8481691f20c3e6d2bfee2e9eccb2a3d249907fcfb198c55b41 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.17" |     version: "2.0.18" | ||||||
|   shared_preferences_android: |   shared_preferences_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_android |       name: shared_preferences_android | ||||||
|       sha256: "955e9736a12ba776bdd261cf030232b30eadfcd9c79b32a3250dd4a494e8c8f7" |       sha256: a51a4f9375097f94df1c6e0a49c0374440d31ab026b59d58a7e7660675879db4 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.15" |     version: "2.0.16" | ||||||
|   shared_preferences_foundation: |   shared_preferences_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_foundation |       name: shared_preferences_foundation | ||||||
|       sha256: "2b55c18636a4edc529fa5cd44c03d3f3100c00513f518c5127c951978efcccd0" |       sha256: "6b84fdf06b32bb336f972d373cd38b63734f3461ba56ac2ba01b56d052796259" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.3" |     version: "2.1.4" | ||||||
|   shared_preferences_linux: |   shared_preferences_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_linux |       name: shared_preferences_linux | ||||||
|       sha256: f8ea038aa6da37090093974ebdcf4397010605fd2ff65c37a66f9d28394cb874 |       sha256: d7fb71e6e20cd3dfffcc823a28da3539b392e53ed5fc5c2b90b55fdaa8a7e8fa | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.3" |     version: "2.1.4" | ||||||
|   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: da9431745ede5ece47bc26d5d73a9d3c6936ef6945c101a5aca46f62e52c1cf3 |       sha256: "824bfd02713e37603b2bdade0842e47d56e7db32b1dcdd1cae533fb88e2913fc" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.0" |     version: "2.1.1" | ||||||
|   shared_preferences_web: |   shared_preferences_web: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_web |       name: shared_preferences_web | ||||||
|       sha256: a4b5bc37fe1b368bbc81f953197d55e12f49d0296e7e412dfe2d2d77d6929958 |       sha256: "6737b757e49ba93de2a233df229d0b6a87728cea1684da828cbc718b65dcf9d7" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.4" |     version: "2.0.5" | ||||||
|   shared_preferences_windows: |   shared_preferences_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_windows |       name: shared_preferences_windows | ||||||
|       sha256: "5eaf05ae77658d3521d0e993ede1af962d4b326cd2153d312df716dc250f00c9" |       sha256: bd014168e8484837c39ef21065b78f305810ceabc1d4f90be6e3b392ce81b46d | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.3" |     version: "2.1.4" | ||||||
|   sky_engine: |   sky_engine: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -670,10 +606,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: sqflite |       name: sqflite | ||||||
|       sha256: "78324387dc81df14f78df06019175a86a2ee0437624166c382e145d0a7fd9a4f" |       sha256: "851d5040552cf911f4cabda08d003eca76b27da3ed0002978272e27c8fbf8ecc" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.2.4+1" |     version: "2.2.5" | ||||||
|   sqflite_common: |   sqflite_common: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -750,66 +686,66 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: url_launcher |       name: url_launcher | ||||||
|       sha256: "698fa0b4392effdc73e9e184403b627362eb5fbf904483ac9defbb1c2191d809" |       sha256: "75f2846facd11168d007529d6cd8fcb2b750186bea046af9711f10b907e1587e" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.1.8" |     version: "6.1.10" | ||||||
|   url_launcher_android: |   url_launcher_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_android |       name: url_launcher_android | ||||||
|       sha256: "3e2f6dfd2c7d9cd123296cab8ef66cfc2c1a13f5845f42c7a0f365690a8a7dd1" |       sha256: "1f4d9ebe86f333c15d318f81dcdc08b01d45da44af74552608455ebdc08d9732" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.0.23" |     version: "6.0.24" | ||||||
|   url_launcher_ios: |   url_launcher_ios: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_ios |       name: url_launcher_ios | ||||||
|       sha256: bb328b24d3bccc20bdf1024a0990ac4f869d57663660de9c936fb8c043edefe3 |       sha256: c9cd648d2f7ab56968e049d4e9116f96a85517f1dd806b96a86ea1018a3a82e5 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.0.18" |     version: "6.1.1" | ||||||
|   url_launcher_linux: |   url_launcher_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_linux |       name: url_launcher_linux | ||||||
|       sha256: "318c42cba924e18180c029be69caf0a1a710191b9ec49bb42b5998fdcccee3cc" |       sha256: e29039160ab3730e42f3d811dc2a6d5f2864b90a70fb765ea60144b03307f682 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.2" |     version: "3.0.3" | ||||||
|   url_launcher_macos: |   url_launcher_macos: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_macos |       name: url_launcher_macos | ||||||
|       sha256: "41988b55570df53b3dd2a7fc90c76756a963de6a8c5f8e113330cb35992e2094" |       sha256: "2dddb3291a57b074dade66b5e07e64401dd2487caefd4e9e2f467138d8c7eb06" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.2" |     version: "3.0.3" | ||||||
|   url_launcher_platform_interface: |   url_launcher_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_platform_interface |       name: url_launcher_platform_interface | ||||||
|       sha256: "4eae912628763eb48fc214522e58e942fd16ce195407dbf45638239523c759a6" |       sha256: "6c9ca697a5ae218ce56cece69d46128169a58aa8653c1b01d26fcd4aad8c4370" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.1" |     version: "2.1.2" | ||||||
|   url_launcher_web: |   url_launcher_web: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_web |       name: url_launcher_web | ||||||
|       sha256: "44d79408ce9f07052095ef1f9a693c258d6373dc3944249374e30eff7219ccb0" |       sha256: "574cfbe2390666003c3a1d129bdc4574aaa6728f0c00a4829a81c316de69dd9b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.14" |     version: "2.0.15" | ||||||
|   url_launcher_windows: |   url_launcher_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_windows |       name: url_launcher_windows | ||||||
|       sha256: b6217370f8eb1fd85c8890c539f5a639a01ab209a36db82c921ebeacefc7a615 |       sha256: "97c9067950a0d09cbd93e2e3f0383d1403989362b97102fbf446473a48079a4b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.3" |     version: "3.0.4" | ||||||
|   uuid: |   uuid: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -830,34 +766,34 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter |       name: webview_flutter | ||||||
|       sha256: f7ec234830f86d0ef2bd664e8460b0038b8c1a83ff076035cad74ac70273753c |       sha256: b6cd42db3ced5411f3d01599906156885b18e4188f7065a8a351eb84bee347e0 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.0.2" |     version: "4.0.6" | ||||||
|   webview_flutter_android: |   webview_flutter_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_android |       name: webview_flutter_android | ||||||
|       sha256: "5f49a6e5fc59e21fcec5e1bbcd401afbee9792a24a4f3d9cef9b5bb0cd1e3767" |       sha256: "5dd3f32b5c2d8f4bf9d05a349e4a65fa718eb137f396f336c3893d558a58fe84" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.2.4" |     version: "3.3.2" | ||||||
|   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: "8b2262dda5d26eabc600a7282a8c16a9473a0c765526afb0ffc33eef912f7968" |       sha256: df6472164b3f4eaf3280422227f361dc8424b106726b7f21d79a8656ba53f71f | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.1" |     version: "2.0.2" | ||||||
|   webview_flutter_wkwebview: |   webview_flutter_wkwebview: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_wkwebview |       name: webview_flutter_wkwebview | ||||||
|       sha256: "92e7e7fa468f1df597fb9d37bcf1f303175cbe147c4dbdf06ecc323d950116eb" |       sha256: "87b6353b40e04f04d5f895a484ad6d92d682d9cce4d2d5b32d2d8aca2448d46e" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.5" |     version: "3.2.0" | ||||||
|   win32: |   win32: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -882,14 +818,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.18.2 <3.0.0" | ||||||
|   flutter: ">=3.4.0-17.0.pre" |   flutter: ">=3.4.0-17.0.pre" | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								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.10.9+115 # When changing this, update the tag in main() accordingly | version: 0.11.8+129 # 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' | ||||||
| @@ -64,7 +64,6 @@ dependencies: | |||||||
| dev_dependencies: | dev_dependencies: | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     sdk: flutter |     sdk: flutter | ||||||
|   flutter_launcher_icons: ^0.11.0 |  | ||||||
|  |  | ||||||
|   # The "flutter_lints" package below contains a set of recommended lints to |   # The "flutter_lints" package below contains a set of recommended lints to | ||||||
|   # encourage good coding practices. The lint set provided by the package is |   # encourage good coding practices. The lint set provided by the package is | ||||||
| @@ -73,12 +72,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 | ||||||
|  |  | ||||||
| @@ -97,6 +90,7 @@ flutter: | |||||||
|    |    | ||||||
|   assets: |   assets: | ||||||
|     - assets/translations/ |     - assets/translations/ | ||||||
|  |     - assets/graphics/ | ||||||
|  |  | ||||||
|   # 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 | ||||||
|   | |||||||