From 09da7b8d68169d1e9d6ca539b653d9aae7f50bb4 Mon Sep 17 00:00:00 2001 From: Sonal Yadav Date: Sun, 22 Jun 2025 19:10:15 +0530 Subject: [PATCH 001/129] Skip image upload to Wikidata (nearby -> green pins) (#6349) * Skip image upload to Wikidata if item already has image * Re-run CI * no more Failed to update Wikidata for green pins --- .../fr/free/nrw/commons/upload/worker/UploadWorker.kt | 8 +++++++- .../free/nrw/commons/wikidata/WikidataEditService.kt | 7 +++++-- .../main/res/values-x-invalidLanguageCode/error.xml | 10 ---------- 3 files changed, 12 insertions(+), 13 deletions(-) delete mode 100644 app/src/main/res/values-x-invalidLanguageCode/error.xml diff --git a/app/src/main/java/fr/free/nrw/commons/upload/worker/UploadWorker.kt b/app/src/main/java/fr/free/nrw/commons/upload/worker/UploadWorker.kt index 6d28085b2..c8a1d9b98 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/worker/UploadWorker.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/worker/UploadWorker.kt @@ -472,7 +472,10 @@ class UploadWorker( if (wikiDataPlace != null) { if (!contribution.hasInvalidLocation()) { var revisionID: Long? = null + val p18WasSkipped = !wikiDataPlace.imageValue.isNullOrBlank() try { + if (!p18WasSkipped) { + // Only set P18 if the place does not already have a picture revisionID = wikidataEditService.createClaim( wikiDataPlace, @@ -489,9 +492,11 @@ class UploadWorker( .subscribeOn(Schedulers.io()) .blockingAwait() Timber.d("Updated WikiItem place ${place.name} with image ${place.pic}") + } } - showSuccessNotification(contribution) } + // Always show success notification, whether P18 was set or skipped + showSuccessNotification(contribution) } catch (exception: Exception) { Timber.e(exception) } @@ -500,6 +505,7 @@ class UploadWorker( wikidataEditService.handleImageClaimResult( contribution.wikidataPlace!!, revisionID, + p18WasSkipped = p18WasSkipped ) } } else { diff --git a/app/src/main/java/fr/free/nrw/commons/wikidata/WikidataEditService.kt b/app/src/main/java/fr/free/nrw/commons/wikidata/WikidataEditService.kt index 0b49c03e9..f4bf23073 100644 --- a/app/src/main/java/fr/free/nrw/commons/wikidata/WikidataEditService.kt +++ b/app/src/main/java/fr/free/nrw/commons/wikidata/WikidataEditService.kt @@ -196,13 +196,16 @@ class WikidataEditService @Inject constructor( return wikidataClient.setClaim(claim, COMMONS_APP_TAG).blockingSingle() } - fun handleImageClaimResult(wikidataItem: WikidataItem, revisionId: Long?) { + fun handleImageClaimResult(wikidataItem: WikidataItem, revisionId: Long?, p18WasSkipped: Boolean = false) { if (revisionId != null) { wikidataEditListener?.onSuccessfulWikidataEdit() showSuccessToast(wikidataItem.name) - } else { + } else if (!p18WasSkipped) { Timber.d("Unable to make wiki data edit for entity %s", wikidataItem) showLongToast(context, context.getString(R.string.wikidata_edit_failure)) + } else { + Timber.d("Wikidata edit skipped for entity %s because P18 already exists", wikidataItem) + // No error shown to user, as this is not a failure } } diff --git a/app/src/main/res/values-x-invalidLanguageCode/error.xml b/app/src/main/res/values-x-invalidLanguageCode/error.xml deleted file mode 100644 index f4e2fe125..000000000 --- a/app/src/main/res/values-x-invalidLanguageCode/error.xml +++ /dev/null @@ -1,10 +0,0 @@ - - - - کامَنٕز گوو رُکِتھ - Oops. کیہہ تام گوو غلط! - ؤنِیوٚ اَسہِ توٚہہِ کیاہ ٲسِیوٚ کران، تہٕ کٕریٚو تہِ اَسہِ سٕتی شیر بذریعہِ برقی خط. یُس مَدَتھ کَرِ اَسہِ اَتھ شہَرنَس منٛز! - شُکریہ! - From d9e89174187d983aabdc9c0f3d1791b39e932538 Mon Sep 17 00:00:00 2001 From: "translatewiki.net" Date: Mon, 23 Jun 2025 14:01:45 +0200 Subject: [PATCH 002/129] Localisation updates from https://translatewiki.net. --- app/src/main/res/values-fi/strings.xml | 2 +- app/src/main/res/values-ps/strings.xml | 4 +- .../values-x-invalidLanguageCode/error.xml | 10 ++++ app/src/main/res/values-xmf/strings.xml | 46 ++++++++++++++++++- 4 files changed, 57 insertions(+), 5 deletions(-) create mode 100644 app/src/main/res/values-x-invalidLanguageCode/error.xml diff --git a/app/src/main/res/values-fi/strings.xml b/app/src/main/res/values-fi/strings.xml index cdb25defc..e65f05825 100644 --- a/app/src/main/res/values-fi/strings.xml +++ b/app/src/main/res/values-fi/strings.xml @@ -180,7 +180,7 @@ Kuvaus: Sydneyn oopperatalo lahden toiselta puolelta katsottuna Luokat: Sydneyn oopperatalo lännestä katsottuna, Sydneyn oopperatalo kaukaa katsottuna Herätä Wikipedia-artikkelit eloon kuvillasi! Tuo kuvasi Wikipediaan. - Wikipedian kuvat tulevat Wikimedia Commonsista. + Wikipedian kuvat ovat peräisin Wikimedia Commonsista. Kuvasi auttavat useita ihmisiä ympäri maailmaa artikkeleiden ymmärtämisessä. Vältä tekijänoikeuksien alaista materiaalia, kuten julisteita, kirjan kansia ja useimpia Internetistä löydettyjä kuvia. Luuletko ymmärtäneesi tämän? diff --git a/app/src/main/res/values-ps/strings.xml b/app/src/main/res/values-ps/strings.xml index 6b3970c88..98bd626e9 100644 --- a/app/src/main/res/values-ps/strings.xml +++ b/app/src/main/res/values-ps/strings.xml @@ -136,7 +136,7 @@ %1$s مور ټولگې نه لري ويکي‌اوتوک خونديځ کې د خپلو انځورونو موندلو لپاره وېشنيزې ورگډې کړئ.\nوېشنيزو ورگډولو لپاره ټاپل پيل کړئ. وېشنيزې - امستنې + اوڼنې نومليکنه ټاکلی انځور دوديز ټاکونکی @@ -216,7 +216,7 @@ پورته کول نژدې په اړه - امستنې + اوڼنې غبرگون وتل بياکتنه diff --git a/app/src/main/res/values-x-invalidLanguageCode/error.xml b/app/src/main/res/values-x-invalidLanguageCode/error.xml new file mode 100644 index 000000000..f4e2fe125 --- /dev/null +++ b/app/src/main/res/values-x-invalidLanguageCode/error.xml @@ -0,0 +1,10 @@ + + + + کامَنٕز گوو رُکِتھ + Oops. کیہہ تام گوو غلط! + ؤنِیوٚ اَسہِ توٚہہِ کیاہ ٲسِیوٚ کران، تہٕ کٕریٚو تہِ اَسہِ سٕتی شیر بذریعہِ برقی خط. یُس مَدَتھ کَرِ اَسہِ اَتھ شہَرنَس منٛز! + شُکریہ! + diff --git a/app/src/main/res/values-xmf/strings.xml b/app/src/main/res/values-xmf/strings.xml index 153d88b54..3f9b0e59f 100644 --- a/app/src/main/res/values-xmf/strings.xml +++ b/app/src/main/res/values-xmf/strings.xml @@ -99,6 +99,8 @@ ფოტოშ გინოღალა გოხოლუას ჩქიმი ეხარგუეფი + რსხილიშ კოპირება + რსხილი კოპირებული რე ბუფერულ შვენას გობჟინაფა ფაილიშ ხასჷლაშ ძირაფა მუკნაჭარა (უციო) @@ -109,6 +111,7 @@ ძალამ მიარე უმწუძინუ ცადება. ქორთხინ, მუხირენ წუთშა ხოლო ქოცადით. მორდება, თე მახვარებუ ბლოკირი რე ვიკიოწკარუეს თქვა გემშიონათ ოკო ჟირფაქტორიანი ავტორიზაციაშ კოდი. + დოდასურაფაშ კოდი ჯღონელი რე თქვანი ელექტრონული ფოშტაშა. ქორთხიინთ, მოჯღონელი კოდი გენშეჸონათ მიშაულარო. მიშულაქ ვემიხუჯინუ ეხარგუა სერიაშ ჯოხო @@ -117,6 +120,7 @@ კატეგორიაშ გიშაგორუა დოგორით ელემენტეფი, ნამუეფით მოჩამილი რე თქვანი სურათის (გვალა, ტაჯ-მაჰალი დო თ.უ.) ჩუალა + გეძინელი მენიუ გოახალაფა ერკებული (ეხარგუეფი ვა რე) @@ -213,6 +217,7 @@ ბეტა ტესტირებას კათაფი ჩართით Beta-შა ჭირინაფა Google Play-ს დო მიღით ორდოიანი ჭირინაფა ახალ ფუნქციეფშა დო ჩილათეფიშ ეშახინტკალო 2ფა კოდი + ელექტრონული ფოშტაშ დოდასურაფაშ კოდი გოკონანო გიშულა? მედიაფაილიშ ჩილათა გიმენკატეგორიეფქ ვეგორინუ @@ -360,6 +365,7 @@ კინოხიანი სურათეფი ორენეფი + კატეგორიეფი მიკოწონებულეფშა გეძინა/ლასუა მიკოწონებულეფი თქვა ვეგეიძინჷნა აკა მიკოწონებული @@ -373,8 +379,10 @@ მოზოჯით ვიკიოწკარუეშა!\n\nგეხარგეთ თქვანი პირველი ფაილი, ქეგუნჭირით კონჭის გეძინა. კატეგორია ვა რე გიშაგორილი სურათეფი კატეგორიზაციაშ უმუშო შხირას მერკეთ გჷმორინაფონი რე. დასურო გონებჷნანო კატეგორიეფიშ მეწურაფაშ უმუშო გაგჷნძარაფა? + ეჭარუა ვა რე გიშაგორილი ეხარგუაშ გოუქვაფა ეხარგუაშ გოგჷნძორაფა + (ნაკორობაშ არძა სურათიშო) ათე არანს გორუა ალობაშ მოთხირი თენა კჷნ დღას ვაბკითხა @@ -383,9 +391,20 @@ ალობაშ მეჩამა გოუქვაფა ღოლამირჷ რე + თინას რენო კათეგორიეფი მეწურაფილი? + გეჸვენჯი სურათი + ქო, მუშენ ვარი ვა რე გუმორინაფილი სურათეფი + ვა რე ეხარგილი სურათეფი + თქვა ვაიღჷნა უკითხირუ გინაფეფი + ვა ვაიღჷნა კითხირებული გინაფეფი + ქოძირით თქვანი ელექტრონული ფოსტა + კითხირებულიშ ძირაფა + უკითხირებუშ ძირაფა სურათიშ ეხარგუაშ ბორჯის ჩილათაქ მოხვადჷ ქორთხინთ ქჷმიცადით … + კოპირაფილი რე + თე სურათიშ გიშატება ავტორი ორენი კამერაშ მოდელი @@ -393,9 +412,32 @@ სერიული ნომერი პროგრამული უნარღელჸუა სურათიშ ინფორმაცია - სელფი - ჸურილი + კატეგორიეფქ ვეგორინუ + ეჭარუეფქ ვეორინუ + ეხარგუაქ გეუქვუ + მუშენ დილასას ოკო %1$s-ქ? + %1$s ეხარგილი რე: %2$s + წუმოძინელო + %1$s ნომინირებული რე ოლასარო. + ვემიხუჯინუ + ვეშილებე ლასუაშ მოთხირი. + სელფი, ნამუთ ვეგჷმირინუაფუ ნამთინ სტატიას + ედომუშამო გინობარდილი რე + უაზრობა, აბსოლუტურო უმუხუჯუ რე ირნერი სტატიას + ახალი ამბეეფიშ სურათი + ნამდგა სურათი ინტერნეტიშე + ლოგო + პანორამაშ დუდიშულაშ აკორცუაფა თიშენ ნამჷ-და თინა რე + კატეგორიეფიშ მოახალებაშ ცადება. + კატეგორიაშ მოახალება + წუმოძინელო + + კატეგორია %1$s გეძინელი რე. + კატეგორიეფი %1$sგეძინელი რე. + + ვემიხუჯინუ კატეგორიეფიშ გეძინაქ. + კატეგორიეფიშ მოახალება მიკოწონებეფი პარამეტრეფი რუმე From 5d7f42d127600f205773214bb877472f259ff552 Mon Sep 17 00:00:00 2001 From: Rohit Verma <101377978+rohit9625@users.noreply.github.com> Date: Tue, 24 Jun 2025 18:26:00 +0530 Subject: [PATCH 003/129] Fix/file usage not working (#6354) * chore: add R8 rules to prevent obfuscating file usage classes * chore: upgrade lifecycle-runtime dependency to resolve lint errors * remove invalid resource directory --- app/proguard-rules.txt | 3 +++ .../main/res/values-x-invalidLanguageCode/error.xml | 10 ---------- gradle/libs.versions.toml | 2 +- 3 files changed, 4 insertions(+), 11 deletions(-) delete mode 100644 app/src/main/res/values-x-invalidLanguageCode/error.xml diff --git a/app/proguard-rules.txt b/app/proguard-rules.txt index 63981633b..21c584ba9 100644 --- a/app/proguard-rules.txt +++ b/app/proguard-rules.txt @@ -66,6 +66,9 @@ # Application classes that will be serialized/deserialized over Gson -keep class com.google.gson.examples.android.model.** { *; } +# Prevent R8 from obfuscating project classes used by Gson for parsing +-keep class fr.free.nrw.commons.fileusages.** { *; } + # Prevent proguard from stripping interface information from TypeAdapterFactory, # JsonSerializer, JsonDeserializer instances (so they can be used in @JsonAdapter) -keep class * implements com.google.gson.TypeAdapterFactory diff --git a/app/src/main/res/values-x-invalidLanguageCode/error.xml b/app/src/main/res/values-x-invalidLanguageCode/error.xml deleted file mode 100644 index f4e2fe125..000000000 --- a/app/src/main/res/values-x-invalidLanguageCode/error.xml +++ /dev/null @@ -1,10 +0,0 @@ - - - - کامَنٕز گوو رُکِتھ - Oops. کیہہ تام گوو غلط! - ؤنِیوٚ اَسہِ توٚہہِ کیاہ ٲسِیوٚ کران، تہٕ کٕریٚو تہِ اَسہِ سٕتی شیر بذریعہِ برقی خط. یُس مَدَتھ کَرِ اَسہِ اَتھ شہَرنَس منٛز! - شُکریہ! - diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index afb3615d2..f38357b04 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -39,7 +39,7 @@ leakcanary = "2.10" livedataTesting = "1.2.0" swipelayout = "1.2.0" viewpagerIndicator = "2.4.1.1" -lifecycleRuntimeKtx = "2.8.4" +lifecycleRuntimeKtx = "2.8.7" loggingInterceptor = "4.10.0" logbackAndroidClassic = "1.1.1-6" material = "1.12.0" From ad7dddaac439b3af4332e43552fc25994a74f219 Mon Sep 17 00:00:00 2001 From: Rohit Verma <101377978+rohit9625@users.noreply.github.com> Date: Wed, 25 Jun 2025 08:54:03 +0530 Subject: [PATCH 004/129] Fix infinite loading circular progress bar after nominating for deletion (#6324) * fix: infinite loading progress bar after nominating for deletion * add logs for testing * refactor: use globalFileUsage instead of achievement to append in reason Fetching achievements is a time consuming operation and globalFileUsage gives the similar result in optimal time * test(ReasonBuilder): fix tests according to new behavior * refactor: remove logs added for testing * test: await for async getReason method call --------- Co-authored-by: Neel Doshi Co-authored-by: Nicolas Raoul --- app/build.gradle.kts | 1 + .../free/nrw/commons/delete/DeleteHelper.kt | 2 - .../free/nrw/commons/delete/ReasonBuilder.kt | 51 +++++++----- .../nrw/commons/media/MediaDetailFragment.kt | 79 +++++++++++-------- .../nrw/commons/mwapi/OkHttpJsonApiClient.kt | 1 + app/src/main/res/values/strings.xml | 3 +- .../nrw/commons/delete/ReasonBuilderTest.kt | 15 ++-- gradle/libs.versions.toml | 2 + 8 files changed, 88 insertions(+), 66 deletions(-) diff --git a/app/build.gradle.kts b/app/build.gradle.kts index 9c53155a7..2e391a24f 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -347,6 +347,7 @@ dependencies { // Kotlin + coroutines implementation(libs.androidx.work.runtime.ktx) implementation(libs.androidx.work.runtime) + implementation(libs.kotlinx.coroutines.rx2) testImplementation(libs.androidx.work.testing) //Glide diff --git a/app/src/main/java/fr/free/nrw/commons/delete/DeleteHelper.kt b/app/src/main/java/fr/free/nrw/commons/delete/DeleteHelper.kt index 09959d0ef..3f9ae47ac 100644 --- a/app/src/main/java/fr/free/nrw/commons/delete/DeleteHelper.kt +++ b/app/src/main/java/fr/free/nrw/commons/delete/DeleteHelper.kt @@ -53,7 +53,6 @@ class DeleteHelper @Inject constructor( media: Media?, reason: String? ): Single? { - if(context == null && media == null) { return null } @@ -86,7 +85,6 @@ class DeleteHelper @Inject constructor( * @return */ private fun delete(media: Media, reason: String): Observable { - Timber.d("thread is delete %s", Thread.currentThread().name) val summary = "Nominating ${media.filename} for deletion." val calendar = Calendar.getInstance() val fileDeleteString = """ diff --git a/app/src/main/java/fr/free/nrw/commons/delete/ReasonBuilder.kt b/app/src/main/java/fr/free/nrw/commons/delete/ReasonBuilder.kt index 09018c249..c4cd73b8a 100644 --- a/app/src/main/java/fr/free/nrw/commons/delete/ReasonBuilder.kt +++ b/app/src/main/java/fr/free/nrw/commons/delete/ReasonBuilder.kt @@ -2,21 +2,19 @@ package fr.free.nrw.commons.delete import android.annotation.SuppressLint import android.content.Context - -import fr.free.nrw.commons.utils.DateUtil -import java.util.Locale - -import javax.inject.Inject -import javax.inject.Singleton - import fr.free.nrw.commons.Media import fr.free.nrw.commons.R -import fr.free.nrw.commons.profile.achievements.FeedbackResponse import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.mwapi.OkHttpJsonApiClient +import fr.free.nrw.commons.utils.DateUtil import fr.free.nrw.commons.utils.ViewUtilWrapper import io.reactivex.Single +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.rx2.rxSingle import timber.log.Timber +import java.util.Locale +import javax.inject.Inject +import javax.inject.Singleton /** * This class handles the reason for deleting a Media object @@ -29,6 +27,8 @@ class ReasonBuilder @Inject constructor( private val viewUtilWrapper: ViewUtilWrapper ) { + private val defaultFileUsagePageSize = 10 + /** * To process the reason and append the media's upload date and uploaded_by_me string * @param media @@ -39,7 +39,7 @@ class ReasonBuilder @Inject constructor( if (media == null || reason == null) { return Single.just("Not known") } - return fetchArticleNumber(media, reason) + return getAndAppendFileUsage(media, reason) } /** @@ -54,27 +54,36 @@ class ReasonBuilder @Inject constructor( } } - private fun fetchArticleNumber(media: Media, reason: String): Single { - return if (checkAccount()) { - okHttpJsonApiClient - .getAchievements(sessionManager.userName) - .map { feedbackResponse -> appendArticlesUsed(feedbackResponse, media, reason) } - } else { - Single.just("") + private fun getAndAppendFileUsage(media: Media, reason: String): Single { + return rxSingle(context = Dispatchers.IO) { + if (!checkAccount()) return@rxSingle "" + + try { + val globalFileUsage = okHttpJsonApiClient.getGlobalFileUsages( + fileName = media.filename, + pageSize = defaultFileUsagePageSize + ) + val globalUsages = globalFileUsage?.query?.pages?.sumOf { it.fileUsage.size } ?: 0 + + appendArticlesUsed(globalUsages, media, reason) + } catch (e: Exception) { + Timber.e(e, "Error fetching file usage") + throw e + } } } /** - * Takes the uploaded_by_me string, the upload date, name of articles using images + * Takes the uploaded_by_me string, the upload date, no. of articles using images * and appends it to the received reason - * @param feedBack object + * @param fileUsages No. of files/articles using this image * @param media whose upload data is to be fetched - * @param reason + * @param reason string to be appended */ @SuppressLint("StringFormatInvalid") - private fun appendArticlesUsed(feedBack: FeedbackResponse, media: Media, reason: String): String { + private fun appendArticlesUsed(fileUsages: Int, media: Media, reason: String): String { val reason1Template = context.getString(R.string.uploaded_by_myself) - return reason + String.format(Locale.getDefault(), reason1Template, prettyUploadedDate(media), feedBack.articlesUsingImages) + return reason + String.format(Locale.getDefault(), reason1Template, prettyUploadedDate(media), fileUsages) .also { Timber.i("New Reason %s", it) } } diff --git a/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt b/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt index 8a4d530c4..f371b733f 100644 --- a/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt @@ -74,7 +74,6 @@ import fr.free.nrw.commons.BuildConfig import fr.free.nrw.commons.CameraPosition import fr.free.nrw.commons.CommonsApplication import fr.free.nrw.commons.CommonsApplication.Companion.instance -import fr.free.nrw.commons.locationpicker.LocationPicker import fr.free.nrw.commons.Media import fr.free.nrw.commons.MediaDataExtractor import fr.free.nrw.commons.R @@ -102,6 +101,7 @@ import fr.free.nrw.commons.explore.depictions.WikidataItemDetailsActivity import fr.free.nrw.commons.kvstore.JsonKvStore import fr.free.nrw.commons.language.AppLanguageLookUpTable import fr.free.nrw.commons.location.LocationServiceManager +import fr.free.nrw.commons.locationpicker.LocationPicker import fr.free.nrw.commons.media.MediaDetailPagerFragment.MediaDetailProvider import fr.free.nrw.commons.profile.ProfileActivity import fr.free.nrw.commons.review.ReviewHelper @@ -116,6 +116,7 @@ import fr.free.nrw.commons.utils.LangCodeUtils.getLocalizedResources import fr.free.nrw.commons.utils.PermissionUtils.PERMISSIONS_STORAGE import fr.free.nrw.commons.utils.PermissionUtils.checkPermissionsAndPerformAction import fr.free.nrw.commons.utils.PermissionUtils.hasPermission +import fr.free.nrw.commons.utils.ViewUtil import fr.free.nrw.commons.utils.ViewUtil.showShortToast import fr.free.nrw.commons.utils.ViewUtilWrapper import fr.free.nrw.commons.wikidata.mwapi.MwQueryPage.Revision @@ -125,6 +126,7 @@ import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.schedulers.Schedulers import org.apache.commons.lang3.StringUtils import timber.log.Timber +import java.lang.String.format import java.util.Date import java.util.Locale import java.util.Objects @@ -1646,7 +1648,7 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C getString(R.string.cancel), { val reason: String = input.text.toString() - onDeleteClickeddialogtext(reason) + onDeleteClickedDialogText(reason) }, {}, input @@ -1700,26 +1702,48 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C resultSingle .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) - .subscribe { _ -> - if (applicationKvStore.getBoolean( - String.format( - NOMINATING_FOR_DELETION_MEDIA, media!!.imageUrl - ), false - ) - ) { - applicationKvStore.remove( - String.format( - NOMINATING_FOR_DELETION_MEDIA, - media!!.imageUrl - ) - ) - callback!!.nominatingForDeletion(index) - } - } + .subscribe(this::handleDeletionResult, this::handleDeletionError); + } + + /** + * Disables Progress Bar and Update delete button text. + */ + private fun disableProgressBar() { + activity?.run { + runOnUiThread(Runnable { + binding.progressBarDeletion.visibility = View.GONE + }) + } ?: return // Prevent NullPointerException when fragment is not attached to activity + } + + private fun handleDeletionResult(success: Boolean) { + if (success) { + binding.nominateDeletion.text = getString(R.string.nominated_for_deletion_btn) + ViewUtil.showLongSnackbar(requireView(), getString(R.string.nominated_for_deletion)) + disableProgressBar() + checkAndClearDeletionFlag() + } else { + disableProgressBar() + } + } + + private fun handleDeletionError(throwable: Throwable) { + throwable.printStackTrace() + disableProgressBar() + checkAndClearDeletionFlag() + } + + private fun checkAndClearDeletionFlag() { + if (applicationKvStore + .getBoolean(format(NOMINATING_FOR_DELETION_MEDIA, media!!.imageUrl), false) + ) { + applicationKvStore.remove(format(NOMINATING_FOR_DELETION_MEDIA, media!!.imageUrl)) + callback!!.nominatingForDeletion(index) + } } @SuppressLint("CheckResult") - private fun onDeleteClickeddialogtext(reason: String) { + private fun onDeleteClickedDialogText(reason: String) { applicationKvStore.putBoolean( String.format( NOMINATING_FOR_DELETION_MEDIA, @@ -1736,22 +1760,7 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C resultSingletext .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) - .subscribe { _ -> - if (applicationKvStore.getBoolean( - String.format( - NOMINATING_FOR_DELETION_MEDIA, media!!.imageUrl - ), false - ) - ) { - applicationKvStore.remove( - String.format( - NOMINATING_FOR_DELETION_MEDIA, - media!!.imageUrl - ) - ) - callback!!.nominatingForDeletion(index) - } - } + .subscribe(this::handleDeletionResult, this::handleDeletionError); } private fun onSeeMoreClicked() { diff --git a/app/src/main/java/fr/free/nrw/commons/mwapi/OkHttpJsonApiClient.kt b/app/src/main/java/fr/free/nrw/commons/mwapi/OkHttpJsonApiClient.kt index 4fa7979d2..a2f92c2e6 100644 --- a/app/src/main/java/fr/free/nrw/commons/mwapi/OkHttpJsonApiClient.kt +++ b/app/src/main/java/fr/free/nrw/commons/mwapi/OkHttpJsonApiClient.kt @@ -281,6 +281,7 @@ class OkHttpJsonApiClient @Inject constructor( FeedbackResponse::class.java ) } catch (e: Exception) { + e.printStackTrace() return@fromCallable FeedbackResponse(0, 0, 0, FeaturedImages(0, 0), 0, "") } } diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 68dba88be..c9a2d16c8 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -431,7 +431,7 @@ I changed my mind, I don\'t want it to be publicly visible anymore Sorry this picture is not interesting for an encyclopedia - Uploaded by myself on %1$s, used in %2$d article(s). + Uploaded by myself on %1$s, used in %2$d article(s) at least. Welcome to Commons!\n Upload your first media by tapping on the add button. @@ -876,4 +876,5 @@ Upload your first media by tapping on the add button. Show in Nearby Created and uploaded by: %1$s Created by %1$s and uploaded by %2$s + Nominated for Deletion diff --git a/app/src/test/kotlin/fr/free/nrw/commons/delete/ReasonBuilderTest.kt b/app/src/test/kotlin/fr/free/nrw/commons/delete/ReasonBuilderTest.kt index e89a02dec..bd67475ee 100644 --- a/app/src/test/kotlin/fr/free/nrw/commons/delete/ReasonBuilderTest.kt +++ b/app/src/test/kotlin/fr/free/nrw/commons/delete/ReasonBuilderTest.kt @@ -5,13 +5,14 @@ import android.content.res.Resources import fr.free.nrw.commons.Media import fr.free.nrw.commons.R import fr.free.nrw.commons.auth.SessionManager +import fr.free.nrw.commons.fileusages.GlobalFileUsagesResponse import fr.free.nrw.commons.mwapi.OkHttpJsonApiClient -import fr.free.nrw.commons.profile.achievements.FeedbackResponse import fr.free.nrw.commons.profile.leaderboard.LeaderboardResponse import fr.free.nrw.commons.profile.leaderboard.UpdateAvatarResponse import fr.free.nrw.commons.utils.ViewUtilWrapper import io.reactivex.Observable import io.reactivex.Single +import kotlinx.coroutines.test.runTest import media import org.junit.Before import org.junit.Test @@ -58,16 +59,16 @@ class ReasonBuilderTest { PowerMockito.`when`(context?.getString(R.string.user_not_logged_in)) .thenReturn("Log-in expired. Please log in again.") - reasonBuilder!!.getReason(mock(Media::class.java), "test") + reasonBuilder!!.getReason(mock(Media::class.java), "test").test().await() verify(sessionManager, times(1))!!.forceLogin(any(Context::class.java)) } @Test - fun getReason() { + fun getReason() = runTest { `when`(sessionManager?.userName).thenReturn("Testuser") `when`(sessionManager?.doesAccountExist()).thenReturn(true) - `when`(okHttpJsonApiClient!!.getAchievements(anyString())) - .thenReturn(Single.just(mock(FeedbackResponse::class.java))) + `when`(okHttpJsonApiClient!!.getGlobalFileUsages(anyString(), anyInt())) + .thenReturn(mock(GlobalFileUsagesResponse::class.java)) `when`(okHttpJsonApiClient!!.getLeaderboard(anyString(), anyString(), anyString(), anyString(), anyString())) .thenReturn(Observable.just(mock(LeaderboardResponse::class.java))) `when`(okHttpJsonApiClient!!.setAvatar(anyString(), anyString())) @@ -75,8 +76,8 @@ class ReasonBuilderTest { val media = media(filename = "test_file", dateUploaded = Date()) - reasonBuilder!!.getReason(media, "test") + reasonBuilder!!.getReason(media, "test").test().await() verify(sessionManager, times(0))!!.forceLogin(any(Context::class.java)) - verify(okHttpJsonApiClient, times(1))!!.getAchievements(anyString()) + verify(okHttpJsonApiClient, times(1))!!.getGlobalFileUsages(anyString(), anyInt()) } } diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index f38357b04..df4b7bb43 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -16,6 +16,7 @@ constraintlayout = "1.1.3" coordinates2country = "1.8" dexcount = "4.0.0" githubTripletPlay = "2.7.2" +kotlinxCoroutinesRx2 = "1.8.0" osmdroidAndroid = "6.1.17" testCore = "1.4.0" coreKtx = "1.9.0" @@ -127,6 +128,7 @@ dagger-compiler = { module = "com.google.dagger:dagger-compiler", version.ref = facebook-fresco = { module = "com.facebook.fresco:fresco", version.ref = "frescoVersion" } glide-compiler = { module = "com.github.bumptech.glide:compiler", version.ref = "glide" } glide = { module = "com.github.bumptech.glide:glide", version.ref = "glide" } +kotlinx-coroutines-rx2 = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-rx2", version.ref = "kotlinxCoroutinesRx2" } photoview = { module = "com.github.chrisbanes:PhotoView", version.ref = "photoviewVersion" } # RxJava and Reactive Programming From b1a8308aaf244c93cdf3c72c0d897ad0f4b9fbcd Mon Sep 17 00:00:00 2001 From: "translatewiki.net" Date: Thu, 26 Jun 2025 14:02:13 +0200 Subject: [PATCH 005/129] Localisation updates from https://translatewiki.net. --- app/src/main/res/values-af/strings.xml | 2 +- app/src/main/res/values-ar/strings.xml | 2 +- app/src/main/res/values-ast/strings.xml | 2 +- app/src/main/res/values-b+sr+Latn/strings.xml | 2 +- app/src/main/res/values-cs/strings.xml | 2 +- app/src/main/res/values-da/strings.xml | 3 ++- app/src/main/res/values-de/strings.xml | 2 +- app/src/main/res/values-el/strings.xml | 2 +- app/src/main/res/values-eo/strings.xml | 2 +- app/src/main/res/values-es/strings.xml | 2 +- app/src/main/res/values-fa/strings.xml | 2 +- app/src/main/res/values-fi/strings.xml | 2 +- app/src/main/res/values-fr/strings.xml | 2 +- app/src/main/res/values-gl/strings.xml | 2 +- app/src/main/res/values-hu/strings.xml | 2 +- app/src/main/res/values-ia/strings.xml | 2 +- app/src/main/res/values-in/strings.xml | 2 +- app/src/main/res/values-io/strings.xml | 2 +- app/src/main/res/values-is/strings.xml | 2 +- app/src/main/res/values-it/strings.xml | 5 +++-- app/src/main/res/values-iw/strings.xml | 3 ++- app/src/main/res/values-ja/strings.xml | 2 +- app/src/main/res/values-krc/strings.xml | 2 +- app/src/main/res/values-kus/strings.xml | 2 +- app/src/main/res/values-lb/strings.xml | 5 +++-- app/src/main/res/values-lt/strings.xml | 2 +- app/src/main/res/values-mk/strings.xml | 2 +- app/src/main/res/values-nb/strings.xml | 2 +- app/src/main/res/values-nl/strings.xml | 3 ++- app/src/main/res/values-nqo/strings.xml | 2 +- app/src/main/res/values-pl/strings.xml | 2 +- app/src/main/res/values-pms/strings.xml | 3 ++- app/src/main/res/values-pt-rBR/strings.xml | 2 +- app/src/main/res/values-pt/strings.xml | 2 +- app/src/main/res/values-ro/strings.xml | 2 +- app/src/main/res/values-ru/strings.xml | 2 +- app/src/main/res/values-sk/strings.xml | 2 +- app/src/main/res/values-sl/strings.xml | 2 +- app/src/main/res/values-sr/strings.xml | 3 ++- app/src/main/res/values-sv/strings.xml | 2 +- app/src/main/res/values-te/strings.xml | 2 +- app/src/main/res/values-tr/strings.xml | 2 +- app/src/main/res/values-uk/strings.xml | 2 +- app/src/main/res/values-vec/strings.xml | 2 +- .../main/res/values-x-invalidLanguageCode/error.xml | 10 ++++++++++ app/src/main/res/values-xmf/strings.xml | 2 +- app/src/main/res/values-zh-rTW/strings.xml | 2 +- app/src/main/res/values-zh/strings.xml | 2 +- 48 files changed, 66 insertions(+), 49 deletions(-) create mode 100644 app/src/main/res/values-x-invalidLanguageCode/error.xml diff --git a/app/src/main/res/values-af/strings.xml b/app/src/main/res/values-af/strings.xml index 8a6a5ccd9..51aaea77e 100644 --- a/app/src/main/res/values-af/strings.xml +++ b/app/src/main/res/values-af/strings.xml @@ -341,7 +341,7 @@ Ek het besef dat dit sleg is vir my privaatheid Ek het van plan verander, ek wil nie hê dit moet meer in die openbaar sigbaar wees nie Jammer, hierdie foto is nie interessant vir \'n ensiklopedie nie - Opgelaai deur myself op %1$s, gebruik in %2$d artikel(s). + Opgelaai deur myself op %1$s, gebruik in %2$d artikel(s). Welkom by Commons!\n\nLaai u eerste media op deur op die byvoegknoppie te tik. Geen kategorieë gekies nie Beelde sonder kategorieë is selde bruikbaar. Is u seker dat u wil indien sonder om kategorieë te kies? diff --git a/app/src/main/res/values-ar/strings.xml b/app/src/main/res/values-ar/strings.xml index 5025b94cb..daa5e9b1c 100644 --- a/app/src/main/res/values-ar/strings.xml +++ b/app/src/main/res/values-ar/strings.xml @@ -469,7 +469,7 @@ أدركت أنه سيئ لخصوصيتي لقد غيرت رأيي، ولا أريد أن يكون مرئيا للجميع بعد الآن آسف هذه الصورة ليست مثيرة للاهتمام لموسوعة - رفعته بنفسي في %1$s، مستخدم في %2$d مقالة(ات). + رفعته بنفسي في %1$s، مستخدم في %2$d مقالة(ات). مرحبا بك في كومنز! \n\nارفع الوسائط الأولى عن طريق النقر على زر الإضافة. لا توجد تصنيفات مختارة نادرًا ما تكون الصور الخالية من التصنيفات قابلة للاستخدام، هل أنت متأكد أنك تريد الإرسال دون تحديد التصنيفات؟ diff --git a/app/src/main/res/values-ast/strings.xml b/app/src/main/res/values-ast/strings.xml index 8d0dba79a..591c21395 100644 --- a/app/src/main/res/values-ast/strings.xml +++ b/app/src/main/res/values-ast/strings.xml @@ -395,7 +395,7 @@ Decatéme que ye malo pa la mio privacidá Cambié d\'opinión, nun quiero que siga siendo visible públicamente Sentímoslo, esta imaxe nun ye interesante pa una enciclopedia - Xubíla yo mesmu\'l %1$s, usada en %2$d artículu(os). + Xubíla yo mesmu\'l %1$s, usada en %2$d artículu(os). ¡Afáyate\'n Commons.\n\nCarga\'l to primer ficheru tocando nel botón amestar. Nun hai categoríes seleicionaes Les imáxenes ensin categoríes raramente puen usase. ¿Seguro que quies siguir ensin escoyer categoría dala? diff --git a/app/src/main/res/values-b+sr+Latn/strings.xml b/app/src/main/res/values-b+sr+Latn/strings.xml index 47e1cb0bd..459f75852 100644 --- a/app/src/main/res/values-b+sr+Latn/strings.xml +++ b/app/src/main/res/values-b+sr+Latn/strings.xml @@ -394,7 +394,7 @@ Shvatio/la sam da je loša za moju privatnost Promenio/la sam mišljenje, ne želim da više javno budu vidljive Ova slika nije zanimljiva za enciklopediju - Otpremio/la sam na %1$s, koristi se u članku/cima — %2$d. + Otpremio/la sam na %1$s, koristi se u članku/cima — %2$d. Dobro došli na Ostavu!\n\nOtpremite prve medije dodirom dugmeta za dodavanje. Kategorije nisu izabrane Slike bez kategorija retko su upotrebljive. Zaista želite da nastavite bez izbora kategorija? diff --git a/app/src/main/res/values-cs/strings.xml b/app/src/main/res/values-cs/strings.xml index 3250b01d7..f3d645530 100644 --- a/app/src/main/res/values-cs/strings.xml +++ b/app/src/main/res/values-cs/strings.xml @@ -448,7 +448,7 @@ Uvědomil/a jsem si, že je to špatné pro mé soukromí Změnil/a jsem názor, nechci, aby to bylo veřejně viditelné Omlouváme se, že tento obrázek není zajímavý pro encyklopedii - Náhráno mnou %1$s, použito v(e) %2$d článku/článcích. + Náhráno mnou %1$s, použito v(e) %2$d článku/článcích. Vítejte na Commons!\n\nNahrajte svá první média klepnutím na tlačítko přidat. Nebyly vybrány žádné kategorie Obrázky bez kategorií jsou používány jen zřídka. Opravdu chcete nahrát obrázek bez výběru kategorií? diff --git a/app/src/main/res/values-da/strings.xml b/app/src/main/res/values-da/strings.xml index bc2b23dbf..043cd871b 100644 --- a/app/src/main/res/values-da/strings.xml +++ b/app/src/main/res/values-da/strings.xml @@ -419,7 +419,7 @@ Jeg indså, det var skidt for mit privatliv Jeg ombestemte mig, jeg ønsker ikke at det skal være offentligt synligt længere Beklager dette billede er ikke interessant for en encyklopædi - Uploadet af mig selv på %1$s , brugt i %2$d artikel(er). + Uploadet af mig selv på %1$s , brugt i mindst %2$d artikel(er). Velkommen til Commons!\n\nUpload dit første medie ved at trykke på Tilføj-knappen. Ingen kategorier valgt Billeder uden kategorier er sjældent brugbare. Er du sikker på, at du vil fortsætte uden at vælge kategorier? @@ -828,4 +828,5 @@ Vis i I nærheden Oprettet og uploadet af: %1$s Oprettet af %1$s og uploadet af %2$s + Nomineret til sletning diff --git a/app/src/main/res/values-de/strings.xml b/app/src/main/res/values-de/strings.xml index 705dfefd0..1c34182f1 100644 --- a/app/src/main/res/values-de/strings.xml +++ b/app/src/main/res/values-de/strings.xml @@ -453,7 +453,7 @@ Ich habe erkannt, dass es schlecht für meine Privatsphäre ist. Ich habe meine Meinung geändert. Ich möchte nicht mehr, dass es öffentlich sichtbar ist. Entschuldigung, aber dieses Bild ist für eine Enzyklopädie nicht relevant. - Von mir selbst hochgeladen am %1$s, verwendet in %2$d Artikel(n). + Von mir selbst hochgeladen am %1$s, verwendet in %2$d Artikel(n). Willkommen bei Commons!\n\nLade deine erste Datei hoch, indem du auf die Hinzufügen-Schaltfläche tippst. Keine Kategorien ausgewählt Bilder ohne Kategorien sind selten nutzbar. Bist du sicher, dass du ohne die Auswahl von Kategorien hochladen möchtest? diff --git a/app/src/main/res/values-el/strings.xml b/app/src/main/res/values-el/strings.xml index da816259e..c89bca41e 100644 --- a/app/src/main/res/values-el/strings.xml +++ b/app/src/main/res/values-el/strings.xml @@ -433,7 +433,7 @@ Συνειδητοποίησα ότι είναι κακό για την ιδιωτικότητά μου Άλλαξα γνώμη, δε θέλω να προβάλλεται πλέον δημόσια Συγγνώμη, αυτή η φωτογραφία δεν είναι ενδιαφέρουσα για μια εγκυκλοπαίδεια - Ανέβηκε από εμένα στο %1$s, χρησιμοποιήθηκε σε %2$d άρθρο/α + Ανέβηκε από εμένα στο %1$s, χρησιμοποιήθηκε σε %2$d άρθρο/α Καλώς ήρθατε στα Commons!\n\nΑνεβάστε τα πρώτα σας πολυμέσα πατώντας το κουμπί της προσθήκης. Δεν επιλέχθηκαν κατηγορίες Οι εικόνες χωρίς κατηγορίες χρησιμοποιούνται σπάνια. Θέλετε πράγματι να συνεχίσετε δίχως να επιλέξετε κατηγορίες; diff --git a/app/src/main/res/values-eo/strings.xml b/app/src/main/res/values-eo/strings.xml index d754c4a91..28f3f0eb8 100644 --- a/app/src/main/res/values-eo/strings.xml +++ b/app/src/main/res/values-eo/strings.xml @@ -408,7 +408,7 @@ Mi eksciis ke ĝi esta malbona por mia privateco Mi ŝanĝis mian preferon; mi ne plu volas ke ĝi estu publike videbla Domaĝe, ĉi tiu bildo ne estas interesa por enciklopedio - Alŝutita de mi je %1$s, uzata en %2$d artikolo(j). + Alŝutita de mi je %1$s, uzata en %2$d artikolo(j). Bonvenon al Komunejo!\n\nAlŝutu vian unuan aŭdvidaĵon per frapeto de la butono \'Aldoni\'. Neniu Elektita Kategorio Bildoj sen kategorioj estas malofte uzebla. Ĉu vi certas, ke vi volas daŭrigi sen elekti kategoriojn? diff --git a/app/src/main/res/values-es/strings.xml b/app/src/main/res/values-es/strings.xml index baed7bc1b..eb8caa3ba 100644 --- a/app/src/main/res/values-es/strings.xml +++ b/app/src/main/res/values-es/strings.xml @@ -464,7 +464,7 @@ Me di cuenta que es malo para mi privacidad Cambié de opinión, no quiero que siga siendo visible públicamente Lo sentimos, esta imagen no es interesante para una enciclopedia - Subida por mí mismo el %1$s, usado en %2$d artículo(s). + Subida por mí mismo el %1$s, usado en %2$d artículo(s). Te damos la bienvenida a Commons.\n\nCarga tu primer archivo mediante el botón Añadir. No hay categorías seleccionadas Las imágenes sin categorías raramente se pueden usar. ¿Seguro que quieres continuar sin seleccionar ninguna categoría? diff --git a/app/src/main/res/values-fa/strings.xml b/app/src/main/res/values-fa/strings.xml index 1cbdf5c47..32ec53638 100644 --- a/app/src/main/res/values-fa/strings.xml +++ b/app/src/main/res/values-fa/strings.xml @@ -386,7 +386,7 @@ فهمیدم که این برای حریم خصوصی من بد است. من تغییر عقیده دادم، نمی‌خواهم دیگر برای همه قابل‌مشاهده باشد. با پوزش، این تصویر برای یک دانشنامه مناسب نیست - بارگذاری‌شده توسط خودم در %1$s؛ استفاده‌شده در %2$d مقاله. + بارگذاری‌شده توسط خودم در %1$s؛ استفاده‌شده در %2$d مقاله. به ویکی‌انبار خوش آمدید!\n\nاولین فایلتان را با فشردن کلید اضافه بارگذاری کنید. رده‌ای انتخاب نشده است تصاویر بدون رده به ندرت قابل‌استفاده هستند. آیا مطمئنید که می‌خواهید بدون انتخاب رده ادامه دهید؟ diff --git a/app/src/main/res/values-fi/strings.xml b/app/src/main/res/values-fi/strings.xml index e65f05825..7561ca533 100644 --- a/app/src/main/res/values-fi/strings.xml +++ b/app/src/main/res/values-fi/strings.xml @@ -404,7 +404,7 @@ Huomasin, että se on haitallinen yksityisyydelleni. Muutin mieleni, en halua että se on enää julkisesti näkyvissä. Valitettavasti tämä kuva ei ole kiinnostava tietosanakirjaan - Minä olen ladannut kohteen %1$s käytetty %2$d artikkelissa. + Minä olen ladannut kohteen %1$s käytetty %2$d artikkelissa. Tervetuloa Commonsiin!\n\nTallenna ensimmäinen mediasi koskettamalla lisäyspainiketta. Luokkia ei valittu Kuvat, jotka eivät ole luokissa, ovat harvoin käyttökelpoisia. Haluatko varmasti jatkaa valitsematta luokkia? diff --git a/app/src/main/res/values-fr/strings.xml b/app/src/main/res/values-fr/strings.xml index 9603f1806..124139b9a 100644 --- a/app/src/main/res/values-fr/strings.xml +++ b/app/src/main/res/values-fr/strings.xml @@ -457,7 +457,7 @@ J’ai pensé que ce n’était pas bon pour ma vie privée J’ai changé d\'avis, je ne désire plus que cel soit visible publiquement Désolé mais cette image n’est pas intéressante pour une encyclopédie - Téléversé par moi-même le %1$s, utilisé dans %2$d article(s). + Téléversé par moi-même le %1$s, utilisé dans %2$d article(s). Bienvenue sur Commons !\n\nTéléversez votre premier média en tapant sur le bouton Ajouter. Aucune catégorie sélectionnée Les images sans catégories sont rarement utilisables. Voulez-vous vraiment continuer sans sélectionner des catégories  appropriées ? diff --git a/app/src/main/res/values-gl/strings.xml b/app/src/main/res/values-gl/strings.xml index 0cf3aed91..bac417647 100644 --- a/app/src/main/res/values-gl/strings.xml +++ b/app/src/main/res/values-gl/strings.xml @@ -394,7 +394,7 @@ Decateime de que prexudica a miña privacidade Cambiei de idea, non quero que siga sendo visible de forma pública Desculpas, esta imaxe non é interesante para unha enciclopedia - Cargada por min o %1$s, usada en %2$d artigo(s). + Cargada por min o %1$s, usada en %2$d artigo(s). Dámoslle a benvida ó Commonsǃ\n\nCargue o seu primeiro ficheiro premendo no botón Engadir. Non hai categorías seleccionadas As imaxes sen categorías só son utilizables en contadas ocasións. Está seguro de que quere continuar sen seleccionar categorías? diff --git a/app/src/main/res/values-hu/strings.xml b/app/src/main/res/values-hu/strings.xml index 629312a57..a52805873 100644 --- a/app/src/main/res/values-hu/strings.xml +++ b/app/src/main/res/values-hu/strings.xml @@ -362,7 +362,7 @@ Észleltem, hogy ez rossz a magánszférámnak Meggondoltam magam, nem szeretném, hogy nyilvánosan látható legyen Sajnos ez a fénykép nem érdekes egy enciklopédia számára - Feltöltve ekkorː %1$s, %2$d cikkben használva. + Feltöltve ekkorː %1$s, %2$d cikkben használva. Üdvözlünk a Commons-ban. \n\nA hozzáadás gombra koppintva feltöltheted első képedet. Nincs kiválasztott kategória A kategória nélküli képek ritkán használhatóak. Biztos, hogy kategória kiválasztása nélkül akarsz továbblépni? diff --git a/app/src/main/res/values-ia/strings.xml b/app/src/main/res/values-ia/strings.xml index 386bddf2e..d139124d2 100644 --- a/app/src/main/res/values-ia/strings.xml +++ b/app/src/main/res/values-ia/strings.xml @@ -407,7 +407,7 @@ Io ha comprendite que es mal pro mi vita private Io ha cambiate de idea, io non vole plus que illo sia publicamente visibile Iste imagine non es interessante pro un encyclopedia - Incargate per me mesme le %1$s, usate in %2$d articulo(s). + Incargate per me mesme le %1$s, usate in %2$d articulo(s). Benvenite a Commons!\n\nPro incargar tu prime file multimedial, tocca le button Adder. Necun categoria seligite Imagines sin categorias es rarmente usabile. Es tu secur de voler continuar sin seliger categorias? diff --git a/app/src/main/res/values-in/strings.xml b/app/src/main/res/values-in/strings.xml index e977e56d5..8a44ef84d 100644 --- a/app/src/main/res/values-in/strings.xml +++ b/app/src/main/res/values-in/strings.xml @@ -410,7 +410,7 @@ Saya menyadari itu buruk untuk privasi saya Saya berubah pikiran, saya tidak ingin itu terlihat publik lagi Maaf gambar ini tidak menarik untuk ensiklopedia - Diunggah saya sendiri pada %1$s, digunakan dalam %2$d artikel. + Diunggah saya sendiri pada %1$s, digunakan dalam %2$d artikel. Unggah media pertama Anda dengan mengetuk tombol. Tidak ada Kategori yang Dipilih Gambar tanpa kategori jarang dapat digunakan. Apakah Anda yakin ingin mengirim tanpa memilih kategori? diff --git a/app/src/main/res/values-io/strings.xml b/app/src/main/res/values-io/strings.xml index f4802280d..1a74c2cc4 100644 --- a/app/src/main/res/values-io/strings.xml +++ b/app/src/main/res/values-io/strings.xml @@ -410,7 +410,7 @@ Me konstatis ke ol esas mala por mea privateso Me chanjis mea ideo: me ne pluse deziras ke ol esos publike videbla Pardonez! Ca imajo ne esas interesanta por ula enciklopedio - Adjuntita da me, che %1$s, uzita en %2$d artiklo/artikli. + Adjuntita da me, che %1$s, uzita en %2$d artiklo/artikli. Bonveno a Commons!\n\nSendez vua unesma arkivo kliktanta sur butono \"adjuntez\" (\'\'add\'\'). Nula kategorio selektita Imaji sen kategorii rare esas uzebla. Ka vu fakte deziras sendar ol sen selektar irga kategorio? diff --git a/app/src/main/res/values-is/strings.xml b/app/src/main/res/values-is/strings.xml index ad215efbc..427e3cec3 100644 --- a/app/src/main/res/values-is/strings.xml +++ b/app/src/main/res/values-is/strings.xml @@ -392,7 +392,7 @@ Ég áttaði mig á að þetta væri slæmt fyrir gagnaleynd mína Ég skipti um skoðun, ég vil ekki að hún sé lengur öllum sýnileg Afsakið, þessi mynd hefur ekkert gildi fyrir alfræðirit - Sent inn af mér sjálfum %1$s, notað í %2$d grein(um). + Sent inn af mér sjálfum %1$s, notað í %2$d grein(um). Velkomin í Commons!\n\nSendu inn fyrstu margmiðlunargögnin þín með því að ýta á viðbætingarhnappinn. Engir flokkar valdir Myndir án flokka eru sjaldnast nýtilegar. Ertu viss um að þú viljir halda áfram án þess að velja flokka? diff --git a/app/src/main/res/values-it/strings.xml b/app/src/main/res/values-it/strings.xml index b923b699b..fde465b06 100644 --- a/app/src/main/res/values-it/strings.xml +++ b/app/src/main/res/values-it/strings.xml @@ -289,7 +289,7 @@ Invia il file di registro agli sviluppatori tramite email per aiutarli a risolvere i problemi con l\'applicazione. Nota: i log potrebbero contenere informazioni di identificazione Nessun browser web trovato per aprire l\'URL Errore! URL non trovato - Proponi per la Cancellazione + Proponi per la cancellazione Questa immagine è stata proposta per la cancellazione. Vedi la pagina web per i dettagli Salta @@ -433,7 +433,7 @@ Mi sono reso conto che viola la mia privacy Ho cambiato idea, non voglio più che sia pubblicamente visibile Spiacente, questa immagine non è interessante per un\'enciclopedia - Caricato da me stesso il %1$s, utilizzato in %2$d voce/i. + Caricato da me stesso il %1$s, utilizzato almeno in %2$d voce/i. Benvenuto su Commons!\n\nCarica il tuo primo file multimediale toccando il pulsante aggiungi. Nessuna categoria selezionata Le immagini senza categorie sono raramente utilizzabili. Sei sicuro di voler continuare senza selezionare le categorie? @@ -836,4 +836,5 @@ Mostra nelle vicinanze Creato e caricato da: %1$s Creato da %1$s e caricato da %2$s + Proposto per la cancellazione diff --git a/app/src/main/res/values-iw/strings.xml b/app/src/main/res/values-iw/strings.xml index 6aecaf1e8..355e12dc6 100644 --- a/app/src/main/res/values-iw/strings.xml +++ b/app/src/main/res/values-iw/strings.xml @@ -444,7 +444,7 @@ הבנתי שזה לא טוב לפרטיות שלי התחרטתי, אינני רוצה עוד שהיא תוצג לציבור סליחה, התמונה אינה מעניינת עבור אנציקלופדיה - הועלה על ידי ב-%1$s, בשימוש ב-%2$d מאמר(ים) + הועלה על ידי ב־%1$s, בשימוש ב־%2$d מאמר(ים) לפחות. ברוך בואך לוויקישיתוף!\n\nכדי להעלות את המדיה הראשונה שלך, נא להקיש על כפתור ההוספה. לא נבחרו קטגוריות תמונות ללא קטגוריות בדרך כלל אינן שימושיות. להמשיך ללא בחירת קטגוריות? @@ -862,4 +862,5 @@ בתצוגת בסביבה נוצר והועלה על־ידי: %1$s נוצר על־ידי %1$s והועלה על־ידי %2$s + העמדה למחיקה diff --git a/app/src/main/res/values-ja/strings.xml b/app/src/main/res/values-ja/strings.xml index 662ddcd23..f051ba27a 100644 --- a/app/src/main/res/values-ja/strings.xml +++ b/app/src/main/res/values-ja/strings.xml @@ -408,7 +408,7 @@ 自分のプライバシーを傷つけると気がつきました 以前とは考えが変わりました。今後は公衆の場で自分の画像を公開したくありません 残念ですがこの画像は百科事典の目的に合いません - 私本人が%1$sにアップロードし、%2$d件の記事で使用されました。 + 私本人が%1$sにアップロードし、%2$d件の記事で使用されました。 コモンズへようこそ!\n\n追加ボタンを押して、ぜひご自分のメディアの初投稿をしましょう。 カテゴリが選択されていません カテゴリを指定しない画像は使用されることがほとんどありません。ほんとうにカテゴリを選択しないまま作業を続けますか? diff --git a/app/src/main/res/values-krc/strings.xml b/app/src/main/res/values-krc/strings.xml index 81d8842ff..f8fb0e45b 100644 --- a/app/src/main/res/values-krc/strings.xml +++ b/app/src/main/res/values-krc/strings.xml @@ -409,7 +409,7 @@ Ташалыгъыма аман болгъанын ангыладым Сагъышымы тюрлендирдим, энди хар кимге кёрюннюгюмю излемейме Кечериксиз, бу сурат энциклопедия ючюн эс бёлюрча тюлдю - Кесим джюклеген %1$s сайтха джюкленди эмда %2$d статьяда хайырландырылды. + Кесим джюклеген %1$s сайтха джюкленди эмда %2$d статьяда хайырландырылды. Гёзеннге Хош Келигиз!\n\nКъош тиекге басыб биринчи медиагъызны джюклегиз. Категорияла Сайланмадыла Категориясыз суратла аз хайырланадыла. Категория сайламай бардырыргъа излегенигизге ишексизмисиз? diff --git a/app/src/main/res/values-kus/strings.xml b/app/src/main/res/values-kus/strings.xml index ede3cbe19..75447a96b 100644 --- a/app/src/main/res/values-kus/strings.xml +++ b/app/src/main/res/values-kus/strings.xml @@ -345,7 +345,7 @@ M tɛn\'ɛs ye di na an bɛ\'ɛd tisi m mɛŋ gu\'udim yɛla M tiaki m pʋtɛ\'ɛr, m pʋ lɛm bɔɔd ye di bɛ zin\'ikanɛ ka sɔ\' wʋsa na nyɛ ya\'asa Gafara footo kaŋa pʋ nar ye di bɛ encyclopedia - M kpɛ\'ɛsidinɛ %1$s m mɛŋ, ka nɔki tʋm %2$d atikil (nam) ni. + M kpɛ\'ɛsidinɛ %1$s m mɛŋ, ka nɔki tʋm %2$d atikil (nam) ni. Commons pʋ\'ʋsidif ken-ken!\n\nKpɛn\'ɛmi fʋ yiiga media dɔlisid fʋn na din paasim la. Fʋ pʋ gaŋ buudsi\'a Fʋ pʋ gaŋ banɛ nwan taaba diff --git a/app/src/main/res/values-lb/strings.xml b/app/src/main/res/values-lb/strings.xml index afca8fcd4..599970d92 100644 --- a/app/src/main/res/values-lb/strings.xml +++ b/app/src/main/res/values-lb/strings.xml @@ -134,7 +134,7 @@ Eroflueden Standardlizenz Viregen Titel a Beschreiwung benotzen - Faarfschema + Faarfscheema Attribution-ShareAlike 4.0 Attribution 4.0 CC Attribution-ShareAlike 3.0 @@ -295,6 +295,7 @@ Virufueren Richteg Äntwert Falsch Äntwert + Ass et an der Rei fir dëse Screenshot eropzelueden? App deelen Dréinen Keng Biller an dëser Géigend @@ -332,7 +333,7 @@ Ech hu gemierkt, datt et schlecht fir meng Privatsphär ass Ech hu meng Meenung geännert, ech wëll net méi, datt et ëffentlech siichtbar ass Pardon, dëst Bild ass net interessant fir eng Enzyklopedie - Vu mir selwer de(n) %1$s eropgelueden, benotzt a(n) %2$d Artikel(en). + Vu mir selwer de(n) %1$s eropgelueden, benotzt a(n) %2$d Artikel(en). Keng Kategorien erausgesicht Biller ouni Kategorië si meeschtens net benotzbar. Sidd Dir sécher, datt Dir weider maache wëllt ouni eng Kategorie auszewielen? Eroplueden ofbriechen diff --git a/app/src/main/res/values-lt/strings.xml b/app/src/main/res/values-lt/strings.xml index 04c1f931f..e148a0a20 100644 --- a/app/src/main/res/values-lt/strings.xml +++ b/app/src/main/res/values-lt/strings.xml @@ -396,7 +396,7 @@ Supratau, kad tai kenkia mano privatumui Persigalvojau, nebenoriu, kad tai būtų viešai matoma Atsiprašome, ši nuotrauka neįdomi enciklopedijai - Įkelta mano %1$s, naudojama %2$d straipsnyje(-iuose). + Įkelta mano %1$s, naudojama %2$d straipsnyje(-iuose). Sveiki atvykę į Vikiteką!\n\nĮkelkite pirmąjį failą bakstelėdami pridėjimo mygtuką. Nepasirinkta jokių kategorijų Vaizdai be kategorijų retai naudojami. Ar tikrai norite tęsti nepasirinkę kategorijų? diff --git a/app/src/main/res/values-mk/strings.xml b/app/src/main/res/values-mk/strings.xml index b3058b81e..a0d1dd6ef 100644 --- a/app/src/main/res/values-mk/strings.xml +++ b/app/src/main/res/values-mk/strings.xml @@ -413,7 +413,7 @@ Сфатив дека ќе ми ја наруши приватноста Се премислив. Не сакам повеќе да биде видлива За жал, сликава не е соодветна за енциклопедија - Подигнато од мене %1$s, кое се користи во %2$d статии. + Подигнато од мене %1$s, кое се користи во %2$d статии. Добре дојдовте на Ризницата!\n\nПодигнете ја вашата прва слика или снимка допирајќи го копчето за додавање. Нема избрано категории Некатегоризираните слики се слабо употребливи. Дали сигурно сакате да продолжите без да ставите категории? diff --git a/app/src/main/res/values-nb/strings.xml b/app/src/main/res/values-nb/strings.xml index 5bb18d8e9..81fb7030b 100644 --- a/app/src/main/res/values-nb/strings.xml +++ b/app/src/main/res/values-nb/strings.xml @@ -408,7 +408,7 @@ Jeg innså at det er dårlig for mitt personvern Jeg ombestemte meg, jeg ønsker ikke at det skal være offentlig synlig lenger Beklager, dette bildet har ingen ensyklopedisk interesse - Lastet opp av meg selv %1$s, brukt i %2$d artikkel/artikler. + Lastet opp av meg selv %1$s, brukt i %2$d artikkel/artikler. Velkommen til Commons!\n\nLast opp din første mediefil ved å trykke på legg til-knappen. Ingen kategorier valgt Bilder uten kategorier er sjelden nyttige. Er du sikker på at du vil fortsette uten å velge kategorier? diff --git a/app/src/main/res/values-nl/strings.xml b/app/src/main/res/values-nl/strings.xml index b800a6c20..c4cad617e 100644 --- a/app/src/main/res/values-nl/strings.xml +++ b/app/src/main/res/values-nl/strings.xml @@ -435,7 +435,7 @@ Ik realiseerde me dat het slecht is voor mijn privacy Ik ben van gedachten veranderd, ik wil niet dat het nog publiekelijk zichtbaar is Sorry, deze foto is niet interessant voor een encyclopedie - Geüpload door mijzelf op %1$s, gebruikt in %2$d artikel(en). + Geüpload door mijzelf op %1$s, gebruikt in %2$d artikel(en). Welkom bij Commons!\n\nUpload je eerste media door op de knop Toevoegen te klikken. Geen categorieën geselecteerd Afbeeldingen zonder categorieën zijn zelden bruikbaar. Weet je zeker dat je verder wilt gaan zonder categorieën te selecteren? @@ -807,6 +807,7 @@ Overleg Schrijf iets over het item \'%1$s\'. Dit is openbaar zichtbaar. ‘%1$s’ bestaat niet meer, er kan nooit een foto van gemaakt worden. + ‘%1$s’ bevindt zich ergens anders. ‘%1$s’ bevindt zich ergens anders. Geef hieronder de juiste plaats aan en noteer, indien mogelijk, de juiste breedte- en lengtegraad. Ander probleem of andere informatie (verklaar hieronder). Uw feedback wordt op de volgende wikipagina geplaatst: <a href=\"https://commons.wikimedia.org/wiki/Commons:Mobile_app/Feedback\">Commons:Mobile app/Feedback</a> diff --git a/app/src/main/res/values-nqo/strings.xml b/app/src/main/res/values-nqo/strings.xml index f7ddc958f..99923ca5f 100644 --- a/app/src/main/res/values-nqo/strings.xml +++ b/app/src/main/res/values-nqo/strings.xml @@ -360,7 +360,7 @@ ߒ ߧߴߊ߬ ߖߊ߬ߕߋ߬ ߟߊ߫ ߒߞߏ߫ ߊ߬ ߡߊ߲ߘߌ߫ ߒ ߖߘߍ߬ߞߊ߬ߣߌ߲߬ ߞߏ ߟߎ߫ ߟߊ߫. ߒ ߓߘߊ߫ ߒ ߡߙߌߦߊ ߡߊߝߊ߬ߟߋ߲߬߸ ߒ ߕߴߊ߬ ߝߵߊ߬ ߦߋ߫ ߜߍߘߐ߫ ߦߋߕߊ߫ ߘߴߏ߬ ߞߐ߫ ߡߎ߰ߡߍ߫ ߤߊߞߍ߬ߕߏ߬ ߔߘߋߞߎ ߘߐ߱ߘߏ߲߬ߣߍ߲߫ ߕߍ߫ ߖߌ߬ߦߊ߬ߓߍ ߣߌ߲߬ ߠߊ߫ - ߡߍ߲ ߠߎ߬ ߟߊߦߟߍ߬ߣߍ߲߬ ߦߋ߫ ߒ ߖߘߍ߬ߞߊߣߌ߲ ߓߟߏ߫ %1$s ߟߊ߫߸ ߣߴߊ߬ ߟߊߓߊ߯ߙߊߣߍ߲߫ ߦߋ߫ %2$d ߞߎߡߘߊ (ߟߎ߬) ߘߐ߫. + ߡߍ߲ ߠߎ߬ ߟߊߦߟߍ߬ߣߍ߲߬ ߦߋ߫ ߒ ߖߘߍ߬ߞߊߣߌ߲ ߓߟߏ߫ %1$s ߟߊ߫߸ ߣߴߊ߬ ߟߊߓߊ߯ߙߊߣߍ߲߫ ߦߋ߫ %2$d ߞߎߡߘߊ (ߟߎ߬) ߘߐ߫. ߌ ߣߌ߫ ߛߣߍ߫ ߞߐߡߐ߲ߛ ߞߊ߲߬߹ \nߌ ߟߊ߫ ߡߋߘߌߦߊ߫ ߝߟߐ ߟߊߦߟߍ߬ ߝߙߊ߬ߟߌ ߞߎ߬ߘߎ ߛߐ߲߬ߞߌ߲ ߠߊ߫. ߦߌߟߡߊ߫ ߡߊ߫ ߓߊߕߐ߬ߡߐ߲߬ ߦߌߟߡߊ߫ ߕߍ߫ ߖߌ߬ߦߊ߬ߓߍ ߡߍ߲ ߠߊ߫߸ ߏ߬ ߟߊߓߊ߯ߙߊ ߜߟߍ߬ߡߊ߲߫. ߌ ߦߴߊ߬ ߝߍ߬ ߞߵߊ߬ ߞߘߊߡߊ߫ ߓߊ߬ ߞߵߊ߬ ߕߘߍ߬ ߌ ߡߊ߫ ߦߌߟߡߊ ߘߏ߫ ߓߊߕߐ߬ߡߐ߲߬؟ diff --git a/app/src/main/res/values-pl/strings.xml b/app/src/main/res/values-pl/strings.xml index 29cbf44c9..198b57c7f 100644 --- a/app/src/main/res/values-pl/strings.xml +++ b/app/src/main/res/values-pl/strings.xml @@ -450,7 +450,7 @@ Zdałem sobie sprawę, że jest to szkodliwe dla mojej prywatności Zmieniłem zdanie, nie chcę, aby było to już publicznie widoczne Przepraszamy, to zdjęcie nie jest interesujące dla encyklopedii - Przesłane przeze mnie na %1$s, używane w artykułach %2$d. + Przesłane przeze mnie na %1$s, używane w artykułach %2$d. Prześlij swoje pierwsze multimedia, dotykając przycisku Dodaj. Nie wybrano kategorii Obrazy bez kategorii rzadko nadają się do użycia. Czy na pewno chcesz kontynuować bez wybierania kategorii? diff --git a/app/src/main/res/values-pms/strings.xml b/app/src/main/res/values-pms/strings.xml index b0ecdcfe1..9b12b3989 100644 --- a/app/src/main/res/values-pms/strings.xml +++ b/app/src/main/res/values-pms/strings.xml @@ -411,7 +411,7 @@ I l\'hai pensa ch\'a l\'era nen bon për mia arservatëssa I l\'hai cangià d\'idèja, i veui pi nen ch\'a sia visìbil ëd fasson pùblica Am dëspias, ma costa plancia a l\'é nen anteressanta për n\'enciclopedìa - Carià da mi-midem ai %1$s, dovrà an %2$d artìcoj. + Carià da mi-midem ai %1$s, dovrà an almanch %2$d artìcoj. Bin-ëvnù su Commons!\n\nCh\'a caria sò prim archivi multimedial an sgnacand an sël boton për gionté. Gnun-a categorìa selessionà Le plance sensa categorìa, a so da ràir dovràbij. É-lo sigur ëd vorèj continué sensa serne dle categorìe? @@ -821,4 +821,5 @@ Smon-e andrinta a Nearby Creà e carià da: %1$s Creà da %1$s e carià da %2$s + Nominà për esse dëscancelà diff --git a/app/src/main/res/values-pt-rBR/strings.xml b/app/src/main/res/values-pt-rBR/strings.xml index 947d65ace..7f45ab484 100644 --- a/app/src/main/res/values-pt-rBR/strings.xml +++ b/app/src/main/res/values-pt-rBR/strings.xml @@ -427,7 +427,7 @@ Eu percebi que é ruim para minha privacidade Eu mudei de ideia, eu não quero mais que seja publicamente visível Desculpe, esta foto não é interessante para uma enciclopédia - Carregada por mim em %1$s, usada em %2$d artigo(s). + Carregada por mim em %1$s, usada em %2$d artigo(s). Bem-vindo ao Commons!\n\nCarregue sua primeira mídia tocando no botão Adicionar. Nenhuma categoria selecionada Imagens sem categorias raramente são utilizáveis. Tem certeza de que deseja continuar sem selecionar categorias? diff --git a/app/src/main/res/values-pt/strings.xml b/app/src/main/res/values-pt/strings.xml index 15c5983c4..f2153caa6 100644 --- a/app/src/main/res/values-pt/strings.xml +++ b/app/src/main/res/values-pt/strings.xml @@ -423,7 +423,7 @@ Apercebi-me de que prejudica a minha privacidade Mudei de ideias, já não quero possa ser vista pelo público Desculpe, esta fotografia não tem interesse para uma enciclopédia - Carregada por mim em %1$s, usada em %2$d artigo(s). + Carregada por mim em %1$s, usada em %2$d artigo(s). Bem-vindo(a) à wiki Commons!\n\nCarregue o seu primeiro ficheiro multimédia tocando no botão Adicionar. Não foi selecionada nenhuma categoria As imagens sem categorias só raramente são utilizáveis. Tem a certeza de que deseja continuar sem selecionar categorias? diff --git a/app/src/main/res/values-ro/strings.xml b/app/src/main/res/values-ro/strings.xml index 550f09968..602fb69bf 100644 --- a/app/src/main/res/values-ro/strings.xml +++ b/app/src/main/res/values-ro/strings.xml @@ -394,7 +394,7 @@ Mi-am dat seama că este rău pentru intimitatea mea M-am răzgândit, nu mai vreau să fie publică Ne pare rău că această imagine nu este interesantă pentru o enciclopedie - Încărcat de mine pe %1$s, folosit în %2$d articol(e). + Încărcat de mine pe %1$s, folosit în %2$d articol(e). Bine ați venit la Commons!\n\nÎncărcați primul dvs. media apăsând pe butonul de adăugare. Nici o categorie selectată Imaginile fără categorii sunt rareori utilizabile. Sigur doriți să continuați fără a selecta categorii? diff --git a/app/src/main/res/values-ru/strings.xml b/app/src/main/res/values-ru/strings.xml index ecef5c74b..7a0e1c247 100644 --- a/app/src/main/res/values-ru/strings.xml +++ b/app/src/main/res/values-ru/strings.xml @@ -470,7 +470,7 @@ Я осознал, что это разглашает мою приватность Я передумал. Больше не хочу чтобы было доступным публично Извините, это изображение не для энциклопедии - Загружено мною на сайт «%1$s» и использовано в {{PLURAL:%2$d|one=одной статье|%2$d статьях}}. + Загружено мною на сайт «%1$s» и использовано в {{PLURAL:%2$d|one=одной статье|%2$d статьях}}. Добро пожаловать на Викисклад!\n\nЗагрузите ваш первый файл, нажав на кнопку добавления. Не выбраны категории Изображения без категорий используются редко. Вы уверены, что хотите продолжить, не выбрав категории? diff --git a/app/src/main/res/values-sk/strings.xml b/app/src/main/res/values-sk/strings.xml index 5def2523d..cbaca17c5 100644 --- a/app/src/main/res/values-sk/strings.xml +++ b/app/src/main/res/values-sk/strings.xml @@ -423,7 +423,7 @@ Uvedomil/a som si, že je to zlé pre moje súkromie Zmenil/a som názor, nechcem byť viac verejne viditeľný/á Prepáčte, tento obrázok pre encyklopédiu nie je zaujímavý - Nahrané mnou %1$s, použité v %2$d článku/článkoch. + Nahrané mnou %1$s, použité v %2$d článku/článkoch. Vitajte na Commons!\n\nNahrajte svoje prvé súbory ťuknutím na tlačidlo pridať. Neboli vybrané žiadne kategórie Obrázky bez kategórií sú zriedka použiteľné. Naozaj chcete nahrať obrázok bez výberu kategórií? diff --git a/app/src/main/res/values-sl/strings.xml b/app/src/main/res/values-sl/strings.xml index d81d5ba73..ebcfba109 100644 --- a/app/src/main/res/values-sl/strings.xml +++ b/app/src/main/res/values-sl/strings.xml @@ -429,7 +429,7 @@ Spoznal sem, da je to slabo za mojo zasebnost Premislil_a sem si, ne želim več biti javno viden_na Ta slika žal ni zanimiva za enciklopedijo. - Sliko sem naložil/a sam/a dne %1$s. Uporablja se v %2$d članku(ih). + Sliko sem naložil/a sam/a dne %1$s. Uporablja se v %2$d članku(ih). Pozdravljeni v Wikimedijini zbirki!\n\nTapnite gumb dodaj in naložite svojo prvo predstavnostno datoteko. Izbrana ni nobena kategorija Slike brez kategorij so le redko uporabne. Ali res želite nadaljevati brez izbire kategorij? diff --git a/app/src/main/res/values-sr/strings.xml b/app/src/main/res/values-sr/strings.xml index cb09f2f80..a41096306 100644 --- a/app/src/main/res/values-sr/strings.xml +++ b/app/src/main/res/values-sr/strings.xml @@ -409,7 +409,7 @@ Схватио/ла сам да је лоша за моју приватност Променио/ла сам мишљење, не желим да више јавно буду видљиве Ова слика није занимљива за енциклопедију - Отпремио/ла сам на %1$s, користи се у чланку/цима — %2$d. + Отпремио/ла сам на %1$s, користи се у чланку/цима — %2$d (бар). Добро дошли на Оставу!\n\nОтпремите прве медије додиром дугмета за додавање. Категорије нису изабране Слике без категорија ретко су употребљиве. Заиста желите да наставите без избора категорија? @@ -751,4 +751,5 @@ Поднапис копиран Направио и отпремио: %1$s Направио %1$s а отпремио %2$s + Номинуј за брисање diff --git a/app/src/main/res/values-sv/strings.xml b/app/src/main/res/values-sv/strings.xml index 28a0896c0..740304f7c 100644 --- a/app/src/main/res/values-sv/strings.xml +++ b/app/src/main/res/values-sv/strings.xml @@ -420,7 +420,7 @@ Jag insåg att den är dålig för mitt privatliv Jag ändrade mig, jag vill inte längre att den ska vara synlig offentligt Tyvärr, denna bild är inte intressant för en encyklopedi - Laddades upp av mig den %1$s och används i %2$d artiklar. + Laddades upp av mig den %1$s och används i %2$d artiklar. Välkommen till Commons!\n\nLadda upp din första mediafil genom att trycka på knappen för att lägga till. Inga kategorier har valts Bilder utan kategorier används sällan. Är du säker på att du vill fortsätta utan att välja kategorier? diff --git a/app/src/main/res/values-te/strings.xml b/app/src/main/res/values-te/strings.xml index ba67a7b35..0d4b323f4 100644 --- a/app/src/main/res/values-te/strings.xml +++ b/app/src/main/res/values-te/strings.xml @@ -377,7 +377,7 @@ నా అంతరంగికతకు ఇది చేటు అని గుర్తించాను నేణు మనసు మార్చుకున్నాను, బహిరంగంగా అందరికీ కనబడేలా ఉండను సారీ, ఈ బొమ్మ విజ్ఞాన సర్వస్వానికి ఆసక్తి కలిగేలా లేదు - %1$s న నేనే ఎక్కించాను, %2$d వ్యాసాల్లో వాడారు. + %1$s న నేనే ఎక్కించాను, %2$d వ్యాసాల్లో వాడారు. కామన్స్‌కు స్వాగతం!\n\nచేర్చు బొత్తాన్ని నొక్కి మీ మొదటి మీడియాను ఎక్కించండి. వర్గాలేమీ ఎంచుకోలేదు వర్గాల్లేని బొమ్మలను అరుదుగా వాడగలం. వర్గాలేమీ ఎంచుకోకుండానే కొనసాగాలని అనుకుంటున్నారా? diff --git a/app/src/main/res/values-tr/strings.xml b/app/src/main/res/values-tr/strings.xml index 2e2b13216..c741e89c4 100644 --- a/app/src/main/res/values-tr/strings.xml +++ b/app/src/main/res/values-tr/strings.xml @@ -442,7 +442,7 @@ Gizliliğimin kötü olduğunu fark ettim Fikrimi değiştirdim, artık herkese görünür olmasını istemiyorum Üzgünüz, bu resim bir ansiklopedi için ilginç değil - Kendi başıma %1$s üzerine yüklendi, %2$d makalede kullanıldı. + Kendi başıma %1$s üzerine yüklendi, %2$d makalede kullanıldı. Commons\'a Hoş Geldiniz!\n\nEkle düğmesine dokunarak ilk medyanızı yükleyin. Kategori Seçilmedi Kategorisiz görüntüler nadiren kullanılabilir. Kategori seçmeden devam etmek istediğinizden emin misiniz? diff --git a/app/src/main/res/values-uk/strings.xml b/app/src/main/res/values-uk/strings.xml index c457636ff..f005f2969 100644 --- a/app/src/main/res/values-uk/strings.xml +++ b/app/src/main/res/values-uk/strings.xml @@ -435,7 +435,7 @@ Мені стало зрозуміло, що це шкодить моїй приватності Моя думка змінилась, я не хочу, щоб це було доступно публічно Перепрошую, це зображення нецікаве для енциклопедії - Завантажено мною на сайт «%1$s» та використано у {{PLURAL:%2$d|one=одній статті|%2$d статтях}}. + Завантажено мною на сайт «%1$s» та використано у {{PLURAL:%2$d|one=одній статті|%2$d статтях}}. Ласкаво просимо до Вікісховища!\n\nЗавантажте Ваш перший медіафайл, натиснувши кнопку додавання. Жодної категорії не вибрано Зображення без категорій рідко використовуються. Ви впевнені, що хочете продовжити без вказаних категорій? diff --git a/app/src/main/res/values-vec/strings.xml b/app/src/main/res/values-vec/strings.xml index 66905e758..17765b3ce 100644 --- a/app/src/main/res/values-vec/strings.xml +++ b/app/src/main/res/values-vec/strings.xml @@ -344,7 +344,7 @@ Me so rendesto conto che ła xé on małe par ła me privacy Go canbià idea, no vujo pì che ła sia vardabiłe da tuti Ne despiaxe, sta imajine no ła xé intaresante par na ençiclopedia. - Cargà da mi el %1$s, doparà in %2$d voxe. + Cargà da mi el %1$s, doparà in %2$d voxe. Benvegnuo so Commons!\n\nCarga el to primo file multimediałe strucando el boton \"xonta\". Nisuna categoria sełesionà Le imajini sensa categorie łe xé poche olte doparabiłi. Sito seguro de vołer enviar sensa sełesionar categorie? diff --git a/app/src/main/res/values-x-invalidLanguageCode/error.xml b/app/src/main/res/values-x-invalidLanguageCode/error.xml new file mode 100644 index 000000000..f4e2fe125 --- /dev/null +++ b/app/src/main/res/values-x-invalidLanguageCode/error.xml @@ -0,0 +1,10 @@ + + + + کامَنٕز گوو رُکِتھ + Oops. کیہہ تام گوو غلط! + ؤنِیوٚ اَسہِ توٚہہِ کیاہ ٲسِیوٚ کران، تہٕ کٕریٚو تہِ اَسہِ سٕتی شیر بذریعہِ برقی خط. یُس مَدَتھ کَرِ اَسہِ اَتھ شہَرنَس منٛز! + شُکریہ! + diff --git a/app/src/main/res/values-xmf/strings.xml b/app/src/main/res/values-xmf/strings.xml index 3f9b0e59f..649cab67d 100644 --- a/app/src/main/res/values-xmf/strings.xml +++ b/app/src/main/res/values-xmf/strings.xml @@ -375,7 +375,7 @@ მიბრჩქინე, ნამჷდა თენა ჩქიმი კონფიდენციალურალაშო გლახა იჸუაფუდჷნ არზი დიბთირე დო შუროთ ვამოკო თინა ოჯარალეთ იძირებედასჷნ მორდება, ათე სურათი ენციკლოპედიაშო ვა რე საინტერესო - ჩქიმი ენახარგა რე %1$s-შა, გჷმორინაფილი რე %2$d სტატიას. + ჩქიმი ენახარგა რე %1$s-შა, გჷმორინაფილი რე %2$d სტატიას. მოზოჯით ვიკიოწკარუეშა!\n\nგეხარგეთ თქვანი პირველი ფაილი, ქეგუნჭირით კონჭის გეძინა. კატეგორია ვა რე გიშაგორილი სურათეფი კატეგორიზაციაშ უმუშო შხირას მერკეთ გჷმორინაფონი რე. დასურო გონებჷნანო კატეგორიეფიშ მეწურაფაშ უმუშო გაგჷნძარაფა? diff --git a/app/src/main/res/values-zh-rTW/strings.xml b/app/src/main/res/values-zh-rTW/strings.xml index 0434441c8..5c35be3e6 100644 --- a/app/src/main/res/values-zh-rTW/strings.xml +++ b/app/src/main/res/values-zh-rTW/strings.xml @@ -436,7 +436,7 @@ 我認為這會干擾到我的隱私 我改變我的主意,我不想要公開可見 真抱歉,此圖片對於百科全書沒有意義 - 我自己在%1$s上上傳,在%2$d篇條目中使用。 + 我自己在%1$s上上傳,在%2$d篇條目中使用。 歡迎來到維基共享資源!\n\n透過觸碰添加按鈕來上傳您的首個多媒體內容。 未選擇分類 不帶分類的圖片很難有機會被利用到,您確定您要不選擇分類來繼續嗎? diff --git a/app/src/main/res/values-zh/strings.xml b/app/src/main/res/values-zh/strings.xml index 930c9e4fb..452e54563 100644 --- a/app/src/main/res/values-zh/strings.xml +++ b/app/src/main/res/values-zh/strings.xml @@ -470,7 +470,7 @@ 我意识到这对我的隐私不利 我改变了主意,我不想再让公众看到它了 对不起,这幅图对百科全书没什么意思 - 由我自己上传在%1$s,使用于%2$d个条目。 + 由我自己上传在%1$s,使用于%2$d个条目。 欢迎使用共享资源!\n\n通过点击添加按钮以上传您的首个媒体。 未提交分类 不带分类的图片很难有机会被利用到,您确定您要不选择分类来继续吗? From 9a94dc2548aa54cf994773ee6308fd0a038b9029 Mon Sep 17 00:00:00 2001 From: Jason-Whitmore Date: Sun, 29 Jun 2025 02:30:10 -0700 Subject: [PATCH 006/129] Fixes Issue 6312: GPS has huge error and does not update (in Nearby) (#6352) * NearbyParentFragment.kt: add helper methods for user location overlays and accuracy data. Before this commit, the code used to create the user location overlays was in multiple places. Additionally, there was no easy way to access the location accuracy. This commit places the user location overlay creation code into helper methods, as well as adding a new location accuracy getter method. These methods can now be used to refactor other parts of the file. * NearbyParentFragment.kt: create method to update user location overlays Before this commit, there was no easy way to update the user location overlays. This commit adds the updateUserLocationOverlays() method, which will properly replace the old user location overlays with new ones. It will also add the overlays if they do not already exist. * NearbyParentFragment.kt: replace old code with calls to updateUserLocationOverlays() This commit completes the refactor and fixes the issue of the user overlays not updating. The new method updateUserLocationOverlays is called to refactor and simplify old code. * Removal of file is not related to the issue, but is needed for project to compile and run. * NearbyParentFragment.kt: fix bug where multiple user location overlays would appear Before this commit, the user could see multiple user location overlays if they paused the app and reopened it when there are no Places/pins on the map. This was caused by a linear search failing to identify the target overlay because it compared Drawables between two Overlays, which was unreliable. This commit contains a better solution for replacing existing user location overlays by adding 2 instance variables to keep track of the overlays. The position of these overlays in the overlay list can then be found by using indexOf() with these instance variables rather than the linear search that was implemented before. Some refactoring was also done. --------- Co-authored-by: Nicolas Raoul --- .../nearby/fragments/NearbyParentFragment.kt | 226 ++++++++++++------ 1 file changed, 154 insertions(+), 72 deletions(-) diff --git a/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt b/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt index 5b52c0ce5..26875927e 100644 --- a/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt @@ -123,6 +123,7 @@ import org.osmdroid.views.CustomZoomButtonsController import org.osmdroid.views.MapView import org.osmdroid.views.overlay.MapEventsOverlay import org.osmdroid.views.overlay.Marker +import org.osmdroid.views.overlay.Overlay import org.osmdroid.views.overlay.ScaleBarOverlay import org.osmdroid.views.overlay.ScaleDiskOverlay import org.osmdroid.views.overlay.TilesOverlay @@ -267,6 +268,9 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), private var dataList: MutableList? = null private var bottomSheetAdapter: BottomSheetAdapter? = null + private var userLocationOverlay: Overlay? = null + private var userLocationErrorOverlay: Overlay? = null + private val galleryPickLauncherForResult = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { result -> controller?.handleActivityResultWithCallback( @@ -724,7 +728,7 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), val targetP = GeoPoint(target.latitude, target.longitude) mapCenter = targetP binding?.map?.controller?.setCenter(targetP) - recenterMarkerToPosition(targetP) + updateUserLocationOverlays(targetP, true) if (!isCameFromExploreMap()) { moveCameraToPosition(targetP) } @@ -1863,6 +1867,8 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), lastKnownLocation = latLng NearbyController.currentLocation = lastKnownLocation presenter!!.updateMapAndList(locationChangeType) + + updateUserLocationOverlays(GeoPoint(latLng.latitude, latLng.longitude), true) } override fun onLocationChangedSignificantly(latLng: LatLng) { @@ -2644,43 +2650,14 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), */ override fun clearAllMarkers() { binding!!.map.overlayManager.clear() - binding!!.map.invalidate() - val geoPoint = mapCenter - if (geoPoint != null) { - val diskOverlay = - ScaleDiskOverlay( - this.context, - geoPoint, 2000, UnitOfMeasure.foot - ) - val circlePaint = Paint() - circlePaint.color = Color.rgb(128, 128, 128) - circlePaint.style = Paint.Style.STROKE - circlePaint.strokeWidth = 2f - diskOverlay.setCirclePaint2(circlePaint) - val diskPaint = Paint() - diskPaint.color = Color.argb(40, 128, 128, 128) - diskPaint.style = Paint.Style.FILL_AND_STROKE - diskOverlay.setCirclePaint1(diskPaint) - diskOverlay.setDisplaySizeMin(900) - diskOverlay.setDisplaySizeMax(1700) - binding!!.map.overlays.add(diskOverlay) - val startMarker = Marker( - binding!!.map - ) - startMarker.position = geoPoint - startMarker.setAnchor( - Marker.ANCHOR_CENTER, - Marker.ANCHOR_BOTTOM - ) - startMarker.icon = - getDrawable( - this.requireContext(), - fr.free.nrw.commons.R.drawable.current_location_marker - ) - startMarker.title = "Your Location" - startMarker.textLabelFontSize = 24 - binding!!.map.overlays.add(startMarker) + + var geoPoint = mapCenter + val lastLatLng = locationManager.getLastLocation() + if (lastLatLng != null) { + geoPoint = GeoPoint(lastLatLng.latitude, lastLatLng.longitude) } + updateUserLocationOverlays(geoPoint, false) + val scaleBarOverlay = ScaleBarOverlay(binding!!.map) scaleBarOverlay.setScaleBarOffset(15, 25) val barPaint = Paint() @@ -2690,6 +2667,7 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), binding!!.map.overlays.add(scaleBarOverlay) binding!!.map.overlays.add(mapEventsOverlay) binding!!.map.setMultiTouchControls(true) + binding!!.map.invalidate() } /** @@ -2700,45 +2678,149 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), private fun recenterMarkerToPosition(geoPoint: GeoPoint?) { geoPoint?.let { binding?.map?.controller?.setCenter(it) - val overlays = binding?.map?.overlays ?: return@let - // Remove markers and disks using index-based removal - var i = 0 - while (i < overlays.size) { - when (overlays[i]) { - is Marker, is ScaleDiskOverlay -> overlays.removeAt(i) - else -> i++ - } - } - - // Add disk overlay - ScaleDiskOverlay(context, it, 2000, UnitOfMeasure.foot).apply { - setCirclePaint2(Paint().apply { - color = Color.rgb(128, 128, 128) - style = Paint.Style.STROKE - strokeWidth = 2f - }) - setCirclePaint1(Paint().apply { - color = Color.argb(40, 128, 128, 128) - style = Paint.Style.FILL_AND_STROKE - }) - setDisplaySizeMin(900) - setDisplaySizeMax(1700) - overlays.add(this) - } - - // Add marker - Marker(binding?.map).apply { - position = it - setAnchor(Marker.ANCHOR_CENTER, Marker.ANCHOR_BOTTOM) - icon = getDrawable(context, R.drawable.current_location_marker) - title = "Your Location" - textLabelFontSize = 24 - overlays.add(this) - } + updateUserLocationOverlays(it, true); } } + /** + * Updates the user current location overlays (both the location and error overlays) by + * replacing any existing location overlays with new overlays at the given GeoPoint. If there + * are no existing location and error overlays, then new overlays are added. + * + * @param geoPoint The GeoPoint representing the user's current location. + * @param invalidate If true, the map overlays will be invalidated after the user + * location overlays are updated/added. If false, the overlays will not be invalidated. + */ + private fun updateUserLocationOverlays(geoPoint: GeoPoint?, invalidate: Boolean) { + geoPoint?.let{ + updateUserLocationOverlay(geoPoint) + updateUserLocationErrorOverlay(geoPoint) + } + + if (invalidate) { + binding!!.map.invalidate() + } + } + + /** + * Updates the user location error overlay by either replacing it with or adding a new one. + * + * If the user location error overlay is null, the new overlay is added. If the + * overlay is not null, it is replaced by the new overlay. + * + * @param geoPoint The GeoPoint representing the user's location + */ + private fun updateUserLocationErrorOverlay(geoPoint: GeoPoint) { + val overlays = binding?.map?.overlays ?: return + + // Multiply accuracy by 2 to get 95% confidence interval + val accuracy = getCurrentLocationAccuracy() * 2 + val overlay = createCurrentLocationErrorOverlay(this.context, geoPoint, + (accuracy).toInt(), UnitOfMeasure.meter) + + val index = overlays.indexOf(userLocationErrorOverlay) + + if (userLocationErrorOverlay == null || index == -1) { + overlays.add(overlay) + } else { + overlays[index] = overlay + } + + userLocationErrorOverlay = overlay + } + + /** + * Updates the user location overlay by either replacing it with or adding a new one. + * + * If the user location overlay is null, the new overlay is added. If the + * overlay is not null, it is replaced by the new overlay. + * + * @param geoPoint The GeoPoint representing the user's location + */ + private fun updateUserLocationOverlay(geoPoint: GeoPoint) { + val overlays = binding?.map?.overlays ?: return + + val overlay = createCurrentLocationOverlay(geoPoint) + + val index = overlays.indexOf(userLocationOverlay) + + if (userLocationOverlay == null || index == -1) { + overlays.add(overlay) + } else { + overlays[index] = overlay + } + + userLocationOverlay = overlay + } + + /** + * @return The accuracy of the current location with a confidence at the 68th percentile. + * Units are in meters. Returning 0 may indicate failure. + */ + private fun getCurrentLocationAccuracy(): Float { + var accuracy = 0f + val lastLocation = locationManager.getLastLocation() + if (lastLocation != null) { + accuracy = lastLocation.accuracy + } + + return accuracy + } + + /** + * Creates the current location overlay + * + * @param geoPoint The GeoPoint where the current location overlay will be placed. + * + * @return The current location overlay as a Marker + */ + private fun createCurrentLocationOverlay(geoPoint: GeoPoint): Marker { + val currentLocationOverlay = Marker( + binding!!.map + ) + currentLocationOverlay.position = geoPoint + currentLocationOverlay.icon = + getDrawable( + this.requireContext(), + fr.free.nrw.commons.R.drawable.current_location_marker + ) + currentLocationOverlay.title = "Your Location" + currentLocationOverlay.textLabelFontSize = 24 + currentLocationOverlay.setAnchor(0.5f, 0.5f) + + return currentLocationOverlay + } + + /** + * Creates the location error overlay to show the user how accurate the current location + * overlay is. The edge of the disk is the 95% confidence interval. + * + * @param context The Android context + * @param point The user's location as a GeoPoint + * @param value The radius of the disk + * @param unitOfMeasure The unit of measurement of the value/disk radius. + * + * @return The location error overlay as a ScaleDiskOverlay. + */ + private fun createCurrentLocationErrorOverlay(context: Context?, point: GeoPoint, value: Int, + unitOfMeasure: UnitOfMeasure): ScaleDiskOverlay { + val scaleDisk = ScaleDiskOverlay(context, point, value, unitOfMeasure) + + scaleDisk.setCirclePaint2(Paint().apply { + color = Color.rgb(128, 128, 128) + style = Paint.Style.STROKE + strokeWidth = 2f + }) + + scaleDisk.setCirclePaint1(Paint().apply { + color = Color.argb(40, 128, 128, 128) + style = Paint.Style.FILL_AND_STROKE + }) + + return scaleDisk + } + private fun moveCameraToPosition(geoPoint: GeoPoint) { binding!!.map.controller.animateTo(geoPoint) } From 27b9d70333de121b10e8ba63b7ce1e81f4320fd6 Mon Sep 17 00:00:00 2001 From: "translatewiki.net" Date: Sun, 29 Jun 2025 20:27:27 +0200 Subject: [PATCH 007/129] Localisation updates from https://translatewiki.net. --- app/src/main/res/values-ar/strings.xml | 3 ++- app/src/main/res/values-b+tg+Cyrl/strings.xml | 3 ++- app/src/main/res/values-bn/strings.xml | 1 + app/src/main/res/values-fa/strings.xml | 3 ++- app/src/main/res/values-fr/strings.xml | 3 ++- app/src/main/res/values-ia/strings.xml | 2 +- app/src/main/res/values-it/strings.xml | 1 + app/src/main/res/values-iw/strings.xml | 4 ++-- app/src/main/res/values-lb/strings.xml | 3 ++- app/src/main/res/values-nl/strings.xml | 3 ++- app/src/main/res/values-ru/strings.xml | 3 ++- .../main/res/values-x-invalidLanguageCode/error.xml | 10 ---------- 12 files changed, 19 insertions(+), 20 deletions(-) delete mode 100644 app/src/main/res/values-x-invalidLanguageCode/error.xml diff --git a/app/src/main/res/values-ar/strings.xml b/app/src/main/res/values-ar/strings.xml index daa5e9b1c..b1d805ec7 100644 --- a/app/src/main/res/values-ar/strings.xml +++ b/app/src/main/res/values-ar/strings.xml @@ -469,7 +469,7 @@ أدركت أنه سيئ لخصوصيتي لقد غيرت رأيي، ولا أريد أن يكون مرئيا للجميع بعد الآن آسف هذه الصورة ليست مثيرة للاهتمام لموسوعة - رفعته بنفسي في %1$s، مستخدم في %2$d مقالة(ات). + تم تحميله بواسطة نفسي على %1$s، وتم استخدامه في %2$d مقالة على الأقل. مرحبا بك في كومنز! \n\nارفع الوسائط الأولى عن طريق النقر على زر الإضافة. لا توجد تصنيفات مختارة نادرًا ما تكون الصور الخالية من التصنيفات قابلة للاستخدام، هل أنت متأكد أنك تريد الإرسال دون تحديد التصنيفات؟ @@ -893,4 +893,5 @@ عرض في المناطق القريبة تم الإنشاء والتحميل بواسطة: %1$s تم إنشاؤه بواسطة %1$s وتم تحميله بواسطة %2$s + مرشح للحذف diff --git a/app/src/main/res/values-b+tg+Cyrl/strings.xml b/app/src/main/res/values-b+tg+Cyrl/strings.xml index 5ab1ab282..3404183dc 100644 --- a/app/src/main/res/values-b+tg+Cyrl/strings.xml +++ b/app/src/main/res/values-b+tg+Cyrl/strings.xml @@ -1,5 +1,6 @@ - - کامَنٕز گوو رُکِتھ - Oops. کیہہ تام گوو غلط! - ؤنِیوٚ اَسہِ توٚہہِ کیاہ ٲسِیوٚ کران، تہٕ کٕریٚو تہِ اَسہِ سٕتی شیر بذریعہِ برقی خط. یُس مَدَتھ کَرِ اَسہِ اَتھ شہَرنَس منٛز! - شُکریہ! - From 6de5a07e0dde9900de2e9f2236894b1f346da60a Mon Sep 17 00:00:00 2001 From: "translatewiki.net" Date: Mon, 30 Jun 2025 14:01:39 +0200 Subject: [PATCH 008/129] Localisation updates from https://translatewiki.net. --- app/src/main/res/values-az/strings.xml | 50 +++++++++++++++++++++++--- 1 file changed, 45 insertions(+), 5 deletions(-) diff --git a/app/src/main/res/values-az/strings.xml b/app/src/main/res/values-az/strings.xml index 1662c2301..c498f23f5 100644 --- a/app/src/main/res/values-az/strings.xml +++ b/app/src/main/res/values-az/strings.xml @@ -44,30 +44,63 @@ (%1$d) Yükləmələrə Başlanılır + + %d yükləmə emal edilir + %d yükləmə emal edilir + + + %d yükləmə + %d yükləmə + + + Bu şəkil %1$s lisenziyası altında yayımlanacaq + Bu şəkillər %1$s lisenziyası altında yayımlanacaq + + + %1$d yükləmə + %1$d yükləmə + + + Paylaşılan məzmun qəbul edilir. Şəklin emalı şəklin ölçüsündən və cihazınızdan asılı olaraq bir az vaxt apara bilər + Paylaşılan məzmun qəbul edilir. Şəkillərin emalı şəklin ölçüsündən və cihazınızdan asılı olaraq bir az vaxt apara bilər + + Kəşf et + Görünüş Ümumi + Rəy Məxfilik Vikianbar Tənzimləmələr + Vikianbara yüklə + Yükləmə davam edir İstifadəçi adı Parol + Commons Beta hesabınıza daxil olun Daxil ol + Şifrəni unutmusunuz? Qeydiyyatdan keç Giriş edilir Zəhmət olmasa, gözləyin… + Başlıqların və təsvirlərin yenilənməsi + Zəhmət olmasa, gözləyin… Uğurlu giriş! Giriş baş tutmadı! Fayl tapılmadı. Xahiş edirik başqa bir fayl üzərində cəhd edin. Maksimum təkrar cəhd limitinə çatdınız! Zəhmət olmasa, yükləməni ləğv edin və yenidən cəhd edin Batareya optimallaşdırılması söndürülsün? - Doğrulama alınmadı, xahiş edirəm yenidən daxil olun + 3-dən çox şəklin yüklənməsi batareyanın optimallaşdırılması söndürüldükdə daha yaxşı işləyir. Rahat yükləmə təcrübəsi üçün Vikianbar tətbiqi üçün batareyanın optimallaşdırılmasını parametrlərdən söndürün. \n\nBatareyanın optimallaşdırılmasını söndürmək üçün mümkün addımlar:\n\nAddım 1: Aşağıdakı \"Parametrlər\" düyməsinə toxunun.\n\nAddım 2: \"Optimallaşdırılmayıb\"dan \"Bütün proqramlar\"a keçin.\n\nAddım 3: \"Commons\" və ya \"fr.free.nrw.commons\" axtarın.\n\nAddım 4: \"Optimallaşdırma\"nı seçin.\n\nAddım 5: \'Bitti\' düyməsini basın. + Doğrulama alınmadı. Zəhmət olmasa, yenidən daxil olun Yükləmə başladı! + Yükləmə növbədə (məhdud əlaqə rejimi aktivdir) %1$s yükləndi! Yükləmənizə baxmaq üçün toxunun - %1$s yüklənməsi başlanır + Fayl yüklənir: %s %1$s yüklənir %1$s yüklənməsi başa çatdı - %1$s faylının yüklənməsi alınmadı + %1$s yükləmək alınmadı + %1$s yüklənməsi dayandırıldı Baxmaq üçün toxun + Baxmaq üçün toxun Son Yükləmələrim Növbəyə alındı Uğursuz @@ -77,19 +110,26 @@ Foto çək Yaxınlıqdakılar Yükləmələrim + Keçidi kopiyala + Link kopyalandı. Paylaş + Faylın səhifəsinə bax Başlıq (Zəruri) + Zəhmət olmasa, bu fayl üçün başlıq təqdim et Açıqlama - Daxil olmaq olmur — şəbəkə xətası + Başlıq + Daxil olmaq mümkün deyil - şəbəkə xətası Çox sayda uğursuz daxil olma. Xahiş edirik bir neçə dəqiqə sonra yenidən cəhd edin. Bağışlayın, bu istifadəçi Vikianbardan bloklanıb Siz iki faktorlu autentifikasiya kodunuzu təqdim etməlisiniz. - Daxil olma uğursuz oldu + E-poçt ünvanınıza giriş doğrulama kodu göndərildi. Daxil olmaq üçün kodu daxil edin. + Daxil olma uğursuz oldu Yüklə Bu dəsti adlandırın Dəyişikliklər Yüklə Kateqoriyalarda axtar + Medianızın təsvir etdiyi elementləri axtarın (dağ, Tac Mahal və s.) Yadda saxla Yenilə Siyahı From 89436b0a753c8cde7eb3a3b4eabcbe155fd11182 Mon Sep 17 00:00:00 2001 From: Ritika Pahwa <83745993+RitikaPahwa4444@users.noreply.github.com> Date: Tue, 1 Jul 2025 11:52:08 +0530 Subject: [PATCH 009/129] Add v5.5.0 to CHANGELOG.md --- CHANGELOG.md | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index cd4e70b6c..72fda2f5d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,15 @@ # Wikimedia Commons for Android +## v5.5.0 + +### What's changed +* Explore images will now be shown based on the map location and not at your current location +* Enhanced Wikidata feedback message +* Green labels in Explore map will no longer be hidden by other pins thumbnails +* Upload wizard's language drop-down now reflects the language used in the pin label +* Users can now pick only one image at a time while using the custom selector +* Bug fixes and stability improvements + ## v5.4.1 ### What's changed From 4befff8f422efe335a09d784b3d754b8d2617536 Mon Sep 17 00:00:00 2001 From: "translatewiki.net" Date: Thu, 3 Jul 2025 14:02:08 +0200 Subject: [PATCH 010/129] Localisation updates from https://translatewiki.net. --- app/src/main/res/values-az/strings.xml | 235 +++++++++++++++++++++++++ app/src/main/res/values-ia/strings.xml | 1 + app/src/main/res/values-it/strings.xml | 2 +- app/src/main/res/values-mk/strings.xml | 3 +- app/src/main/res/values-sh/error.xml | 3 +- app/src/main/res/values-sh/strings.xml | 15 +- app/src/main/res/values-uk/strings.xml | 53 +++++- app/src/main/res/values-zh/strings.xml | 3 +- 8 files changed, 301 insertions(+), 14 deletions(-) diff --git a/app/src/main/res/values-az/strings.xml b/app/src/main/res/values-az/strings.xml index c498f23f5..b5f736415 100644 --- a/app/src/main/res/values-az/strings.xml +++ b/app/src/main/res/values-az/strings.xml @@ -135,31 +135,266 @@ Siyahı (Hələ yükləmə yoxdur) %1$s axtarışına uyğun kateqoriya tapılmadı + %1$s uyğun heç bir Vikidata elementi tapılmadı + %1$s alt sinifləri yoxdur + %1$s üst sinfi yoxdur + Şəkillərinizi Vikianbarda daha tapıla bilən etmək üçün kateqoriyalar əlavə edin.\nKateqoriyalar əlavə etmək üçün yazmağa başlayın. Kateqoriyalar Nizamlamalar Qeydiyyatdan keç Seçilmiş şəkillər + Xüsusi Seçici Kateqoriya + Yoxlanış Haqqında + Vikianbar proqramı Vikimedia icmasının qrant alanları və könüllüləri tərəfindən yaradılmış və dəstəklənən açıq mənbəli proqramdır. Vikimedia Fondu proqramın yaradılması, inkişafı və ya texniki xidmətində iştirak etmir. + Xəta hesabatları və təkliflər üçün yeni <a href=\"%1$s\">GitHub sorğusu</a> yaradın. Məxfilik siyasəti + Töhfə verənlər Haqqında + Rəy göndər (e-poçt vasitəsilə) + Heç bir e-poçt tətbiqi quraşdırılmayıb + Son istifadə olunan kateqoriyalar + İlk sinxronizasiya gözlənilir… + Siz hələ heç bir şəkil yükləməmisiniz. Yenidən cəhd edin + İmtina + Bu şəkli yükləməklə bildirirəm ki, bu mənim şəxsi işimdir, onda müəllif hüququ ilə qorunan material və ya selfilər yoxdur və <a href=\"https://commons.wikimedia.org/wiki/Commons:Policies_and_guidelines\">Vikianbar qaydalarına</a> riayət edir. Endir + Defolt lisenziya + Əvvəlki başlıq və təsvirdən istifadə et + Tema CC BY 3.0 + Vikianbar Vikipediyada istifadə olunan şəkillərin əksəriyyətinə ev sahibliyi edir. + Şəkilləriniz bütün dünyada insanları maarifləndirməyə kömək edir! + Tamamilə özünüz tərəfindən çəkilmiş və ya yaradılmış şəkilləri yükləyin: + Təbii obyektlər (çiçəklər, heyvanlar, dağlar) + Faydalı obyektlər (velosipedlər, qatar stansiyaları) + Məşhur insanlar (icra başçınız, tanış olduğunuz olimpiya idmançıları) + Zəhmət olmasa, YÜKLƏMƏYİN: + Selfi və ya dostlarınızın şəkilləri + İnternetdə tapdığınız şəkillər + Özəl proqramların ekran görüntüləri + Yükləmə nümunəsi: + Başlıq: Sidney Opera Evi + Təsvir: Sidney Opera Evinin körfəzdən görünüşü + Kateqoriyalar: Sidney Opera Evi Qərb istiqamətindən, Sidney Opera Evinin uzaqdan mənzərələri + Şəkilləriniz ilə töhfə verin. Vikipediya məqalələrinin canlanmasına kömək edin! + Vikipediyadakı şəkillər Vikianbardan götürülüb. + Şəkilləriniz bütün dünyada insanları maarifləndirməyə kömək edir. + İnternetdən tapdığınız müəllif hüquqları ilə qorunan materialları, eləcə də afişaların, kitab üzlüklərinin və s. şəkillərini yükləməkdən çəkinin. + Sizcə bunu başa düşdünüz? + Bəli! Əlavə məlumat Kateqoriyalar Yüklənir... Heç biri seçilməmişdir + Başlıq yoxdur + Təsvir yoxdur + Müzakirə yoxdur Naməlum lisenziya Yenilə + Yaddaşa giriş icazəsi tələb olunur + Tələb olunan icazə: Xarici yaddaşı oxumaq. Bu olmadan proqram qalereyanıza daxil ola bilməz. + Tələb olunan icazə: Xarici yaddaşa yazmaq. Bu olmadan proqram kameranıza/qalereyanıza daxil ola bilməz. + Məkan icazəsi tələb olunur + Tətbiqdaxili çəkilişlər üçün məkanı qeyd et + Cihaz kamerası onu qeyd etmədiyi halda, məkanı tətbiqdaxili çəkilişlərlə əlavə etmək üçün bunu aktiv edin + Oldu + Xəbərdarlıq + Dublikat fayl adı tapıldı + Yüklə Bəli Xeyr + Başlıq Başlıq + Təsvirlər + Təsvir Müzakirə Müəllif + Yüklənmə tarixi Lisenziya + Koordinatlar + Heç biri göstərilməyib + Beta Tester ol + Google Play-də beta kanalımıza qoşulun və yeni funksiyalara və xəta həllərinə erkən giriş əldə edin + 2FA kodu + E-poçt doğrulama kodu + Həqiqətən çıxış etmək istəyirsiniz? + Media şəkli uğursuz oldu + Heç bir alt kateqoriya tapılmadı + Heç bir üst kateqoriya tapılmadı + Zao dağı + Lamalar + Göy qurşağı körpüsü + Lalə + Vikipediyaya xoş gəlmisiniz + Müəlliflik hüquqlarına xoş gəlmisiniz + Sidney Opera Evi + İmtina + + Bağla + Ana səhifə + Yüklə + Yaxınlıqdakılar + Haqqında + Parametrlər + Rəy + GitHub vasitəsilə rəy + Çıxış + Təlimat + Bildirişlər + Yoxla + təsvir tapılmadı + Commons fayl səhifəsi + Vikidata elementi + Vikipediya məqaləsi + Zəhmət olmasa, medianı mümkün qədər təsvir edin: Harada çəkilib? Nəyi təsvir edir? Kontekst nədir? Obyektləri və ya şəxsləri təsvir edin. Asanlıqla təxmin edilə bilməyən məlumatları, məsələn, bir mənzərədirsə, günün saatını qeyd edin. Əgər media qeyri-adi bir şey göstərirsə, zəhmət olmasa, onu izah edin. + Zəhmət olmasa şəklin qısa təsvirini yazın. İlk başlıq şəkil üçün Başlıq kimi istifadə olunacaq. 255 simvolla məhdudlaşır. + Bu şəkil ilə bağlı potensial problemlər: + Şəkil çox qaranlıqdır. + Şəkil bulanıqdır. + Şəkil artıq Vikianbarda var. + Bu şəkil başqa yerdə çəkilib. + Zəhmət olmasa, yalnız özünüz çəkdiyiniz şəkilləri yükləyin. Başqalarının Facebook hesablarında tapdığınız şəkilləri yükləməyin. + Hələ də bu şəkli yükləmək istəyirsiniz? + Bağlantı xətası + Yükləmə prosesi aktiv internetə çıxış tələb edir. Şəbəkə bağlantınızı yoxlayın. + Şəkildə tapılan problemlər + Zəhmət olmasa, yalnız özünüz çəkdiyiniz şəkilləri yükləyin. İnternetdən tapdığınız şəkilləri yükləməyin. + Tətbiqdaxili kadrları yadda saxla + Tətbiqdaxili kamera ilə çəkilmiş şəkilləri cihazın yaddaşında saxla + Hesabınıza daxil olun + Jurnal faylını göndərin + Proqramla bağlı problemlərin aradan qaldırılmasına kömək etmək üçün jurnal faylını tərtibatçılara e-poçt vasitəsilə göndərin. Qeyd: jurnallar potensial olaraq şəxsi məlumatları ehtiva edə bilər + URL-i açmaq üçün heç bir veb brauzer tapılmadı + Xəta! URL tapılmadı + Silinməyə namizəd göstər + Bu şəkil silinməyə namizəddir. + Ətraflı məlumat üçün veb səhifəsinə bax + Ötür + Daxil ol + Daxil olma mərhələsini keçmək istədiyinizə əminsiniz? + Gələcəkdə şəkilləri yükləmək üçün daxil olmalısınız. + Bu funksiyadan istifadə etmək üçün zəhmət olmasa daxil olun + Vikimətni mübadilə buferinə köçür + Vikimətn mübadilə buferinə köçürüldü + Yaxınlıqdakılar düzgün işləməyə bilər, məkan əlçatan deyil. + İnternet əlçatan deyil. Yalnız keşlənmiş yerlər göstərilir. + Məkan girişi rədd edildi. Bu funksiyadan istifadə etmək üçün yerinizi manual təyin edin. + Yaxınlıqdakı yerlərin siyahısını göstərmək üçün icazə tələb olunur + Yaxınlıqdakı şəkillərin siyahısını göstərmək üçün icazə tələb olunur + İstiqamətlər Vikidata Vikipediya + Vikianbar + Bizi qiymətləndir + TSS + İstifadəçi təlimatı + Təlimatı ötür + İnternet əlçatan deyil + Bildirişləri əldə edərkən xəta baş verdi + Şəkli nəzərdən keçirmək üçün gətirilərkən xəta baş verdi. Yenidən cəhd etmək üçün yeniləmə düyməsini basın. + Heç bir bildiriş tapılmadı + Tərcümə et + Dillər + Tərcümələri təqdim etmək istədiyiniz dili seçin + Davam et + İmtina + Yenidən cəhd et + Bunlar sizə yaxın yerlərdir və onların Vikipediya məqalələrini təsvir etmək üçün şəkillərə ehtiyac duyurlar.\n\n\'BU SAHƏDƏ AXTAR\' düyməsinə klikləməklə, xəritə kilidləyib həmin məkan ətrafında yaxınlıqda axtarış başlaya bilərsiniz. + Bu yerə şəkil lazımdır. + Bu yerin artıq şəkli var. + Bu yer artıq mövcud deyil. + Şəkil tapılmadı! + Şəkillər yüklənərkən xəta baş verdi. + %1$s tərəfindən yüklənilib + Bloklanıb + Sizə Vikianbarı redaktə etmək qadağan edilib + Günün Şəkli + Axtar + Vikianbarda axtar + Axtar + Son axtarışlar: + Bu yaxınlarda axtarılanlar + Son dil sorğuları + Kateqoriyalar yüklənərkən xəta baş verdi. + Təsvirləri yükləyərkən xəta baş verdi. + Media + Kateqoriyalar + Elementlər + Seçilmiş + Mobil tətbiq vasitəsilə yüklənib + Xəritə + Şəkil Vikidatada %1$s elementinə əlavə edildi + Müvafiq Vikidata obyektini yeniləmək alınmadı! + Divar kağızı kimi təyin et + Divar kağızı uğurla təyin edildi! + Sorğu + Bu şəkli yükləmək olar? + Sual + Nəticə + Silinməyi tələb edilən şəkilləri yükləməyə davam etsəniz, hesabınız blok olunacaq. Sorğunu bitirmək istədiyinizə əminsiniz? + Yüklədiyiniz şəkillərin %1$s+ ədədi silinib. Silinməyi tələb edilən şəkilləri yükləməyə davam etsəniz, hesabınız blok olunacaq.\n\nTəlimata yenidən baxmaq və sonra hansı növ şəkilləri yükləməli və ya yükləməməli olduğunuzu öyrənməyə kömək etmək üçün testdən keçmək istəyirsiniz? + Selfielərin o qədər də məlumatlandırıcı dəyəri yoxdur. Haqqınızda Vikipediya məqaləsi yoxdursa, zəhmət olmasa, şəklinizi yükləməyin. + Abidələrin və mənzərələrin şəkillərini əksər ölkələrdə yükləmək olar. Nəzərə alın ki, xaricdəki müvəqqəti incəsənət abidələri tez-tez müəllif hüquqları ilə qorunur və yükləmək düzgün deyil. + Veb saytların skrinşotları törəmə əsərlər hesab edilir və veb-saytdakı müəllif hüququna tabedir. Bunlar müəllifin icazəsindən sonra istifadə edilə bilər. Belə icazə olmadan, onların işi əsasında yaratdığınız hər hansı əsər qanuni olaraq orijinal müəllifə məxsus lisenziyasız nüsxə sayılır. + Vikianbarın məqsədlərindən biri keyfiyyətli şəkillər toplamaqdır. Ona görə də bulanıq şəkillər yüklənməməlidir. Həmişə yaxşı işıqlandırma ilə gözəl şəkillər çəkməyə çalışın. + Texnologiya və ya mədəniyyəti təsvir edən şəkillər üçün Vikianbarda həmişə yer var. + Cavabların %1$s ədədi düzgündür. Təbriklər! + Suala cavab vermək üçün iki variantdan birini seçin + Giriş müddəti bitib. Zəhmət olmasa, yenidən daxil olun. + Bunu dostlarınızla paylaşın! + Davam et + Düzgün Cavab + Yanlış Cavab + Bu skrinşotu yükləmək olar? + Proqramı paylaş + Fırlat + Yaxınlıqdakı yerləri yükləmək mümkün olmadı + Bu ərazidə şəkillər yoxdur + Yaxınlıqda yer tapılmadı + Yaxınlıqdakı abidələri əldə edərkən xəta baş verdi. + Son axtarışlar yoxdur + Axtarış tarixçəsini silmək istədiyindən əminsiniz? + Bu yükləməni ləğv etmək istədiyinizə əminsiniz? + Bu axtarışı silmək istəyirsiniz? + Axtarış tarixçəsi silindi + Silinməyə namizəd göstər + Sil + Nailiyyətlər + Profil + Rozetlər + Statistika + Qəbul edilən təşəkkürlər + Seçilmiş şəkillər + \"Yaxınlıqdakı yerlər\" vasitəsilə şəkillər + Səviyyə %d + %s (Səviyyə %s) + Yüklənən şəkillər + Geri qaytarılan şəkillər + İstifadə olunan şəkillər + Nailiyyətlərinizi dostlarınızla paylaşın! + Bu tələbləri yerinə yetirdikcə səviyyəniz yüksəlir. \"Statistika\" bölməsindəki elementlər sizin səviyyənizi nəzərə almır. + minimum tələb olunur: + İstənilən yükləmə proqramı vasitəsilə Vikianbara yüklədiyiniz şəkillərin sayı + Vikianbara yüklədiyiniz və silinməyən şəkillərin faizi + Vikianbara yüklədiyiniz və Vikimedia məqalələrində istifadə olunan şəkillərin sayı + Xəta baş verdi! + Vikianbar bildirişi + Fərdi müəllif adından istifadə et + Fotoşəkilləri yükləyərkən istifadəçi adınız əvəzinə fərdi müəllif adından istifadə edin + Fərdi müəllif adı + Töhfələr + Yaxınlıqdakılar + Bildirişlər + Bildirişlər (oxunmuş) + Yaxınlıqdakı bildirişini göstər + Şəkillərə ehtiyacı olan ən yaxın yerlər üçün tətbiqdaxili bildiriş göstərin + Siyahı + Yaddaş icazəsi + Şəkilləri yükləmək üçün cihazınızın xarici yaddaşına giriş icazəsi lazımdır. + Artıq şəkillərə ehtiyacı olan ən yaxın yeri görməyəcəksiniz. Bununla belə, istəyirsinizsə, bu bildirişi Parametrlərdə yenidən aktivləşdirə bilərsiniz. Təşəkkür uğurla göndərildi Təşəkkür göndərilə bilmədi %1$s Təşəkkür göndərilir: Xəta diff --git a/app/src/main/res/values-ia/strings.xml b/app/src/main/res/values-ia/strings.xml index 452f4f9b6..9e169f8f6 100644 --- a/app/src/main/res/values-ia/strings.xml +++ b/app/src/main/res/values-ia/strings.xml @@ -816,4 +816,5 @@ Monstrar in A proximitate Create e incargate per: %1$s Create per %1$s e incargate per %2$s + Nominate pro deletion diff --git a/app/src/main/res/values-it/strings.xml b/app/src/main/res/values-it/strings.xml index 9ca05c344..ce121ab5d 100644 --- a/app/src/main/res/values-it/strings.xml +++ b/app/src/main/res/values-it/strings.xml @@ -772,7 +772,7 @@ Impara come scrivere una didascalia utile Vedi i tuoi risultati Modifica Immagine - Modifica Posizione + Modifica posizione Posizione aggiornata! Rimuovi posizione Rimuovi avviso di posizione diff --git a/app/src/main/res/values-mk/strings.xml b/app/src/main/res/values-mk/strings.xml index a0d1dd6ef..6a06f04e7 100644 --- a/app/src/main/res/values-mk/strings.xml +++ b/app/src/main/res/values-mk/strings.xml @@ -413,7 +413,7 @@ Сфатив дека ќе ми ја наруши приватноста Се премислив. Не сакам повеќе да биде видлива За жал, сликава не е соодветна за енциклопедија - Подигнато од мене %1$s, кое се користи во %2$d статии. + Подигнато од мене на %1$s, и се користи во барем {{PLURAL:%2$d|one=една статија|%2$d статии}}. Добре дојдовте на Ризницата!\n\nПодигнете ја вашата прва слика или снимка допирајќи го копчето за додавање. Нема избрано категории Некатегоризираните слики се слабо употребливи. Дали сигурно сакате да продолжите без да ставите категории? @@ -825,4 +825,5 @@ Прикажи во „Во близина“ Создал: %1$s Создал %1$s, а подигнал %2$s + Предложено за бришење diff --git a/app/src/main/res/values-sh/error.xml b/app/src/main/res/values-sh/error.xml index ea9b8e670..28d2bbe0a 100644 --- a/app/src/main/res/values-sh/error.xml +++ b/app/src/main/res/values-sh/error.xml @@ -1,9 +1,10 @@ - Commons se srušio + Aplikacija Ostava prestala je raditi Ups. Nešto nije u redu! Recite nam šta radite, pa podijelite s nama putem e-pošte. Pomoći će nam da ih popravimo! Hvala vam! diff --git a/app/src/main/res/values-sh/strings.xml b/app/src/main/res/values-sh/strings.xml index 5997677ef..97271b945 100644 --- a/app/src/main/res/values-sh/strings.xml +++ b/app/src/main/res/values-sh/strings.xml @@ -1,5 +1,6 @@ @@ -93,12 +94,12 @@ Pretraži kategorije Snimi Preučitaj - Popis + Lista (Još uvijek nema postavljenih datoteka) Nema kategorija što odgovoraju %1$s Stavite kategorije slikama kako biste olakšali korisnicima njihovo pronalaženje na Ostavi.\n\nDa biste stavili kategoriju, počnite sa pisanjem njenog imena. Kategorije - Podešavanja + Postavke Registracija Izabrane slike Kategorija @@ -125,7 +126,7 @@ Autorstvo 4.0 Autorstvo-Dijeliti pod istim uslovima 3.0 Autorstvo 3.0 - Ostava je udomljač najvećine slika što se koriste na Wikipediji. + Wikimedijina ostava skladišti većinu slika koje se upotrebljavaju na Wikipediji. Vaše slike pomažu u obrazovanju ljudi širom svijeta! Postavljajte slike koje su u potpunosti Vaše djelo: Objekti iz prirode (cvijeće, životinje, planine) @@ -188,7 +189,7 @@ Postavi U blizini O privitku - Podešavanja + Postavke Povratna informacija Odjava Uputstva @@ -214,7 +215,7 @@ U budućnosti ćete se morati prijaviti kako biste postavili slike. Prijavite se da biste koristili ovu funkciju „U blizini“ možda ne radi kako treba. Lokacija nije dostupna. - Potrebna je dozvola za popis liste lokacija u blizini + Potrebna je dozvola za prikaz liste lokacija u blizini Upute Wikidata Wikipedia @@ -226,7 +227,7 @@ Nema obavijesti Prevedi Jezici - Odaberite za koji jezik bi želeli da pravite prijevode + Izaberite jezik na koji biste željeli prevoditi Nastavi Otkaži Pokušaj ponovo @@ -261,6 +262,6 @@ Poništi Podaci o lokaciji pomažu Wikiurednicima da pronađu vašu sliku, čineći je mnogo korisnijom.\nVaše nedavne postavljene slike nemaju lokaciju.\nPredlažemo da uključite lokaciju u postavkama priloga kamere.\nHvala vam na učitavanju! Detalji - Razina priložnika + Nivo API-ja Android verzija diff --git a/app/src/main/res/values-uk/strings.xml b/app/src/main/res/values-uk/strings.xml index f005f2969..8eeceec22 100644 --- a/app/src/main/res/values-uk/strings.xml +++ b/app/src/main/res/values-uk/strings.xml @@ -132,6 +132,8 @@ Оберіть фото Поблизу Мої завантаження + Скопіювати посилання + Посилання скопійовано в буфер обміну Поширити Переглянути сторінку файлу Підпис (обов\'язково) @@ -142,6 +144,7 @@ Надто багато невдалих спроб. Будь ласка, спробуйте знову через кілька хвилин. Вибачте, цього користувача було заблоковано на Вікісховищі Ви повинні надати код двофакторної автентифікації. + Код підтвердження входу надіслано на вашу адресу електронної пошти. Будь ласка, введіть код для входу. Не вдалося увійти Завантажити Назвіть цю серію @@ -150,6 +153,7 @@ Пошук категорій Пошук об\'єктів, що зображено у цьому медіа (напр. гори, Тадж-Махал, тощо) Зберегти + Додаткове меню Оновити Список (Ще нема завантажень) @@ -246,6 +250,7 @@ Станьте бета-тестером Підпишіться на наш бета-канал на Google Play і отримайте ранній доступ до нових функцій та виправлень баґів Код 2FA + Код підтвердження електронної пошти Ви справді хочете вийти із системи? Помилка медіазображення Підкатегорій не знайдено @@ -306,6 +311,7 @@ Скопіювати вікі-текст у буфер обміну Вікі-текст скопійовано у буфер обміну Функція «Поблизу» може працювати некоректно, «Розташування» недоступне. + Інтернет недоступний. Показуються лише кешовані місця. Доступ до місцезнаходження заборонено. Щоб скористатися цією функцією, будь ласка, вкажіть своє місцезнаходження вручну. Потрібний дозвіл для показу списку місць поблизу Потрібний дозвіл для показу зображень місць поблизу @@ -389,11 +395,13 @@ Вилучити Досягнення Профіль + Значки Статистика Отримані подяки Вибрані зображення Зображення місць поблизу - Рівень + Рівень %d + %s (Рівень %s) Завантажені зображення Не відхилені зображення Використані зображення @@ -425,6 +433,7 @@ На Вашому пристрої не знайдено сумісного додатка з картами. Будь ласка, встановіть додаток з картами, якщо хочете скористатись цією функцією. Зображення Місця + Категорії Додати/вилучити закладки Закладки Ви не додали жодної закладки @@ -435,7 +444,7 @@ Мені стало зрозуміло, що це шкодить моїй приватності Моя думка змінилась, я не хочу, щоб це було доступно публічно Перепрошую, це зображення нецікаве для енциклопедії - Завантажено мною на сайт «%1$s» та використано у {{PLURAL:%2$d|one=одній статті|%2$d статтях}}. + Завантажено мною на сайт «%1$s» та використано у принаймні {{PLURAL:%2$d|one=одній статті|%2$d статтях}}. Ласкаво просимо до Вікісховища!\n\nЗавантажте Ваш перший медіафайл, натиснувши кнопку додавання. Жодної категорії не вибрано Зображення без категорій рідко використовуються. Ви впевнені, що хочете продовжити без вказаних категорій? @@ -506,6 +515,7 @@ У вас немає непрочитаних сповіщень Немає прочитаних сповіщень Поширення журналів + Перевірте свою скриньку електронної пошти Перегляд прочитаних Перегляд непрочитаних Сталася помилка при завантаженні зображень @@ -725,7 +735,7 @@ Для належної роботи мапи поблизу мають відображати стан PHONE STATE Внесок користувача: %s Досягнення користувача: %s - Переглянути сторінку користувача + Переглянути профіль користувача Редагувати описи Редагувати категорії Розширені параметри @@ -810,8 +820,45 @@ Повідомити у Вікідані про проблему з цим елементом Будь ласка, додайте якийсь коментар Обговорення + Напишіть щось про елемент «%1$s». Це буде видно публічно. «%1$s» більше не існує, фотографій цього більше не можливо зробити. + «%1$s» розташовано в іншому місці. «%1$s» — це інше місце. Будь ласка, вкажіть правильне місце нижче і, якщо можна, напишіть правильні широту і довготу. Інша проблема або інформація (поясніть нижче). Ваші відгуки розміщуються на ось цій вікісторінці: <a href=\"https://commons.wikimedia.org/wiki/Commons:Mobile_app/Feedback\">Commons:Mobile app/Feedback</a> + Ви впевнені, що бажаєте скасувати усі вивантаження? + Скасування усіх вивантажень… + Вивантаження + В очікуванні + Не вдалося + Не вдалося завантажити дані про місце + Вилучити папку + Підтвердити вилучення + Ви впевнені, що бажаєте вилучити папку %1$s, у якій міститься %2$d елемент(ів)? + Вилучити + Скасувати + Папку %1$s успішно вилучено + Не вдалося вилучити папку %1$s + Помилка при вилученні вмісту папки: %1$s + Не вдалося отримати шлях до папки для ID контейнера: %1$d + У цього місця ще немає зображень, сфотографуйте його! + Це місце уже має зображення. + Перевіряємо, чи це місце має зображення. + Помилка при завантаженні + Не знайдено використань + Вікісховище + Інші вікі + Використання файлу + Обліковий запис + Знищити обліковий запис + Попередження про знищення облікового запису + Зникнення є <b>останнім заходом</b>, і <b>його слід використовувати лише тоді, коли ви хочете назавжди припинити редагування</b>, а також щоб приховати якомога більше своїх минулих зв\'язків.<br/><br/>Вилучення облікового запису у Вікісховищі робиться шляхом зміни імені вашого облікового запису, щоб інші не могли розпізнати ваші внески; це називають зникненням облікового запису. <b>Зникнення не гарантує повної анонімності і не вилучає внесок у проєкти</b>. + Підпис + Підпис скопійовано до буферу обміну + Вітаємо, всі зображення в цьому альбомі або завантажені, або позначені як не для завантаження. + Показати в розділі «Дослідити» + Показати в розділі «Поблизу» + Створено та завантажено: %1$s + Створено %1$s та завантажено %2$s + Номіновано на вилучення diff --git a/app/src/main/res/values-zh/strings.xml b/app/src/main/res/values-zh/strings.xml index 452e54563..8244bdc30 100644 --- a/app/src/main/res/values-zh/strings.xml +++ b/app/src/main/res/values-zh/strings.xml @@ -470,7 +470,7 @@ 我意识到这对我的隐私不利 我改变了主意,我不想再让公众看到它了 对不起,这幅图对百科全书没什么意思 - 由我自己上传在%1$s,使用于%2$d个条目。 + 由我自己上传在%1$s,使用于至少%2$d个条目。 欢迎使用共享资源!\n\n通过点击添加按钮以上传您的首个媒体。 未提交分类 不带分类的图片很难有机会被利用到,您确定您要不选择分类来继续吗? @@ -880,4 +880,5 @@ 显示在附近 创建并上传者: %1$s 由%1$s创建并由%2$s上传 + 已提名删除 From 3bd0ec446610ab9df1ee17360a3840522382dd60 Mon Sep 17 00:00:00 2001 From: Paul Hawke Date: Fri, 4 Jul 2025 06:18:52 -0500 Subject: [PATCH 011/129] Convert top level "Utils" class to kotlin (#6364) * Unused class removed * Convert BasePresenter to kotlin * Removed redundent class * Move the Utils class into the utils package * Inline the creation of a page title object * Move license utilities into their own file * Inline app rating since its only ever used in 1 place * Moved GeoCoordinates utilities into their own class * Moved Monuments related utils into their own class * Moved screen capture into its own util class * Moved handleWebUrl to its own utility class * Moved fixExtension to its own class * Moved clipboard copy into its own utility class * Renames class to match remaining utility method * Convert UnderlineUtils to kotlin * Converted the copy-to-clipboard utility to kotlin * Converted license name and url lookup to kotlin * Converted fixExtension to kotlin * Convert handleGeoCoordinates to kotlin * Monument utils converted to kotlin * Convert then inline screeen capture in kotlin * Convert handleWebUrl to kotlin --- .../java/fr/free/nrw/commons/AboutActivity.kt | 66 ++--- .../fr/free/nrw/commons/BasePresenter.java | 18 -- .../java/fr/free/nrw/commons/BasePresenter.kt | 10 + .../java/fr/free/nrw/commons/License.java | 79 ------ .../main/java/fr/free/nrw/commons/Media.kt | 5 +- .../java/fr/free/nrw/commons/MvpView.java | 8 - .../main/java/fr/free/nrw/commons/Utils.java | 264 ------------------ .../free/nrw/commons/WelcomePagerAdapter.java | 7 +- .../fr/free/nrw/commons/auth/LoginActivity.kt | 6 +- .../nrw/commons/campaigns/CampaignView.kt | 4 +- .../commons/campaigns/CampaignsPresenter.kt | 2 +- .../nrw/commons/campaigns/ICampaignsView.kt | 3 +- .../category/CategoryDetailsActivity.kt | 10 +- .../contributions/ContributionsContract.kt | 1 - .../contributions/ContributionsFragment.kt | 14 +- .../ContributionsListContract.kt | 2 +- .../ContributionsListFragment.kt | 8 +- .../WikidataItemDetailsActivity.java | 9 +- .../explore/map/ExploreMapFragment.java | 7 +- .../locationpicker/LocationPickerActivity.kt | 6 +- .../nrw/commons/media/MediaDetailFragment.kt | 19 +- .../media/MediaDetailPagerFragment.java | 6 +- .../fragments/CommonPlaceClickActions.kt | 7 +- .../nearby/fragments/NearbyParentFragment.kt | 31 +- .../notification/NotificationActivity.kt | 4 +- .../nrw/commons/profile/ProfileActivity.kt | 28 +- .../achievements/AchievementsFragment.kt | 4 +- .../nrw/commons/settings/SettingsFragment.kt | 10 +- .../nrw/commons/upload/PageContentsCreator.kt | 4 +- .../fr/free/nrw/commons/upload/UploadItem.kt | 4 +- .../upload/license/MediaLicenseFragment.kt | 16 +- .../upload/license/MediaLicensePresenter.kt | 10 +- .../UploadMediaDetailsContract.kt | 2 +- .../free/nrw/commons/utils/ClipboardUtils.kt | 20 ++ .../fr/free/nrw/commons/utils/FixExtension.kt | 38 +++ .../free/nrw/commons/utils/GeoCoordinates.kt | 27 ++ .../fr/free/nrw/commons/utils/Licenses.kt | 31 ++ .../fr/free/nrw/commons/utils/Monuments.kt | 39 +++ .../free/nrw/commons/utils/UnderlineUtils.kt | 19 ++ .../fr/free/nrw/commons/utils/UrlUtils.kt | 33 +++ .../kotlin/fr/free/nrw/commons/UtilsTest.kt | 7 +- .../ContributionsFragmentUnitTests.kt | 7 - .../upload/MediaLicensePresenterTest.kt | 10 +- .../commons/utils/UtilsFixExtensionTest.kt | 1 - 44 files changed, 387 insertions(+), 519 deletions(-) delete mode 100644 app/src/main/java/fr/free/nrw/commons/BasePresenter.java create mode 100644 app/src/main/java/fr/free/nrw/commons/BasePresenter.kt delete mode 100644 app/src/main/java/fr/free/nrw/commons/License.java delete mode 100644 app/src/main/java/fr/free/nrw/commons/MvpView.java delete mode 100644 app/src/main/java/fr/free/nrw/commons/Utils.java create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/ClipboardUtils.kt create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/FixExtension.kt create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/GeoCoordinates.kt create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/Licenses.kt create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/Monuments.kt create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/UnderlineUtils.kt create mode 100644 app/src/main/java/fr/free/nrw/commons/utils/UrlUtils.kt diff --git a/app/src/main/java/fr/free/nrw/commons/AboutActivity.kt b/app/src/main/java/fr/free/nrw/commons/AboutActivity.kt index 143f5e569..ebbb4097a 100644 --- a/app/src/main/java/fr/free/nrw/commons/AboutActivity.kt +++ b/app/src/main/java/fr/free/nrw/commons/AboutActivity.kt @@ -1,7 +1,9 @@ package fr.free.nrw.commons import android.annotation.SuppressLint +import android.content.ActivityNotFoundException import android.content.Intent +import android.content.Intent.ACTION_VIEW import android.net.Uri import android.os.Bundle import android.view.Menu @@ -16,6 +18,9 @@ import fr.free.nrw.commons.theme.BaseActivity import fr.free.nrw.commons.utils.ConfigUtils.getVersionNameWithSha import fr.free.nrw.commons.utils.DialogUtil.showAlertDialog import java.util.Collections +import androidx.core.net.toUri +import fr.free.nrw.commons.utils.handleWebUrl +import fr.free.nrw.commons.utils.setUnderlinedText /** * Represents about screen of this app @@ -59,30 +64,12 @@ class AboutActivity : BaseActivity() { binding!!.aboutImprove.setHtmlText(improveText) binding!!.aboutVersion.text = applicationContext.getVersionNameWithSha() - Utils.setUnderlinedText( - binding!!.aboutFaq, R.string.about_faq, - applicationContext - ) - Utils.setUnderlinedText( - binding!!.aboutRateUs, R.string.about_rate_us, - applicationContext - ) - Utils.setUnderlinedText( - binding!!.aboutUserGuide, R.string.user_guide, - applicationContext - ) - Utils.setUnderlinedText( - binding!!.aboutPrivacyPolicy, R.string.about_privacy_policy, - applicationContext - ) - Utils.setUnderlinedText( - binding!!.aboutTranslate, R.string.about_translate, - applicationContext - ) - Utils.setUnderlinedText( - binding!!.aboutCredits, R.string.about_credits, - applicationContext - ) + binding!!.aboutFaq.setUnderlinedText(R.string.about_faq) + binding!!.aboutRateUs.setUnderlinedText(R.string.about_rate_us) + binding!!.aboutUserGuide.setUnderlinedText(R.string.user_guide) + binding!!.aboutPrivacyPolicy.setUnderlinedText(R.string.about_privacy_policy) + binding!!.aboutTranslate.setUnderlinedText(R.string.about_translate) + binding!!.aboutCredits.setUnderlinedText(R.string.about_credits) /* To set listeners, we can create a separate method and use lambda syntax. @@ -106,47 +93,56 @@ class AboutActivity : BaseActivity() { fun launchFacebook(view: View?) { val intent: Intent try { - intent = Intent(Intent.ACTION_VIEW, Uri.parse(Urls.FACEBOOK_APP_URL)) + intent = Intent(ACTION_VIEW, Urls.FACEBOOK_APP_URL.toUri()) intent.setPackage(Urls.FACEBOOK_PACKAGE_NAME) startActivity(intent) } catch (e: Exception) { - Utils.handleWebUrl(this, Uri.parse(Urls.FACEBOOK_WEB_URL)) + handleWebUrl(this, Urls.FACEBOOK_WEB_URL.toUri()) } } fun launchGithub(view: View?) { val intent: Intent try { - intent = Intent(Intent.ACTION_VIEW, Uri.parse(Urls.GITHUB_REPO_URL)) + intent = Intent(ACTION_VIEW, Urls.GITHUB_REPO_URL.toUri()) intent.setPackage(Urls.GITHUB_PACKAGE_NAME) startActivity(intent) } catch (e: Exception) { - Utils.handleWebUrl(this, Uri.parse(Urls.GITHUB_REPO_URL)) + handleWebUrl(this, Urls.GITHUB_REPO_URL.toUri()) } } fun launchWebsite(view: View?) { - Utils.handleWebUrl(this, Uri.parse(Urls.WEBSITE_URL)) + handleWebUrl(this, Urls.WEBSITE_URL.toUri()) } fun launchRatings(view: View?) { - Utils.rateApp(this) + try { + startActivity( + Intent( + ACTION_VIEW, + (Urls.PLAY_STORE_PREFIX + packageName).toUri() + ) + ) + } catch (_: ActivityNotFoundException) { + handleWebUrl(this, (Urls.PLAY_STORE_URL_PREFIX + packageName).toUri()) + } } fun launchCredits(view: View?) { - Utils.handleWebUrl(this, Uri.parse(Urls.CREDITS_URL)) + handleWebUrl(this, Urls.CREDITS_URL.toUri()) } fun launchUserGuide(view: View?) { - Utils.handleWebUrl(this, Uri.parse(Urls.USER_GUIDE_URL)) + handleWebUrl(this, Urls.USER_GUIDE_URL.toUri()) } fun launchPrivacyPolicy(view: View?) { - Utils.handleWebUrl(this, Uri.parse(BuildConfig.PRIVACY_POLICY_URL)) + handleWebUrl(this, BuildConfig.PRIVACY_POLICY_URL.toUri()) } fun launchFrequentlyAskedQuesions(view: View?) { - Utils.handleWebUrl(this, Uri.parse(Urls.FAQ_URL)) + handleWebUrl(this, Urls.FAQ_URL.toUri()) } override fun onCreateOptionsMenu(menu: Menu): Boolean { @@ -193,7 +189,7 @@ class AboutActivity : BaseActivity() { val positiveButtonRunnable = Runnable { val langCode = instance.languageLookUpTable!!.getCodes()[spinner.selectedItemPosition] - Utils.handleWebUrl(this@AboutActivity, Uri.parse(Urls.TRANSLATE_WIKI_URL + langCode)) + handleWebUrl(this@AboutActivity, (Urls.TRANSLATE_WIKI_URL + langCode).toUri()) } showAlertDialog( this, diff --git a/app/src/main/java/fr/free/nrw/commons/BasePresenter.java b/app/src/main/java/fr/free/nrw/commons/BasePresenter.java deleted file mode 100644 index 2653b3711..000000000 --- a/app/src/main/java/fr/free/nrw/commons/BasePresenter.java +++ /dev/null @@ -1,18 +0,0 @@ -package fr.free.nrw.commons; - -import androidx.annotation.NonNull; - -/** - * Base presenter, enforcing contracts to atach and detach view - */ -public interface BasePresenter { - /** - * Until a view is attached, it is open to listen events from the presenter - */ - void onAttachView(@NonNull T view); - - /** - * Detaching a view makes sure that the view no more receives events from the presenter - */ - void onDetachView(); -} diff --git a/app/src/main/java/fr/free/nrw/commons/BasePresenter.kt b/app/src/main/java/fr/free/nrw/commons/BasePresenter.kt new file mode 100644 index 000000000..085307c3e --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/BasePresenter.kt @@ -0,0 +1,10 @@ +package fr.free.nrw.commons + +/** + * Base presenter, enforcing contracts to attach and detach view + */ +interface BasePresenter { + fun onAttachView(view: T) + + fun onDetachView() +} diff --git a/app/src/main/java/fr/free/nrw/commons/License.java b/app/src/main/java/fr/free/nrw/commons/License.java deleted file mode 100644 index 1fea236ee..000000000 --- a/app/src/main/java/fr/free/nrw/commons/License.java +++ /dev/null @@ -1,79 +0,0 @@ -package fr.free.nrw.commons; - -import androidx.annotation.Nullable; - -/** - * represents Licence object - */ -public class License { - private String key; - private String template; - private String url; - private String name; - - /** - * Constructs a new instance of License. - * - * @param key license key - * @param template license template - * @param url license URL - * @param name licence name - * - * @throws RuntimeException if License.key or Licence.template is null - */ - public License(String key, String template, String url, String name) { - if (key == null) { - throw new RuntimeException("License.key must not be null"); - } - if (template == null) { - throw new RuntimeException("License.template must not be null"); - } - this.key = key; - this.template = template; - this.url = url; - this.name = name; - } - - /** - * Gets the license key. - * @return license key as a String. - */ - public String getKey() { - return key; - } - - /** - * Gets the license template. - * @return license template as a String. - */ - public String getTemplate() { - return template; - } - - /** - * Gets the license name. If name is null, return license key. - * @return license name as string. if name null, license key as String - */ - public String getName() { - if (name == null) { - // hack - return getKey(); - } else { - return name; - } - } - - /** - * Gets the license URL - * - * @param language license language - * @return URL - */ - public @Nullable String getUrl(String language) { - if (url == null) { - return null; - } else { - return url.replace("$lang", language); - } - } -} diff --git a/app/src/main/java/fr/free/nrw/commons/Media.kt b/app/src/main/java/fr/free/nrw/commons/Media.kt index 7bd8e95fd..dbe722e91 100644 --- a/app/src/main/java/fr/free/nrw/commons/Media.kt +++ b/app/src/main/java/fr/free/nrw/commons/Media.kt @@ -1,7 +1,9 @@ package fr.free.nrw.commons import android.os.Parcelable +import fr.free.nrw.commons.BuildConfig.COMMONS_URL import fr.free.nrw.commons.location.LatLng +import fr.free.nrw.commons.wikidata.model.WikiSite import fr.free.nrw.commons.wikidata.model.page.PageTitle import kotlinx.parcelize.IgnoredOnParcel import kotlinx.parcelize.Parcelize @@ -173,7 +175,8 @@ class Media constructor( * Gets file page title * @return New media page title */ - val pageTitle: PageTitle get() = Utils.getPageTitle(filename!!) + val pageTitle: PageTitle + get() = PageTitle(filename!!, WikiSite(COMMONS_URL)) /** * Returns wikicode to use the media file on a MediaWiki site diff --git a/app/src/main/java/fr/free/nrw/commons/MvpView.java b/app/src/main/java/fr/free/nrw/commons/MvpView.java deleted file mode 100644 index 7485b2aaf..000000000 --- a/app/src/main/java/fr/free/nrw/commons/MvpView.java +++ /dev/null @@ -1,8 +0,0 @@ -package fr.free.nrw.commons; - -/** - * Base interface for all the views - */ -public interface MvpView { - void showMessage(String message); -} diff --git a/app/src/main/java/fr/free/nrw/commons/Utils.java b/app/src/main/java/fr/free/nrw/commons/Utils.java deleted file mode 100644 index 8d0f8b530..000000000 --- a/app/src/main/java/fr/free/nrw/commons/Utils.java +++ /dev/null @@ -1,264 +0,0 @@ -package fr.free.nrw.commons; - -import android.content.ClipData; -import android.content.ClipboardManager; -import android.content.Context; -import android.content.Intent; -import android.graphics.Bitmap; -import android.net.Uri; -import android.text.SpannableString; -import android.text.style.UnderlineSpan; -import android.view.View; -import android.widget.TextView; - -import androidx.annotation.NonNull; -import androidx.browser.customtabs.CustomTabColorSchemeParams; -import androidx.browser.customtabs.CustomTabsIntent; -import androidx.core.content.ContextCompat; - -import java.util.Calendar; -import java.util.Date; -import fr.free.nrw.commons.wikidata.model.WikiSite; -import fr.free.nrw.commons.wikidata.model.page.PageTitle; - -import java.util.Locale; -import java.util.regex.Pattern; - -import fr.free.nrw.commons.location.LatLng; -import fr.free.nrw.commons.settings.Prefs; -import fr.free.nrw.commons.utils.ViewUtil; -import timber.log.Timber; - -public class Utils { - - public static PageTitle getPageTitle(@NonNull String title) { - return new PageTitle(title, new WikiSite(BuildConfig.COMMONS_URL)); - } - - /** - * Generates licence name with given ID - * @param license License ID - * @return Name of license - */ - public static int licenseNameFor(String license) { - switch (license) { - case Prefs.Licenses.CC_BY_3: - return R.string.license_name_cc_by; - case Prefs.Licenses.CC_BY_4: - return R.string.license_name_cc_by_four; - case Prefs.Licenses.CC_BY_SA_3: - return R.string.license_name_cc_by_sa; - case Prefs.Licenses.CC_BY_SA_4: - return R.string.license_name_cc_by_sa_four; - case Prefs.Licenses.CC0: - return R.string.license_name_cc0; - } - throw new IllegalStateException("Unrecognized license value: " + license); - } - - /** - * Generates license url with given ID - * @param license License ID - * @return Url of license - */ - - - @NonNull - public static String licenseUrlFor(String license) { - switch (license) { - case Prefs.Licenses.CC_BY_3: - return "https://creativecommons.org/licenses/by/3.0/"; - case Prefs.Licenses.CC_BY_4: - return "https://creativecommons.org/licenses/by/4.0/"; - case Prefs.Licenses.CC_BY_SA_3: - return "https://creativecommons.org/licenses/by-sa/3.0/"; - case Prefs.Licenses.CC_BY_SA_4: - return "https://creativecommons.org/licenses/by-sa/4.0/"; - case Prefs.Licenses.CC0: - return "https://creativecommons.org/publicdomain/zero/1.0/"; - default: - throw new IllegalStateException("Unrecognized license value: " + license); - } - } - - /** - * Adds extension to filename. Converts to .jpg if system provides .jpeg, adds .jpg if no extension detected - * @param title File name - * @param extension Correct extension - * @return File with correct extension - */ - public static String fixExtension(String title, String extension) { - Pattern jpegPattern = Pattern.compile("\\.jpeg$", Pattern.CASE_INSENSITIVE); - - // People are used to ".jpg" more than ".jpeg" which the system gives us. - if (extension != null && extension.toLowerCase(Locale.ENGLISH).equals("jpeg")) { - extension = "jpg"; - } - title = jpegPattern.matcher(title).replaceFirst(".jpg"); - if (extension != null && !title.toLowerCase(Locale.getDefault()) - .endsWith("." + extension.toLowerCase(Locale.ENGLISH))) { - title += "." + extension; - } - - // If extension is still null, make it jpg. (Hotfix for https://github.com/commons-app/apps-android-commons/issues/228) - // If title has an extension in it, if won't be true - if (extension == null && title.lastIndexOf(".")<=0) { - extension = "jpg"; - title += "." + extension; - } - - return title; - } - - /** - * Launches intent to rate app - * @param context - */ - public static void rateApp(Context context) { - final String appPackageName = context.getPackageName(); - try { - context.startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse(Urls.PLAY_STORE_PREFIX + appPackageName))); - } - catch (android.content.ActivityNotFoundException anfe) { - handleWebUrl(context, Uri.parse(Urls.PLAY_STORE_URL_PREFIX + appPackageName)); - } - } - - /** - * Opens Custom Tab Activity with in-app browser for the specified URL. - * Launches intent for web URL - * @param context - * @param url - */ - public static void handleWebUrl(Context context, Uri url) { - Timber.d("Launching web url %s", url.toString()); - - final CustomTabColorSchemeParams color = new CustomTabColorSchemeParams.Builder() - .setToolbarColor(ContextCompat.getColor(context, R.color.primaryColor)) - .setSecondaryToolbarColor(ContextCompat.getColor(context, R.color.primaryDarkColor)) - .build(); - - CustomTabsIntent.Builder builder = new CustomTabsIntent.Builder(); - builder.setDefaultColorSchemeParams(color); - builder.setExitAnimations(context, android.R.anim.slide_in_left, android.R.anim.slide_out_right); - CustomTabsIntent customTabsIntent = builder.build(); - // Clear previous browser tasks, so that back/exit buttons work as intended. - customTabsIntent.intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK); - customTabsIntent.launchUrl(context, url); - } - - /** - * Util function to handle geo coordinates. It no longer depends on google maps and any app - * capable of handling the map intent can handle it - * - * @param context The context for launching intent - * @param latLng The latitude and longitude of the location - */ - public static void handleGeoCoordinates(final Context context, final LatLng latLng) { - handleGeoCoordinates(context, latLng, 16); - } - - /** - * Util function to handle geo coordinates with specified zoom level. It no longer depends on - * google maps and any app capable of handling the map intent can handle it - * - * @param context The context for launching intent - * @param latLng The latitude and longitude of the location - * @param zoomLevel The zoom level - */ - public static void handleGeoCoordinates(final Context context, final LatLng latLng, - final double zoomLevel) { - final Intent mapIntent = new Intent(Intent.ACTION_VIEW, latLng.getGmmIntentUri(zoomLevel)); - if (mapIntent.resolveActivity(context.getPackageManager()) != null) { - context.startActivity(mapIntent); - } else { - ViewUtil.showShortToast(context, context.getString(R.string.map_application_missing)); - } - } - - /** - * To take screenshot of the screen and return it in Bitmap format - * - * @param view - * @return - */ - public static Bitmap getScreenShot(View view) { - View screenView = view.getRootView(); - screenView.setDrawingCacheEnabled(true); - Bitmap drawingCache = screenView.getDrawingCache(); - if (drawingCache != null) { - Bitmap bitmap = Bitmap.createBitmap(drawingCache); - screenView.setDrawingCacheEnabled(false); - return bitmap; - } - return null; - } - - /* - *Copies the content to the clipboard - * - */ - public static void copy(String label,String text, Context context){ - ClipboardManager clipboard = (ClipboardManager) context.getSystemService(Context.CLIPBOARD_SERVICE); - ClipData clip = ClipData.newPlainText(label, text); - clipboard.setPrimaryClip(clip); - } - - /** - * This method sets underlined string text to a TextView - * - * @param textView TextView associated with string resource - * @param stringResourceName string resource name - * @param context - */ - public static void setUnderlinedText(TextView textView, int stringResourceName, Context context) { - SpannableString content = new SpannableString(context.getString(stringResourceName)); - content.setSpan(new UnderlineSpan(), 0, content.length(), 0); - textView.setText(content); - } - - /** - * For now we are enabling the monuments only when the date lies between 1 Sept & 31 OCt - * @param date - * @return - */ - public static boolean isMonumentsEnabled(final Date date) { - if (date.getMonth() == 8) { - return true; - } - return false; - } - - /** - * Util function to get the start date of wlm monument - * For this release we are hardcoding it to be 1st September - * @return - */ - public static String getWLMStartDate() { - return "1 Sep"; - } - - /*** - * Util function to get the end date of wlm monument - * For this release we are hardcoding it to be 31st October - * @return - */ - public static String getWLMEndDate() { - return "30 Sep"; - } - - /*** - * Function to get the current WLM year - * It increments at the start of September in line with the other WLM functions - * (No consideration of locales for now) - * @param calendar - * @return - */ - public static int getWikiLovesMonumentsYear(Calendar calendar) { - int year = calendar.get(Calendar.YEAR); - if (calendar.get(Calendar.MONTH) < Calendar.SEPTEMBER) { - year -= 1; - } - return year; - } -} diff --git a/app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.java b/app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.java index 8fd3fc704..a9b7381df 100644 --- a/app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.java +++ b/app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.java @@ -1,5 +1,7 @@ package fr.free.nrw.commons; +import static fr.free.nrw.commons.utils.UrlUtilsKt.handleWebUrl; + import android.net.Uri; import android.view.LayoutInflater; import android.view.View; @@ -7,6 +9,7 @@ import android.view.ViewGroup; import android.widget.TextView; import androidx.viewpager.widget.PagerAdapter; +import fr.free.nrw.commons.utils.UnderlineUtils; public class WelcomePagerAdapter extends PagerAdapter { private static final int[] PAGE_LAYOUTS = new int[]{ @@ -46,8 +49,8 @@ public class WelcomePagerAdapter extends PagerAdapter { if (position == PAGE_LAYOUTS.length - 1) { // Add link to more information TextView moreInfo = layout.findViewById(R.id.welcomeInfo); - Utils.setUnderlinedText(moreInfo, R.string.welcome_help_button_text, container.getContext()); - moreInfo.setOnClickListener(view -> Utils.handleWebUrl( + UnderlineUtils.setUnderlinedText(moreInfo, R.string.welcome_help_button_text); + moreInfo.setOnClickListener(view -> handleWebUrl( container.getContext(), Uri.parse("https://commons.wikimedia.org/wiki/Help:Contents") )); diff --git a/app/src/main/java/fr/free/nrw/commons/auth/LoginActivity.kt b/app/src/main/java/fr/free/nrw/commons/auth/LoginActivity.kt index 840bc7ca3..a606d639f 100644 --- a/app/src/main/java/fr/free/nrw/commons/auth/LoginActivity.kt +++ b/app/src/main/java/fr/free/nrw/commons/auth/LoginActivity.kt @@ -25,7 +25,6 @@ import androidx.core.content.ContextCompat import fr.free.nrw.commons.BuildConfig import fr.free.nrw.commons.CommonsApplication import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.login.LoginCallback import fr.free.nrw.commons.auth.login.LoginClient import fr.free.nrw.commons.auth.login.LoginResult @@ -38,6 +37,7 @@ import fr.free.nrw.commons.utils.ActivityUtils.startActivityWithFlags import fr.free.nrw.commons.utils.ConfigUtils.isBetaFlavour import fr.free.nrw.commons.utils.SystemThemeUtils import fr.free.nrw.commons.utils.ViewUtil.hideKeyboard +import fr.free.nrw.commons.utils.handleWebUrl import io.reactivex.disposables.CompositeDisposable import timber.log.Timber import java.util.Locale @@ -254,10 +254,10 @@ class LoginActivity : AccountAuthenticatorActivity() { } private fun forgotPassword() = - Utils.handleWebUrl(this, Uri.parse(BuildConfig.FORGOT_PASSWORD_URL)) + handleWebUrl(this, Uri.parse(BuildConfig.FORGOT_PASSWORD_URL)) private fun onPrivacyPolicyClicked() = - Utils.handleWebUrl(this, Uri.parse(BuildConfig.PRIVACY_POLICY_URL)) + handleWebUrl(this, Uri.parse(BuildConfig.PRIVACY_POLICY_URL)) private fun signUp() = startActivity(Intent(this, SignupActivity::class.java)) diff --git a/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignView.kt b/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignView.kt index 7a4720177..7a30ff5c4 100644 --- a/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignView.kt +++ b/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignView.kt @@ -7,7 +7,6 @@ import android.view.LayoutInflater import android.view.View import androidx.core.content.ContextCompat import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.campaigns.models.Campaign import fr.free.nrw.commons.contributions.MainActivity import fr.free.nrw.commons.databinding.LayoutCampaginBinding @@ -16,6 +15,7 @@ import fr.free.nrw.commons.utils.CommonsDateUtil.getIso8601DateFormatShort import fr.free.nrw.commons.utils.DateUtil.getExtraShortDateString import fr.free.nrw.commons.utils.SwipableCardView import fr.free.nrw.commons.utils.ViewUtil.showLongToast +import fr.free.nrw.commons.utils.handleWebUrl import timber.log.Timber import java.text.ParseException @@ -74,7 +74,7 @@ class CampaignView : SwipableCardView { if (it.isWLMCampaign) { ((context) as MainActivity).showNearby() } else { - Utils.handleWebUrl(context, Uri.parse(it.link)) + handleWebUrl(context, Uri.parse(it.link)) } } } diff --git a/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignsPresenter.kt b/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignsPresenter.kt index 4743e0e54..53013c1ae 100644 --- a/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignsPresenter.kt +++ b/app/src/main/java/fr/free/nrw/commons/campaigns/CampaignsPresenter.kt @@ -26,7 +26,7 @@ class CampaignsPresenter @Inject constructor( private val okHttpJsonApiClient: OkHttpJsonApiClient?, @param:Named(IO_THREAD) private val ioScheduler: Scheduler, @param:Named(MAIN_THREAD) private val mainThreadScheduler: Scheduler -) : BasePresenter { +) : BasePresenter { private var view: ICampaignsView? = null private var disposable: Disposable? = null private var campaign: Campaign? = null diff --git a/app/src/main/java/fr/free/nrw/commons/campaigns/ICampaignsView.kt b/app/src/main/java/fr/free/nrw/commons/campaigns/ICampaignsView.kt index 62a19aaac..1cbf7da1f 100644 --- a/app/src/main/java/fr/free/nrw/commons/campaigns/ICampaignsView.kt +++ b/app/src/main/java/fr/free/nrw/commons/campaigns/ICampaignsView.kt @@ -1,11 +1,10 @@ package fr.free.nrw.commons.campaigns -import fr.free.nrw.commons.MvpView import fr.free.nrw.commons.campaigns.models.Campaign /** * Interface which defines the view contracts of the campaign view */ -interface ICampaignsView : MvpView { +interface ICampaignsView { fun showCampaigns(campaign: Campaign?) } diff --git a/app/src/main/java/fr/free/nrw/commons/category/CategoryDetailsActivity.kt b/app/src/main/java/fr/free/nrw/commons/category/CategoryDetailsActivity.kt index a42d26fd6..e15d9baf3 100644 --- a/app/src/main/java/fr/free/nrw/commons/category/CategoryDetailsActivity.kt +++ b/app/src/main/java/fr/free/nrw/commons/category/CategoryDetailsActivity.kt @@ -13,9 +13,9 @@ import androidx.fragment.app.FragmentManager import androidx.lifecycle.Lifecycle import androidx.lifecycle.lifecycleScope import androidx.lifecycle.repeatOnLifecycle +import fr.free.nrw.commons.BuildConfig.COMMONS_URL import fr.free.nrw.commons.Media import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.ViewPagerAdapter import fr.free.nrw.commons.databinding.ActivityCategoryDetailsBinding import fr.free.nrw.commons.explore.categories.media.CategoriesMediaFragment @@ -23,6 +23,9 @@ import fr.free.nrw.commons.explore.categories.parent.ParentCategoriesFragment import fr.free.nrw.commons.explore.categories.sub.SubCategoriesFragment import fr.free.nrw.commons.media.MediaDetailPagerFragment import fr.free.nrw.commons.theme.BaseActivity +import fr.free.nrw.commons.utils.handleWebUrl +import fr.free.nrw.commons.wikidata.model.WikiSite +import fr.free.nrw.commons.wikidata.model.page.PageTitle import kotlinx.coroutines.launch import javax.inject.Inject @@ -199,8 +202,9 @@ class CategoryDetailsActivity : BaseActivity(), override fun onOptionsItemSelected(item: MenuItem): Boolean { return when (item.itemId) { R.id.menu_browser_current_category -> { - val title = Utils.getPageTitle(CATEGORY_PREFIX + categoryName) - Utils.handleWebUrl(this, Uri.parse(title.canonicalUri)) + val title = PageTitle(CATEGORY_PREFIX + categoryName, WikiSite(COMMONS_URL)) + + handleWebUrl(this, Uri.parse(title.canonicalUri)) true } diff --git a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsContract.kt b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsContract.kt index 7027950e3..0e039729f 100644 --- a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsContract.kt +++ b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsContract.kt @@ -9,7 +9,6 @@ import fr.free.nrw.commons.BasePresenter interface ContributionsContract { interface View { - fun showMessage(localizedMessage: String) fun getContext(): Context? } diff --git a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsFragment.kt b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsFragment.kt index 3992d35dd..541cc6e56 100644 --- a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsFragment.kt @@ -30,7 +30,6 @@ import androidx.work.WorkManager import fr.free.nrw.commons.MapController.NearbyPlacesInfo import fr.free.nrw.commons.Media import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.campaigns.CampaignView import fr.free.nrw.commons.campaigns.CampaignsPresenter @@ -64,6 +63,9 @@ import fr.free.nrw.commons.utils.LengthUtils.formatDistanceBetween import fr.free.nrw.commons.utils.NetworkUtils.isInternetConnectionEstablished import fr.free.nrw.commons.utils.PermissionUtils.hasPermission import fr.free.nrw.commons.utils.ViewUtil.showLongToast +import fr.free.nrw.commons.utils.isMonumentsEnabled +import fr.free.nrw.commons.utils.wLMEndDate +import fr.free.nrw.commons.utils.wLMStartDate import io.reactivex.Observable import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.disposables.CompositeDisposable @@ -242,8 +244,8 @@ class ContributionsFragment : CommonsDaggerSupportFragment(), FragmentManager.On private fun initWLMCampaign() { wlmCampaign = Campaign( getString(R.string.wlm_campaign_title), - getString(R.string.wlm_campaign_description), Utils.getWLMStartDate().toString(), - Utils.getWLMEndDate().toString(), NearbyParentFragment.WLM_URL, true + getString(R.string.wlm_campaign_description), wLMStartDate, + wLMEndDate, NearbyParentFragment.WLM_URL, true ) } @@ -729,7 +731,7 @@ class ContributionsFragment : CommonsDaggerSupportFragment(), FragmentManager.On * of campaigns on the campaigns card */ private fun fetchCampaigns() { - if (Utils.isMonumentsEnabled(Date())) { + if (isMonumentsEnabled) { if (binding != null) { binding!!.campaignsView.setCampaign(wlmCampaign) binding!!.campaignsView.visibility = View.VISIBLE @@ -743,10 +745,6 @@ class ContributionsFragment : CommonsDaggerSupportFragment(), FragmentManager.On } } - override fun showMessage(message: String) { - Toast.makeText(context, message, Toast.LENGTH_SHORT).show() - } - override fun showCampaigns(campaign: Campaign?) { if (campaign != null && !isUserProfile) { if (binding != null) { diff --git a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListContract.kt b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListContract.kt index c6b8dd8a8..0c8c822ad 100644 --- a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListContract.kt +++ b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListContract.kt @@ -15,7 +15,7 @@ class ContributionsListContract { fun showNoContributionsUI(shouldShow: Boolean) } - interface UserActionListener : BasePresenter { + interface UserActionListener : BasePresenter { fun refreshList(swipeRefreshLayout: SwipeRefreshLayout?) } } diff --git a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListFragment.kt b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListFragment.kt index 9ecb35b24..b86cd6dc9 100644 --- a/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/contributions/ContributionsListFragment.kt @@ -29,7 +29,6 @@ import androidx.recyclerview.widget.SimpleItemAnimator import fr.free.nrw.commons.Media import fr.free.nrw.commons.MediaDataExtractor import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.contributions.WikipediaInstructionsDialogFragment.Companion.newInstance import fr.free.nrw.commons.databinding.FragmentContributionsListBinding @@ -41,6 +40,8 @@ import fr.free.nrw.commons.profile.ProfileActivity import fr.free.nrw.commons.utils.DialogUtil.showAlertDialog import fr.free.nrw.commons.utils.SystemThemeUtils import fr.free.nrw.commons.utils.ViewUtil.showShortToast +import fr.free.nrw.commons.utils.copyToClipboard +import fr.free.nrw.commons.utils.handleWebUrl import fr.free.nrw.commons.wikidata.model.WikiSite import org.apache.commons.lang3.StringUtils import javax.inject.Inject @@ -527,14 +528,13 @@ class ContributionsListFragment : CommonsDaggerSupportFragment(), ContributionsL */ override fun onConfirmClicked(contribution: Contribution?, copyWikicode: Boolean) { if (copyWikicode) { - val wikicode = contribution!!.media.wikiCode - Utils.copy("wikicode", wikicode, context) + requireContext().copyToClipboard("wikicode", contribution!!.media.wikiCode) } val url = languageWikipediaSite!!.mobileUrl() + "/wiki/" + (contribution!!.wikidataPlace ?.getWikipediaPageTitle()) - Utils.handleWebUrl(context, Uri.parse(url)) + handleWebUrl(requireContext(), Uri.parse(url)) } fun getContributionStateAt(position: Int): Int { diff --git a/app/src/main/java/fr/free/nrw/commons/explore/depictions/WikidataItemDetailsActivity.java b/app/src/main/java/fr/free/nrw/commons/explore/depictions/WikidataItemDetailsActivity.java index cf7269123..89593d07e 100644 --- a/app/src/main/java/fr/free/nrw/commons/explore/depictions/WikidataItemDetailsActivity.java +++ b/app/src/main/java/fr/free/nrw/commons/explore/depictions/WikidataItemDetailsActivity.java @@ -1,5 +1,7 @@ package fr.free.nrw.commons.explore.depictions; +import static fr.free.nrw.commons.utils.UrlUtilsKt.handleWebUrl; + import android.content.Context; import android.content.Intent; import android.net.Uri; @@ -8,16 +10,11 @@ import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; -import android.widget.FrameLayout; -import androidx.appcompat.widget.Toolbar; import androidx.fragment.app.Fragment; import androidx.fragment.app.FragmentManager; -import androidx.viewpager.widget.ViewPager; import com.google.android.material.snackbar.Snackbar; -import com.google.android.material.tabs.TabLayout; import fr.free.nrw.commons.Media; import fr.free.nrw.commons.R; -import fr.free.nrw.commons.Utils; import fr.free.nrw.commons.ViewPagerAdapter; import fr.free.nrw.commons.bookmarks.items.BookmarkItemsDao; import fr.free.nrw.commons.category.CategoryImagesCallback; @@ -249,7 +246,7 @@ public class WikidataItemDetailsActivity extends BaseActivity implements MediaDe case R.id.browser_actions_menu_items: String entityId=getIntent().getStringExtra("entityId"); Uri uri = Uri.parse("https://www.wikidata.org/wiki/" + entityId); - Utils.handleWebUrl(this, uri); + handleWebUrl(this, uri); return true; case R.id.menu_bookmark_current_item: diff --git a/app/src/main/java/fr/free/nrw/commons/explore/map/ExploreMapFragment.java b/app/src/main/java/fr/free/nrw/commons/explore/map/ExploreMapFragment.java index 2b3aa9f3c..364f4d53a 100644 --- a/app/src/main/java/fr/free/nrw/commons/explore/map/ExploreMapFragment.java +++ b/app/src/main/java/fr/free/nrw/commons/explore/map/ExploreMapFragment.java @@ -2,7 +2,9 @@ package fr.free.nrw.commons.explore.map; import static fr.free.nrw.commons.location.LocationServiceManager.LocationChangeType.LOCATION_SIGNIFICANTLY_CHANGED; import static fr.free.nrw.commons.location.LocationServiceManager.LocationChangeType.LOCATION_SLIGHTLY_CHANGED; +import static fr.free.nrw.commons.utils.GeoCoordinatesKt.handleGeoCoordinates; import static fr.free.nrw.commons.utils.MapUtils.ZOOM_LEVEL; +import static fr.free.nrw.commons.utils.UrlUtilsKt.handleWebUrl; import android.Manifest.permission; import android.annotation.SuppressLint; @@ -36,7 +38,6 @@ import fr.free.nrw.commons.BaseMarker; import fr.free.nrw.commons.MapController; import fr.free.nrw.commons.Media; import fr.free.nrw.commons.R; -import fr.free.nrw.commons.Utils; import fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsDao; import fr.free.nrw.commons.contributions.MainActivity; import fr.free.nrw.commons.databinding.FragmentExploreMapBinding; @@ -639,13 +640,13 @@ public class ExploreMapFragment extends CommonsDaggerSupportFragment */ private void passInfoToSheet(final Place place) { binding.bottomSheetDetailsBinding.directionsButton.setOnClickListener( - view -> Utils.handleGeoCoordinates(getActivity(), + view -> handleGeoCoordinates(requireActivity(), place.getLocation(), binding.mapView.getZoomLevelDouble())); binding.bottomSheetDetailsBinding.commonsButton.setVisibility( place.hasCommonsLink() ? View.VISIBLE : View.GONE); binding.bottomSheetDetailsBinding.commonsButton.setOnClickListener( - view -> Utils.handleWebUrl(getContext(), place.siteLinks.getCommonsLink())); + view -> handleWebUrl(getContext(), place.siteLinks.getCommonsLink())); int index = 0; for (Media media : mediaList) { diff --git a/app/src/main/java/fr/free/nrw/commons/locationpicker/LocationPickerActivity.kt b/app/src/main/java/fr/free/nrw/commons/locationpicker/LocationPickerActivity.kt index 2a7b7713b..a8b6ddf26 100644 --- a/app/src/main/java/fr/free/nrw/commons/locationpicker/LocationPickerActivity.kt +++ b/app/src/main/java/fr/free/nrw/commons/locationpicker/LocationPickerActivity.kt @@ -30,7 +30,6 @@ import fr.free.nrw.commons.CameraPosition import fr.free.nrw.commons.CommonsApplication import fr.free.nrw.commons.Media import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.auth.csrf.CsrfTokenClient import fr.free.nrw.commons.coordinates.CoordinateEditHelper @@ -45,6 +44,7 @@ import fr.free.nrw.commons.upload.mediaDetails.UploadMediaDetailFragment.Compani import fr.free.nrw.commons.upload.mediaDetails.UploadMediaDetailFragment.Companion.LAST_ZOOM import fr.free.nrw.commons.utils.DialogUtil import fr.free.nrw.commons.utils.MapUtils.ZOOM_LEVEL +import fr.free.nrw.commons.utils.handleGeoCoordinates import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.schedulers.Schedulers import org.osmdroid.tileprovider.tilesource.TileSourceFactory @@ -432,8 +432,8 @@ class LocationPickerActivity : BaseActivity(), LocationPermissionCallback { position?.let { mapView?.zoomLevelDouble?.let { zoomLevel -> - Utils.handleGeoCoordinates(this, it, zoomLevel) - } ?: Utils.handleGeoCoordinates(this, it) + handleGeoCoordinates(this, it, zoomLevel) + } ?: handleGeoCoordinates(this, it) } } diff --git a/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt b/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt index f371b733f..5980e1fb5 100644 --- a/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/media/MediaDetailFragment.kt @@ -77,7 +77,7 @@ import fr.free.nrw.commons.CommonsApplication.Companion.instance import fr.free.nrw.commons.Media import fr.free.nrw.commons.MediaDataExtractor import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils +import fr.free.nrw.commons.utils.UnderlineUtils import fr.free.nrw.commons.actions.ThanksClient import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.auth.csrf.InvalidLoginTokenException @@ -119,6 +119,10 @@ import fr.free.nrw.commons.utils.PermissionUtils.hasPermission import fr.free.nrw.commons.utils.ViewUtil import fr.free.nrw.commons.utils.ViewUtil.showShortToast import fr.free.nrw.commons.utils.ViewUtilWrapper +import fr.free.nrw.commons.utils.copyToClipboard +import fr.free.nrw.commons.utils.handleGeoCoordinates +import fr.free.nrw.commons.utils.handleWebUrl +import fr.free.nrw.commons.utils.setUnderlinedText import fr.free.nrw.commons.wikidata.mwapi.MwQueryPage.Revision import io.reactivex.Observable import io.reactivex.Single @@ -316,8 +320,7 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C _binding = FragmentMediaDetailBinding.inflate(inflater, container, false) val view: View = binding.root - - Utils.setUnderlinedText(binding.seeMore, R.string.nominated_see_more, requireContext()) + binding.seeMore.setUnderlinedText(R.string.nominated_see_more) if (isCategoryImage) { binding.authorLinearLayout.visibility = View.VISIBLE @@ -909,7 +912,7 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C private fun onMediaDetailLicenceClicked() { val url: String? = media!!.licenseUrl if (!StringUtils.isBlank(url) && activity != null) { - Utils.handleWebUrl(activity, Uri.parse(url)) + handleWebUrl(requireContext(), Uri.parse(url)) } else { viewUtil.showShortToast(requireActivity(), getString(R.string.null_url)) } @@ -917,17 +920,17 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C private fun onMediaDetailCoordinatesClicked() { if (media!!.coordinates != null && activity != null) { - Utils.handleGeoCoordinates(activity, media!!.coordinates) + handleGeoCoordinates(requireContext(), media!!.coordinates!!) } } private fun onCopyWikicodeClicked() { val data: String = "[[" + media!!.filename + "|thumb|" + media!!.fallbackDescription + "]]" - Utils.copy("wikiCode", data, context) + requireContext().copyToClipboard("wikiCode", data) Timber.d("Generated wikidata copy code: %s", data) - Toast.makeText(context, getString(R.string.wikicode_copied), Toast.LENGTH_SHORT) + Toast.makeText(requireContext(), getString(R.string.wikicode_copied), Toast.LENGTH_SHORT) .show() } @@ -1765,7 +1768,7 @@ class MediaDetailFragment : CommonsDaggerSupportFragment(), CategoryEditHelper.C private fun onSeeMoreClicked() { if (binding.nominatedDeletionBanner.visibility == View.VISIBLE && activity != null) { - Utils.handleWebUrl(activity, Uri.parse(media!!.pageTitle.mobileUri)) + handleWebUrl(requireContext(), Uri.parse(media!!.pageTitle.mobileUri)) } } diff --git a/app/src/main/java/fr/free/nrw/commons/media/MediaDetailPagerFragment.java b/app/src/main/java/fr/free/nrw/commons/media/MediaDetailPagerFragment.java index cba582a35..324d5867b 100644 --- a/app/src/main/java/fr/free/nrw/commons/media/MediaDetailPagerFragment.java +++ b/app/src/main/java/fr/free/nrw/commons/media/MediaDetailPagerFragment.java @@ -1,6 +1,6 @@ package fr.free.nrw.commons.media; -import static fr.free.nrw.commons.Utils.handleWebUrl; +import static fr.free.nrw.commons.utils.UrlUtilsKt.handleWebUrl; import android.os.Handler; import android.os.Looper; @@ -31,7 +31,7 @@ import com.google.android.material.snackbar.Snackbar; import fr.free.nrw.commons.CommonsApplication; import fr.free.nrw.commons.Media; import fr.free.nrw.commons.R; -import fr.free.nrw.commons.Utils; +import fr.free.nrw.commons.utils.ClipboardUtils; import fr.free.nrw.commons.auth.SessionManager; import fr.free.nrw.commons.bookmarks.models.Bookmark; import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesContentProvider; @@ -216,7 +216,7 @@ public class MediaDetailPagerFragment extends CommonsDaggerSupportFragment imple return true; case R.id.menu_copy_link: String uri = m.getPageTitle().getCanonicalUri(); - Utils.copy("shareLink", uri, requireContext()); + ClipboardUtils.copy("shareLink", uri, requireContext()); Timber.d("Copied share link to clipboard: %s", uri); Toast.makeText(requireContext(), getString(R.string.menu_link_copied), Toast.LENGTH_SHORT).show(); diff --git a/app/src/main/java/fr/free/nrw/commons/nearby/fragments/CommonPlaceClickActions.kt b/app/src/main/java/fr/free/nrw/commons/nearby/fragments/CommonPlaceClickActions.kt index 202f2c305..5f4d0ab13 100644 --- a/app/src/main/java/fr/free/nrw/commons/nearby/fragments/CommonPlaceClickActions.kt +++ b/app/src/main/java/fr/free/nrw/commons/nearby/fragments/CommonPlaceClickActions.kt @@ -10,12 +10,13 @@ import androidx.activity.result.ActivityResultLauncher import androidx.appcompat.app.AlertDialog import androidx.appcompat.widget.PopupMenu import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.LoginActivity import fr.free.nrw.commons.contributions.ContributionController import fr.free.nrw.commons.kvstore.JsonKvStore import fr.free.nrw.commons.nearby.Place import fr.free.nrw.commons.utils.ActivityUtils +import fr.free.nrw.commons.utils.handleGeoCoordinates +import fr.free.nrw.commons.utils.handleWebUrl import fr.free.nrw.commons.wikidata.WikidataConstants import timber.log.Timber import javax.inject.Inject @@ -104,7 +105,7 @@ class CommonPlaceClickActions fun onDirectionsClicked(): (Place) -> Unit = { - Utils.handleGeoCoordinates(activity, it.getLocation()) + handleGeoCoordinates(activity, it.getLocation()) } private fun storeSharedPrefs(selectedPlace: Place) { @@ -113,7 +114,7 @@ class CommonPlaceClickActions } private fun openWebView(link: Uri): Boolean { - Utils.handleWebUrl(activity, link) + handleWebUrl(activity, link) return true } diff --git a/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt b/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt index 26875927e..a0dcead07 100644 --- a/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/nearby/fragments/NearbyParentFragment.kt @@ -58,12 +58,10 @@ import fr.free.nrw.commons.CommonsApplication import fr.free.nrw.commons.MapController.NearbyPlacesInfo import fr.free.nrw.commons.Media import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsDao import fr.free.nrw.commons.contributions.ContributionController import fr.free.nrw.commons.contributions.MainActivity import fr.free.nrw.commons.contributions.MainActivity.ActiveFragment -import fr.free.nrw.commons.customselector.ui.selector.ImageLoader import fr.free.nrw.commons.databinding.FragmentNearbyParentBinding import fr.free.nrw.commons.di.CommonsDaggerSupportFragment import fr.free.nrw.commons.filepicker.FilePicker @@ -76,7 +74,6 @@ import fr.free.nrw.commons.location.LocationServiceManager.LocationChangeType import fr.free.nrw.commons.location.LocationUpdateListener import fr.free.nrw.commons.media.MediaClient import fr.free.nrw.commons.media.MediaDetailPagerFragment -import fr.free.nrw.commons.media.MediaDetailPagerFragment.MediaDetailProvider import fr.free.nrw.commons.navtab.NavTab import fr.free.nrw.commons.nearby.BottomSheetAdapter import fr.free.nrw.commons.nearby.BottomSheetAdapter.ItemClickListener @@ -105,6 +102,10 @@ import fr.free.nrw.commons.utils.NearbyFABUtils.removeAnchorFromFAB import fr.free.nrw.commons.utils.NetworkUtils.isInternetConnectionEstablished import fr.free.nrw.commons.utils.SystemThemeUtils import fr.free.nrw.commons.utils.ViewUtil.showLongToast +import fr.free.nrw.commons.utils.copyToClipboard +import fr.free.nrw.commons.utils.handleGeoCoordinates +import fr.free.nrw.commons.utils.handleWebUrl +import fr.free.nrw.commons.utils.isMonumentsEnabled import fr.free.nrw.commons.wikidata.WikidataConstants import fr.free.nrw.commons.wikidata.WikidataEditListener import fr.free.nrw.commons.wikidata.WikidataEditListener.WikidataP18EditListener @@ -140,7 +141,6 @@ import java.util.UUID import java.util.concurrent.TimeUnit import javax.inject.Inject import javax.inject.Named -import javax.sql.DataSource import kotlin.concurrent.Volatile @@ -467,7 +467,7 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), } } _isDarkTheme = systemThemeUtils?.isDeviceInNightMode() == true - if (Utils.isMonumentsEnabled(Date())) { + if (isMonumentsEnabled) { binding?.rlContainerWlmMonthMessage?.visibility = View.VISIBLE } else { binding?.rlContainerWlmMonthMessage?.visibility = View.GONE @@ -836,7 +836,7 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), loadAnimations() setBottomSheetCallbacks() addActionToTitle() - if (!Utils.isMonumentsEnabled(Date())) { + if (!isMonumentsEnabled) { NearbyFilterState.setWlmSelected(false) } } @@ -1017,11 +1017,10 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), */ private fun addActionToTitle() { binding!!.bottomSheetDetails.title.setOnLongClickListener { view -> - Utils.copy( - "place", binding!!.bottomSheetDetails.title.text.toString(), - context + requireContext().copyToClipboard( + "place", binding!!.bottomSheetDetails.title.text.toString() ) - Toast.makeText(context, fr.free.nrw.commons.R.string.text_copy, Toast.LENGTH_SHORT) + Toast.makeText(requireContext(), fr.free.nrw.commons.R.string.text_copy, Toast.LENGTH_SHORT) .show() true } @@ -1580,7 +1579,7 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), searchLatLng, false, true, - Utils.isMonumentsEnabled(Date()), + isMonumentsEnabled, customQuery ) } @@ -1633,7 +1632,7 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), searchLatLng, false, true, - Utils.isMonumentsEnabled(Date()), + isMonumentsEnabled, customQuery ) } @@ -2854,14 +2853,14 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), R.drawable.ic_directions_black_24dp -> { selectedPlace?.let { - Utils.handleGeoCoordinates(this.context, it.getLocation()) + handleGeoCoordinates(requireContext(), it.getLocation()) binding?.map?.zoomLevelDouble ?: 0.0 } } R.drawable.ic_wikidata_logo_24dp -> { selectedPlace?.siteLinks?.wikidataLink?.let { - Utils.handleWebUrl(this.context, it) + handleWebUrl(requireContext(), it) } } @@ -2879,13 +2878,13 @@ class NearbyParentFragment : CommonsDaggerSupportFragment(), R.drawable.ic_wikipedia_logo_24dp -> { selectedPlace?.siteLinks?.wikipediaLink?.let { - Utils.handleWebUrl(this.context, it) + handleWebUrl(requireContext(), it) } } R.drawable.ic_commons_icon_vector -> { selectedPlace?.siteLinks?.commonsLink?.let { - Utils.handleWebUrl(this.context, it) + handleWebUrl(requireContext(), it) } } diff --git a/app/src/main/java/fr/free/nrw/commons/notification/NotificationActivity.kt b/app/src/main/java/fr/free/nrw/commons/notification/NotificationActivity.kt index 1547f89ad..76975964b 100644 --- a/app/src/main/java/fr/free/nrw/commons/notification/NotificationActivity.kt +++ b/app/src/main/java/fr/free/nrw/commons/notification/NotificationActivity.kt @@ -13,7 +13,6 @@ import androidx.recyclerview.widget.LinearLayoutManager import com.google.android.material.snackbar.Snackbar import fr.free.nrw.commons.CommonsApplication import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.auth.csrf.InvalidLoginTokenException import fr.free.nrw.commons.databinding.ActivityNotificationBinding @@ -22,6 +21,7 @@ import fr.free.nrw.commons.notification.models.NotificationType import fr.free.nrw.commons.theme.BaseActivity import fr.free.nrw.commons.utils.NetworkUtils import fr.free.nrw.commons.utils.ViewUtil +import fr.free.nrw.commons.utils.handleWebUrl import io.reactivex.Observable import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.schedulers.Schedulers @@ -197,7 +197,7 @@ class NotificationActivity : BaseActivity() { private fun handleUrl(url: String?) { if (url.isNullOrEmpty()) return - Utils.handleWebUrl(this, Uri.parse(url)) + handleWebUrl(this, Uri.parse(url)) } private fun setItems(notificationList: List?) { diff --git a/app/src/main/java/fr/free/nrw/commons/profile/ProfileActivity.kt b/app/src/main/java/fr/free/nrw/commons/profile/ProfileActivity.kt index 164842c9a..105cf1860 100644 --- a/app/src/main/java/fr/free/nrw/commons/profile/ProfileActivity.kt +++ b/app/src/main/java/fr/free/nrw/commons/profile/ProfileActivity.kt @@ -3,16 +3,16 @@ package fr.free.nrw.commons.profile import android.content.Context import android.content.Intent import android.graphics.Bitmap -import android.net.Uri import android.os.Bundle import android.util.Log -import android.view.* +import android.view.Menu +import android.view.MenuItem +import android.view.View import android.widget.ImageView import android.widget.TextView import androidx.core.content.FileProvider import androidx.fragment.app.Fragment import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.ViewPagerAdapter import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.contributions.ContributionsFragment @@ -23,7 +23,7 @@ import fr.free.nrw.commons.theme.BaseActivity import fr.free.nrw.commons.utils.DialogUtil import java.io.File import java.io.FileOutputStream -import java.util.* +import java.util.Locale import javax.inject.Inject /** @@ -133,7 +133,7 @@ class ProfileActivity : BaseActivity() { return when (item.itemId) { R.id.share_app_icon -> { val rootView = window.decorView.findViewById(android.R.id.content) - val screenShot = Utils.getScreenShot(rootView) + val screenShot = getScreenShot(rootView) if (screenShot == null) { Log.e("ERROR", "ScreenShot is null") return false @@ -212,6 +212,24 @@ class ProfileActivity : BaseActivity() { binding.tabLayout.visibility = if (isVisible) View.VISIBLE else View.GONE } + /** + * To take screenshot of the screen and return it in Bitmap format + * + * @param view + * @return + */ + fun getScreenShot(view: View): Bitmap? { + val screenView = view.rootView + screenView.isDrawingCacheEnabled = true + val drawingCache = screenView.drawingCache + if (drawingCache != null) { + val bitmap = Bitmap.createBitmap(drawingCache) + screenView.isDrawingCacheEnabled = false + return bitmap + } + return null + } + companion object { const val KEY_USERNAME = "username" const val KEY_SHOULD_SHOW_CONTRIBUTIONS = "shouldShowContributions" diff --git a/app/src/main/java/fr/free/nrw/commons/profile/achievements/AchievementsFragment.kt b/app/src/main/java/fr/free/nrw/commons/profile/achievements/AchievementsFragment.kt index f967b8619..8f23674ca 100644 --- a/app/src/main/java/fr/free/nrw/commons/profile/achievements/AchievementsFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/profile/achievements/AchievementsFragment.kt @@ -15,7 +15,6 @@ import com.google.android.material.badge.BadgeDrawable import com.google.android.material.badge.BadgeUtils import com.google.android.material.badge.ExperimentalBadgeUtils import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.auth.SessionManager import fr.free.nrw.commons.databinding.FragmentAchievementsBinding import fr.free.nrw.commons.di.CommonsDaggerSupportFragment @@ -27,6 +26,7 @@ import fr.free.nrw.commons.utils.ConfigUtils.isBetaFlavour import fr.free.nrw.commons.utils.DialogUtil.showAlertDialog import fr.free.nrw.commons.utils.ViewUtil.showDismissibleSnackBar import fr.free.nrw.commons.utils.ViewUtil.showLongToast +import fr.free.nrw.commons.utils.handleWebUrl import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.schedulers.Schedulers import org.apache.commons.lang3.StringUtils @@ -524,7 +524,7 @@ class AchievementsFragment : CommonsDaggerSupportFragment(){ getString(R.string.ok), getString(R.string.read_help_link), {}, - { Utils.handleWebUrl(requireContext(), Uri.parse(helpLinkUrl)) }, + { handleWebUrl(requireContext(), Uri.parse(helpLinkUrl)) }, null ) } diff --git a/app/src/main/java/fr/free/nrw/commons/settings/SettingsFragment.kt b/app/src/main/java/fr/free/nrw/commons/settings/SettingsFragment.kt index 092f057e9..387dd4672 100644 --- a/app/src/main/java/fr/free/nrw/commons/settings/SettingsFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/settings/SettingsFragment.kt @@ -33,9 +33,7 @@ import com.karumi.dexter.MultiplePermissionsReport import com.karumi.dexter.PermissionToken import com.karumi.dexter.listener.PermissionRequest import com.karumi.dexter.listener.multi.MultiplePermissionsListener -import fr.free.nrw.commons.BuildConfig.MOBILE_META_URL import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.activity.SingleWebViewActivity import fr.free.nrw.commons.campaigns.CampaignView import fr.free.nrw.commons.contributions.ContributionController @@ -53,6 +51,7 @@ import fr.free.nrw.commons.utils.DialogUtil import fr.free.nrw.commons.utils.PermissionUtils import fr.free.nrw.commons.utils.StringUtil import fr.free.nrw.commons.utils.ViewUtil +import fr.free.nrw.commons.utils.handleWebUrl import java.util.Locale import javax.inject.Inject import javax.inject.Named @@ -239,7 +238,10 @@ class SettingsFragment : PreferenceFragmentCompat() { val betaTesterPreference: Preference? = findPreference("becomeBetaTester") betaTesterPreference?.setOnPreferenceClickListener { - Utils.handleWebUrl(requireActivity(), Uri.parse(getString(R.string.beta_opt_in_link))) + handleWebUrl( + requireActivity(), + Uri.parse(getString(R.string.beta_opt_in_link)) + ) true } @@ -296,7 +298,7 @@ class SettingsFragment : PreferenceFragmentCompat() { getString(R.string.ok), getString(R.string.read_help_link), { }, - { Utils.handleWebUrl(requireContext(), Uri.parse(GET_CONTENT_PICKER_HELP_URL)) }, + { handleWebUrl(requireContext(), Uri.parse(GET_CONTENT_PICKER_HELP_URL)) }, null ) } diff --git a/app/src/main/java/fr/free/nrw/commons/upload/PageContentsCreator.kt b/app/src/main/java/fr/free/nrw/commons/upload/PageContentsCreator.kt index 0c4ded8b2..2de17f849 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/PageContentsCreator.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/PageContentsCreator.kt @@ -1,11 +1,11 @@ package fr.free.nrw.commons.upload import android.content.Context -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.contributions.Contribution import fr.free.nrw.commons.filepicker.UploadableFile.DateTimeWithSource import fr.free.nrw.commons.settings.Prefs.Licenses import fr.free.nrw.commons.utils.ConfigUtils.getVersionNameWithSha +import fr.free.nrw.commons.utils.getWikiLovesMonumentsYear import org.apache.commons.lang3.StringUtils import java.text.SimpleDateFormat import java.util.Calendar @@ -49,7 +49,7 @@ class PageContentsCreator @Inject constructor(private val context: Context) { String.format( Locale.ENGLISH, "{{Wiki Loves Monuments %d|1= %s}}\n", - Utils.getWikiLovesMonumentsYear(Calendar.getInstance()), + getWikiLovesMonumentsYear(Calendar.getInstance()), contribution.countryCode ) ) diff --git a/app/src/main/java/fr/free/nrw/commons/upload/UploadItem.kt b/app/src/main/java/fr/free/nrw/commons/upload/UploadItem.kt index f357cd112..6d2321def 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/UploadItem.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/UploadItem.kt @@ -1,10 +1,10 @@ package fr.free.nrw.commons.upload import android.net.Uri -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.filepicker.MimeTypeMapWrapper.Companion.getExtensionFromMimeType import fr.free.nrw.commons.nearby.Place import fr.free.nrw.commons.utils.ImageUtils +import fr.free.nrw.commons.utils.fixExtension class UploadItem( var mediaUri: Uri?, @@ -32,7 +32,7 @@ class UploadItem( * languages have been entered, the first language is used. */ val filename: String - get() = Utils.fixExtension( + get() = fixExtension( uploadMediaDetails[0].captionText, getExtensionFromMimeType(mimeType) ) diff --git a/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicenseFragment.kt b/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicenseFragment.kt index 0415d3270..65826a505 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicenseFragment.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicenseFragment.kt @@ -16,11 +16,13 @@ import android.widget.AdapterView import android.widget.ArrayAdapter import android.widget.TextView import fr.free.nrw.commons.R -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.databinding.FragmentMediaLicenseBinding import fr.free.nrw.commons.upload.UploadActivity import fr.free.nrw.commons.upload.UploadBaseFragment import fr.free.nrw.commons.utils.DialogUtil.showAlertDialog +import fr.free.nrw.commons.utils.handleWebUrl +import fr.free.nrw.commons.utils.toLicenseName +import fr.free.nrw.commons.utils.toLicenseUrl import timber.log.Timber import javax.inject.Inject @@ -126,20 +128,20 @@ class MediaLicenseFragment : UploadBaseFragment(), MediaLicenseContract.View { } override fun setSelectedLicense(license: String?) { - var position = licenses!!.indexOf(getString(Utils.licenseNameFor(license))) + var position = license?.let { licenses!!.indexOf(getString(it.toLicenseName())) } ?: -1 // Check if position is valid if (position < 0) { Timber.d("Invalid position: %d. Using default licenses", position) position = licenses!!.size - 1 - } else { - Timber.d("Position: %d %s", position, getString(Utils.licenseNameFor(license))) } binding.spinnerLicenseList.setSelection(position) } override fun updateLicenseSummary(selectedLicense: String?, numberOfItems: Int) { - val licenseHyperLink = "" + - getString(Utils.licenseNameFor(selectedLicense)) + "
" + if (selectedLicense == null) return + + val licenseHyperLink = "" + + getString(selectedLicense.toLicenseName()) + "
" setTextViewHTML( binding.tvShareLicenseSummary, resources @@ -184,7 +186,7 @@ class MediaLicenseFragment : UploadBaseFragment(), MediaLicenseContract.View { } private fun launchBrowser(hyperLink: String) = - Utils.handleWebUrl(context, Uri.parse(hyperLink)) + handleWebUrl(requireContext(), Uri.parse(hyperLink)) override fun onDestroyView() { presenter.onDetachView() diff --git a/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicensePresenter.kt b/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicensePresenter.kt index 25d1a2324..df75019b2 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicensePresenter.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/license/MediaLicensePresenter.kt @@ -1,9 +1,9 @@ package fr.free.nrw.commons.upload.license -import fr.free.nrw.commons.Utils import fr.free.nrw.commons.kvstore.JsonKvStore import fr.free.nrw.commons.repository.UploadRepository import fr.free.nrw.commons.settings.Prefs +import fr.free.nrw.commons.utils.toLicenseName import timber.log.Timber import java.lang.reflect.Method import java.lang.reflect.Proxy @@ -34,12 +34,14 @@ class MediaLicensePresenter @Inject constructor( val licenses = repository.getLicenses() view.setLicenses(licenses) - var selectedLicense = defaultKVStore.getString( + //CC_BY_SA_4 is the default one used by the commons web app + var selectedLicense: String = defaultKVStore.getString( Prefs.DEFAULT_LICENSE, Prefs.Licenses.CC_BY_SA_4 - ) //CC_BY_SA_4 is the default one used by the commons web app + ) ?: Prefs.Licenses.CC_BY_SA_4 + try { //I have to make sure that the stored default license was not one of the deprecated one's - Utils.licenseNameFor(selectedLicense) + selectedLicense.toLicenseName() } catch (exception: IllegalStateException) { Timber.e(exception) selectedLicense = Prefs.Licenses.CC_BY_SA_4 diff --git a/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaDetailsContract.kt b/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaDetailsContract.kt index c368b96ac..d6d774208 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaDetailsContract.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaDetailsContract.kt @@ -54,7 +54,7 @@ interface UploadMediaDetailsContract { fun showBadImagePopup(errorCode: Int, index: Int, uploadItem: UploadItem) } - interface UserActionListener : BasePresenter { + interface UserActionListener : BasePresenter { fun setupBasicKvStoreFactory(factory: (String) -> BasicKvStore) fun receiveImage( diff --git a/app/src/main/java/fr/free/nrw/commons/utils/ClipboardUtils.kt b/app/src/main/java/fr/free/nrw/commons/utils/ClipboardUtils.kt new file mode 100644 index 000000000..64d3636f0 --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/ClipboardUtils.kt @@ -0,0 +1,20 @@ +package fr.free.nrw.commons.utils + +import android.content.ClipData +import android.content.ClipboardManager +import android.content.Context +import android.content.Context.CLIPBOARD_SERVICE + +object ClipboardUtils { + // Convenience for Java usages - remove when they are converted. + @JvmStatic + fun copy(label: String?, text: String?, context: Context) { + context.copyToClipboard(label, text) + } +} + +fun Context.copyToClipboard(label: String?, text: String?) { + with(getSystemService(CLIPBOARD_SERVICE) as ClipboardManager) { + setPrimaryClip(ClipData.newPlainText(label, text)) + } +} \ No newline at end of file diff --git a/app/src/main/java/fr/free/nrw/commons/utils/FixExtension.kt b/app/src/main/java/fr/free/nrw/commons/utils/FixExtension.kt new file mode 100644 index 000000000..b9e3988a3 --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/FixExtension.kt @@ -0,0 +1,38 @@ +package fr.free.nrw.commons.utils + +import java.util.Locale +import java.util.regex.Pattern + +private val jpegPattern = Pattern.compile("\\.jpeg$", Pattern.CASE_INSENSITIVE) + +/** + * Adds extension to filename. Converts to .jpg if system provides .jpeg, adds .jpg if no extension detected + * @param theTitle File name + * @param ext Correct extension + * @return File with correct extension + */ +fun fixExtension(theTitle: String, ext: String?): String { + var result = theTitle + var extension = ext + + // People are used to ".jpg" more than ".jpeg" which the system gives us. + if (extension != null && extension.lowercase() == "jpeg") { + extension = "jpg" + } + + result = jpegPattern.matcher(result).replaceFirst(".jpg") + if (extension != null && + !result.lowercase(Locale.getDefault()).endsWith("." + extension.lowercase()) + ) { + result += ".$extension" + } + + // If extension is still null, make it jpg. (Hotfix for https://github.com/commons-app/apps-android-commons/issues/228) + // If title has an extension in it, if won't be true + if (extension == null && result.lastIndexOf(".") <= 0) { + extension = "jpg" + result += ".$extension" + } + + return result +} diff --git a/app/src/main/java/fr/free/nrw/commons/utils/GeoCoordinates.kt b/app/src/main/java/fr/free/nrw/commons/utils/GeoCoordinates.kt new file mode 100644 index 000000000..513e36f10 --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/GeoCoordinates.kt @@ -0,0 +1,27 @@ +package fr.free.nrw.commons.utils + +import android.content.Context +import android.content.Intent +import fr.free.nrw.commons.R +import fr.free.nrw.commons.location.LatLng +import fr.free.nrw.commons.utils.ViewUtil.showShortToast + +/** + * Util function to handle geo coordinates with specified zoom level. It no longer depends on + * google maps and any app capable of handling the map intent can handle it + * + * @param context The context for launching intent + * @param latLng The latitude and longitude of the location + * @param zoomLevel The zoom level + */ +fun handleGeoCoordinates( + context: Context, latLng: LatLng, + zoomLevel: Double = 16.0 +) { + val mapIntent = Intent(Intent.ACTION_VIEW, latLng.getGmmIntentUri(zoomLevel)) + if (mapIntent.resolveActivity(context.packageManager) != null) { + context.startActivity(mapIntent) + } else { + showShortToast(context, context.getString(R.string.map_application_missing)) + } +} diff --git a/app/src/main/java/fr/free/nrw/commons/utils/Licenses.kt b/app/src/main/java/fr/free/nrw/commons/utils/Licenses.kt new file mode 100644 index 000000000..065a14718 --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/Licenses.kt @@ -0,0 +1,31 @@ +package fr.free.nrw.commons.utils + +import fr.free.nrw.commons.R +import fr.free.nrw.commons.settings.Prefs + +/** + * Generates licence name with given ID + * @return Name of license + */ +fun String.toLicenseName(): Int = when (this) { + Prefs.Licenses.CC_BY_3 -> R.string.license_name_cc_by + Prefs.Licenses.CC_BY_4 -> R.string.license_name_cc_by_four + Prefs.Licenses.CC_BY_SA_3 -> R.string.license_name_cc_by_sa + Prefs.Licenses.CC_BY_SA_4 -> R.string.license_name_cc_by_sa_four + Prefs.Licenses.CC0 -> R.string.license_name_cc0 + else -> throw IllegalStateException("Unrecognized license value: $this") +} + +/** + * Generates license url with given ID + * @return Url of license + */ +fun String.toLicenseUrl(): String = when (this) { + Prefs.Licenses.CC_BY_3 -> "https://creativecommons.org/licenses/by/3.0/" + Prefs.Licenses.CC_BY_4 -> "https://creativecommons.org/licenses/by/4.0/" + Prefs.Licenses.CC_BY_SA_3 -> "https://creativecommons.org/licenses/by-sa/3.0/" + Prefs.Licenses.CC_BY_SA_4 -> "https://creativecommons.org/licenses/by-sa/4.0/" + Prefs.Licenses.CC0 -> "https://creativecommons.org/publicdomain/zero/1.0/" + else -> throw IllegalStateException("Unrecognized license value: $this") +} + diff --git a/app/src/main/java/fr/free/nrw/commons/utils/Monuments.kt b/app/src/main/java/fr/free/nrw/commons/utils/Monuments.kt new file mode 100644 index 000000000..d5f5736f5 --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/Monuments.kt @@ -0,0 +1,39 @@ +package fr.free.nrw.commons.utils + +import java.util.Calendar +import java.util.Date + +/** + * Get the start date of wlm monument + * For this release we are hardcoding it to be 1st September + * @return + */ +const val wLMStartDate: String = "1 Sep" + +/*** + * Get the end date of wlm monument + * For this release we are hardcoding it to be 31st October + * @return + */ +const val wLMEndDate: String = "30 Sep" + +/** + * For now we are enabling the monuments only when the date lies between 1 Sept & 31 OCt + */ +val isMonumentsEnabled: Boolean + get() = Date().month == 8 + +/*** + * Function to get the current WLM year + * It increments at the start of September in line with the other WLM functions + * (No consideration of locales for now) + * @param calendar + * @return + */ +fun getWikiLovesMonumentsYear(calendar: Calendar): Int { + var year = calendar[Calendar.YEAR] + if (calendar[Calendar.MONTH] < Calendar.SEPTEMBER) { + year -= 1 + } + return year +} diff --git a/app/src/main/java/fr/free/nrw/commons/utils/UnderlineUtils.kt b/app/src/main/java/fr/free/nrw/commons/utils/UnderlineUtils.kt new file mode 100644 index 000000000..75760d4ab --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/UnderlineUtils.kt @@ -0,0 +1,19 @@ +package fr.free.nrw.commons.utils + +import android.widget.TextView +import androidx.core.text.buildSpannedString +import androidx.core.text.underline + +object UnderlineUtils { + // Convenience method for Java usages - remove when those classes are converted + @JvmStatic + fun setUnderlinedText(textView: TextView, stringResourceName: Int) { + textView.setUnderlinedText(stringResourceName) + } +} + +fun TextView.setUnderlinedText(stringResourceName: Int) { + text = buildSpannedString { + underline { append(context.getString(stringResourceName)) } + } +} diff --git a/app/src/main/java/fr/free/nrw/commons/utils/UrlUtils.kt b/app/src/main/java/fr/free/nrw/commons/utils/UrlUtils.kt new file mode 100644 index 000000000..4843cf0aa --- /dev/null +++ b/app/src/main/java/fr/free/nrw/commons/utils/UrlUtils.kt @@ -0,0 +1,33 @@ +package fr.free.nrw.commons.utils + +import android.content.Context +import android.content.Intent +import android.net.Uri +import androidx.browser.customtabs.CustomTabColorSchemeParams +import androidx.browser.customtabs.CustomTabsIntent +import androidx.core.content.ContextCompat +import fr.free.nrw.commons.R +import timber.log.Timber + +/** + * Opens Custom Tab Activity with in-app browser for the specified URL. + * Launches intent for web URL + */ +fun handleWebUrl(context: Context, url: Uri) { + Timber.d("Launching web url %s", url.toString()) + + val color = CustomTabColorSchemeParams.Builder() + .setToolbarColor(ContextCompat.getColor(context, R.color.primaryColor)) + .setSecondaryToolbarColor(ContextCompat.getColor(context, R.color.primaryDarkColor)) + .build() + + val customTabsIntent = CustomTabsIntent.Builder() + .setDefaultColorSchemeParams(color) + .setExitAnimations( + context, android.R.anim.slide_in_left, android.R.anim.slide_out_right + ).build() + + // Clear previous browser tasks, so that back/exit buttons work as intended. + customTabsIntent.intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK) + customTabsIntent.launchUrl(context, url) +} diff --git a/app/src/test/kotlin/fr/free/nrw/commons/UtilsTest.kt b/app/src/test/kotlin/fr/free/nrw/commons/UtilsTest.kt index e9e68a3ad..d9151335a 100644 --- a/app/src/test/kotlin/fr/free/nrw/commons/UtilsTest.kt +++ b/app/src/test/kotlin/fr/free/nrw/commons/UtilsTest.kt @@ -1,5 +1,6 @@ package fr.free.nrw.commons +import fr.free.nrw.commons.utils.getWikiLovesMonumentsYear import org.junit.Test import org.junit.jupiter.api.Assertions import java.util.Calendar @@ -9,20 +10,20 @@ class UtilsTest { fun wikiLovesMonumentsYearBeforeSeptember() { val cal = Calendar.getInstance() cal.set(2022, Calendar.FEBRUARY, 1) - Assertions.assertEquals(2021, Utils.getWikiLovesMonumentsYear(cal)) + Assertions.assertEquals(2021, getWikiLovesMonumentsYear(cal)) } @Test fun wikiLovesMonumentsYearInSeptember() { val cal = Calendar.getInstance() cal.set(2022, Calendar.SEPTEMBER, 1) - Assertions.assertEquals(2022, Utils.getWikiLovesMonumentsYear(cal)) + Assertions.assertEquals(2022, getWikiLovesMonumentsYear(cal)) } @Test fun wikiLovesMonumentsYearAfterSeptember() { val cal = Calendar.getInstance() cal.set(2022, Calendar.DECEMBER, 1) - Assertions.assertEquals(2022, Utils.getWikiLovesMonumentsYear(cal)) + Assertions.assertEquals(2022, getWikiLovesMonumentsYear(cal)) } } diff --git a/app/src/test/kotlin/fr/free/nrw/commons/contributions/ContributionsFragmentUnitTests.kt b/app/src/test/kotlin/fr/free/nrw/commons/contributions/ContributionsFragmentUnitTests.kt index 4fd5689da..e3f1c86cc 100644 --- a/app/src/test/kotlin/fr/free/nrw/commons/contributions/ContributionsFragmentUnitTests.kt +++ b/app/src/test/kotlin/fr/free/nrw/commons/contributions/ContributionsFragmentUnitTests.kt @@ -248,13 +248,6 @@ class ContributionsFragmentUnitTests { fragment.onDestroyView() } - @Test - @Throws(Exception::class) - fun testShowMessage() { - Shadows.shadowOf(Looper.getMainLooper()).idle() - fragment.showMessage("") - } - @Test @Throws(Exception::class) fun testShowCampaigns() { diff --git a/app/src/test/kotlin/fr/free/nrw/commons/upload/MediaLicensePresenterTest.kt b/app/src/test/kotlin/fr/free/nrw/commons/upload/MediaLicensePresenterTest.kt index 68b1c95cb..b448df6d2 100644 --- a/app/src/test/kotlin/fr/free/nrw/commons/upload/MediaLicensePresenterTest.kt +++ b/app/src/test/kotlin/fr/free/nrw/commons/upload/MediaLicensePresenterTest.kt @@ -1,11 +1,12 @@ package fr.free.nrw.commons.upload import com.nhaarman.mockitokotlin2.verify -import fr.free.nrw.commons.Utils +import fr.free.nrw.commons.utils.UnderlineUtils import fr.free.nrw.commons.kvstore.JsonKvStore import fr.free.nrw.commons.repository.UploadRepository import fr.free.nrw.commons.upload.license.MediaLicenseContract import fr.free.nrw.commons.upload.license.MediaLicensePresenter +import fr.free.nrw.commons.utils.toLicenseName import org.junit.After import org.junit.Before import org.junit.Test @@ -25,7 +26,7 @@ import org.robolectric.RobolectricTestRunner */ @RunWith(RobolectricTestRunner::class) -@PrepareForTest(Utils::class) +@PrepareForTest(UnderlineUtils::class) class MediaLicensePresenterTest { @Mock internal lateinit var repository: UploadRepository @@ -39,7 +40,7 @@ class MediaLicensePresenterTest { @InjectMocks lateinit var mediaLicensePresenter: MediaLicensePresenter - private lateinit var mockedUtil: MockedStatic + private lateinit var mockedUtil: MockedStatic /** * initial setup test environemnt @@ -49,8 +50,7 @@ class MediaLicensePresenterTest { fun setUp() { MockitoAnnotations.openMocks(this) mediaLicensePresenter.onAttachView(view) - mockedUtil = Mockito.mockStatic(Utils::class.java) - `when`(Utils.licenseNameFor(ArgumentMatchers.anyString())).thenReturn(1) + mockedUtil = Mockito.mockStatic(UnderlineUtils::class.java) } @After diff --git a/app/src/test/kotlin/fr/free/nrw/commons/utils/UtilsFixExtensionTest.kt b/app/src/test/kotlin/fr/free/nrw/commons/utils/UtilsFixExtensionTest.kt index 24e9b233a..f4e6a771b 100644 --- a/app/src/test/kotlin/fr/free/nrw/commons/utils/UtilsFixExtensionTest.kt +++ b/app/src/test/kotlin/fr/free/nrw/commons/utils/UtilsFixExtensionTest.kt @@ -1,6 +1,5 @@ package fr.free.nrw.commons.utils -import fr.free.nrw.commons.Utils.fixExtension import org.junit.Assert.assertEquals import org.junit.Test From f98b49608e00331e9f659e104393b7cba35af525 Mon Sep 17 00:00:00 2001 From: Sonal Yadav Date: Sat, 5 Jul 2025 09:45:57 +0530 Subject: [PATCH 012/129] fix popup from appearing in nearby (#6359) Co-authored-by: Nicolas Raoul --- .../nrw/commons/upload/mediaDetails/UploadMediaPresenter.kt | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaPresenter.kt b/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaPresenter.kt index 77999cf2f..4d565adb2 100644 --- a/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaPresenter.kt +++ b/app/src/main/java/fr/free/nrw/commons/upload/mediaDetails/UploadMediaPresenter.kt @@ -107,7 +107,10 @@ class UploadMediaPresenter @Inject constructor( view.showProgress(false) val gpsCoords = uploadItem.gpsCoords val hasImageCoordinates = gpsCoords != null && gpsCoords.imageCoordsExists - if (hasImageCoordinates && place == null) { + + // Only check for nearby places if image has coordinates AND no place was pre-selected + // This prevents the popup from appearing when uploading from Nearby feature + if (hasImageCoordinates && place == null && uploadItem.place == null) { checkNearbyPlaces(uploadItem) } }, { throwable: Throwable? -> From 65f41beed8247a8e2ba6986dc990fb238df820a3 Mon Sep 17 00:00:00 2001 From: VoidRaven Date: Sat, 5 Jul 2025 11:35:54 +0530 Subject: [PATCH 013/129] Update bug-report.yml to use Type:bug label as per issue #6356 (#6363) * Update bug-report.yml to use Type:bug label as per issue #6356 * Update bug-report.yml to use type: bug and labels: ['Type: bug'] as per issue #6356 * Update bug-report.yml to use type: Bug and revert labels to ['bug'] as per mentor's feedback for issue #6356 * Remove labels field from bug-report.yml as per feedback for issue #6356 --------- Co-authored-by: Ritika Pahwa <83745993+RitikaPahwa4444@users.noreply.github.com> --- .github/ISSUE_TEMPLATE/bug-report.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/ISSUE_TEMPLATE/bug-report.yml b/.github/ISSUE_TEMPLATE/bug-report.yml index f92f51a43..a4682fd3c 100644 --- a/.github/ISSUE_TEMPLATE/bug-report.yml +++ b/.github/ISSUE_TEMPLATE/bug-report.yml @@ -1,7 +1,7 @@ name: "\U0001F41E Bug report" description: Create a report to help us improve. title: "[Bug]: " -labels: ["bug"] +type: Bug # Retained to categorize the issue as per organization-level type body: - type: markdown attributes: From 66395b98719e76d3c6f6546678a361486cc27a97 Mon Sep 17 00:00:00 2001 From: Paul Hawke Date: Sun, 6 Jul 2025 19:50:16 -0500 Subject: [PATCH 014/129] convert top level classes to kotlin (#6368) * Converted welcome activity / pager to kotlin * Removed unused interface * Convert ViewPagerAdapter to kotlin and enforce that all tabs must have a title that comes from strings.xml * Convert OkHttpConnectionFactory and remove an exception class nobody was using * Convert MapController to kotlin along with fixing nullability in a few places --- .idea/codeStyles/Project.xml | 1 + .../fr/free/nrw/commons/MapController.java | 30 ---- .../java/fr/free/nrw/commons/MapController.kt | 46 ++++++ .../nrw/commons/OkHttpConnectionFactory.java | 154 ------------------ .../nrw/commons/OkHttpConnectionFactory.kt | 122 ++++++++++++++ .../java/fr/free/nrw/commons/ViewHolder.java | 7 - .../fr/free/nrw/commons/ViewPagerAdapter.java | 68 -------- .../fr/free/nrw/commons/ViewPagerAdapter.kt | 44 +++++ .../fr/free/nrw/commons/WelcomeActivity.java | 109 ------------- .../fr/free/nrw/commons/WelcomeActivity.kt | 78 +++++++++ .../free/nrw/commons/WelcomePagerAdapter.java | 77 --------- .../free/nrw/commons/WelcomePagerAdapter.kt | 70 ++++++++ .../category/CategoryDetailsActivity.kt | 17 +- .../nrw/commons/contributions/MainActivity.kt | 4 +- .../nrw/commons/explore/ExploreFragment.java | 22 +-- .../nrw/commons/explore/SearchActivity.java | 20 +-- .../WikidataItemDetailsActivity.java | 20 +-- .../commons/explore/map/ExploreMapCalls.java | 3 + .../explore/map/ExploreMapPresenter.java | 9 +- .../free/nrw/commons/media/MediaInterface.kt | 2 +- .../commons/navtab/MoreBottomSheetFragment.kt | 4 +- .../free/nrw/commons/nearby/NearbyPlaces.java | 26 ++- .../nrw/commons/profile/ProfileActivity.kt | 35 ++-- .../commons/upload/UploadProgressActivity.kt | 13 +- app/src/main/res/values/strings.xml | 2 + .../free/nrw/commons/MockWebServerTest.java | 110 ------------- .../fr/free/nrw/commons/MockWebServerTest.kt | 110 +++++++++++++ .../free/nrw/commons/TestConnectionFactory.kt | 3 +- .../commons/auth/csrf/CsrfTokenClientTest.kt | 4 +- 29 files changed, 555 insertions(+), 655 deletions(-) delete mode 100644 app/src/main/java/fr/free/nrw/commons/MapController.java create mode 100644 app/src/main/java/fr/free/nrw/commons/MapController.kt delete mode 100644 app/src/main/java/fr/free/nrw/commons/OkHttpConnectionFactory.java create mode 100644 app/src/main/java/fr/free/nrw/commons/OkHttpConnectionFactory.kt delete mode 100644 app/src/main/java/fr/free/nrw/commons/ViewHolder.java delete mode 100644 app/src/main/java/fr/free/nrw/commons/ViewPagerAdapter.java create mode 100644 app/src/main/java/fr/free/nrw/commons/ViewPagerAdapter.kt delete mode 100644 app/src/main/java/fr/free/nrw/commons/WelcomeActivity.java create mode 100644 app/src/main/java/fr/free/nrw/commons/WelcomeActivity.kt delete mode 100644 app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.java create mode 100644 app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.kt delete mode 100644 app/src/test/kotlin/fr/free/nrw/commons/MockWebServerTest.java create mode 100644 app/src/test/kotlin/fr/free/nrw/commons/MockWebServerTest.kt diff --git a/.idea/codeStyles/Project.xml b/.idea/codeStyles/Project.xml index 5c297a65e..ea0cb3b07 100644 --- a/.idea/codeStyles/Project.xml +++ b/.idea/codeStyles/Project.xml @@ -16,6 +16,7 @@