mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-31 13:33:28 +01:00 
			
		
		
		
	Compare commits
	
		
			95 Commits
		
	
	
		
			v0.13.16-b
			...
			v0.13.23-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 2a5118a2cf | ||
|  | 347c56da55 | ||
|  | 8073723e1f | ||
|  | c8e90a755d | ||
|  | aeb0a5d8ea | ||
|  | 09fe7f3ecd | ||
|  | a549411589 | ||
|  | f426b5e118 | ||
|  | 12a8ef5e31 | ||
|  | 8210279a4c | ||
|  | 999d13b80d | ||
|  | 0573c0b270 | ||
|  | 5a36a7980c | ||
|  | 6bf9b5297f | ||
|  | fdc6b0ff00 | ||
|  | 73c20a53d2 | ||
|  | 16ae8d8e4d | ||
|  | 3f8cfae64e | ||
|  | 6861a71efb | ||
|  | 2a603f410f | ||
|  | 6d22788f92 | ||
|  | e36e6bbaca | ||
|  | ac3a13ed73 | ||
|  | 202f7df5cb | ||
|  | 1b902b1a18 | ||
|  | 1765d399c8 | ||
|  | 5dd79707f1 | ||
|  | 14755134bf | ||
|  | cccde7e135 | ||
|  | 3dafd643c0 | ||
|  | 76f8cd4102 | ||
|  | a8bfb03f58 | ||
|  | 10ead4f3e0 | ||
|  | af5a6857ba | ||
|  | d9225fd639 | ||
|  | 995d44551c | ||
|  | 68c0224b98 | ||
|  | 7767468d5d | ||
|  | f9f83d8243 | ||
|  | cc4cec829f | ||
|  | f665bf1eb2 | ||
|  | bccd52054e | ||
|  | 3a4c782aab | ||
|  | 6e047e96fa | ||
|  | 42f8753166 | ||
|  | b6a64129b3 | ||
|  | bb5bd23263 | ||
|  | 58361a0493 | ||
|  | 7671ab95f9 | ||
|  | 41f102c0ce | ||
|  | 5cee527d6f | ||
|  | c2eb13d5e5 | ||
|  | ed89e20826 | ||
|  | e50c7554e3 | ||
|  | 00354f10a3 | ||
|  | 031d0b7444 | ||
|  | 2ced06367b | ||
|  | 9ef3279bae | ||
|  | b055486d74 | ||
|  | c70f2c481e | ||
|  | e4b26be01f | ||
|  | b27bdc63c1 | ||
|  | 543e7d8cdc | ||
|  | 852decbe7d | ||
|  | d2150320fa | ||
|  | 9c723c7522 | ||
|  | ac5660de88 | ||
|  | e6467452a6 | ||
|  | 04b49c2e61 | ||
|  | 324773ba58 | ||
|  | d885ca5db7 | ||
|  | e093fc28b0 | ||
|  | 579bc94847 | ||
|  | 53dba06cc3 | ||
|  | 1c390a7f04 | ||
|  | 6b16857186 | ||
|  | 785bba1f45 | ||
|  | 3befcbc16d | ||
|  | be300608a9 | ||
|  | f05902925d | ||
|  | d32e7acc8f | ||
|  | fda9e6195a | ||
|  | b5d91adb21 | ||
|  | 1b52cb6233 | ||
|  | 5669634b44 | ||
|  | d6bb365cf1 | ||
|  | 53b7cfb9bf | ||
|  | 2558c851c0 | ||
|  | a8aa63daa3 | ||
|  | d587e9f708 | ||
|  | e2a8f40e3e | ||
|  | 1cc4241fac | ||
|  | bb98dfe0b3 | ||
|  | 6395dd820b | ||
|  | e909c03356 | 
							
								
								
									
										297
									
								
								assets/translations/bs.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										297
									
								
								assets/translations/bs.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,297 @@ | ||||
|  { | ||||
|    "invalidURLForSource": "Nije važeći URL aplikacije {}", | ||||
|    "noReleaseFound": "Nije moguće pronaći odgovarajuće izdanje", | ||||
|    "noVersionFound": "Nije moguće odrediti verziju izdanja", | ||||
|    "urlMatchesNoSource": "URL se ne podudara s poznatim izvorom", | ||||
|    "cantInstallOlderVersion": "Nije moguće instalirati stariju verziju aplikacije", | ||||
|    "appIdMismatch": "ID preuzetog paketa se ne podudara s postojećim ID-om aplikacije", | ||||
|    "functionNotImplemented": "Ova klasa nije implementirala ovu funkciju", | ||||
|    "placeholder": "Rezervirano mjesto", | ||||
|    "someErrors": "Došlo je do nekih grešaka", | ||||
|    "unexpectedError": "Neočekivana greška", | ||||
|    "ok": "Dobro", | ||||
|    "and": "i", | ||||
|    "startedBgUpdateTask": "Započeo je pozadinski zadatak provjere ažuriranja", | ||||
|    "bgUpdateIgnoreAfterIs": "ignoreAfter pozadinskog zadataka je  {}", | ||||
|    "startedActualBGUpdateCheck": "Započela je stvarna provjera ažuriranja", | ||||
|    "bgUpdateTaskFinished": "Završen zadatak provjere ažuriranja", | ||||
|    "firstRun": "Ovo je prvi put da pokrećete Obtainium", | ||||
|    "settingUpdateCheckIntervalTo": "Podešavanje intervala ažuriranja na {}", | ||||
|    "githubPATLabel": "GitHub token za lični pristup (eng. PAT, povećava ograničenje stope)", | ||||
|    "githubPATHint": "PAT mora biti u ovom formatu: korisničko_ime:token", | ||||
|    "githubPATFormat": "korisničko_ime:token", | ||||
|    "includePrereleases": "Uključi preliminarna izdanja", | ||||
|    "fallbackToOlderReleases": "Povratak na starija izdanja", | ||||
|    "filterReleaseTitlesByRegEx": "Filtrirajte naslove izdanja prema regularnom izrazu", | ||||
|    "invalidRegEx": "Nevažeći regularni izraz", | ||||
|    "noDescription": "Bez opisa", | ||||
|    "cancel": "Otkaži", | ||||
|    "continue": "Nastavite", | ||||
|    "requiredInBrackets": "(obavezno)", | ||||
|    "dropdownNoOptsError": "GREŠKA: PADAJUĆI MENI MORA IMATI NAJMANJE JEDNU OPCIJU", | ||||
|    "colour": "Boja", | ||||
|    "githubStarredRepos": "GitHub repo-i sa zvjezdicom", | ||||
|    "uname": "Korisničko ime", | ||||
|    "wrongArgNum": "Naveden je pogrešan broj argumenata", | ||||
|    "xIsTrackOnly": "{} je samo za praćenje", | ||||
|    "source": "Izvor", | ||||
|    "app": "Aplikacija. ", | ||||
|    "appsFromSourceAreTrackOnly": "Aplikacije iz ovog izvora su 'Samo za praćenje'.", | ||||
|    "youPickedTrackOnly": "Odabrali ste opciju „Samo za praćenje”.", | ||||
|    "trackOnlyAppDescription": "Aplikacija će se pratiti radi ažuriranja, ali Obtainium neće moći da je preuzme ili instalira.", | ||||
|    "cancelled": "Otkazano", | ||||
|    "appAlreadyAdded": "Aplikacija je već dodana", | ||||
|    "alreadyUpToDateQuestion": "Aplikacija je već ažurirana?", | ||||
|    "addApp": "Dodaj aplikaciju", | ||||
|    "appSourceURL": "Izvorni URL aplikacije", | ||||
|    "error": "Greška", | ||||
|    "add": "Dodaj", | ||||
|    "searchSomeSourcesLabel": "Pretraživanje (samo neki izvori)", | ||||
|    "search": "Pretraživanje", | ||||
|    "additionalOptsFor": "Dodatne opcije za {}", | ||||
|    "supportedSourcesBelow": "Podržani izvori:", | ||||
|    "trackOnlyInBrackets": "(Samo za praćenje)", | ||||
|    "searchableInBrackets": "(Može se pretraživati)", | ||||
|    "appsString": "Aplikacije", | ||||
|    "noApps": "Nema aplikacija", | ||||
|    "noAppsForFilter": "Nema aplikacija za filter", | ||||
|    "byX": "Autor {}", | ||||
|    "percentProgress": "Napredak: {}%", | ||||
|    "pleaseWait": "Molimo sačekajte", | ||||
|    "updateAvailable": "Ažuriranje dostupno", | ||||
|    "estimateInBracketsShort": "(Procjena)", | ||||
|    "notInstalled": "Nije instalirano", | ||||
|    "estimateInBrackets": "(Procjena)", | ||||
|    "selectAll": "Označi sve", | ||||
|    "deselectN": "Poništi odabir {}", | ||||
|    "xWillBeRemovedButRemainInstalled": "{} će biti uklonjen iz Obtainiuma, ali će ostati instaliran na uređaju.", | ||||
|    "removeSelectedAppsQuestion": "Želite li ukloniti odabrane aplikacije?", | ||||
|    "removeSelectedApps": "Ukloni odabrane aplikacije", | ||||
|    "updateX": "Nadogradi {}", | ||||
|    "installX": "Instaliraj {}", | ||||
|    "markXTrackOnlyAsUpdated": "Označi {}\n(samo za praćenje)\nkao ažurirano", | ||||
|    "changeX": "Promjena {}", | ||||
|    "installUpdateApps": "Instalirajte/ažurirajte aplikacije", | ||||
|    "installUpdateSelectedApps": "Instalirajte/ažurirajte odabrane aplikacije", | ||||
|    "markXSelectedAppsAsUpdated": "Označite {} odabrane aplikacije kao ažurirane?", | ||||
|    "no": "Ne", | ||||
|    "yes": "Da", | ||||
|    "markSelectedAppsUpdated": "Označi odabrane aplikacije kao ažurirane", | ||||
|    "pinToTop": "Prikvači na vrh", | ||||
|    "unpinFromTop": "Otkvači sa vrha", | ||||
|    "resetInstallStatusForSelectedAppsQuestion": "Resetujte status instalacije za odabrane aplikacije?", | ||||
|    "installStatusOfXWillBeResetExplanation": "Status instalacije bilo koje odabrane aplikacije će se resetovati.\n\nTo može pomoći kada je verzija aplikacije prikazana u Obtainiumu netačna zbog neuspjelih ažuriranja ili drugih problema.", | ||||
|    "shareSelectedAppURLs": "Podijeli odabrane URL-ove aplikacija", | ||||
|    "resetInstallStatus": "Resetujte status instalacije", | ||||
|    "more": "Više", | ||||
|    "removeOutdatedFilter": "Uklonite zastarjeli filter aplikacija", | ||||
|    "showOutdatedOnly": "Prikaži samo zastarjele aplikacije", | ||||
|    "filter": "Filtriranje", | ||||
|    "filterActive": "Filtriranje", | ||||
|    "filterApps": "Filtriraj aplikacije", | ||||
|    "appName": "Naziv aplikacije", | ||||
|    "author": "Autor", | ||||
|    "upToDateApps": "Ažurirane aplikacije", | ||||
|    "nonInstalledApps": "Neinstalirane aplikacije", | ||||
|    "importExport": "Uvoz/izvoz", | ||||
|    "settings": "Postavke", | ||||
|    "exportedTo": "Izvezeno u {}", | ||||
|    "obtainiumExport": "Obtainium Export", | ||||
|    "invalidInput": "Neispravan unos.", | ||||
|    "importedX": "Uvezeno {}", | ||||
|    "obtainiumImport": "Obtainium uvoz", | ||||
|    "importFromURLList": "Uvoz iz URL liste", | ||||
|    "searchQuery": "Pretraga za: ", | ||||
|    "appURLList": "Lista URL adresa aplikacija", | ||||
|    "line": "Linija", | ||||
|    "searchX": "Pretraživanje {}", | ||||
|    "noResults": "Nema rezultata", | ||||
|    "importX": "Uvoz {}", | ||||
|    "importedAppsIdDisclaimer": "Uvezene aplikacije mogu se pogrešno prikazati kao „Nije instalirano”.\nDa biste to riješili, ponovo ih instalirajte putem aplikacije Obtainium.\nTo ne bi trebalo uticati na podatke aplikacije.\n\nUtječe samo na URL i metode uvoza treće strane.", | ||||
|    "importErrors": "Uvezi greške", | ||||
|    "importedXOfYApps": "{} od {} aplikacija uvezeno.", | ||||
|    "followingURLsHadErrors": "Sljedeći URL-ovi su imali greške:", | ||||
|    "okay": "Dobro", | ||||
|    "selectURL": "Odaberite URL", | ||||
|    "selectURLs": "Odaberite URL-ove", | ||||
|    "pick": "Odaberi", | ||||
|    "theme": "Tema", | ||||
|    "dark": "Tamna", | ||||
|    "light": "Svijetla", | ||||
|    "followSystem": "Pratite sistem", | ||||
|    "obtainium": "Obtainium", | ||||
|    "materialYou": "Material You", | ||||
|    "useBlackTheme": "Koristite čisto crnu tamnu temu", | ||||
|    "appSortBy": "Aplikacije sortirane po", | ||||
|    "authorName": "Autor/Ime", | ||||
|    "nameAuthor": "Ime/Autor", | ||||
|    "asAdded": "Kao što je dodano", | ||||
|    "appSortOrder": "Redoslijed sortiranja aplikacija", | ||||
|    "ascending": "Uzlazno", | ||||
|    "descending": "Silazno", | ||||
|    "bgUpdateCheckInterval": "Interval provjere ažuriranja u pozadini", | ||||
|    "neverManualOnly": "Nikada - samo ručno", | ||||
|    "appearance": "Izgled", | ||||
|    "showWebInAppView": "Prikaži izvornu web stranicu u prikazu aplikacije", | ||||
|    "pinUpdates": "Prikvačite ažuriranja na vrh prikaza aplikacija", | ||||
|    "updates": "Nadogradnje", | ||||
|    "sourceSpecific": "Specifično za izvor", | ||||
|    "appSource": "Izvor aplikacije", | ||||
|    "noLogs": "Nema evidencije", | ||||
|    "appLogs": "Evidencije aplikacija", | ||||
|    "close": "Zatvori", | ||||
|    "share": "Podijeli", | ||||
|    "appNotFound": "Aplikacija nije pronađena", | ||||
|    "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||
|    "pickAnAPK": "Odaberite APK", | ||||
|    "appHasMoreThanOnePackage": "{} ima više od jednog paketa:", | ||||
|    "deviceSupportsXArch": "Vaš uređaj podržava {} arhitekturu procesora.", | ||||
|    "deviceSupportsFollowingArchs": "Vaš uređaj podržava sljedeće arhitekture procesora:", | ||||
|    "warning": "Upozorenje", | ||||
|    "sourceIsXButPackageFromYPrompt": "Izvor aplikacije je '{}', ali paket za izdavanje dolazi iz '{}'. Želite li nastaviti?", | ||||
|    "updatesAvailable": "Dostupna ažuriranja", | ||||
|    "updatesAvailableNotifDescription": "Obavještava korisnika da su ažuriranja dostupna za jednu ili više aplikacija koje prati Obtainium", | ||||
|    "noNewUpdates": "Nema novih ažuriranja.", | ||||
|    "xHasAnUpdate": "{} ima ažuriranje.", | ||||
|    "appsUpdated": "Aplikacije su ažurirane", | ||||
|    "appsUpdatedNotifDescription": "Obavještava korisnika da su u pozadini primijenjena ažuriranja na jednu ili više aplikacija", | ||||
|    "xWasUpdatedToY": "{} je ažuriran na {}.", | ||||
|    "errorCheckingUpdates": "Greška pri provjeri ažuriranja", | ||||
|    "errorCheckingUpdatesNotifDescription": "Obavijest koja se prikazuje kada provjera sigurnosnog ažuriranja ne uspije", | ||||
|    "appsRemoved": "Aplikacije su uklonjene", | ||||
|    "appsRemovedNotifDescription": "Obavještava korisnika da je jedna ili više aplikacija uklonjeno zbog grešaka prilikom učitavanja", | ||||
|    "xWasRemovedDueToErrorY": "{} je uklonjen zbog ove greške: {}", | ||||
|    "completeAppInstallation": "Dovršite instalaciju aplikacije", | ||||
|    "obtainiumMustBeOpenToInstallApps": "Obtainium mora biti otvoren za instalaciju aplikacija", | ||||
|    "completeAppInstallationNotifDescription": "Traži od korisnika da se vrati u Obtainium kako bi dovršio instalaciju aplikacije", | ||||
|    "checkingForUpdates": "Tražim moguće nadogradnje", | ||||
|    "checkingForUpdatesNotifDescription": "Privremeno obavještenje koje se pojavljuje prilikom provjere ažuriranja", | ||||
|    "pleaseAllowInstallPerm": "Dozvolite Obtainiumu da instalira aplikacije", | ||||
|    "trackOnly": "Samo za praćenje", | ||||
|    "errorWithHttpStatusCode": "Greška {}", | ||||
|    "versionCorrectionDisabled": "Ispravka verzije je onemogućena (izgleda da plugin ne radi)", | ||||
|    "unknown": "Nepoznato", | ||||
|    "none": "Ništa", | ||||
|    "never": "Nikad", | ||||
|    "latestVersionX": "Najnovija verzija: {}", | ||||
|    "installedVersionX": "Instalirana verzija: {}", | ||||
|    "lastUpdateCheckX": "Posljednja provjera ažuriranja: {}", | ||||
|    "remove": "Izbriši", | ||||
|    "yesMarkUpdated": "Da, označi kao ažurirano", | ||||
|    "fdroid": "F-Droid Official", | ||||
|    "appIdOrName": "ID ili ime aplikacije", | ||||
|    "appId": "Apl ID", | ||||
|    "appWithIdOrNameNotFound": "Nije pronađena aplikacija s tim ID-om ili imenom", | ||||
|    "reposHaveMultipleApps": "Repo-i mogu sadržavati više aplikacija", | ||||
|    "fdroidThirdPartyRepo": "F-Droid Repo treće strane", | ||||
|    "steam": "Steam", | ||||
|    "steamMobile": "Steam Mobile", | ||||
|    "steamChat": "Razgovor na Steamu (chat)", | ||||
|    "install": "Instaliraj", | ||||
|    "markInstalled": "Označi kao instalirano", | ||||
|    "update": "Nadogradi", | ||||
|    "markUpdated": "Označi kao ažurirano", | ||||
|    "additionalOptions": "Dodatne opcije", | ||||
|    "disableVersionDetection": "Onemogući detekciju verzije", | ||||
|    "noVersionDetectionExplanation": "Ova opcija bi se trebala koristiti samo za aplikacije gdje detekcija verzije ne radi ispravno.", | ||||
|    "downloadingX": "Preuzimanje {}", | ||||
|    "downloadNotifDescription": "Obavještava korisnika o napretku u preuzimanju aplikacije", | ||||
|    "noAPKFound": "APK nije pronađen", | ||||
|    "noVersionDetection": "Nema detekcije verzije", | ||||
|    "categorize": "Kategoriziraj", | ||||
|    "categories": "Kategorije", | ||||
|    "category": "Kategorija", | ||||
|    "noCategory": "Nema kategorije", | ||||
|    "noCategories": "Nema kategorija", | ||||
|    "deleteCategoriesQuestion": "Želite li izbrisati kategorije?", | ||||
|    "categoryDeleteWarning": "Sve aplikacije u izbrisanim kategorijama će biti postavljene kao nekategorisane.", | ||||
|    "addCategory": "Dodaj kategoriju", | ||||
|    "label": "Oznaka", | ||||
|    "language": "Jezik", | ||||
|    "copiedToClipboard": "Podaci kopirani u međuspremnik", | ||||
|    "storagePermissionDenied": "Dozvola za pohranu je odbijena", | ||||
|    "selectedCategorizeWarning": "Ovo će zamijeniti sve postojeće postavke kategorije za odabrane aplikacije.", | ||||
|    "filterAPKsByRegEx": "Filtrirajte APK-ove prema regularnom izrazu", | ||||
|    "removeFromObtainium": "Ukloni iz Obtainiuma", | ||||
|    "uninstallFromDevice": "Deinstaliraj s uređaja", | ||||
|    "onlyWorksWithNonVersionDetectApps": "Radi samo za aplikacije s onemogućenom detekcijom verzije.", | ||||
|    "releaseDateAsVersion": "Koristi datum izdanja kao verziju", | ||||
|    "releaseDateAsVersionExplanation": "Ova opcija bi se trebala koristiti samo za aplikacije gdje detekcija verzije ne radi ispravno, ali je datum izdavanja dostupan.", | ||||
|    "changes": "Promjene", | ||||
|    "releaseDate": "Datum izdavanja", | ||||
|    "importFromURLsInFile": "Uvoz iz URL-ova u datoteci (kao što je OPML)", | ||||
|    "versionDetection": "Otkrivanje verzije", | ||||
|    "standardVersionDetection": "Detekcija standardne verzije", | ||||
|    "groupByCategory": "Grupiši po kategoriji", | ||||
|    "autoApkFilterByArch": "Pokušajte filtrirati APK-ove po arhitekturi procesora ako je moguće", | ||||
|    "overrideSource": "Premosti izvor", | ||||
|    "dontShowAgain": "Ne prikazuj ovo ponovo", | ||||
|    "dontShowTrackOnlyWarnings": "Ne prikazuj upozorenja „Samo za  praćenje”", | ||||
|    "dontShowAPKOriginWarnings": "Ne prikazuj upozorenja o porijeklu APK-a", | ||||
|    "moveNonInstalledAppsToBottom": "Premjesti neinstalirane aplikacije na dno prikaza aplikacija", | ||||
|    "gitlabPATLabel": "GitLab token za lični pristup\n(Omogućava pretraživanje i bolje otkrivanje APK-a)", | ||||
|    "about": "O nama", | ||||
|    "requiresCredentialsInSettings": "Za ovo su potrebni dodatni akreditivi (u Postavkama)", | ||||
|    "checkOnStart": "Provjerite ima li novosti pri pokretanju", | ||||
|    "tryInferAppIdFromCode": "Pokušati otkriti ID aplikacije iz izvornog koda", | ||||
|    "removeOnExternalUninstall": "Automatski ukloni eksterno deinstalirane aplikacije", | ||||
|    "pickHighestVersionCode": "Automatski odaberite najviši kôd verzije APK-a", | ||||
|    "checkUpdateOnDetailPage": "Provjerite ima li novosti pri otvaranju stranice s detaljima aplikacije", | ||||
|    "disablePageTransitions": "Ugasite animaciju prijelaza stranice", | ||||
|    "reversePageTransitions": "Reverzne animacije prijelaza stranice", | ||||
|    "minStarCount": "Minimum Star Count", | ||||
|    "addInfoBelow": "Add this info below.", | ||||
|    "addInfoInSettings": "Add this info in the Settings.", | ||||
|    "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|    "gitlabSourceNote": "GitLab APK extraction may not work without an API key.", | ||||
|    "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|    "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|    "removeAppQuestion": { | ||||
|       "one": "Želite li ukloniti aplikaciju?", | ||||
|       "other": "Želite li ukloniti aplikacije?" | ||||
|    }, | ||||
|    "tooManyRequestsTryAgainInMinutes": { | ||||
|       "one": "Previše zahtjeva (ograničena broj zahteva) - pokušajte ponovo za {} minutu", | ||||
|       "other": "Previše zahtjeva (ograničena cijena) - pokušajte ponovo za {} min." | ||||
|    }, | ||||
|    "bgUpdateGotErrorRetryInMinutes": { | ||||
|       "one": "Provjera ažuriranja u pozadini naišla je na {}, zakazuje se ponovni pokušaj za {} minutu", | ||||
|       "other": "Provjera ažuriranja u pozadini naišla je na {}, zakazuje se ponovni pokušaj za {} min." | ||||
|    }, | ||||
|    "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|       "one": "Provjera ažuriranja u pozadini je pronašla {} ažuriranje - korisnik će biti obavješten ako je to potrebno", | ||||
|       "other": "Provjera ažuriranja u pozadini je pronašla {} ažuriranja - korisnik će biti obavješten ako je to potrebno" | ||||
|    }, | ||||
|    "apps": { | ||||
|       "one": "{} aplikacija", | ||||
|       "other": "{} aplikacije" | ||||
|    }, | ||||
|    "url": { | ||||
|       "one": "{} URL", | ||||
|       "other": "{} URL-ovi" | ||||
|    }, | ||||
|    "minute": { | ||||
|       "one": "{} minuta", | ||||
|       "other": "min." | ||||
|    }, | ||||
|    "hour": { | ||||
|       "one": "{} sat", | ||||
|       "other": "{} sat/i" | ||||
|    }, | ||||
|    "day": { | ||||
|       "one": "{} dan", | ||||
|       "other": "{} dana" | ||||
|    }, | ||||
|    "clearedNLogsBeforeXAfterY": { | ||||
|       "one": "Izbrisan {n} log (prije = {before}, nakon = {after})", | ||||
|       "other": "Izbrisano {n} log-ova (prije = {before}, nakon = {after})" | ||||
|    }, | ||||
|    "xAndNMoreUpdatesAvailable": { | ||||
|       "one": "{} i još 1 aplikacija ima ažuriranja.", | ||||
|       "other": "{} i još {} aplikacija imaju ažuriranja." | ||||
|    }, | ||||
|    "xAndNMoreUpdatesInstalled": { | ||||
|       "one": "{} i još 1 aplikacija je ažurirana.", | ||||
|       "other": "{} i još {} aplikacija je ažurirano." | ||||
|    } | ||||
| } | ||||
| @@ -233,10 +233,19 @@ | ||||
|     "about": "Über", | ||||
|     "requiresCredentialsInSettings": "Benötigt zusätzliche Anmeldedaten (in den Einstellungen)", | ||||
|     "checkOnStart": "Überprüfe einmalig beim Start", | ||||
|     "tryInferAppIdFromCode": "Try inferring App ID from source code", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "tryInferAppIdFromCode": "Versuche, die App-ID aus dem Quellcode zu ermitteln", | ||||
|     "removeOnExternalUninstall": "Automatisches Entfernen von extern deinstallierten Apps", | ||||
|     "pickHighestVersionCode": "Automatische Auswahl des APK mit höchstem Versionscode", | ||||
|     "checkUpdateOnDetailPage": "Nach Updates suchen, wenn eine App-Detailseite geöffnet wird", | ||||
|     "disablePageTransitions": "Animationen für Seitenübergänge deaktivieren", | ||||
|     "reversePageTransitions": "Umgekehrte Animationen für Seitenübergänge", | ||||
|     "minStarCount": "Minimale Anzahl von Sternen", | ||||
|     "addInfoBelow": "Fügen Sie diese Informationen unten hinzu.", | ||||
|     "addInfoInSettings": "Fügen Sie diese Info in den Einstellungen hinzu.", | ||||
|     "githubSourceNote": "Die GitHub-Ratenbegrenzung kann mit einem API-Schlüssel umgangen werden.", | ||||
|     "gitlabSourceNote": "GitLab APK-Extraktion funktioniert möglicherweise nicht ohne API-Schlüssel", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App entfernen?", | ||||
|         "other": "Apps entfernen?" | ||||
|   | ||||
| @@ -237,6 +237,15 @@ | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "disablePageTransitions": "Disable page transition animations", | ||||
|     "reversePageTransitions": "Reverse page transition animations", | ||||
|     "minStarCount": "Minimum Star Count", | ||||
|     "addInfoBelow": "Add this info below.", | ||||
|     "addInfoInSettings": "Add this info in the Settings.", | ||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|     "gitlabSourceNote": "GitLab APK extraction may not work without an API key.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Remove App?", | ||||
|         "other": "Remove Apps?" | ||||
|   | ||||
| @@ -237,6 +237,15 @@ | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "disablePageTransitions": "Disable page transition animations", | ||||
|     "reversePageTransitions": "Reverse page transition animations", | ||||
|     "minStarCount": "Minimum Star Count", | ||||
|     "addInfoBelow": "Add this info below.", | ||||
|     "addInfoInSettings": "Add this info in the Settings.", | ||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|     "gitlabSourceNote": "GitLab APK extraction may not work without an API key.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "¿Eliminar Aplicación?", | ||||
|         "other": "¿Eliminar Aplicaciones?" | ||||
|   | ||||
| @@ -237,6 +237,15 @@ | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "disablePageTransitions": "Disable page transition animations", | ||||
|     "reversePageTransitions": "Reverse page transition animations", | ||||
|     "minStarCount": "Minimum Star Count", | ||||
|     "addInfoBelow": "Add this info below.", | ||||
|     "addInfoInSettings": "Add this info in the Settings.", | ||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|     "gitlabSourceNote": "GitLab APK extraction may not work without an API key.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "برنامه حذف شود؟", | ||||
|         "other": "برنامه ها حذف شوند؟" | ||||
|   | ||||
| @@ -45,7 +45,7 @@ | ||||
|     "addApp": "Ajouter une application", | ||||
|     "appSourceURL": "URL de la source de l'application", | ||||
|     "error": "Erreur", | ||||
|     "add": "Ajoutée", | ||||
|     "add": "Ajouter", | ||||
|     "searchSomeSourcesLabel": "Rechercher (certaines sources uniquement)", | ||||
|     "search": "Rechercher", | ||||
|     "additionalOptsFor": "Options supplémentaires pour {}", | ||||
| @@ -237,6 +237,15 @@ | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "disablePageTransitions": "Disable page transition animations", | ||||
|     "reversePageTransitions": "Reverse page transition animations", | ||||
|     "minStarCount": "Minimum Star Count", | ||||
|     "addInfoBelow": "Add this info below.", | ||||
|     "addInfoInSettings": "Add this info in the Settings.", | ||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|     "gitlabSourceNote": "GitLab APK extraction may not work without an API key.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Supprimer l'application ?", | ||||
|         "other": "Supprimer les applications ?" | ||||
|   | ||||
| @@ -17,7 +17,7 @@ | ||||
|     "bgUpdateTaskFinished": "A háttérfrissítés ellenőrzési feladat befejeződött", | ||||
|     "firstRun": "Ez az Obtainium első futása", | ||||
|     "settingUpdateCheckIntervalTo": "A frissítési intervallum beállítása erre: {}", | ||||
|     "githubPATLabel": "GitHub személyes hozzáférési token (megnöveli a díjkorlátot)", | ||||
|     "githubPATLabel": "GitHub Personal Access Token (megnöveli a díjkorlátot)", | ||||
|     "githubPATHint": "A PAT-nak a következő formátumban kell lennie: felhasználónév:token", | ||||
|     "githubPATFormat": "felhasználónév:token", | ||||
|     "includePrereleases": "Tartalmazza az előzetes kiadásokat", | ||||
| @@ -228,14 +228,23 @@ | ||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítsen meg 'Csak nyomon követés' figyelmeztetést", | ||||
|     "dontShowAPKOriginWarnings": "Ne jelenítsen meg az APK eredetére vonatkozó figyelmeztetéseket", | ||||
|     "moveNonInstalledAppsToBottom": "Helyezze át a nem telepített appokat az App nézet aljára", | ||||
|     "gitlabPATLabel": "GitLab Personal Access Token\n(Engedélyezi a Keresést and Better APK Discovery)", | ||||
|     "gitlabPATLabel": "GitLab Personal Access Token\n(Engedélyezi a Keresést és jobb APK felfedezés)", | ||||
|     "about": "Rólunk", | ||||
|     "requiresCredentialsInSettings": "Ehhez további hitelesítő adatokra van szükség (a Beállításokban)", | ||||
|     "checkOnStart": "Egyszer az indításkor", | ||||
|     "tryInferAppIdFromCode": "Próbálja kikövetkeztetni az app azonosítót a forráskódból", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "removeOnExternalUninstall": "A külsőleg eltávolított appok auto. eltávolítása", | ||||
|     "pickHighestVersionCode": "A legmagasabb verziószámú APK auto. kiválasztása", | ||||
|     "checkUpdateOnDetailPage": "Frissítések keresése az app részleteit tartalmazó oldal megnyitásakor", | ||||
|     "disablePageTransitions": "Lap áttűnési animációk tiltása", | ||||
|     "reversePageTransitions": "Fordított lap áttűnési animációk", | ||||
|     "minStarCount": "Minimális csillag szám", | ||||
|     "addInfoBelow": "Adja hozzá ezt az infót alább.", | ||||
|     "addInfoInSettings": "Adja hozzá ezt az infót a Beállításokban.", | ||||
|     "githubSourceNote": "A GitHub értékelési korlátozása elkerülhető API-kulcs használatával.", | ||||
|     "gitlabSourceNote": "Előfordulhat, hogy a GitLab APK kibontása nem működik API-kulcs nélkül.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Eltávolítja az alkalmazást?", | ||||
|         "other": "Eltávolítja az alkalmazást?" | ||||
|   | ||||
| @@ -237,6 +237,15 @@ | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "disablePageTransitions": "Disable page transition animations", | ||||
|     "reversePageTransitions": "Reverse page transition animations", | ||||
|     "minStarCount": "Minimum Star Count", | ||||
|     "addInfoBelow": "Add this info below.", | ||||
|     "addInfoInSettings": "Add this info in the Settings.", | ||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||
|     "gitlabSourceNote": "GitLab APK extraction may not work without an API key.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Rimuovere l'app?", | ||||
|         "other": "Rimuovere le app?" | ||||
|   | ||||
| @@ -135,7 +135,7 @@ | ||||
|     "showWebInAppView": "アプリページにソースのWebページを表示する", | ||||
|     "pinUpdates": "アップデートがあるアプリをトップに固定する", | ||||
|     "updates": "アップデート", | ||||
|     "sourceSpecific": "Github アクセストークン", | ||||
|     "sourceSpecific": "ソース別の設定", | ||||
|     "appSource": "アプリのソース", | ||||
|     "noLogs": "ログはありません", | ||||
|     "appLogs": "アプリのログ", | ||||
| @@ -227,16 +227,25 @@ | ||||
|     "overrideSource": "ソースの上書き", | ||||
|     "dontShowAgain": "二度と表示しない", | ||||
|     "dontShowTrackOnlyWarnings": "「追跡のみ」の警告を表示しない", | ||||
|     "dontShowAPKOriginWarnings": "APK Originの警告を表示しない", | ||||
|     "dontShowAPKOriginWarnings": "APKのダウンロード元の警告を表示しない", | ||||
|     "moveNonInstalledAppsToBottom": "未インストールのアプリをアプリ一覧の下部に移動させる", | ||||
|     "gitlabPATLabel": "GitLab パーソナルアクセストークン\n(検索を有効化する and Better APK Discovery)", | ||||
|     "gitlabPATLabel": "GitLab パーソナルアクセストークン\n(検索とより良いAPK検出の有効化)", | ||||
|     "about": "概要", | ||||
|     "requiresCredentialsInSettings": "これには追加の認証が必要です (設定にて)", | ||||
|     "checkOnStart": "Check for updates on startup", | ||||
|     "tryInferAppIdFromCode": "Try inferring App ID from source code", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "checkOnStart": "起動時にアップデートを確認する", | ||||
|     "tryInferAppIdFromCode": "ソースコードからApp IDを推測する", | ||||
|     "removeOnExternalUninstall": "外部でアンインストールされたアプリを自動的に削除する", | ||||
|     "pickHighestVersionCode": "最も高いバージョンコードのAPKを自動的に選択する", | ||||
|     "checkUpdateOnDetailPage": "アプリの詳細ページを開く際にアップデートを確認する", | ||||
|     "disablePageTransitions": "ページ遷移アニメーションを無効化する", | ||||
|     "reversePageTransitions": "ページ遷移アニメーションを反転する", | ||||
|     "minStarCount": "最小スター数", | ||||
|     "addInfoBelow": "下部でこの情報を追加してください。", | ||||
|     "addInfoInSettings": "設定でこの情報を追加してください。", | ||||
|     "githubSourceNote": "GitHubのレート制限はAPIキーを使うことで回避できます。", | ||||
|     "gitlabSourceNote": "GitLabのAPK抽出はAPIキーがないと動作しない場合があります。", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "アプリを削除しますか?", | ||||
|         "other": "アプリを削除しますか?" | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
|     "noVersionFound": "Nie można określić wersji wydania", | ||||
|     "urlMatchesNoSource": "Adres URL nie pasuje do znanego źródła", | ||||
|     "cantInstallOlderVersion": "Nie można zainstalować starszej wersji aplikacji", | ||||
|     "appIdMismatch": "Pobrany identyfikator pakietu nie pasuje do istniejącego identyfikatora aplikacji", | ||||
|     "appIdMismatch": "Pobrane ID pakietu nie pasuje do istniejącego ID aplikacji", | ||||
|     "functionNotImplemented": "Ta klasa nie zaimplementowała tej funkcji", | ||||
|     "placeholder": "Placeholder", | ||||
|     "someErrors": "Wystąpiły pewne błędy", | ||||
| @@ -25,8 +25,8 @@ | ||||
|     "bgUpdateTaskFinished": "Zakończono zadanie sprawdzania aktualizacji w tle", | ||||
|     "firstRun": "Jest to pierwsze uruchomienie Obtainium", | ||||
|     "settingUpdateCheckIntervalTo": "Ustawianie interwału aktualizacji na {}", | ||||
|     "githubPATLabel": "Osobisty Token Dostępu GitHub (zwiększa limit zapytań)", | ||||
|     "githubPATHint": "Wymagany format OTD: użytkownik:token", | ||||
|     "githubPATLabel": "Osobisty token dostępu GitHub (zwiększa limit zapytań)", | ||||
|     "githubPATHint": "Wymagany format: użytkownik:token", | ||||
|     "githubPATFormat": "użytkownik:token", | ||||
|     "includePrereleases": "Uwzględnij wersje wstępne", | ||||
|     "fallbackToOlderReleases": "Powracaj do starszych wersji", | ||||
| @@ -230,17 +230,26 @@ | ||||
|     "autoApkFilterByArch": "Spróbuj filtrować pliki APK według architektury procesora, jeśli to możliwe", | ||||
|     "overrideSource": "Nadpisz źródło", | ||||
|     "dontShowAgain": "Nie pokazuj tego ponownie", | ||||
|     "dontShowTrackOnlyWarnings": "Nie wyświetlaj ostrzeżeń „Tylko obserwowana”", | ||||
|     "dontShowTrackOnlyWarnings": "Nie pokazuj ostrzeżeń „Tylko obserwowana”", | ||||
|     "dontShowAPKOriginWarnings": "Nie pokazuj ostrzeżeń o pochodzeniu APK", | ||||
|     "moveNonInstalledAppsToBottom": "Przenieś niezainstalowane aplikacje na dół widoku aplikacji", | ||||
|     "gitlabPATLabel": "Osobisty Token Dostępu GitLab\n(umożliwia wyszukiwanie and Better APK Discovery)", | ||||
|     "gitlabPATLabel": "Osobisty token dostępu GitLab\n(Umożliwia wyszukiwanie i lepsze wykrywanie APK)", | ||||
|     "about": "Więcej informacji", | ||||
|     "requiresCredentialsInSettings": "Wymaga to dodatkowych poświadczeń (w Ustawieniach)", | ||||
|     "checkOnStart": "Sprawdź raz przy starcie", | ||||
|     "checkOnStart": "Sprawdź aktualizacje przy uruchomieniu", | ||||
|     "tryInferAppIdFromCode": "Spróbuj wywnioskować identyfikator aplikacji z kodu źródłowego", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "removeOnExternalUninstall": "Automatyczne usuń odinstalowane zewnętrznie aplikacje", | ||||
|     "pickHighestVersionCode": "Automatycznie wybierz najwyższy kod wersji APK", | ||||
|     "checkUpdateOnDetailPage": "Sprawdzaj aktualizacje podczas otwierania strony szczegółów aplikacji", | ||||
|     "disablePageTransitions": "Wyłącz animacje przejścia między stronami", | ||||
|     "reversePageTransitions": "Odwróć animacje przejścia pomiędzy stronami", | ||||
|     "minStarCount": "Minimalna ilość gwiazdek", | ||||
|     "addInfoBelow": "Dodaj tę informację poniżej.", | ||||
|     "addInfoInSettings": "Dodaj tę informację w Ustawieniach.", | ||||
|     "githubSourceNote": "Limit żądań GitHub można ominąć za pomocą klucza API.", | ||||
|     "gitlabSourceNote": "Pozyskiwanie pliku APK z GitLab może nie działać bez klucza API.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Usunąć aplikację?", | ||||
|         "other": "Usunąć aplikacje?" | ||||
|   | ||||
| @@ -22,7 +22,7 @@ | ||||
|     "githubPATFormat": "имя_пользователя:токен", | ||||
|     "includePrereleases": "Включить предварительные релизы", | ||||
|     "fallbackToOlderReleases": "Откатиться к более старым версиям", | ||||
|     "filterReleaseTitlesByRegEx": "Фильтровать заголовки релизов с помощью регулярного выражения", | ||||
|     "filterReleaseTitlesByRegEx": "Фильтровать заголовки релизов\nс помощью регулярного выражения", | ||||
|     "invalidRegEx": "Неверное регулярное выражение", | ||||
|     "noDescription": "Нет описания", | ||||
|     "cancel": "Отмена", | ||||
| @@ -33,7 +33,7 @@ | ||||
|     "githubStarredRepos": "Помеченные звездочкой репозитории на GitHub", | ||||
|     "uname": "Имя пользователя", | ||||
|     "wrongArgNum": "Неправильное количество предоставленных аргументов", | ||||
|     "xIsTrackOnly": "{} является приложением только для отслеживания", | ||||
|     "xIsTrackOnly": "{} только для отслеживания", | ||||
|     "source": "Источник", | ||||
|     "app": "Приложение", | ||||
|     "appsFromSourceAreTrackOnly": "Приложения из этого источника являются 'только для отслеживания'.", | ||||
| @@ -100,7 +100,7 @@ | ||||
|     "invalidInput": "Неверный ввод", | ||||
|     "importedX": "Импортировано {}", | ||||
|     "obtainiumImport": "Импорт в Obtainium", | ||||
|     "importFromURLList": "Импорт из списка URL-адреса", | ||||
|     "importFromURLList": "Импорт из списка URL-адресов", | ||||
|     "searchQuery": "Поисковый запрос", | ||||
|     "appURLList": "Список URL приложений", | ||||
|     "line": "Строка", | ||||
| @@ -116,9 +116,9 @@ | ||||
|     "selectURLs": "Выбрать URL-адреса", | ||||
|     "pick": "Выбрать", | ||||
|     "theme": "Тема", | ||||
|     "dark": "Темный", | ||||
|     "light": "Светлый", | ||||
|     "followSystem": "Следовать системе", | ||||
|     "dark": "Темная", | ||||
|     "light": "Светлая", | ||||
|     "followSystem": "Как в системе", | ||||
|     "obtainium": "Obtainium", | ||||
|     "materialYou": "Material You", | ||||
|     "useBlackTheme": "Использовать чёрную тему", | ||||
| @@ -183,7 +183,7 @@ | ||||
|     "appId": "ID приложения", | ||||
|     "appWithIdOrNameNotFound": "Приложение с таким ID или названием не было найдено", | ||||
|     "reposHaveMultipleApps": "В хранилище может быть несколько приложений", | ||||
|     "fdroidThirdPartyRepo": "Хранилище F-Droid сторонних разработчиков", | ||||
|     "fdroidThirdPartyRepo": "Сторонние репозитории F-Droid", | ||||
|     "steam": "Steam", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam Chat", | ||||
| @@ -211,7 +211,7 @@ | ||||
|     "copiedToClipboard": "Скопировано в буфер обмена", | ||||
|     "storagePermissionDenied": "Отказано в доступе к хранилищу", | ||||
|     "selectedCategorizeWarning": "Это заменит все текущие настройки категорий для выбранных приложений.", | ||||
|     "filterAPKsByRegEx": "Фильтровать APK-файлы с помощью регулярного выражения", | ||||
|     "filterAPKsByRegEx": "Фильтровать APK-файлы с помощью\nрегулярного выражения", | ||||
|     "removeFromObtainium": "Удалить из Obtainium", | ||||
|     "uninstallFromDevice": "Удалить с устройства", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Работает только для приложений с отключенным определением версии.", | ||||
| @@ -219,7 +219,7 @@ | ||||
|     "releaseDateAsVersionExplanation": "Этот параметр следует использовать только для приложений, в которых определение версии не работает правильно, но имеется дата выпуска.", | ||||
|     "changes": "Изменения", | ||||
|     "releaseDate": "Дата выпуска", | ||||
|     "importFromURLsInFile": "Импортировать из URL-адресов в файл (например, OPML)", | ||||
|     "importFromURLsInFile": "Импорт URL-адресов из файла (например, OPML)", | ||||
|     "versionDetection": "Определение версии", | ||||
|     "standardVersionDetection": "Стандартное определение версии", | ||||
|     "groupByCategory": "Группировать по категориям", | ||||
| @@ -229,14 +229,23 @@ | ||||
|     "dontShowTrackOnlyWarnings": "Не показывать предупреждения о только отслеживаемых приложениях", | ||||
|     "dontShowAPKOriginWarnings": "Не показывать предупреждения об источнике APK-файлов", | ||||
|     "moveNonInstalledAppsToBottom": "Переместить неустановленные приложения вниз списка", | ||||
|     "gitlabPATLabel": "Персональный токен доступа GitLab\n(Включает поиск and Better APK Discovery)", | ||||
|     "gitlabPATLabel": "Персональный токен доступа GitLab\n(Включает поиск и улучшает обнаружение APK)", | ||||
|     "about": "О приложении", | ||||
|     "requiresCredentialsInSettings": "Для этого требуются дополнительные учетные данные (в настройках)", | ||||
|     "checkOnStart": "Проверять наличие обновлений при запуске", | ||||
|     "tryInferAppIdFromCode": "Попытаться определить ID приложения из исходного кода", | ||||
|     "removeOnExternalUninstall": "Автоматически удалять удаленные извне приложения", | ||||
|     "removeOnExternalUninstall": "Автоматически убирать из списка удаленные извне приложения", | ||||
|     "pickHighestVersionCode": "Автовыбор кода наивысшей версии APK", | ||||
|     "checkUpdateOnDetailPage": "Проверять наличие обновлений при открытии страницы представления приложения", | ||||
|     "disablePageTransitions": "Отключить анимацию перехода между страницами", | ||||
|     "reversePageTransitions": "Реверс анимации перехода между страницами", | ||||
|     "minStarCount": "Минимальное количество звёзд", | ||||
|     "addInfoBelow": "Добавьте эту информацию ниже.", | ||||
|     "addInfoInSettings": "Добавьте эту информацию в Настройки.", | ||||
|     "githubSourceNote": "Лимит запросов GitHub можно обойти, используя ключ API.", | ||||
|     "gitlabSourceNote": "Извлечение APK из GitLab может не работать без ключа API.", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Удалить приложение?", | ||||
|         "other": "Удалить приложения?" | ||||
| @@ -258,8 +267,8 @@ | ||||
|         "other": "{} Приложений" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} Ссылка", | ||||
|         "other": "{} Ссылки" | ||||
|         "one": "{} URL-адрес", | ||||
|         "other": "{} URL-адреса" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Минута", | ||||
|   | ||||
| @@ -107,7 +107,7 @@ | ||||
|     "searchX": "搜索 {}", | ||||
|     "noResults": "无结果", | ||||
|     "importX": "导入 {}", | ||||
|     "importedAppsIdDisclaimer": "导入的应用可能错误地显示为“未安装”。\n请通过 Obtainium 重新安装这些应用来解决此问题。", | ||||
|     "importedAppsIdDisclaimer": "导入的应用可能会错误地显示为“未安装”状态。\n请通过 Obtainium 重新安装这些应用来解决此问题。", | ||||
|     "importErrors": "导入错误", | ||||
|     "importedXOfYApps": "已导入 {} 中的 {} 个应用。", | ||||
|     "followingURLsHadErrors": "下列 URL 存在错误:", | ||||
| @@ -229,14 +229,23 @@ | ||||
|     "dontShowTrackOnlyWarnings": "不显示“仅追踪”模式警告", | ||||
|     "dontShowAPKOriginWarnings": "不显示 APK 文件来源警告", | ||||
|     "moveNonInstalledAppsToBottom": "将未安装应用置底", | ||||
|     "gitlabPATLabel": "GitLab 个人访问令牌\n(用于搜索应用 and Better APK Discovery)", | ||||
|     "gitlabPATLabel": "GitLab 个人访问令牌\n(启用搜索功能并增强 APK 发现)", | ||||
|     "about": "相关文档", | ||||
|     "requiresCredentialsInSettings": "此功能需要额外的凭据(在“设置”中添加)", | ||||
|     "checkOnStart": "启动时进行一次检查", | ||||
|     "tryInferAppIdFromCode": "尝试从源代码推断应用 ID", | ||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", | ||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", | ||||
|     "removeOnExternalUninstall": "自动删除已卸载的外部应用", | ||||
|     "pickHighestVersionCode": "自动选择版本号最高的 APK 文件", | ||||
|     "checkUpdateOnDetailPage": "打开应用详情页时检查更新", | ||||
|     "disablePageTransitions": "禁用页面过渡动画效果", | ||||
|     "reversePageTransitions": "反转页面过渡动画效果", | ||||
|     "minStarCount": "最小星标数", | ||||
|     "addInfoBelow": "在下方添加此凭据。", | ||||
|     "addInfoInSettings": "在“设置”中添加此凭据。", | ||||
|     "githubSourceNote": "使用访问令牌可避免触发 GitHub 的 API 请求限制。", | ||||
|     "gitlabSourceNote": "未使用访问令牌时可能无法从 GitLab 获取 APK 文件。", | ||||
|     "sortByFileNamesNotLinks": "Sort by file names instead of full links", | ||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "是否删除应用?", | ||||
|         "other": "是否删除应用?" | ||||
|   | ||||
| @@ -1,41 +1,19 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:obtainium/app_sources/github.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class Codeberg extends AppSource { | ||||
|   GitHub gh = GitHub(); | ||||
|   Codeberg() { | ||||
|     host = 'codeberg.org'; | ||||
|  | ||||
|     additionalSourceSpecificSettingFormItems = []; | ||||
|  | ||||
|     additionalSourceAppSpecificSettingFormItems = [ | ||||
|       [ | ||||
|         GeneratedFormSwitch('includePrereleases', | ||||
|             label: tr('includePrereleases'), defaultValue: false) | ||||
|       ], | ||||
|       [ | ||||
|         GeneratedFormSwitch('fallbackToOlderReleases', | ||||
|             label: tr('fallbackToOlderReleases'), defaultValue: true) | ||||
|       ], | ||||
|       [ | ||||
|         GeneratedFormTextField('filterReleaseTitlesByRegEx', | ||||
|             label: tr('filterReleaseTitlesByRegEx'), | ||||
|             required: false, | ||||
|             additionalValidators: [ | ||||
|               (value) { | ||||
|                 return regExValidator(value); | ||||
|               } | ||||
|             ]) | ||||
|       ] | ||||
|     ]; | ||||
|     additionalSourceAppSpecificSettingFormItems = | ||||
|         gh.additionalSourceAppSpecificSettingFormItems; | ||||
|  | ||||
|     canSearch = true; | ||||
|     searchQuerySettingFormItems = gh.searchQuerySettingFormItems; | ||||
|   } | ||||
|  | ||||
|   var gh = GitHub(); | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|     RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+'); | ||||
| @@ -68,10 +46,12 @@ class Codeberg extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, List<String>>> search(String query) async { | ||||
|   Future<Map<String, List<String>>> search(String query, | ||||
|       {Map<String, dynamic> querySettings = const {}}) async { | ||||
|     return gh.searchCommon( | ||||
|         query, | ||||
|         'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100', | ||||
|         'data'); | ||||
|         'data', | ||||
|         querySettings: querySettings); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -72,7 +72,8 @@ class FDroid extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, List<String>>> search(String query) async { | ||||
|   Future<Map<String, List<String>>> search(String query, | ||||
|       {Map<String, dynamic> querySettings = const {}}) async { | ||||
|     Response res = await sourceRequest( | ||||
|         'https://search.$host/?q=${Uri.encodeQueryComponent(query)}'); | ||||
|     if (res.statusCode == 200) { | ||||
|   | ||||
| @@ -16,7 +16,7 @@ class GitHub extends AppSource { | ||||
|     host = 'github.com'; | ||||
|     appIdInferIsOptional = true; | ||||
|  | ||||
|     additionalSourceSpecificSettingFormItems = [ | ||||
|     sourceConfigSettingFormItems = [ | ||||
|       GeneratedFormTextField('github-creds', | ||||
|           label: tr('githubPATLabel'), | ||||
|           password: true, | ||||
| @@ -75,10 +75,35 @@ class GitHub extends AppSource { | ||||
|                 return regExValidator(value); | ||||
|               } | ||||
|             ]) | ||||
|       ], | ||||
|       [ | ||||
|         GeneratedFormTextField('filterReleaseNotesByRegEx', | ||||
|             label: tr('filterReleaseNotesByRegEx'), | ||||
|             required: false, | ||||
|             additionalValidators: [ | ||||
|               (value) { | ||||
|                 return regExValidator(value); | ||||
|               } | ||||
|             ]) | ||||
|       ] | ||||
|     ]; | ||||
|  | ||||
|     canSearch = true; | ||||
|     searchQuerySettingFormItems = [ | ||||
|       GeneratedFormTextField('minStarCount', | ||||
|           label: tr('minStarCount'), | ||||
|           defaultValue: '0', | ||||
|           additionalValidators: [ | ||||
|             (value) { | ||||
|               try { | ||||
|                 int.parse(value ?? '0'); | ||||
|               } catch (e) { | ||||
|                 return tr('invalidInput'); | ||||
|               } | ||||
|               return null; | ||||
|             } | ||||
|           ]) | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -92,7 +117,7 @@ class GitHub extends AppSource { | ||||
|     for (var path in possibleBuildGradleLocations) { | ||||
|       try { | ||||
|         var res = await sourceRequest( | ||||
|             '${await convertStandardUrlToAPIUrl(standardUrl)}/contents/$path'); | ||||
|             '${await convertStandardUrlToAPIUrl(standardUrl, additionalSettings)}/contents/$path'); | ||||
|         if (res.statusCode == 200) { | ||||
|           try { | ||||
|             var body = jsonDecode(res.body); | ||||
| @@ -140,19 +165,30 @@ class GitHub extends AppSource { | ||||
|     return url.substring(0, match.end); | ||||
|   } | ||||
|  | ||||
|   Future<String> getCredentialPrefixIfAny() async { | ||||
|   Future<String> getCredentialPrefixIfAny( | ||||
|       Map<String, dynamic> additionalSettings) async { | ||||
|     SettingsProvider settingsProvider = SettingsProvider(); | ||||
|     await settingsProvider.initializeSettings(); | ||||
|     String? creds = settingsProvider | ||||
|         .getSettingString(additionalSourceSpecificSettingFormItems[0].key); | ||||
|     var sourceConfig = | ||||
|         await getSourceConfigValues(additionalSettings, settingsProvider); | ||||
|     String? creds = sourceConfig['github-creds']; | ||||
|     return creds != null && creds.isNotEmpty ? '$creds@' : ''; | ||||
|   } | ||||
|  | ||||
|   Future<String> getAPIHost() async => | ||||
|       'https://${await getCredentialPrefixIfAny()}api.$host'; | ||||
|   @override | ||||
|   Future<String?> getSourceNote() async { | ||||
|     if (!hostChanged && (await getCredentialPrefixIfAny({})).isEmpty) { | ||||
|       return '${tr('githubSourceNote')} ${hostChanged ? tr('addInfoBelow') : tr('addInfoInSettings')}'; | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   Future<String> convertStandardUrlToAPIUrl(String standardUrl) async => | ||||
|       '${await getAPIHost()}/repos${standardUrl.substring('https://$host'.length)}'; | ||||
|   Future<String> getAPIHost(Map<String, dynamic> additionalSettings) async => | ||||
|       'https://${await getCredentialPrefixIfAny(additionalSettings)}api.$host'; | ||||
|  | ||||
|   Future<String> convertStandardUrlToAPIUrl( | ||||
|           String standardUrl, Map<String, dynamic> additionalSettings) async => | ||||
|       '${await getAPIHost(additionalSettings)}/repos${standardUrl.substring('https://$host'.length)}'; | ||||
|  | ||||
|   @override | ||||
|   String? changeLogPageFromStandardUrl(String standardUrl) => | ||||
| @@ -170,6 +206,12 @@ class GitHub extends AppSource { | ||||
|                 true | ||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||
|             : null; | ||||
|     String? regexNotesFilter = | ||||
|         (additionalSettings['filterReleaseNotesByRegEx'] as String?) | ||||
|                     ?.isNotEmpty == | ||||
|                 true | ||||
|             ? additionalSettings['filterReleaseNotesByRegEx'] | ||||
|             : null; | ||||
|     Response res = await sourceRequest(requestUrl); | ||||
|     if (res.statusCode == 200) { | ||||
|       var releases = jsonDecode(res.body) as List<dynamic>; | ||||
| @@ -211,8 +253,8 @@ class GitHub extends AppSource { | ||||
|                 (nameA as String).substring(matchA!.start, matchA.end), | ||||
|                 (nameB as String).substring(matchB!.start, matchB.end)); | ||||
|           } else { | ||||
|             return getReleaseDateFromRelease(a)! | ||||
|                 .compareTo(getReleaseDateFromRelease(b)!); | ||||
|             return (getReleaseDateFromRelease(a) ?? DateTime(1)) | ||||
|                 .compareTo(getReleaseDateFromRelease(b) ?? DateTime(0)); | ||||
|           } | ||||
|         } | ||||
|       }); | ||||
| @@ -238,6 +280,11 @@ class GitHub extends AppSource { | ||||
|             !RegExp(regexFilter).hasMatch(nameToFilter.trim())) { | ||||
|           continue; | ||||
|         } | ||||
|         if (regexNotesFilter != null && | ||||
|             !RegExp(regexNotesFilter) | ||||
|                 .hasMatch(((releases[i]['body'] as String?) ?? '').trim())) { | ||||
|           continue; | ||||
|         } | ||||
|         var apkUrls = getReleaseAPKUrls(releases[i]); | ||||
|         if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) { | ||||
|           continue; | ||||
| @@ -296,7 +343,7 @@ class GitHub extends AppSource { | ||||
|   ) async { | ||||
|     return await getLatestAPKDetailsCommon2(standardUrl, additionalSettings, | ||||
|         (bool useTagUrl) async { | ||||
|       return '${await convertStandardUrlToAPIUrl(standardUrl)}/${useTagUrl ? 'tags' : 'releases'}?per_page=100'; | ||||
|       return '${await convertStandardUrlToAPIUrl(standardUrl, additionalSettings)}/${useTagUrl ? 'tags' : 'releases'}?per_page=100'; | ||||
|     }, (Response res) { | ||||
|       rateLimitErrorCheck(res); | ||||
|     }); | ||||
| @@ -310,20 +357,26 @@ class GitHub extends AppSource { | ||||
|  | ||||
|   Future<Map<String, List<String>>> searchCommon( | ||||
|       String query, String requestUrl, String rootProp, | ||||
|       {Function(Response)? onHttpErrorCode}) async { | ||||
|       {Function(Response)? onHttpErrorCode, | ||||
|       Map<String, dynamic> querySettings = const {}}) async { | ||||
|     Response res = await sourceRequest(requestUrl); | ||||
|     if (res.statusCode == 200) { | ||||
|       int minStarCount = querySettings['minStarCount'] != null | ||||
|           ? int.parse(querySettings['minStarCount']) | ||||
|           : 0; | ||||
|       Map<String, List<String>> urlsWithDescriptions = {}; | ||||
|       for (var e in (jsonDecode(res.body)[rootProp] as List<dynamic>)) { | ||||
|         urlsWithDescriptions.addAll({ | ||||
|           e['html_url'] as String: [ | ||||
|             e['full_name'] as String, | ||||
|             ((e['archived'] == true ? '[ARCHIVED] ' : '') + | ||||
|                 (e['description'] != null | ||||
|                     ? e['description'] as String | ||||
|                     : tr('noDescription'))) | ||||
|           ] | ||||
|         }); | ||||
|         if ((e['stargazers_count'] ?? e['stars_count'] ?? 0) >= minStarCount) { | ||||
|           urlsWithDescriptions.addAll({ | ||||
|             e['html_url'] as String: [ | ||||
|               e['full_name'] as String, | ||||
|               ((e['archived'] == true ? '[ARCHIVED] ' : '') + | ||||
|                   (e['description'] != null | ||||
|                       ? e['description'] as String | ||||
|                       : tr('noDescription'))) | ||||
|             ] | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|       return urlsWithDescriptions; | ||||
|     } else { | ||||
| @@ -335,13 +388,14 @@ class GitHub extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, List<String>>> search(String query) async { | ||||
|   Future<Map<String, List<String>>> search(String query, | ||||
|       {Map<String, dynamic> querySettings = const {}}) async { | ||||
|     return searchCommon( | ||||
|         query, | ||||
|         '${await getAPIHost()}/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100', | ||||
|         '${await getAPIHost({})}/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100', | ||||
|         'items', onHttpErrorCode: (Response res) { | ||||
|       rateLimitErrorCheck(res); | ||||
|     }); | ||||
|     }, querySettings: querySettings); | ||||
|   } | ||||
|  | ||||
|   rateLimitErrorCheck(Response res) { | ||||
|   | ||||
| @@ -16,7 +16,7 @@ class GitLab extends AppSource { | ||||
|     host = 'gitlab.com'; | ||||
|     canSearch = true; | ||||
|  | ||||
|     additionalSourceSpecificSettingFormItems = [ | ||||
|     sourceConfigSettingFormItems = [ | ||||
|       GeneratedFormTextField('gitlab-creds', | ||||
|           label: tr('gitlabPATLabel'), | ||||
|           password: true, | ||||
| @@ -60,17 +60,27 @@ class GitLab extends AppSource { | ||||
|     return url.substring(0, match.end); | ||||
|   } | ||||
|  | ||||
|   Future<String?> getPATIfAny() async { | ||||
|   Future<String?> getPATIfAny(Map<String, dynamic> additionalSettings) async { | ||||
|     SettingsProvider settingsProvider = SettingsProvider(); | ||||
|     await settingsProvider.initializeSettings(); | ||||
|     String? creds = settingsProvider | ||||
|         .getSettingString(additionalSourceSpecificSettingFormItems[0].key); | ||||
|     var sourceConfig = | ||||
|         await getSourceConfigValues(additionalSettings, settingsProvider); | ||||
|     String? creds = sourceConfig['gitlab-creds']; | ||||
|     return creds != null && creds.isNotEmpty ? creds : null; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, List<String>>> search(String query) async { | ||||
|     String? PAT = await getPATIfAny(); | ||||
|   Future<String?> getSourceNote() async { | ||||
|     if ((await getPATIfAny({})) == null) { | ||||
|       return '${tr('gitlabSourceNote')} ${hostChanged ? tr('addInfoBelow') : tr('addInfoInSettings')}'; | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, List<String>>> search(String query, | ||||
|       {Map<String, dynamic> querySettings = const {}}) async { | ||||
|     String? PAT = await getPATIfAny({}); | ||||
|     if (PAT == null) { | ||||
|       throw CredsNeededError(name); | ||||
|     } | ||||
| @@ -102,7 +112,7 @@ class GitLab extends AppSource { | ||||
|   ) async { | ||||
|     bool fallbackToOlderReleases = | ||||
|         additionalSettings['fallbackToOlderReleases'] == true; | ||||
|     String? PAT = await getPATIfAny(); | ||||
|     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); | ||||
|     Iterable<APKDetails> apkDetailsList = []; | ||||
|     if (PAT != null) { | ||||
|       var names = GitHub().getAppNames(standardUrl); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| @@ -85,6 +86,15 @@ bool _isNumeric(String s) { | ||||
| } | ||||
|  | ||||
| class HTML extends AppSource { | ||||
|   HTML() { | ||||
|     additionalSourceAppSpecificSettingFormItems = [ | ||||
|       [ | ||||
|         GeneratedFormSwitch('sortByFileNamesNotLinks', | ||||
|             label: tr('sortByFileNamesNotLinks')) | ||||
|       ] | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   // TODO: implement requestHeaders choice, hardcoded for now | ||||
|   Map<String, String>? get requestHeaders => { | ||||
| @@ -111,7 +121,9 @@ class HTML extends AppSource { | ||||
|           .where((element) => | ||||
|               Uri.parse(element).path.toLowerCase().endsWith('.apk')) | ||||
|           .toList(); | ||||
|       links.sort((a, b) => compareAlphaNumeric(a, b)); | ||||
|       links.sort((a, b) => additionalSettings['sortByFileNamesNotLinks'] == true | ||||
|           ? compareAlphaNumeric(a.split('/').last, b.split('/').last) | ||||
|           : compareAlphaNumeric(a, b)); | ||||
|       if (additionalSettings['apkFilterRegEx'] != null) { | ||||
|         var reg = RegExp(additionalSettings['apkFilterRegEx']); | ||||
|         links = links.where((element) => reg.hasMatch(element)).toList(); | ||||
|   | ||||
| @@ -38,16 +38,14 @@ class VLC extends AppSource { | ||||
|       } | ||||
|       String? targetUrl = 'https://$dwUrlBase/$version/'; | ||||
|       Response res2 = await sourceRequest(targetUrl); | ||||
|       String mirrorDwBase = | ||||
|           'https://plug-mirror.rcac.purdue.edu/vlc/vlc-android/$version/'; | ||||
|       List<String> apkUrls = []; | ||||
|       if (res2.statusCode == 200) { | ||||
|         apkUrls = parse(res2.body) | ||||
|             .querySelectorAll('a') | ||||
|             .map((e) => e.attributes['href']) | ||||
|             .map((e) => e.attributes['href']?.split('/').last) | ||||
|             .where((h) => | ||||
|                 h != null && h.isNotEmpty && h.toLowerCase().endsWith('.apk')) | ||||
|             .map((e) => mirrorDwBase + e!) | ||||
|             .map((e) => targetUrl + e!) | ||||
|             .toList(); | ||||
|       } else { | ||||
|         throw getObtainiumHttpError(res2); | ||||
| @@ -59,4 +57,20 @@ class VLC extends AppSource { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String> apkUrlPrefetchModifier( | ||||
|       String apkUrl, String standardUrl) async { | ||||
|     Response res = await sourceRequest(apkUrl); | ||||
|     if (res.statusCode == 200) { | ||||
|       String? apkUrl = | ||||
|           parse(res.body).querySelector('#alt_link')?.attributes['href']; | ||||
|       if (apkUrl == null) { | ||||
|         throw NoAPKError(); | ||||
|       } | ||||
|       return apkUrl; | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import 'dart:math'; | ||||
|  | ||||
| import 'package:hsluv/hsluv.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:obtainium/components/generated_form_modal.dart'; | ||||
| @@ -132,19 +133,19 @@ class GeneratedForm extends StatefulWidget { | ||||
|   State<GeneratedForm> createState() => _GeneratedFormState(); | ||||
| } | ||||
|  | ||||
| // Generates a random light color | ||||
| // Courtesy of ChatGPT 😭 (with a bugfix 🥳) | ||||
| // Generates a color in the HSLuv (Pastel) color space | ||||
| // https://pub.dev/documentation/hsluv/latest/hsluv/Hsluv/hpluvToRgb.html | ||||
| Color generateRandomLightColor() { | ||||
|   // Create a random number generator | ||||
|   final Random random = Random(); | ||||
|  | ||||
|   // Generate random hue, saturation, and value values | ||||
|   final double hue = random.nextDouble() * 360; | ||||
|   final double saturation = 0.5 + random.nextDouble() * 0.5; | ||||
|   final double value = 0.9 + random.nextDouble() * 0.1; | ||||
|  | ||||
|   // Create a HSV color with the random values | ||||
|   return HSVColor.fromAHSV(1.0, hue, saturation, value).toColor(); | ||||
|   final randomSeed = Random().nextInt(120); | ||||
|   // https://en.wikipedia.org/wiki/Golden_angle | ||||
|   final goldenAngle = 180 * (3 - sqrt(5)); | ||||
|   // Generate next golden angle hue | ||||
|   final double hue = randomSeed * goldenAngle; | ||||
|   // Map from HPLuv color space to RGB, use constant saturation=100, lightness=70 | ||||
|   final List<double> rgbValuesDbl = Hsluv.hpluvToRgb([hue, 100, 70]); | ||||
|   // Map RBG values from 0-1 to 0-255: | ||||
|   final List<int> rgbValues = rgbValuesDbl.map((rgb) => (rgb * 255).toInt()).toList(); | ||||
|   return Color.fromARGB(255, rgbValues[0], rgbValues[1], rgbValues[2]); | ||||
| } | ||||
|  | ||||
| class _GeneratedFormState extends State<GeneratedForm> { | ||||
| @@ -368,6 +369,36 @@ class _GeneratedFormState extends State<GeneratedForm> { | ||||
|                           )); | ||||
|                     }) ?? | ||||
|                     [const SizedBox.shrink()], | ||||
|                 (values[widget.items[r][e].key] | ||||
|                 as Map<String, MapEntry<int, bool>>?) | ||||
|                     ?.values | ||||
|                     .where((e) => e.value) | ||||
|                     .length == 1 | ||||
|                     ? Padding( | ||||
|                     padding: const EdgeInsets.symmetric(horizontal: 4), | ||||
|                     child: IconButton( | ||||
|                       onPressed: () { | ||||
|                         setState(() { | ||||
|                           var temp = values[widget.items[r][e].key] | ||||
|                               as Map<String, MapEntry<int, bool>>; | ||||
|                           // get selected category str where bool is true | ||||
|                           final oldEntry = temp.entries.firstWhere((entry) => entry.value.value); | ||||
|                           // generate new color, ensure it is not the same | ||||
|                           int newColor = oldEntry.value.key; | ||||
|                           while(oldEntry.value.key == newColor) { | ||||
|                             newColor = generateRandomLightColor().value; | ||||
|                           } | ||||
|                           // Update entry with new color, remain selected | ||||
|                           temp.update(oldEntry.key, (old) => MapEntry(newColor, old.value)); | ||||
|                           values[widget.items[r][e].key] = temp; | ||||
|                           someValueChanged(); | ||||
|                         }); | ||||
|                       }, | ||||
|                       icon: const Icon(Icons.format_color_fill_rounded), | ||||
|                       visualDensity: VisualDensity.compact, | ||||
|                       tooltip: tr('colour'), | ||||
|                     )) | ||||
|                     : const SizedBox.shrink(), | ||||
|                 (values[widget.items[r][e].key] | ||||
|                                 as Map<String, MapEntry<int, bool>>?) | ||||
|                             ?.values | ||||
|   | ||||
| @@ -11,7 +11,8 @@ class GeneratedFormModal extends StatefulWidget { | ||||
|       this.initValid = false, | ||||
|       this.message = '', | ||||
|       this.additionalWidgets = const [], | ||||
|       this.singleNullReturnButton}); | ||||
|       this.singleNullReturnButton, | ||||
|       this.primaryActionColour}); | ||||
|  | ||||
|   final String title; | ||||
|   final String message; | ||||
| @@ -19,6 +20,7 @@ class GeneratedFormModal extends StatefulWidget { | ||||
|   final bool initValid; | ||||
|   final List<Widget> additionalWidgets; | ||||
|   final String? singleNullReturnButton; | ||||
|   final Color? primaryActionColour; | ||||
|  | ||||
|   @override | ||||
|   State<GeneratedFormModal> createState() => _GeneratedFormModalState(); | ||||
| @@ -71,6 +73,10 @@ class _GeneratedFormModalState extends State<GeneratedFormModal> { | ||||
|                 : widget.singleNullReturnButton!)), | ||||
|         widget.singleNullReturnButton == null | ||||
|             ? TextButton( | ||||
|                 style: widget.primaryActionColour == null | ||||
|                     ? null | ||||
|                     : TextButton.styleFrom( | ||||
|                         foregroundColor: widget.primaryActionColour), | ||||
|                 onPressed: !valid | ||||
|                     ? null | ||||
|                     : () { | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import 'package:android_package_installer/android_package_installer.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:obtainium/providers/logs_provider.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| @@ -101,7 +102,14 @@ showError(dynamic e, BuildContext context) { | ||||
|             title: Text(e is MultiAppMultiError | ||||
|                 ? tr('someErrors') | ||||
|                 : tr('unexpectedError')), | ||||
|             content: Text(e.toString()), | ||||
|             content: GestureDetector( | ||||
|                 onLongPress: () { | ||||
|                   Clipboard.setData(ClipboardData(text: e.toString())); | ||||
|                   ScaffoldMessenger.of(context).showSnackBar(SnackBar( | ||||
|                     content: Text(tr('copiedToClipboard')), | ||||
|                   )); | ||||
|                 }, | ||||
|                 child: Text(e.toString())), | ||||
|             actions: [ | ||||
|               TextButton( | ||||
|                   onPressed: () { | ||||
|   | ||||
| @@ -3,6 +3,7 @@ import 'dart:math'; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/pages/home.dart'; | ||||
| import 'package:obtainium/providers/apps_provider.dart'; | ||||
| @@ -21,7 +22,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart'; | ||||
| // ignore: implementation_imports | ||||
| import 'package:easy_localization/src/localization.dart'; | ||||
|  | ||||
| const String currentVersion = '0.13.16'; | ||||
| const String currentVersion = '0.13.23'; | ||||
| const String currentReleaseTag = | ||||
|     'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES | ||||
|  | ||||
| @@ -39,6 +40,7 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | ||||
|   MapEntry(Locale('es'), 'Español'), | ||||
|   MapEntry(Locale('pl'), 'Polski'), | ||||
|   MapEntry(Locale('ru'), 'Русский язык'), | ||||
|   MapEntry(Locale('bs'), 'Bosanski'), | ||||
| ]; | ||||
| const fallbackLocale = Locale('en'); | ||||
| const localeDir = 'assets/translations'; | ||||
| @@ -100,10 +102,15 @@ Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async { | ||||
|       await appsProvider.checkUpdates( | ||||
|           ignoreAppsCheckedAfter: ignoreAfter, throwErrorsForRetry: true); | ||||
|     } catch (e) { | ||||
|       if (e is RateLimitError || e is SocketException) { | ||||
|       if (e is RateLimitError || e is ClientException) { | ||||
|         var remainingMinutes = e is RateLimitError ? e.remainingMinutes : 15; | ||||
|         logs.add(plural('bgUpdateGotErrorRetryInMinutes', remainingMinutes, | ||||
|             args: [e.toString(), remainingMinutes.toString()])); | ||||
|         logs.add( | ||||
|             plural('bgUpdateGotErrorRetryInMinutes', remainingMinutes, args: [ | ||||
|           e is ClientException | ||||
|               ? '${(e).message}, ${e.uri?.path}' | ||||
|               : e.toString(), | ||||
|           remainingMinutes.toString() | ||||
|         ])); | ||||
|         AndroidAlarmManager.oneShot(Duration(minutes: remainingMinutes), | ||||
|             Random().nextInt(pow(2, 31) as int), bgUpdateCheck, params: { | ||||
|           'ignoreAfterMicroseconds': nextIgnoreAfter.microsecondsSinceEpoch | ||||
|   | ||||
| @@ -16,7 +16,7 @@ class GitHubStars implements MassAppUrlSource { | ||||
|   Future<Map<String, List<String>>> getOnePageOfUserStarredUrlsWithDescriptions( | ||||
|       String username, int page) async { | ||||
|     Response res = await get(Uri.parse( | ||||
|         'https://${await GitHub().getCredentialPrefixIfAny()}api.github.com/users/$username/starred?per_page=100&page=$page')); | ||||
|         'https://${await GitHub().getCredentialPrefixIfAny({})}api.github.com/users/$username/starred?per_page=100&page=$page')); | ||||
|     if (res.statusCode == 200) { | ||||
|       Map<String, List<String>> urlsWithDescriptions = {}; | ||||
|       for (var e in (jsonDecode(res.body) as List<dynamic>)) { | ||||
|   | ||||
| @@ -41,6 +41,7 @@ class _AddAppPageState extends State<AddAppPage> { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     AppsProvider appsProvider = context.read<AppsProvider>(); | ||||
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); | ||||
|  | ||||
|     bool doingSomething = gettingAppInfo || searching; | ||||
|  | ||||
| @@ -85,8 +86,7 @@ class _AddAppPageState extends State<AddAppPage> { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     Future<bool> getTrackOnlyConfirmationIfNeeded( | ||||
|         bool userPickedTrackOnly, SettingsProvider settingsProvider, | ||||
|     Future<bool> getTrackOnlyConfirmationIfNeeded(bool userPickedTrackOnly, | ||||
|         {bool ignoreHideSetting = false}) async { | ||||
|       var useTrackOnly = userPickedTrackOnly || pickedSource!.enforceTrackOnly; | ||||
|       if (useTrackOnly && | ||||
| @@ -138,11 +138,9 @@ class _AddAppPageState extends State<AddAppPage> { | ||||
|         gettingAppInfo = true; | ||||
|       }); | ||||
|       try { | ||||
|         var settingsProvider = context.read<SettingsProvider>(); | ||||
|         var userPickedTrackOnly = additionalSettings['trackOnly'] == true; | ||||
|         App? app; | ||||
|         if ((await getTrackOnlyConfirmationIfNeeded( | ||||
|                 userPickedTrackOnly, settingsProvider)) && | ||||
|         if ((await getTrackOnlyConfirmationIfNeeded(userPickedTrackOnly)) && | ||||
|             (await getReleaseDateAsVersionConfirmationIfNeeded( | ||||
|                 userPickedTrackOnly))) { | ||||
|           var trackOnly = pickedSource!.enforceTrackOnly || userPickedTrackOnly; | ||||
| @@ -410,7 +408,13 @@ class _AddAppPageState extends State<AddAppPage> { | ||||
|             ), | ||||
|             GeneratedForm( | ||||
|                 key: Key(pickedSource.runtimeType.toString()), | ||||
|                 items: pickedSource!.combinedAppSpecificSettingFormItems, | ||||
|                 items: [ | ||||
|                   ...pickedSource!.combinedAppSpecificSettingFormItems, | ||||
|                   ...(pickedSourceOverride != null | ||||
|                       ? pickedSource!.sourceConfigSettingFormItems | ||||
|                           .map((e) => [e]) | ||||
|                       : []) | ||||
|                 ], | ||||
|                 onValueChanges: (values, valid, isBuilding) { | ||||
|                   if (!isBuilding) { | ||||
|                     setState(() { | ||||
| @@ -504,6 +508,18 @@ class _AddAppPageState extends State<AddAppPage> { | ||||
|                                   HTML().runtimeType.toString())) | ||||
|                         getHTMLSourceOverrideDropdown(), | ||||
|                       if (shouldShowSearchBar()) getSearchBarRow(), | ||||
|                       if (pickedSource != null) | ||||
|                         FutureBuilder( | ||||
|                             builder: (ctx, val) { | ||||
|                               return val.data != null && val.data!.isNotEmpty | ||||
|                                   ? Text( | ||||
|                                       val.data!, | ||||
|                                       style: | ||||
|                                           Theme.of(context).textTheme.bodySmall, | ||||
|                                     ) | ||||
|                                   : const SizedBox(); | ||||
|                             }, | ||||
|                             future: pickedSource?.getSourceNote()), | ||||
|                       const SizedBox( | ||||
|                         height: 16, | ||||
|                       ), | ||||
|   | ||||
| @@ -57,6 +57,11 @@ class _AppPageState extends State<AppPage> { | ||||
|         app?.app.additionalSettings['versionDetection'] == | ||||
|             'standardVersionDetection'; | ||||
|  | ||||
|     bool installedVersionIsEstimate = trackOnly || | ||||
|         (app?.app.installedVersion != null && | ||||
|             app?.app.additionalSettings['versionDetection'] == | ||||
|                 'noVersionDetection'); | ||||
|  | ||||
|     getInfoColumn() => Column( | ||||
|           mainAxisAlignment: MainAxisAlignment.center, | ||||
|           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
| @@ -92,9 +97,7 @@ class _AppPageState extends State<AppPage> { | ||||
|                         app?.app.latestVersion ?? tr('unknown') | ||||
|                       ])}\n${tr('installedVersionX', args: [ | ||||
|                         app?.app.installedVersion ?? tr('none') | ||||
|                       ])}${trackOnly ? ' ${tr('estimateInBrackets')}\n\n${tr('xIsTrackOnly', args: [ | ||||
|                           tr('app') | ||||
|                         ])}' : ''}', | ||||
|                       ])}${installedVersionIsEstimate ? '\n${tr('estimateInBrackets')}' : ''}', | ||||
|                   textAlign: TextAlign.end, | ||||
|                   style: Theme.of(context).textTheme.bodyLarge!, | ||||
|                 ), | ||||
| @@ -105,11 +108,14 @@ class _AppPageState extends State<AppPage> { | ||||
|               Column( | ||||
|                 children: [ | ||||
|                   const SizedBox( | ||||
|                     height: 4, | ||||
|                     height: 16, | ||||
|                   ), | ||||
|                   Text( | ||||
|                     tr('noVersionDetection'), | ||||
|                     '${trackOnly ? '${tr('xIsTrackOnly', args: [ | ||||
|                             tr('app') | ||||
|                           ])}\n' : ''}${tr('noVersionDetection')}', | ||||
|                     style: Theme.of(context).textTheme.labelSmall, | ||||
|                     textAlign: TextAlign.center, | ||||
|                   ) | ||||
|                 ], | ||||
|               ), | ||||
| @@ -425,8 +431,7 @@ class _AppPageState extends State<AppPage> { | ||||
|                               ? getResetInstallStatusButton() | ||||
|                               : getInstallOrUpdateButton()), | ||||
|                       const SizedBox(width: 16.0), | ||||
|                       Expanded( | ||||
|                           child: TextButton( | ||||
|                       IconButton( | ||||
|                         onPressed: app?.downloadProgress != null | ||||
|                             ? null | ||||
|                             : () { | ||||
| @@ -439,13 +444,9 @@ class _AppPageState extends State<AppPage> { | ||||
|                                   } | ||||
|                                 }); | ||||
|                               }, | ||||
|                         style: TextButton.styleFrom( | ||||
|                             foregroundColor: | ||||
|                                 Theme.of(context).colorScheme.error, | ||||
|                             surfaceTintColor: | ||||
|                                 Theme.of(context).colorScheme.error), | ||||
|                         child: Text(tr('remove')), | ||||
|                       )), | ||||
|                         tooltip: tr('remove'), | ||||
|                         icon: const Icon(Icons.delete_outline), | ||||
|                       ), | ||||
|                     ])), | ||||
|             if (app?.downloadProgress != null) | ||||
|               Padding( | ||||
|   | ||||
| @@ -543,20 +543,19 @@ class AppsPageState extends State<AppsPage> { | ||||
|                         : FontWeight.normal)), | ||||
|             trailing: listedApps[index].downloadProgress != null | ||||
|                 ? SizedBox( | ||||
|                     width: 90, | ||||
|                     child: Text( | ||||
|                       listedApps[index].downloadProgress! >= 0 | ||||
|                           ? tr('percentProgress', args: [ | ||||
|                               listedApps[index] | ||||
|                                   .downloadProgress! | ||||
|                                   .toInt() | ||||
|                                   .toString() | ||||
|                             ]) | ||||
|                           : tr('pleaseWait'), | ||||
|                       textAlign: (listedApps[index].downloadProgress! >= 0) | ||||
|                           ? TextAlign.start | ||||
|                           : TextAlign.end, | ||||
|                     )) | ||||
|                     listedApps[index].downloadProgress! >= 0 | ||||
|                         ? tr('percentProgress', args: [ | ||||
|                             listedApps[index] | ||||
|                                 .downloadProgress! | ||||
|                                 .toInt() | ||||
|                                 .toString() | ||||
|                           ]) | ||||
|                         : tr('pleaseWait'), | ||||
|                     textAlign: (listedApps[index].downloadProgress! >= 0) | ||||
|                         ? TextAlign.start | ||||
|                         : TextAlign.end, | ||||
|                   )) | ||||
|                 : trailingRow, | ||||
|             onTap: () { | ||||
|               if (selectedAppIds.isNotEmpty) { | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import 'package:obtainium/pages/apps.dart'; | ||||
| import 'package:obtainium/pages/import_export.dart'; | ||||
| import 'package:obtainium/pages/settings.dart'; | ||||
| import 'package:obtainium/providers/apps_provider.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class HomePage extends StatefulWidget { | ||||
| @@ -26,6 +27,7 @@ class NavigationPageItem { | ||||
|  | ||||
| class _HomePageState extends State<HomePage> { | ||||
|   List<int> selectedIndexHistory = []; | ||||
|   bool isReversing = false; | ||||
|   int prevAppCount = -1; | ||||
|   bool prevIsLoading = true; | ||||
|  | ||||
| @@ -41,8 +43,18 @@ class _HomePageState extends State<HomePage> { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     AppsProvider appsProvider = context.watch<AppsProvider>(); | ||||
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); | ||||
|  | ||||
|     setIsReversing(int targetIndex) { | ||||
|       bool reversing = selectedIndexHistory.isNotEmpty && | ||||
|           selectedIndexHistory.last > targetIndex; | ||||
|       setState(() { | ||||
|         isReversing = reversing; | ||||
|       }); | ||||
|     } | ||||
|  | ||||
|     switchToPage(int index) async { | ||||
|       setIsReversing(index); | ||||
|       if (index == 0) { | ||||
|         while ((pages[0].widget.key as GlobalKey<AppsPageState>).currentState != | ||||
|             null) { | ||||
| @@ -79,6 +91,12 @@ class _HomePageState extends State<HomePage> { | ||||
|         child: Scaffold( | ||||
|           backgroundColor: Theme.of(context).colorScheme.surface, | ||||
|           body: PageTransitionSwitcher( | ||||
|             duration: Duration( | ||||
|                 milliseconds: | ||||
|                     settingsProvider.disablePageTransitions ? 0 : 300), | ||||
|             reverse: settingsProvider.reversePageTransitions | ||||
|                 ? !isReversing | ||||
|                 : isReversing, | ||||
|             transitionBuilder: ( | ||||
|               Widget child, | ||||
|               Animation<double> animation, | ||||
| @@ -104,13 +122,16 @@ class _HomePageState extends State<HomePage> { | ||||
|                 .toList(), | ||||
|             onDestinationSelected: (int index) async { | ||||
|               HapticFeedback.selectionClick(); | ||||
|               await switchToPage(index); | ||||
|               switchToPage(index); | ||||
|             }, | ||||
|             selectedIndex: | ||||
|                 selectedIndexHistory.isEmpty ? 0 : selectedIndexHistory.last, | ||||
|           ), | ||||
|         ), | ||||
|         onWillPop: () async { | ||||
|           setIsReversing(selectedIndexHistory.length >= 2 | ||||
|               ? selectedIndexHistory.reversed.toList()[1] | ||||
|               : 0); | ||||
|           if (selectedIndexHistory.isNotEmpty) { | ||||
|             setState(() { | ||||
|               selectedIndexHistory.removeLast(); | ||||
|   | ||||
| @@ -182,7 +182,8 @@ class _ImportExportPageState extends State<ImportExportPage> { | ||||
|                   [ | ||||
|                     GeneratedFormTextField('searchQuery', | ||||
|                         label: tr('searchQuery')) | ||||
|                   ] | ||||
|                   ], | ||||
|                   ...source.searchQuerySettingFormItems.map((e) => [e]) | ||||
|                 ], | ||||
|               ); | ||||
|             }); | ||||
| @@ -191,8 +192,8 @@ class _ImportExportPageState extends State<ImportExportPage> { | ||||
|           setState(() { | ||||
|             importInProgress = true; | ||||
|           }); | ||||
|           var urlsWithDescriptions = | ||||
|               await source.search(values['searchQuery'] as String); | ||||
|           var urlsWithDescriptions = await source | ||||
|               .search(values['searchQuery'] as String, querySettings: values); | ||||
|           if (urlsWithDescriptions.isNotEmpty) { | ||||
|             var selectedUrls = | ||||
|                 // ignore: use_build_context_synchronously | ||||
|   | ||||
| @@ -1,5 +1,3 @@ | ||||
| import 'dart:math'; | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:obtainium/components/custom_app_bar.dart'; | ||||
| @@ -21,21 +19,6 @@ class SettingsPage extends StatefulWidget { | ||||
|   State<SettingsPage> createState() => _SettingsPageState(); | ||||
| } | ||||
|  | ||||
| // Generates a random light color | ||||
| // Courtesy of ChatGPT 😭 (with a bugfix 🥳) | ||||
| Color generateRandomLightColor() { | ||||
|   // Create a random number generator | ||||
|   final Random random = Random(); | ||||
|  | ||||
|   // Generate random hue, saturation, and value values | ||||
|   final double hue = random.nextDouble() * 360; | ||||
|   final double saturation = 0.5 + random.nextDouble() * 0.5; | ||||
|   final double value = 0.9 + random.nextDouble() * 0.1; | ||||
|  | ||||
|   // Create a HSV color with the random values | ||||
|   return HSVColor.fromAHSV(1.0, hue, saturation, value).toColor(); | ||||
| } | ||||
|  | ||||
| class _SettingsPageState extends State<SettingsPage> { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
| @@ -183,9 +166,9 @@ class _SettingsPageState extends State<SettingsPage> { | ||||
|         }); | ||||
|  | ||||
|     var sourceSpecificFields = sourceProvider.sources.map((e) { | ||||
|       if (e.additionalSourceSpecificSettingFormItems.isNotEmpty) { | ||||
|       if (e.sourceConfigSettingFormItems.isNotEmpty) { | ||||
|         return GeneratedForm( | ||||
|             items: e.additionalSourceSpecificSettingFormItems.map((e) { | ||||
|             items: e.sourceConfigSettingFormItems.map((e) { | ||||
|               e.defaultValue = settingsProvider.getSettingString(e.key); | ||||
|               return [e]; | ||||
|             }).toList(), | ||||
| @@ -396,6 +379,39 @@ class _SettingsPageState extends State<SettingsPage> { | ||||
|                                     }) | ||||
|                               ], | ||||
|                             ), | ||||
|                             height16, | ||||
|                             Row( | ||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                               children: [ | ||||
|                                 Flexible( | ||||
|                                     child: Text(tr('disablePageTransitions'))), | ||||
|                                 Switch( | ||||
|                                     value: | ||||
|                                         settingsProvider.disablePageTransitions, | ||||
|                                     onChanged: (value) { | ||||
|                                       settingsProvider.disablePageTransitions = | ||||
|                                           value; | ||||
|                                     }) | ||||
|                               ], | ||||
|                             ), | ||||
|                             height16, | ||||
|                             Row( | ||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                               children: [ | ||||
|                                 Flexible( | ||||
|                                     child: Text(tr('reversePageTransitions'))), | ||||
|                                 Switch( | ||||
|                                     value: | ||||
|                                         settingsProvider.reversePageTransitions, | ||||
|                                     onChanged: settingsProvider | ||||
|                                             .disablePageTransitions | ||||
|                                         ? null | ||||
|                                         : (value) { | ||||
|                                             settingsProvider | ||||
|                                                 .reversePageTransitions = value; | ||||
|                                           }) | ||||
|                               ], | ||||
|                             ), | ||||
|                             height32, | ||||
|                             Text( | ||||
|                               tr('categories'), | ||||
|   | ||||
| @@ -117,7 +117,7 @@ class AppsProvider with ChangeNotifier { | ||||
|     foregroundStream = FGBGEvents.stream.asBroadcastStream(); | ||||
|     foregroundSubscription = foregroundStream?.listen((event) async { | ||||
|       isForeground = event == FGBGType.foreground; | ||||
|       if (isForeground) await refreshInstallStatuses(); | ||||
|       if (isForeground) await loadApps(); | ||||
|     }); | ||||
|     () async { | ||||
|       var cacheDirs = await getExternalCacheDirectories(); | ||||
| @@ -144,6 +144,25 @@ class AppsProvider with ChangeNotifier { | ||||
|     }(); | ||||
|   } | ||||
|  | ||||
|   Future<File> downloadFileWithRetry( | ||||
|       String url, String fileNameNoExt, Function? onProgress, | ||||
|       {bool useExisting = true, | ||||
|       Map<String, String>? headers, | ||||
|       int retries = 3}) async { | ||||
|     try { | ||||
|       return await downloadFile(url, fileNameNoExt, onProgress, | ||||
|           useExisting: useExisting, headers: headers); | ||||
|     } catch (e) { | ||||
|       if (retries > 0 && e is ClientException) { | ||||
|         await Future.delayed(const Duration(seconds: 5)); | ||||
|         return await downloadFileWithRetry(url, fileNameNoExt, onProgress, | ||||
|             useExisting: useExisting, headers: headers, retries: (retries - 1)); | ||||
|       } else { | ||||
|         rethrow; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<File> downloadFile( | ||||
|       String url, String fileNameNoExt, Function? onProgress, | ||||
|       {bool useExisting = true, Map<String, String>? headers}) async { | ||||
| @@ -236,8 +255,9 @@ class AppsProvider with ChangeNotifier { | ||||
|       notificationsProvider?.cancel(notif.id); | ||||
|       int? prevProg; | ||||
|       var fileNameNoExt = '${app.id}-${downloadUrl.hashCode}'; | ||||
|       var downloadedFile = await downloadFile(downloadUrl, fileNameNoExt, | ||||
|           headers: source.requestHeaders, (double? progress) { | ||||
|       var downloadedFile = await downloadFileWithRetry( | ||||
|           downloadUrl, fileNameNoExt, headers: source.requestHeaders, | ||||
|           (double? progress) { | ||||
|         int? prog = progress?.ceil(); | ||||
|         if (apps[app.id] != null) { | ||||
|           apps[app.id]!.downloadProgress = progress; | ||||
| @@ -341,20 +361,33 @@ class AppsProvider with ChangeNotifier { | ||||
|  | ||||
|   Future<void> installXApkDir(DownloadedXApkDir dir, | ||||
|       {bool silent = false}) async { | ||||
|     // We don't know which APKs in an XAPK are supported by the user's device | ||||
|     // So we try installing all of them and assume success if at least one installed | ||||
|     // If 0 APKs installed, throw the first install error encountered | ||||
|     try { | ||||
|       var somethingInstalled = false; | ||||
|       Object? firstError; | ||||
|       for (var file in dir.extracted | ||||
|           .listSync(recursive: true, followLinks: false) | ||||
|           .whereType<File>()) { | ||||
|         if (file.path.toLowerCase().endsWith('.apk')) { | ||||
|           somethingInstalled = somethingInstalled || | ||||
|               await installApk(DownloadedApk(dir.appId, file), silent: silent); | ||||
|           try { | ||||
|             somethingInstalled = somethingInstalled || | ||||
|                 await installApk(DownloadedApk(dir.appId, file), | ||||
|                     silent: silent); | ||||
|           } catch (e) { | ||||
|             logs.add( | ||||
|                 'Could not install APK from XAPK \'${file.path}\': ${e.toString()}'); | ||||
|             firstError ??= e; | ||||
|           } | ||||
|         } else if (file.path.toLowerCase().endsWith('.obb')) { | ||||
|           await moveObbFile(file, dir.appId); | ||||
|         } | ||||
|       } | ||||
|       if (somethingInstalled) { | ||||
|         dir.file.delete(recursive: true); | ||||
|       } else if (firstError != null) { | ||||
|         throw firstError; | ||||
|       } | ||||
|     } finally { | ||||
|       dir.extracted.delete(recursive: true); | ||||
| @@ -725,21 +758,12 @@ class AppsProvider with ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|     var sp = SourceProvider(); | ||||
|     List<List<String>> errors = []; | ||||
|     List<FileSystemEntity> newApps = (await getAppsDir()) | ||||
|     List<App?> newApps = (await getAppsDir()) // Parse Apps from JSON | ||||
|         .listSync() | ||||
|         .where((item) => item.path.toLowerCase().endsWith('.json')) | ||||
|         .toList(); | ||||
|     for (var e in newApps) { | ||||
|         .map((e) { | ||||
|       try { | ||||
|         var app = App.fromJson(jsonDecode(File(e.path).readAsStringSync())); | ||||
|         try { | ||||
|           var info = await getInstalledInfo(app.id); | ||||
|           sp.getSource(app.url, overrideSource: app.overrideSource); | ||||
|           apps[app.id] = AppInMemory(app, null, info); | ||||
|           notifyListeners(); | ||||
|         } catch (e) { | ||||
|           errors.add([app.id, app.finalName, e.toString()]); | ||||
|         } | ||||
|         return App.fromJson(jsonDecode(File(e.path).readAsStringSync())); | ||||
|       } catch (err) { | ||||
|         if (err is FormatException) { | ||||
|           logs.add('Corrupt JSON when loading App (will be ignored): $e'); | ||||
| @@ -748,28 +772,40 @@ class AppsProvider with ChangeNotifier { | ||||
|           rethrow; | ||||
|         } | ||||
|       } | ||||
|     }).toList(); | ||||
|     for (var app in newApps) { | ||||
|       // Put Apps into memory to list them (fast) | ||||
|       if (app != null) { | ||||
|         try { | ||||
|           sp.getSource(app.url, overrideSource: app.overrideSource); | ||||
|           apps.update( | ||||
|               app.id, | ||||
|               (value) => | ||||
|                   AppInMemory(app, value.downloadProgress, value.installedInfo), | ||||
|               ifAbsent: () => AppInMemory(app, null, null)); | ||||
|         } catch (e) { | ||||
|           errors.add([app.id, app.finalName, e.toString()]); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     notifyListeners(); | ||||
|     if (errors.isNotEmpty) { | ||||
|       removeApps(errors.map((e) => e[0]).toList()); | ||||
|       NotificationsProvider().notify( | ||||
|           AppsRemovedNotification(errors.map((e) => [e[1], e[2]]).toList())); | ||||
|     } | ||||
|     loadingApps = false; | ||||
|     notifyListeners(); | ||||
|  | ||||
|     refreshInstallStatuses(useExistingInstalledInfo: true); | ||||
|   } | ||||
|  | ||||
|   Future<void> refreshInstallStatuses( | ||||
|       {bool useExistingInstalledInfo = false}) async { | ||||
|     if (await doesInstalledAppsPluginWork()) { | ||||
|       for (var app in apps.values) { | ||||
|         // Check install status for each App (slow) | ||||
|         apps[app.app.id]?.installedInfo = await getInstalledInfo(app.app.id); | ||||
|         notifyListeners(); | ||||
|       } | ||||
|       // Reconcile version differences | ||||
|       List<App> modifiedApps = []; | ||||
|       for (var app in apps.values) { | ||||
|         var moddedApp = getCorrectedInstallStatusAppIfPossible( | ||||
|             app.app, | ||||
|             useExistingInstalledInfo | ||||
|                 ? app.installedInfo | ||||
|                 : await getInstalledInfo(app.app.id)); | ||||
|         var moddedApp = | ||||
|             getCorrectedInstallStatusAppIfPossible(app.app, app.installedInfo); | ||||
|         if (moddedApp != null) { | ||||
|           modifiedApps.add(moddedApp); | ||||
|         } | ||||
| @@ -780,6 +816,7 @@ class AppsProvider with ChangeNotifier { | ||||
|             .where((a) => a.installedVersion == null) | ||||
|             .map((e) => e.id) | ||||
|             .toList(); | ||||
|         // After reconciliation, delete externally uninstalled Apps if needed | ||||
|         if (removedAppIds.isNotEmpty) { | ||||
|           var settingsProvider = SettingsProvider(); | ||||
|           await settingsProvider.initializeSettings(); | ||||
| @@ -789,6 +826,9 @@ class AppsProvider with ChangeNotifier { | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     loadingApps = false; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> saveApps(List<App> apps, | ||||
| @@ -852,6 +892,7 @@ class AppsProvider with ChangeNotifier { | ||||
|         context: context, | ||||
|         builder: (BuildContext ctx) { | ||||
|           return GeneratedFormModal( | ||||
|             primaryActionColour: Theme.of(context).colorScheme.error, | ||||
|             title: plural('removeAppQuestion', apps.length), | ||||
|             items: !showUninstallOption | ||||
|                 ? [] | ||||
|   | ||||
| @@ -291,4 +291,22 @@ class SettingsProvider with ChangeNotifier { | ||||
|     prefs?.setBool('checkUpdateOnDetailPage', show); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   bool get disablePageTransitions { | ||||
|     return prefs?.getBool('disablePageTransitions') ?? false; | ||||
|   } | ||||
|  | ||||
|   set disablePageTransitions(bool show) { | ||||
|     prefs?.setBool('disablePageTransitions', show); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   bool get reversePageTransitions { | ||||
|     return prefs?.getBool('reversePageTransitions') ?? false; | ||||
|   } | ||||
|  | ||||
|   set reversePageTransitions(bool show) { | ||||
|     prefs?.setBool('reversePageTransitions', show); | ||||
|     notifyListeners(); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -28,6 +28,7 @@ import 'package:obtainium/app_sources/vlc.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/mass_app_sources/githubstars.dart'; | ||||
| import 'package:obtainium/providers/settings_provider.dart'; | ||||
|  | ||||
| class AppNames { | ||||
|   late String author; | ||||
| @@ -424,19 +425,40 @@ abstract class AppSource { | ||||
|   } | ||||
|  | ||||
|   // Some Sources may have additional settings at the Source level (not specific to Apps) - these use SettingsProvider | ||||
|   List<GeneratedFormItem> additionalSourceSpecificSettingFormItems = []; | ||||
|   // If the source has been overridden, we expect the user to define one-time values as additional settings - don't use the stored values | ||||
|   List<GeneratedFormItem> sourceConfigSettingFormItems = []; | ||||
|   Future<Map<String, String>> getSourceConfigValues( | ||||
|       Map<String, dynamic> additionalSettings, | ||||
|       SettingsProvider settingsProvider) async { | ||||
|     Map<String, String> results = {}; | ||||
|     for (var e in sourceConfigSettingFormItems) { | ||||
|       var val = hostChanged | ||||
|           ? additionalSettings[e.key] | ||||
|           : settingsProvider.getSettingString(e.key); | ||||
|       if (val != null) { | ||||
|         results[e.key] = val; | ||||
|       } | ||||
|     } | ||||
|     return results; | ||||
|   } | ||||
|  | ||||
|   String? changeLogPageFromStandardUrl(String standardUrl) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   Future<String?> getSourceNote() async { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   Future<String> apkUrlPrefetchModifier( | ||||
|       String apkUrl, String standardUrl) async { | ||||
|     return apkUrl; | ||||
|   } | ||||
|  | ||||
|   bool canSearch = false; | ||||
|   Future<Map<String, List<String>>> search(String query) { | ||||
|   List<GeneratedFormItem> searchQuerySettingFormItems = []; | ||||
|   Future<Map<String, List<String>>> search(String query, | ||||
|       {Map<String, dynamic> querySettings = const {}}) { | ||||
|     throw NotImplementedError(); | ||||
|   } | ||||
|  | ||||
| @@ -447,8 +469,11 @@ abstract class AppSource { | ||||
| } | ||||
|  | ||||
| ObtainiumError getObtainiumHttpError(Response res) { | ||||
|   return ObtainiumError(res.reasonPhrase ?? | ||||
|       tr('errorWithHttpStatusCode', args: [res.statusCode.toString()])); | ||||
|   return ObtainiumError((res.reasonPhrase != null && | ||||
|           res.reasonPhrase != null && | ||||
|           res.reasonPhrase!.isNotEmpty) | ||||
|       ? res.reasonPhrase! | ||||
|       : tr('errorWithHttpStatusCode', args: [res.statusCode.toString()])); | ||||
| } | ||||
|  | ||||
| abstract class MassAppUrlSource { | ||||
| @@ -594,9 +619,7 @@ class SourceProvider { | ||||
|     } | ||||
|     String apkVersion = apk.version.replaceAll('/', '-'); | ||||
|     var name = currentApp != null ? currentApp.name.trim() : ''; | ||||
|     name = name.isNotEmpty | ||||
|         ? name | ||||
|         : apk.names.name[0].toUpperCase() + apk.names.name.substring(1); | ||||
|     name = name.isNotEmpty ? name : apk.names.name; | ||||
|     return App( | ||||
|         currentApp?.id ?? | ||||
|             ((!source.appIdInferIsOptional || | ||||
| @@ -606,7 +629,7 @@ class SourceProvider { | ||||
|                 : null) ?? | ||||
|             generateTempID(standardUrl, additionalSettings), | ||||
|         standardUrl, | ||||
|         apk.names.author[0].toUpperCase() + apk.names.author.substring(1), | ||||
|         apk.names.author, | ||||
|         name, | ||||
|         currentApp?.installedVersion, | ||||
|         apkVersion, | ||||
|   | ||||
							
								
								
									
										88
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										88
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -5,18 +5,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: android_alarm_manager_plus | ||||
|       sha256: "80f963d47cb7ab0818144c7b0668aea4c038f9cb8626626e89a4ea77375defb7" | ||||
|       sha256: c20d91a9096596f66274bf8172321c278f9cba8091638f80205fe66d31587fa5 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.0.1" | ||||
|     version: "3.0.2" | ||||
|   android_intent_plus: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: android_intent_plus | ||||
|       sha256: "2c87d8330ba5deef5fe20e77f4d178190b3b24531dce08368030ab4be40a9d4e" | ||||
|       sha256: f72ae20bb37108694f442e7ae6acbd28b453ca62ce86842f6787b784355abfe6 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.1" | ||||
|     version: "4.0.2" | ||||
|   android_package_installer: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -158,10 +158,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: device_info_plus | ||||
|       sha256: "2c35b6d1682b028e42d07b3aee4b98fa62996c10bc12cb651ec856a80d6a761b" | ||||
|       sha256: "86add5ef97215562d2e090535b0a16f197902b10c369c558a100e74ea06e8659" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "9.0.2" | ||||
|     version: "9.0.3" | ||||
|   device_info_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -222,10 +222,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: file_picker | ||||
|       sha256: b1729fc96627dd44012d0a901558177418818d6bd428df59dcfeb594e5f66432 | ||||
|       sha256: "21145c9c268d54b1f771d8380c195d2d6f655e0567dc1ca2f9c134c02c819e0a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.3.2" | ||||
|     version: "5.3.3" | ||||
|   flutter: | ||||
|     dependency: "direct main" | ||||
|     description: flutter | ||||
| @@ -243,10 +243,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_fgbg | ||||
|       sha256: d37511eef6afb7e2e3f2278ec6498bb12c650ed517c81bcd09452d910e8b9174 | ||||
|       sha256: "08c4d2fd229e3df26083d5aecc3dea9ff4f2d188f8cd57aaf2b3f047bd08a047" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.2.2" | ||||
|     version: "0.3.0" | ||||
|   flutter_launcher_icons: | ||||
|     dependency: "direct dev" | ||||
|     description: | ||||
| @@ -326,6 +326,14 @@ packages: | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "8.2.2" | ||||
|   hsluv: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: hsluv | ||||
|       sha256: f33e63b0c24ceee0f6492874424aa8edc671ef9a20cc889e4b969284d8f02eb1 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.1.3" | ||||
|   html: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -482,42 +490,42 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_android | ||||
|       sha256: "2cec049d282c7f13c594b4a73976b0b4f2d7a1838a6dd5aaf7bd9719196bee86" | ||||
|       sha256: "5d44fc3314d969b84816b569070d7ace0f1dea04bd94a83f74c4829615d22ad8" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.27" | ||||
|     version: "2.1.0" | ||||
|   path_provider_foundation: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_foundation | ||||
|       sha256: "916731ccbdce44d545414dd9961f26ba5fbaa74bcbb55237d8e65a623a8c7297" | ||||
|       sha256: "1b744d3d774e5a879bb76d6cd1ecee2ba2c6960c03b1020cd35212f6aa267ac5" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.4" | ||||
|     version: "2.3.0" | ||||
|   path_provider_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_linux | ||||
|       sha256: ffbb8cc9ed2c9ec0e4b7a541e56fd79b138e8f47d2fb86815f15358a349b3b57 | ||||
|       sha256: ba2b77f0c52a33db09fc8caf85b12df691bf28d983e84cf87ff6d693cfa007b3 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.11" | ||||
|     version: "2.2.0" | ||||
|   path_provider_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_platform_interface | ||||
|       sha256: "57585299a729335f1298b43245842678cb9f43a6310351b18fb577d6e33165ec" | ||||
|       sha256: bced5679c7df11190e1ddc35f3222c858f328fff85c3942e46e7f5589bf9eb84 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.6" | ||||
|     version: "2.1.0" | ||||
|   path_provider_windows: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_windows | ||||
|       sha256: "1cb68ba4cd3a795033de62ba1b7b4564dace301f952de6bfb3cd91b202b6ee96" | ||||
|       sha256: ee0e0d164516b90ae1f970bdf29f726f1aa730d7cfc449ecc74c495378b705da | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.7" | ||||
|     version: "2.2.0" | ||||
|   permission_handler: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -530,10 +538,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: permission_handler_android | ||||
|       sha256: c0c9754479a4c4b1c1f3862ddc11930c9b3f03bef2816bb4ea6eed1e13551d6f | ||||
|       sha256: "2ffaf52a21f64ac9b35fe7369bb9533edbd4f698e5604db8645b1064ff4cf221" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "10.3.2" | ||||
|     version: "10.3.3" | ||||
|   permission_handler_apple: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -578,10 +586,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: plugin_platform_interface | ||||
|       sha256: "6a2128648c854906c53fa8e33986fc0247a1116122f9534dd20e3ab9e16a32bc" | ||||
|       sha256: "43798d895c929056255600343db8f049921cbec94d31ec87f1dc5c16c01935dd" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.4" | ||||
|     version: "2.1.5" | ||||
|   pointycastle: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -602,18 +610,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: share_plus | ||||
|       sha256: ed3fcea4f789ed95913328e629c0c53e69e80e08b6c24542f1b3576046c614e8 | ||||
|       sha256: "6cec740fa0943a826951223e76218df002804adb588235a8910dc3d6b0654e11" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "7.0.2" | ||||
|     version: "7.1.0" | ||||
|   share_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: share_plus_platform_interface | ||||
|       sha256: "0c6e61471bd71b04a138b8b588fa388e66d8b005e6f2deda63371c5c505a0981" | ||||
|       sha256: "357412af4178d8e11d14f41723f80f12caea54cf0d5cd29af9dcdab85d58aea7" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.2.1" | ||||
|     version: "3.3.0" | ||||
|   shared_preferences: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -687,18 +695,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: sqflite | ||||
|       sha256: b4d6710e1200e96845747e37338ea8a819a12b51689a3bcf31eff0003b37a0b9 | ||||
|       sha256: "591f1602816e9c31377d5f008c2d9ef7b8aca8941c3f89cc5fd9d84da0c38a9a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.8+4" | ||||
|     version: "2.3.0" | ||||
|   sqflite_common: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_common | ||||
|       sha256: "8f7603f3f8f126740bc55c4ca2d1027aab4b74a1267a3e31ce51fe40e3b65b8f" | ||||
|       sha256: "1b92f368f44b0dee2425bb861cfa17b6f6cf3961f762ff6f941d20b33355660a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.5+1" | ||||
|     version: "2.5.0" | ||||
|   stack_trace: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -775,10 +783,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_android | ||||
|       sha256: "15f5acbf0dce90146a0f5a2c4a002b1814a6303c4c5c075aa2623b2d16156f03" | ||||
|       sha256: "78cb6dea3e93148615109e58e42c35d1ffbf5ef66c44add673d0ab75f12ff3af" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.0.36" | ||||
|     version: "6.0.37" | ||||
|   url_launcher_ios: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -855,10 +863,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_android | ||||
|       sha256: "27ad6a99c4b2d5e1ffd2b993a10f738b6b4979f139b4d64c34ac511595fcd748" | ||||
|       sha256: d936a09fbfd08cb78f7329e0bbacf6158fbdfe24ffc908b22444c07d295eb193 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.9.0" | ||||
|     version: "3.9.2" | ||||
|   webview_flutter_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -871,18 +879,18 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_wkwebview | ||||
|       sha256: "369fdf6160944a7db660ff15fa048c2bd681b09557907beaef1f95e8557d21dc" | ||||
|       sha256: "5fa098f28b606f699e8ca52d9e4e11edbbfef65189f5f77ae92703ba5408fd25" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.7.0" | ||||
|     version: "3.7.2" | ||||
|   win32: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: win32 | ||||
|       sha256: dfdf0136e0aa7a1b474ea133e67cb0154a0acd2599c4f3ada3b49d38d38793ee | ||||
|       sha256: f2add6fa510d3ae152903412227bda57d0d5a8da61d2c39c1fb022c9429a41c0 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.0.5" | ||||
|     version: "5.0.6" | ||||
|   win32_registry: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|   | ||||
| @@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev | ||||
| # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | ||||
| # In Windows, build-name is used as the major, minor, and patch parts | ||||
| # of the product and file versions while build-number is used as the build suffix. | ||||
| version: 0.13.16+180 # When changing this, update the tag in main() accordingly | ||||
| version: 0.13.23+187 # When changing this, update the tag in main() accordingly | ||||
|  | ||||
| environment: | ||||
|   sdk: '>=2.18.2 <3.0.0' | ||||
| @@ -37,7 +37,7 @@ dependencies: | ||||
|   # Use with the CupertinoIcons class for iOS style icons. | ||||
|   cupertino_icons: ^1.0.5 | ||||
|   path_provider: ^2.0.11 | ||||
|   flutter_fgbg: ^0.2.0 # Try removing reliance on this | ||||
|   flutter_fgbg: ^0.3.0 # Try removing reliance on this | ||||
|   flutter_local_notifications: ^15.1.0+1 | ||||
|   provider: ^6.0.3 | ||||
|   http: ^1.0.0 | ||||
| @@ -64,7 +64,7 @@ dependencies: | ||||
|   android_intent_plus: ^4.0.0 | ||||
|   flutter_markdown: ^0.6.14 | ||||
|   flutter_archive: ^5.0.0 | ||||
|  | ||||
|   hsluv: ^1.1.3 | ||||
|  | ||||
| dev_dependencies: | ||||
|   flutter_test: | ||||
|   | ||||
		Reference in New Issue
	
	Block a user