diff --git a/.github/ISSUE_TEMPLATE/quest-suggestion.md b/.github/ISSUE_TEMPLATE/quest-suggestion.md index 0aad34abc6..47cc6d9e87 100644 --- a/.github/ISSUE_TEMPLATE/quest-suggestion.md +++ b/.github/ISSUE_TEMPLATE/quest-suggestion.md @@ -31,7 +31,7 @@ If you are not sure about how one condition applies to your suggestion or you ha ### Ideas for implementation - + **Element selection:** diff --git a/ARCHITECTURE.md b/ARCHITECTURE.md deleted file mode 100644 index 42bf671179..0000000000 --- a/ARCHITECTURE.md +++ /dev/null @@ -1,17 +0,0 @@ -# Quest data flow - -StreetComplete queries external services to find quest candidates. - -Every quest type has defined some properties, including [Overpass](https://wiki.openstreetmap.org/wiki/Overpass_API) queries. The Overpass instance is queried and responds with OpenStreetMap data matching rules for quest candidates. - -Query responses are used to build a local quest database, displayed to the user as markers on the map. Once a user solves a quest, the solution is stored in the local database as a diff. Changes are [uploaded to OSM](https://wiki.openstreetmap.org/wiki/API_v0.6) as soon as possible. Changes are made in the OSM database using credentials provided by the user. Edits are grouped into changesets by quest types. - -The definition of quests in the program may be simple, with just few parameters and made with reusable blocks, like [quest asking whatever toilet is paid](https://github.com/westnordost/StreetComplete/blob/master/app/src/main/java/de/westnordost/streetcomplete/quests/toilets_fee/AddToiletsFee.kt). Some definitions are highly complicated, defining special interfaces, using [country specific data](https://github.com/westnordost/StreetComplete/tree/master/res/country_metadata) or involve special processing of data. [The quest asking about house number](https://github.com/westnordost/StreetComplete/tree/master/app/src/main/java/de/westnordost/streetcomplete/quests/housenumber) is a good example of a quest handling quite complex situations. For starters, not the entire world has the same numbering system – some countries have block-based addressing or addresses with [more than one](https://wiki.openstreetmap.org/wiki/Key:addr:conscriptionnumber) assigned house number. - -Some quests may be based on other data sources. The [note quest](https://github.com/westnordost/StreetComplete/tree/master/app/src/main/java/de/westnordost/streetcomplete/quests/note_discussion) is based on data directly downloaded from the [OSM API](https://wiki.openstreetmap.org/wiki/API_v0.6#Map_Notes_API). The [oneway quest](https://github.com/westnordost/StreetComplete/tree/master/app/src/main/java/de/westnordost/streetcomplete/quests/oneway) is using [an external list of roads likely to be a oneway](https://github.com/ENT8R/oneway-data-api). - -The note quest is also special because part of the answer – photos made by users – is uploaded to a [special photo service](https://github.com/exploide/sc-photo-service), as OSM notes do not allow hosting of images directly on OSM servers. - -# Map - -SC downloads the [vector tiles](https://github.com/tilezen/vector-datasource) used for displaying its map from an external source and renders them using the library [tangram-es](https://github.com/tangrams/tangram-es). diff --git a/app/build.gradle.kts b/app/build.gradle.kts index d70bc6265f..c4e5268eea 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -86,12 +86,6 @@ configurations { all { exclude(group = "net.sf.kxml", module = "kxml2") } - - compile.configure { - exclude(group = "org.jetbrains", module = "annotations") - exclude(group = "com.intellij", module = "annotations") - exclude(group = "org.intellij", module = "annotations") - } } dependencies { @@ -146,11 +140,10 @@ dependencies { // finding a name for a feature without a name tag implementation("de.westnordost:osmfeatures-android:1.1") // talking with the OSM API - implementation("de.westnordost:osmapi-overpass:1.1") - implementation("de.westnordost:osmapi-map:1.2") - implementation("de.westnordost:osmapi-changesets:1.2") - implementation("de.westnordost:osmapi-notes:1.1") - implementation("de.westnordost:osmapi-user:1.1") + implementation("de.westnordost:osmapi-map:1.3") + implementation("de.westnordost:osmapi-changesets:1.3") + implementation("de.westnordost:osmapi-notes:1.2") + implementation("de.westnordost:osmapi-user:1.2") // widgets implementation("androidx.viewpager2:viewpager2:1.0.0") diff --git a/app/src/androidTest/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDaoTest.kt b/app/src/androidTest/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDaoTest.kt index a366ce1f1c..3b62199961 100644 --- a/app/src/androidTest/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDaoTest.kt +++ b/app/src/androidTest/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDaoTest.kt @@ -4,6 +4,7 @@ import org.junit.Before import org.junit.Test import de.westnordost.streetcomplete.data.ApplicationDbTestCase +import de.westnordost.streetcomplete.ktx.containsExactlyInAnyOrder import de.westnordost.streetcomplete.util.Tile import de.westnordost.streetcomplete.util.TilesRect @@ -76,6 +77,6 @@ class DownloadedTilesDaoTest : ApplicationDbTestCase() { check = dao.get(r(0, 0, 6, 6), 0) assertTrue(check.isEmpty()) } - + private fun r(left: Int, top: Int, right: Int, bottom: Int) = TilesRect(left, top, right, bottom) } diff --git a/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDaoTest.kt b/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDaoTest.kt index 50539b26a2..88f92c73c2 100644 --- a/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDaoTest.kt +++ b/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDaoTest.kt @@ -204,13 +204,13 @@ private val TEST_QUEST_TYPE = TestQuestType() private val TEST_QUEST_TYPE2 = TestQuestType2() private fun create( - questType: OsmElementQuestType<*> = TEST_QUEST_TYPE, - elementType: Element.Type = Element.Type.NODE, - elementId: Long = 1, - status: QuestStatus = QuestStatus.NEW, - geometry: ElementGeometry = ElementPointGeometry(OsmLatLon(5.0, 5.0)), - changes: StringMapChanges? = null, - changesSource: String? = null + questType: OsmElementQuestType<*> = TEST_QUEST_TYPE, + elementType: Element.Type = Element.Type.NODE, + elementId: Long = 1, + status: QuestStatus = QuestStatus.NEW, + geometry: ElementGeometry = ElementPointGeometry(OsmLatLon(5.0, 5.0)), + changes: StringMapChanges? = null, + changesSource: String? = null ) = OsmQuest( null, questType, elementType, elementId, status, changes, changesSource, Date(), geometry ) diff --git a/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/TestQuestType.kt b/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/TestQuestType.kt index 80010c8548..b58a88b79d 100644 --- a/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/TestQuestType.kt +++ b/app/src/androidTest/java/de/westnordost/streetcomplete/data/osm/osmquest/TestQuestType.kt @@ -1,18 +1,17 @@ package de.westnordost.streetcomplete.data.osm.osmquest -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.quests.AbstractQuestAnswerFragment open class TestQuestType : OsmElementQuestType { override fun getTitle(tags: Map) = 0 - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit) = false override fun isApplicableTo(element: Element):Boolean? = null override fun applyAnswerTo(answer: String, changes: StringMapChangesBuilder) {} override val icon = 0 override fun createForm(): AbstractQuestAnswerFragment = object : AbstractQuestAnswerFragment() {} override val commitMessage = "" + override fun getApplicableElements(mapData: MapDataWithGeometry) = emptyList() } diff --git a/app/src/main/java/de/westnordost/osmapi/map/LightweightOsmMapDataFactory.kt b/app/src/main/java/de/westnordost/osmapi/map/LightweightOsmMapDataFactory.kt new file mode 100644 index 0000000000..7ee72cb8ae --- /dev/null +++ b/app/src/main/java/de/westnordost/osmapi/map/LightweightOsmMapDataFactory.kt @@ -0,0 +1,28 @@ +package de.westnordost.osmapi.map + +import de.westnordost.osmapi.changesets.Changeset +import de.westnordost.osmapi.map.data.* +import java.util.* + +/** Same as OsmMapDataFactory only that it throws away the Changeset data included in the OSM + * response */ +class LightweightOsmMapDataFactory : MapDataFactory { + override fun createNode( + id: Long, version: Int, lat: Double, lon: Double, tags: MutableMap?, + changeset: Changeset?, dateEdited: Date? + ): Node = OsmNode(id, version, lat, lon, tags, null, dateEdited) + + override fun createWay( + id: Long, version: Int, nodes: MutableList, tags: MutableMap?, + changeset: Changeset?, dateEdited: Date? + ): Way = OsmWay(id, version, nodes, tags, null, dateEdited) + + override fun createRelation( + id: Long, version: Int, members: MutableList, + tags: MutableMap?, changeset: Changeset?, dateEdited: Date? + ): Relation = OsmRelation(id, version, members, tags, null, dateEdited) + + override fun createRelationMember( + ref: Long, role: String?, type: Element.Type + ): RelationMember = OsmRelationMember(ref, role, type) +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/osmapi/map/MapData.kt b/app/src/main/java/de/westnordost/osmapi/map/MapData.kt new file mode 100644 index 0000000000..5e8718a822 --- /dev/null +++ b/app/src/main/java/de/westnordost/osmapi/map/MapData.kt @@ -0,0 +1,85 @@ +package de.westnordost.osmapi.map + +import de.westnordost.osmapi.map.data.* +import de.westnordost.osmapi.map.handler.MapDataHandler +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry +import de.westnordost.streetcomplete.util.MultiIterable + +interface MapDataWithGeometry : MapData { + fun getNodeGeometry(id: Long): ElementPointGeometry? + fun getWayGeometry(id: Long): ElementGeometry? + fun getRelationGeometry(id: Long): ElementGeometry? + + fun getGeometry(elementType: Element.Type, id: Long): ElementGeometry? = when(elementType) { + Element.Type.NODE -> getNodeGeometry(id) + Element.Type.WAY -> getWayGeometry(id) + Element.Type.RELATION -> getRelationGeometry(id) + } +} + +interface MapData : Iterable { + val nodes: Collection + val ways: Collection + val relations: Collection + val boundingBox: BoundingBox? + + fun getNode(id: Long): Node? + fun getWay(id: Long): Way? + fun getRelation(id: Long): Relation? +} + +open class MutableMapData : MapData, MapDataHandler { + + protected val nodesById: MutableMap = mutableMapOf() + protected val waysById: MutableMap = mutableMapOf() + protected val relationsById: MutableMap = mutableMapOf() + override var boundingBox: BoundingBox? = null + protected set + + override fun handle(bounds: BoundingBox) { boundingBox = bounds } + override fun handle(node: Node) { nodesById[node.id] = node } + override fun handle(way: Way) { waysById[way.id] = way } + override fun handle(relation: Relation) { relationsById[relation.id] = relation } + + override val nodes get() = nodesById.values + override val ways get() = waysById.values + override val relations get() = relationsById.values + + override fun getNode(id: Long) = nodesById[id] + override fun getWay(id: Long) = waysById[id] + override fun getRelation(id: Long) = relationsById[id] + + fun addAll(elements: Iterable) { + for (element in elements) { + when(element) { + is Node -> nodesById[element.id] = element + is Way -> waysById[element.id] = element + is Relation -> relationsById[element.id] = element + } + } + } + + override fun iterator(): Iterator { + val elements = MultiIterable() + elements.add(nodes) + elements.add(ways) + elements.add(relations) + return elements.iterator() + } +} + +fun MapData.isRelationComplete(id: Long): Boolean = + getRelation(id)?.members?.all { member -> + when (member.type!!) { + Element.Type.NODE -> getNode(member.ref) != null + Element.Type.WAY -> getWay(member.ref) != null && isWayComplete(member.ref) + /* not being recursive here is deliberate. sub-relations are considered not relevant + for the element geometry in StreetComplete (and OSM API call to get a "complete" + relation also does not include sub-relations) */ + Element.Type.RELATION -> getRelation(member.ref) != null + } + } ?: false + +fun MapData.isWayComplete(id: Long): Boolean = + getWay(id)?.nodeIds?.all { getNode(it) != null } ?: false \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/osmapi/map/MapDataApi.kt b/app/src/main/java/de/westnordost/osmapi/map/MapDataApi.kt new file mode 100644 index 0000000000..4596e00808 --- /dev/null +++ b/app/src/main/java/de/westnordost/osmapi/map/MapDataApi.kt @@ -0,0 +1,22 @@ +package de.westnordost.osmapi.map + +import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.streetcomplete.data.MapDataApi + +fun MapDataApi.getMap(bounds: BoundingBox): MapData { + val result = MutableMapData() + getMap(bounds, result) + return result +} + +fun MapDataApi.getWayComplete(id: Long): MapData { + val result = MutableMapData() + getWayComplete(id, result) + return result +} + +fun MapDataApi.getRelationComplete(id: Long): MapData { + val result = MutableMapData() + getRelationComplete(id, result) + return result +} diff --git a/app/src/main/java/de/westnordost/streetcomplete/ApplicationComponent.kt b/app/src/main/java/de/westnordost/streetcomplete/ApplicationComponent.kt index ca2a49b1e8..e7d2bf1a50 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/ApplicationComponent.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/ApplicationComponent.kt @@ -90,7 +90,7 @@ interface ApplicationComponent { fun inject(undoButtonFragment: UndoButtonFragment) fun inject(uploadButtonFragment: UploadButtonFragment) fun inject(answersCounterFragment: AnswersCounterFragment) - fun inject(questDownloadProgressFragment: QuestDownloadProgressFragment) + fun inject(downloadProgressFragment: DownloadProgressFragment) fun inject(questStatisticsByCountryFragment: QuestStatisticsByCountryFragment) fun inject(questStatisticsByQuestTypeFragment: QuestStatisticsByQuestTypeFragment) fun inject(privacyStatementFragment: PrivacyStatementFragment) diff --git a/app/src/main/java/de/westnordost/streetcomplete/ApplicationConstants.java b/app/src/main/java/de/westnordost/streetcomplete/ApplicationConstants.java index e61d48213f..e79b5dcef2 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/ApplicationConstants.java +++ b/app/src/main/java/de/westnordost/streetcomplete/ApplicationConstants.java @@ -8,25 +8,21 @@ public class ApplicationConstants QUESTTYPE_TAG_KEY = NAME + ":quest_type"; public final static double - MAX_DOWNLOADABLE_AREA_IN_SQKM = 20, - MIN_DOWNLOADABLE_AREA_IN_SQKM = 1; - - public final static double MIN_DOWNLOADABLE_RADIUS_IN_METERS = 600; + MAX_DOWNLOADABLE_AREA_IN_SQKM = 12.0, + MIN_DOWNLOADABLE_AREA_IN_SQKM = 0.1; public final static String DATABASE_NAME = "streetcomplete.db"; - public final static int QUEST_TILE_ZOOM = 14; + public final static int QUEST_TILE_ZOOM = 16; public final static int NOTE_MIN_ZOOM = 15; - /** How many quests to download when pressing manually on "download quests" */ - public final static int MANUAL_DOWNLOAD_QUEST_TYPE_COUNT = 10; - /** a "best before" duration for quests. Quests will not be downloaded again for any tile * before the time expired */ - public static final long REFRESH_QUESTS_AFTER = 7L*24*60*60*1000; // 1 week in ms - /** the duration after which quests will be deleted from the database if unsolved */ - public static final long DELETE_UNSOLVED_QUESTS_AFTER = 1L*30*24*60*60*1000; // 1 months in ms + public static final long REFRESH_QUESTS_AFTER = 3L*24*60*60*1000; // 3 days in ms + /** the duration after which quests (and quest meta data) will be deleted from the database if + * unsolved and not refreshed in the meantime */ + public static final long DELETE_UNSOLVED_QUESTS_AFTER = 14*24*60*60*1000; // 14 days in ms /** the max age of the undo history - one cannot undo changes older than X */ public static final long MAX_QUEST_UNDO_HISTORY_AGE = 24*60*60*1000; // 1 day in ms diff --git a/app/src/main/java/de/westnordost/streetcomplete/MainActivity.java b/app/src/main/java/de/westnordost/streetcomplete/MainActivity.java index 8ca5393fb0..124cf88438 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/MainActivity.java +++ b/app/src/main/java/de/westnordost/streetcomplete/MainActivity.java @@ -14,6 +14,7 @@ import android.os.Build; import android.os.Bundle; import androidx.annotation.AnyThread; +import androidx.annotation.Nullable; import androidx.core.content.ContextCompat; import androidx.fragment.app.Fragment; import androidx.localbroadcastmanager.content.LocalBroadcastManager; @@ -28,9 +29,6 @@ import android.widget.TextView; import android.widget.Toast; -import org.jetbrains.annotations.NotNull; -import org.jetbrains.annotations.Nullable; - import javax.inject.Inject; import de.westnordost.osmapi.common.errors.OsmApiException; @@ -40,12 +38,13 @@ import de.westnordost.osmapi.map.data.LatLon; import de.westnordost.osmapi.map.data.OsmLatLon; import de.westnordost.streetcomplete.controls.NotificationButtonFragment; +import de.westnordost.streetcomplete.data.download.DownloadItem; import de.westnordost.streetcomplete.data.notifications.Notification; import de.westnordost.streetcomplete.data.notifications.NotificationsSource; import de.westnordost.streetcomplete.data.quest.Quest; import de.westnordost.streetcomplete.data.quest.QuestAutoSyncer; import de.westnordost.streetcomplete.data.quest.QuestController; -import de.westnordost.streetcomplete.data.download.QuestDownloadProgressListener; +import de.westnordost.streetcomplete.data.download.DownloadProgressListener; import de.westnordost.streetcomplete.data.download.QuestDownloadController; import de.westnordost.streetcomplete.data.quest.QuestType; import de.westnordost.streetcomplete.data.quest.UnsyncedChangesCountSource; @@ -193,7 +192,7 @@ private void handleGeoUri() questDownloadController.setShowNotification(false); uploadController.addUploadProgressListener(uploadProgressListener); - questDownloadController.addQuestDownloadProgressListener(downloadProgressListener); + questDownloadController.addDownloadProgressListener(downloadProgressListener); if(!hasAskedForLocation && !prefs.getBoolean(Prefs.LAST_LOCATION_REQUEST_DENIED, false)) { @@ -253,7 +252,7 @@ public boolean dispatchKeyEvent(KeyEvent event) { questDownloadController.setShowNotification(true); uploadController.removeUploadProgressListener(uploadProgressListener); - questDownloadController.removeQuestDownloadProgressListener(downloadProgressListener); + questDownloadController.removeDownloadProgressListener(downloadProgressListener); } @Override public void onConfigurationChanged(@NonNull Configuration newConfig) { @@ -346,14 +345,14 @@ else if(e instanceof OsmAuthorizationException) /* ----------------------------- Download Progress listener -------------------------------- */ - private final QuestDownloadProgressListener downloadProgressListener - = new QuestDownloadProgressListener() + private final DownloadProgressListener downloadProgressListener + = new DownloadProgressListener() { @AnyThread @Override public void onStarted() {} - @Override public void onFinished(@NotNull QuestType questType) {} + @Override public void onFinished(@NonNull DownloadItem item) {} - @Override public void onStarted(@NotNull QuestType questType) {} + @Override public void onStarted(@NonNull DownloadItem item) {} @AnyThread @Override public void onError(@NonNull final Exception e) { @@ -386,7 +385,7 @@ else if(e instanceof OsmAuthorizationException) /* --------------------------------- NotificationButtonFragment.Listener ---------------------------------- */ - @Override public void onClickShowNotification(@NotNull Notification notification) + @Override public void onClickShowNotification(@NonNull Notification notification) { Fragment f = getSupportFragmentManager().findFragmentById(R.id.notifications_container_fragment); ((NotificationsContainerFragment) f).showNotification(notification); @@ -399,7 +398,7 @@ else if(e instanceof OsmAuthorizationException) ensureLoggedIn(); } - @Override public void onCreatedNote(@NotNull Point screenPosition) + @Override public void onCreatedNote(@NonNull Point screenPosition) { ensureLoggedIn(); } diff --git a/app/src/main/java/de/westnordost/streetcomplete/Prefs.java b/app/src/main/java/de/westnordost/streetcomplete/Prefs.java index 0fdd8b7884..2ba0189185 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/Prefs.java +++ b/app/src/main/java/de/westnordost/streetcomplete/Prefs.java @@ -15,7 +15,6 @@ public class Prefs KEEP_SCREEN_ON = "display.keepScreenOn", UNGLUE_HINT_TIMES_SHOWN = "unglueHint.shown", THEME_SELECT = "theme.select", - OVERPASS_URL = "overpass_url", RESURVEY_INTERVALS = "quests.resurveyIntervals"; @@ -38,6 +37,7 @@ public class Prefs LAST_PICKED_PREFIX = "imageListLastPicked.", LAST_LOCATION_REQUEST_DENIED = "location.denied", LAST_VERSION = "lastVersion", + LAST_VERSION_DATA = "lastVersion_data", HAS_SHOWN_TUTORIAL = "hasShownTutorial"; public static final String HAS_SHOWN_UNDO_FUCKUP_WARNING = "alert.undo_fuckup_warning"; diff --git a/app/src/main/java/de/westnordost/streetcomplete/StreetCompleteApplication.java b/app/src/main/java/de/westnordost/streetcomplete/StreetCompleteApplication.java index 61e4e9de98..3b3549443f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/StreetCompleteApplication.java +++ b/app/src/main/java/de/westnordost/streetcomplete/StreetCompleteApplication.java @@ -11,6 +11,8 @@ import androidx.appcompat.app.AppCompatDelegate; import de.westnordost.countryboundaries.CountryBoundaries; import de.westnordost.osmfeatures.FeatureDictionary; +import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesDao; +import de.westnordost.streetcomplete.settings.ResurveyIntervalsUpdater; import de.westnordost.streetcomplete.util.CrashReportExceptionHandler; public class StreetCompleteApplication extends Application @@ -18,6 +20,8 @@ public class StreetCompleteApplication extends Application @Inject FutureTask countryBoundariesFuture; @Inject FutureTask featuresDictionaryFuture; @Inject CrashReportExceptionHandler crashReportExceptionHandler; + @Inject ResurveyIntervalsUpdater resurveyIntervalsUpdater; + @Inject DownloadedTilesDao downloadedTilesDao; @Inject SharedPreferences prefs; private static final String PRELOAD_TAG = "Preload"; @@ -36,6 +40,18 @@ public void onCreate() Prefs.Theme theme = Prefs.Theme.valueOf(prefs.getString(Prefs.THEME_SELECT, "AUTO")); AppCompatDelegate.setDefaultNightMode(theme.appCompatNightMode); + + resurveyIntervalsUpdater.update(); + + String lastVersion = prefs.getString(Prefs.LAST_VERSION_DATA, null); + if (!BuildConfig.VERSION_NAME.equals(lastVersion)) + { + prefs.edit().putString(Prefs.LAST_VERSION_DATA, BuildConfig.VERSION_NAME).apply(); + if (lastVersion != null) + { + onNewVersion(); + } + } } /** Load some things in the background that are needed later */ @@ -56,4 +72,9 @@ private void preload() Log.i(PRELOAD_TAG, "Loaded features dictionary"); }).start(); } + + private void onNewVersion() { + // on each new version, invalidate quest cache + downloadedTilesDao.removeAll(); + } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/controls/QuestDownloadProgressFragment.kt b/app/src/main/java/de/westnordost/streetcomplete/controls/DownloadProgressFragment.kt similarity index 77% rename from app/src/main/java/de/westnordost/streetcomplete/controls/QuestDownloadProgressFragment.kt rename to app/src/main/java/de/westnordost/streetcomplete/controls/DownloadProgressFragment.kt index d6feadc413..8e8afe60cf 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/controls/QuestDownloadProgressFragment.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/controls/DownloadProgressFragment.kt @@ -6,9 +6,9 @@ import android.view.View import androidx.fragment.app.Fragment import de.westnordost.streetcomplete.Injector import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.download.QuestDownloadProgressListener -import de.westnordost.streetcomplete.data.download.QuestDownloadProgressSource -import de.westnordost.streetcomplete.data.quest.QuestType +import de.westnordost.streetcomplete.data.download.DownloadItem +import de.westnordost.streetcomplete.data.download.DownloadProgressListener +import de.westnordost.streetcomplete.data.download.DownloadProgressSource import de.westnordost.streetcomplete.ktx.toPx import de.westnordost.streetcomplete.ktx.toast import kotlinx.coroutines.CoroutineScope @@ -17,10 +17,10 @@ import kotlinx.coroutines.cancel import kotlinx.coroutines.launch import javax.inject.Inject -class QuestDownloadProgressFragment : Fragment(R.layout.fragment_quest_download_progress), +class DownloadProgressFragment : Fragment(R.layout.fragment_download_progress), CoroutineScope by CoroutineScope(Dispatchers.Main) { - @Inject internal lateinit var downloadProgressSource: QuestDownloadProgressSource + @Inject internal lateinit var downloadProgressSource: DownloadProgressSource private val mainHandler = Handler(Looper.getMainLooper()) @@ -28,7 +28,7 @@ class QuestDownloadProgressFragment : Fragment(R.layout.fragment_quest_download_ private val animateOutRunnable = Runnable { animateOutProgressView() } - private val downloadProgressListener = object : QuestDownloadProgressListener { + private val downloadProgressListener = object : DownloadProgressListener { private var startedButNoQuestsYet = false override fun onStarted() { @@ -36,12 +36,12 @@ class QuestDownloadProgressFragment : Fragment(R.layout.fragment_quest_download_ launch(Dispatchers.Main) { animateInProgressView() } } - override fun onStarted(questType: QuestType<*>) { + override fun onStarted(item: DownloadItem) { startedButNoQuestsYet = false - launch(Dispatchers.Main) { progressView.enqueueIcon(resources.getDrawable(questType.icon)) } + launch(Dispatchers.Main) { progressView.enqueueIcon(resources.getDrawable(item.iconResId)) } } - override fun onFinished(questType: QuestType<*>) { + override fun onFinished(item: DownloadItem) { launch(Dispatchers.Main) { progressView.pollIcon() } } @@ -63,12 +63,12 @@ class QuestDownloadProgressFragment : Fragment(R.layout.fragment_quest_download_ override fun onStart() { super.onStart() updateDownloadProgress() - downloadProgressSource.addQuestDownloadProgressListener(downloadProgressListener) + downloadProgressSource.addDownloadProgressListener(downloadProgressListener) } override fun onStop() { super.onStop() - downloadProgressSource.removeQuestDownloadProgressListener(downloadProgressListener) + downloadProgressSource.removeDownloadProgressListener(downloadProgressListener) } override fun onDestroy() { @@ -101,9 +101,9 @@ class QuestDownloadProgressFragment : Fragment(R.layout.fragment_quest_download_ private fun updateDownloadProgress() { if (downloadProgressSource.isDownloadInProgress) { showProgressView() - val questType = downloadProgressSource.currentDownloadingQuestType - if (questType != null) { - progressView.setIcon(resources.getDrawable(questType.icon)) + val item = downloadProgressSource.currentDownloadItem + if (item != null) { + progressView.setIcon(resources.getDrawable(item.iconResId)) } } else { hideProgressView() diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/OsmApiModule.kt b/app/src/main/java/de/westnordost/streetcomplete/data/OsmApiModule.kt index 561b3ff027..aed029757c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/OsmApiModule.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/OsmApiModule.kt @@ -1,12 +1,10 @@ package de.westnordost.streetcomplete.data -import android.content.SharedPreferences import dagger.Module import dagger.Provides import de.westnordost.osmapi.OsmConnection -import de.westnordost.osmapi.overpass.OverpassMapDataDao +import de.westnordost.osmapi.map.LightweightOsmMapDataFactory import de.westnordost.streetcomplete.ApplicationConstants -import de.westnordost.streetcomplete.Prefs import de.westnordost.streetcomplete.data.user.OAuthStore import oauth.signpost.OAuthConsumer import javax.inject.Singleton @@ -14,14 +12,7 @@ import javax.inject.Singleton @Module object OsmApiModule { - const val OSM_API_URL = "https://api.openstreetmap.org/api/0.6/" - const val OVERPASS_API_URL = "https://lz4.overpass-api.de/api/" - - // see https://wiki.openstreetmap.org/wiki/Overpass_API/Overpass_QL#timeout: - // default value is 180 seconds - // give additional 4 seconds to get and process refusal from Overpass - // or maybe a bit late response rather than trigger timeout exception - private const val OVERPASS_QUERY_TIMEOUT_IN_MILISECONDS = (180 + 4) * 1000 + private const val OSM_API_URL = "https://api.openstreetmap.org/api/0.6/" /** Returns the osm connection singleton used for all daos with the saved oauth consumer */ @Provides @Singleton fun osmConnection(oAuthStore: OAuthStore): OsmConnection { @@ -33,21 +24,13 @@ object OsmApiModule { return OsmConnection(OSM_API_URL, ApplicationConstants.USER_AGENT, consumer) } - @Provides @Singleton - fun overpassMapDataDao(prefs: SharedPreferences): OverpassMapDataDao { - val timeout = OVERPASS_QUERY_TIMEOUT_IN_MILISECONDS - val overpassConnection = OsmConnection( - prefs.getString(Prefs.OVERPASS_URL, OVERPASS_API_URL), - ApplicationConstants.USER_AGENT, - null, - timeout - ) - return OverpassMapDataDao(overpassConnection) - } - @Provides fun userDao(osm: OsmConnection): UserApi = UserApi(osm) @Provides fun notesDao(osm: OsmConnection): NotesApi = NotesApi(osm) - @Provides fun mapDataDao(osm: OsmConnection): MapDataApi = MapDataApi(osm) + @Provides fun mapDataDao(osm: OsmConnection): MapDataApi { + // generally we are not interested in certain data returned by the OSM API, so we use a + // map data factory that does not include that data + return MapDataApi(osm, LightweightOsmMapDataFactory()) + } } \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/OsmapiTypealiases.kt b/app/src/main/java/de/westnordost/streetcomplete/data/OsmapiTypealiases.kt index 6b301b29d9..6fee6dd706 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/OsmapiTypealiases.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/OsmapiTypealiases.kt @@ -2,7 +2,6 @@ package de.westnordost.streetcomplete.data import de.westnordost.osmapi.map.MapDataDao import de.westnordost.osmapi.notes.NotesDao -import de.westnordost.osmapi.overpass.OverpassMapDataDao import de.westnordost.osmapi.user.PermissionsDao import de.westnordost.osmapi.user.UserDao @@ -10,4 +9,3 @@ typealias NotesApi = NotesDao typealias PermissionsApi = PermissionsDao typealias MapDataApi = MapDataDao typealias UserApi = UserDao -typealias OverpassMapDataApi = OverpassMapDataDao \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/StreetCompleteSQLiteOpenHelper.kt b/app/src/main/java/de/westnordost/streetcomplete/data/StreetCompleteSQLiteOpenHelper.kt index 72deeb973e..f405448b5c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/StreetCompleteSQLiteOpenHelper.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/StreetCompleteSQLiteOpenHelper.kt @@ -29,6 +29,7 @@ import de.westnordost.streetcomplete.ktx.hasColumn import de.westnordost.streetcomplete.quests.road_name.data.RoadNamesTable import de.westnordost.streetcomplete.quests.oneway_suspects.AddSuspectedOneway import de.westnordost.streetcomplete.quests.oneway_suspects.data.WayTrafficFlowTable +import java.util.* @Singleton class StreetCompleteSQLiteOpenHelper(context: Context, dbName: String) : SQLiteOpenHelper(context, dbName, null, DB_VERSION) { @@ -201,9 +202,22 @@ import de.westnordost.streetcomplete.quests.oneway_suspects.data.WayTrafficFlowT DELETE FROM ${RelationTable.NAME} """.trimIndent()) } + + if (oldVersion < 17 && newVersion >= 17) { + db.execSQL(""" + ALTER TABLE ${RoadNamesTable.NAME} + ADD COLUMN ${RoadNamesTable.Columns.LAST_UPDATE} int NOT NULL default ${Date().time}; + """.trimIndent()) + } + + if (oldVersion < 18 && newVersion >= 18) { + // QUEST_TILE_ZOOM changed + db.execSQL("DELETE FROM ${DownloadedTilesTable.NAME}") + } + // for later changes to the DB // ... } } -private const val DB_VERSION = 16 +private const val DB_VERSION = 18 diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/AActiveRadiusStrategy.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/AActiveRadiusStrategy.kt deleted file mode 100644 index 298e77cc30..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/AActiveRadiusStrategy.kt +++ /dev/null @@ -1,73 +0,0 @@ -package de.westnordost.streetcomplete.data.download - -import android.util.Log - -import de.westnordost.streetcomplete.ApplicationConstants -import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesDao -import de.westnordost.streetcomplete.data.visiblequests.OrderedVisibleQuestTypesProvider -import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.osmapi.map.data.LatLon -import de.westnordost.streetcomplete.data.quest.VisibleQuestsSource -import de.westnordost.streetcomplete.util.area -import de.westnordost.streetcomplete.util.enclosingBoundingBox -import de.westnordost.streetcomplete.util.enclosingTilesRect -import kotlin.math.max - -/** Quest auto download strategy that observes that a minimum amount of quests in a predefined - * radius around the user is not undercut */ -abstract class AActiveRadiusStrategy( - private val visibleQuestsSource: VisibleQuestsSource, - private val downloadedTilesDao: DownloadedTilesDao, - private val questTypesProvider: OrderedVisibleQuestTypesProvider -) : QuestAutoDownloadStrategy { - - protected abstract val minQuestsInActiveRadiusPerKm2: Int - protected abstract val activeRadii: IntArray - protected abstract val downloadRadius: Int - - private fun mayDownloadHere(pos: LatLon, radius: Int, questTypeNames: List): Boolean { - val bbox = pos.enclosingBoundingBox(radius.toDouble()) - - // nothing more to download - val tiles = bbox.enclosingTilesRect(ApplicationConstants.QUEST_TILE_ZOOM) - val questExpirationTime = ApplicationConstants.REFRESH_QUESTS_AFTER - val ignoreOlderThan = max(0, System.currentTimeMillis() - questExpirationTime) - val alreadyDownloaded = downloadedTilesDao.get(tiles, ignoreOlderThan).toSet() - val notAlreadyDownloaded = mutableListOf() - for (questTypeName in questTypeNames) { - if (!alreadyDownloaded.contains(questTypeName)) notAlreadyDownloaded.add(questTypeName) - } - - if (notAlreadyDownloaded.isEmpty()) { - Log.i(TAG, "Not downloading quests because everything has been downloaded already in ${radius}m radius") - return false - } - - if (alreadyDownloaded.isNotEmpty()) { - val areaInKm2 = bbox.area() / 1000.0 / 1000.0 - // got enough quests in vicinity - val visibleQuests = visibleQuestsSource.getAllVisibleCount(bbox, alreadyDownloaded) - if (visibleQuests / areaInKm2 > minQuestsInActiveRadiusPerKm2) { - Log.i(TAG, "Not downloading quests because there are enough quests in ${radius}m radius") - return false - } - } - - return true - } - - override fun mayDownloadHere(pos: LatLon): Boolean { - val questTypeNames = questTypesProvider.get().map { it.javaClass.simpleName } - return activeRadii.any { radius -> - mayDownloadHere(pos, radius, questTypeNames) - } - } - - override fun getDownloadBoundingBox(pos: LatLon): BoundingBox { - return pos.enclosingBoundingBox(downloadRadius.toDouble()) - } - - companion object { - private const val TAG = "AutoQuestDownload" - } -} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/AVariableRadiusStrategy.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/AVariableRadiusStrategy.kt new file mode 100644 index 0000000000..6b4c13ac95 --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/AVariableRadiusStrategy.kt @@ -0,0 +1,73 @@ +package de.westnordost.streetcomplete.data.download + +import android.util.Log + +import de.westnordost.streetcomplete.ApplicationConstants +import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesDao +import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.data.LatLon +import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesType +import de.westnordost.streetcomplete.data.quest.VisibleQuestsSource +import de.westnordost.streetcomplete.util.* +import kotlin.math.PI +import kotlin.math.max +import kotlin.math.min +import kotlin.math.sqrt + +/** Quest auto download strategy decides how big of an area to download based on the quest density */ +abstract class AVariableRadiusStrategy( + private val visibleQuestsSource: VisibleQuestsSource, + private val downloadedTilesDao: DownloadedTilesDao +) : QuestAutoDownloadStrategy { + + protected abstract val maxDownloadAreaInKm2: Double + protected abstract val desiredQuestCountInVicinity: Int + + override fun getDownloadBoundingBox(pos: LatLon): BoundingBox? { + val tileZoom = ApplicationConstants.QUEST_TILE_ZOOM + + val thisTile = pos.enclosingTile(tileZoom) + val hasMissingQuestsForThisTile = hasMissingQuestsFor(thisTile.toTilesRect()) + + // if at the location where we are, there is nothing yet, first download the tiniest + // possible bbox (~ 360x360m) so that we can estimate the quest density + if (hasMissingQuestsForThisTile) { + Log.i(TAG, "Downloading tiny area around user") + return thisTile.asBoundingBox(tileZoom) + } + + // otherwise, see if anything is missing in a variable radius, based on quest density + val density = getQuestDensityFor(thisTile.asBoundingBox(tileZoom)) + val maxRadius = sqrt( maxDownloadAreaInKm2 * 1000 * 1000 / PI ) + + var radius = if (density > 0) sqrt( desiredQuestCountInVicinity / ( PI * density )) else maxRadius + + radius = min( radius, maxRadius) + + val activeBoundingBox = pos.enclosingBoundingBox(radius) + if (hasMissingQuestsFor(activeBoundingBox.enclosingTilesRect(tileZoom))) { + Log.i(TAG, "Downloading in radius of ${radius.toInt()} meters around user") + return activeBoundingBox + } + Log.i(TAG, "All downloaded in radius of ${radius.toInt()} meters around user") + return null + } + + /** return the quest density in quests per m² for this given [boundingBox]*/ + private fun getQuestDensityFor(boundingBox: BoundingBox): Double { + val areaInKm = boundingBox.area() + val visibleQuestCount = visibleQuestsSource.getAllVisibleCount(boundingBox) + return visibleQuestCount / areaInKm + } + + /** return if there are any quests in the given tiles rect that haven't been downloaded yet */ + private fun hasMissingQuestsFor(tilesRect: TilesRect): Boolean { + val questExpirationTime = ApplicationConstants.REFRESH_QUESTS_AFTER + val ignoreOlderThan = max(0, System.currentTimeMillis() - questExpirationTime) + return !downloadedTilesDao.get(tilesRect, ignoreOlderThan).contains(DownloadedTilesType.QUESTS) + } + + companion object { + private const val TAG = "AutoQuestDownload" + } +} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadModule.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadModule.kt index 3ff3f54c89..ed2c952088 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadModule.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadModule.kt @@ -6,6 +6,6 @@ import dagger.Provides @Module object DownloadModule { @Provides - fun downloadProgressSource(downloadController: QuestDownloadController): QuestDownloadProgressSource = + fun downloadProgressSource(downloadController: QuestDownloadController): DownloadProgressSource = downloadController } \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressListener.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressListener.kt new file mode 100644 index 0000000000..46a692b003 --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressListener.kt @@ -0,0 +1,14 @@ +package de.westnordost.streetcomplete.data.download + +import androidx.annotation.DrawableRes + +interface DownloadProgressListener { + fun onStarted() {} + fun onStarted(item: DownloadItem) {} + fun onFinished(item: DownloadItem) {} + fun onError(e: Exception) {} + fun onFinished() {} + fun onSuccess() {} +} + +data class DownloadItem(@DrawableRes val iconResId: Int, val title: String) \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressRelay.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressRelay.kt new file mode 100644 index 0000000000..5a04766281 --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressRelay.kt @@ -0,0 +1,22 @@ +package de.westnordost.streetcomplete.data.download + +import java.util.concurrent.CopyOnWriteArrayList + +class DownloadProgressRelay : DownloadProgressListener { + + private val listeners = CopyOnWriteArrayList() + + override fun onStarted() { listeners.forEach { it.onStarted() } } + override fun onError(e: Exception) { listeners.forEach { it.onError(e) } } + override fun onSuccess() { listeners.forEach { it.onSuccess() } } + override fun onFinished() { listeners.forEach { it.onFinished() } } + override fun onStarted(item: DownloadItem) { listeners.forEach { it.onStarted(item) } } + override fun onFinished(item: DownloadItem) {listeners.forEach { it.onFinished(item) } } + + fun addListener(listener: DownloadProgressListener) { + listeners.add(listener) + } + fun removeListener(listener: DownloadProgressListener) { + listeners.remove(listener) + } +} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressSource.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressSource.kt new file mode 100644 index 0000000000..af1e068754 --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/DownloadProgressSource.kt @@ -0,0 +1,10 @@ +package de.westnordost.streetcomplete.data.download + +interface DownloadProgressSource { + val isPriorityDownloadInProgress: Boolean + val isDownloadInProgress: Boolean + val currentDownloadItem: DownloadItem? + + fun addDownloadProgressListener(listener: DownloadProgressListener) + fun removeDownloadProgressListener(listener: DownloadProgressListener) +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/MobileDataAutoDownloadStrategy.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/MobileDataAutoDownloadStrategy.kt index 6785aa1e7a..74ad616c4d 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/MobileDataAutoDownloadStrategy.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/MobileDataAutoDownloadStrategy.kt @@ -8,12 +8,9 @@ import de.westnordost.streetcomplete.data.visiblequests.OrderedVisibleQuestTypes /** Download strategy if user is on mobile data */ class MobileDataAutoDownloadStrategy @Inject constructor( visibleQuestsSource: VisibleQuestsSource, - downloadedTilesDao: DownloadedTilesDao, - questTypesProvider: OrderedVisibleQuestTypesProvider -) : AActiveRadiusStrategy(visibleQuestsSource, downloadedTilesDao, questTypesProvider) { + downloadedTilesDao: DownloadedTilesDao +) : AVariableRadiusStrategy(visibleQuestsSource, downloadedTilesDao) { - override val questTypeDownloadCount = 5 - override val minQuestsInActiveRadiusPerKm2 = 12 - override val activeRadii = intArrayOf(200) - override val downloadRadius = 600 + override val maxDownloadAreaInKm2 = 6.0 // that's a radius of about 1.4 km + override val desiredQuestCountInVicinity = 500 } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestAutoDownloadStrategy.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestAutoDownloadStrategy.kt index f17996b2df..32bdba8f03 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestAutoDownloadStrategy.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestAutoDownloadStrategy.kt @@ -4,12 +4,7 @@ import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.osmapi.map.data.LatLon interface QuestAutoDownloadStrategy { - /** returns the number of quest types to retrieve in one run */ - val questTypeDownloadCount: Int - - /** returns true if quests should be downloaded automatically at this position now */ - fun mayDownloadHere(pos: LatLon): Boolean - - /** returns the bbox that should be downloaded at this position (if mayDownloadHere returned true) */ - fun getDownloadBoundingBox(pos: LatLon): BoundingBox + /** returns the bbox that should be downloaded at this position or null if nothing should be + * downloaded now */ + fun getDownloadBoundingBox(pos: LatLon): BoundingBox? } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadController.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadController.kt index 8e17982a15..637a231760 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadController.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadController.kt @@ -7,7 +7,6 @@ import android.content.ServiceConnection import android.os.IBinder import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.streetcomplete.ApplicationConstants -import de.westnordost.streetcomplete.data.quest.QuestType import de.westnordost.streetcomplete.util.enclosingTilesRect import javax.inject.Inject import javax.inject.Singleton @@ -15,7 +14,7 @@ import javax.inject.Singleton /** Controls quest downloading */ @Singleton class QuestDownloadController @Inject constructor( private val context: Context -): QuestDownloadProgressSource { +): DownloadProgressSource { private var downloadServiceIsBound: Boolean = false private var downloadService: QuestDownloadService.Interface? = null @@ -29,7 +28,7 @@ import javax.inject.Singleton downloadService = null } } - private val downloadProgressRelay = QuestDownloadProgressRelay() + private val downloadProgressRelay = DownloadProgressRelay() /** @return true if a quest download triggered by the user is running */ override val isPriorityDownloadInProgress: Boolean get() = @@ -39,9 +38,9 @@ import javax.inject.Singleton override val isDownloadInProgress: Boolean get() = downloadService?.isDownloadInProgress == true - /** @return the quest type that is currently being downloaded or null if nothing is downloaded */ - override val currentDownloadingQuestType: QuestType<*>? get() = - downloadService?.currentDownloadingQuestType + /** @return the item that is currently being downloaded or null if nothing is downloaded */ + override val currentDownloadItem: DownloadItem? get() = + downloadService?.currentDownloadItem var showNotification: Boolean get() = downloadService?.showDownloadNotification == true @@ -52,17 +51,15 @@ import javax.inject.Singleton } /** Download quests in at least the given bounding box asynchronously. The next-bigger rectangle - * in a (z14) tiles grid that encloses the given bounding box will be downloaded. + * in a (z16) tiles grid that encloses the given bounding box will be downloaded. * * @param bbox the minimum area to download - * @param maxQuestTypesToDownload download at most the given number of quest types. null for - * unlimited * @param isPriority whether this shall be a priority download (cancels previous downloads and * puts itself in the front) */ - fun download(bbox: BoundingBox, maxQuestTypesToDownload: Int? = null, isPriority: Boolean = false) { + fun download(bbox: BoundingBox, isPriority: Boolean = false) { val tilesRect = bbox.enclosingTilesRect(ApplicationConstants.QUEST_TILE_ZOOM) - context.startService(QuestDownloadService.createIntent(context, tilesRect, maxQuestTypesToDownload, isPriority)) + context.startService(QuestDownloadService.createIntent(context, tilesRect, isPriority)) } private fun bindServices() { @@ -77,10 +74,10 @@ import javax.inject.Singleton downloadServiceIsBound = false } - override fun addQuestDownloadProgressListener(listener: QuestDownloadProgressListener) { + override fun addDownloadProgressListener(listener: DownloadProgressListener) { downloadProgressRelay.addListener(listener) } - override fun removeQuestDownloadProgressListener(listener: QuestDownloadProgressListener) { + override fun removeDownloadProgressListener(listener: DownloadProgressListener) { downloadProgressRelay.removeListener(listener) } } \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressListener.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressListener.kt deleted file mode 100644 index b77e46e193..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressListener.kt +++ /dev/null @@ -1,12 +0,0 @@ -package de.westnordost.streetcomplete.data.download - -import de.westnordost.streetcomplete.data.quest.QuestType - -interface QuestDownloadProgressListener { - fun onStarted() {} - fun onStarted(questType: QuestType<*>) {} - fun onFinished(questType: QuestType<*>) {} - fun onError(e: Exception) {} - fun onFinished() {} - fun onSuccess() {} -} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressRelay.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressRelay.kt deleted file mode 100644 index 2bfa617f30..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressRelay.kt +++ /dev/null @@ -1,23 +0,0 @@ -package de.westnordost.streetcomplete.data.download - -import de.westnordost.streetcomplete.data.quest.QuestType -import java.util.concurrent.CopyOnWriteArrayList - -class QuestDownloadProgressRelay : QuestDownloadProgressListener { - - private val listeners = CopyOnWriteArrayList() - - override fun onStarted() { listeners.forEach { it.onStarted() } } - override fun onError(e: Exception) { listeners.forEach { it.onError(e) } } - override fun onSuccess() { listeners.forEach { it.onSuccess() } } - override fun onFinished() { listeners.forEach { it.onFinished() } } - override fun onStarted(questType: QuestType<*>) { listeners.forEach { it.onStarted(questType) } } - override fun onFinished(questType: QuestType<*>) {listeners.forEach { it.onFinished(questType) } } - - fun addListener(listener: QuestDownloadProgressListener) { - listeners.add(listener) - } - fun removeListener(listener: QuestDownloadProgressListener) { - listeners.remove(listener) - } -} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressSource.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressSource.kt deleted file mode 100644 index 1e4dea32d0..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadProgressSource.kt +++ /dev/null @@ -1,12 +0,0 @@ -package de.westnordost.streetcomplete.data.download - -import de.westnordost.streetcomplete.data.quest.QuestType - -interface QuestDownloadProgressSource { - val isPriorityDownloadInProgress: Boolean - val isDownloadInProgress: Boolean - val currentDownloadingQuestType: QuestType<*>? - - fun addQuestDownloadProgressListener(listener: QuestDownloadProgressListener) - fun removeQuestDownloadProgressListener(listener: QuestDownloadProgressListener) -} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadService.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadService.kt index 2a5ba72792..42f837cccc 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadService.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloadService.kt @@ -7,7 +7,6 @@ import android.os.IBinder import android.util.Log import de.westnordost.streetcomplete.ApplicationConstants import de.westnordost.streetcomplete.Injector -import de.westnordost.streetcomplete.data.quest.QuestType import de.westnordost.streetcomplete.util.TilesRect import java.util.concurrent.atomic.AtomicBoolean import javax.inject.Inject @@ -18,9 +17,7 @@ import javax.inject.Provider * * Generally, starting a new download cancels the old one. This is a feature; Consideration: * If the user requests a new area to be downloaded, he'll generally be more interested in his last - * request than any request he made earlier and he wants that as fast as possible. (Downloading - * in-parallel is not possible with Overpass, only one request a time is allowed on the public - * instance) + * request than any request he made earlier and he wants that as fast as possible. * * The service can be bound to snoop into the state of the downloading process: * * To receive progress callbacks @@ -37,21 +34,27 @@ class QuestDownloadService : SingleIntentService(TAG) { private val binder: IBinder = Interface() // listener - private var progressListenerRelay = object : QuestDownloadProgressListener { + private var progressListenerRelay = object : DownloadProgressListener { override fun onStarted() { progressListener?.onStarted() } override fun onError(e: Exception) { progressListener?.onError(e) } - override fun onSuccess() { progressListener?.onSuccess() } - override fun onFinished() { progressListener?.onFinished() } - override fun onStarted(questType: QuestType<*>) { - currentQuestType = questType - progressListener?.onStarted(questType) + override fun onSuccess() { + isDownloading = false + progressListener?.onSuccess() } - override fun onFinished(questType: QuestType<*>) { - currentQuestType = null - progressListener?.onFinished(questType) + override fun onFinished() { + isDownloading = false + progressListener?.onFinished() + } + override fun onStarted(item: DownloadItem) { + currentDownloadItem = item + progressListener?.onStarted(item) + } + override fun onFinished(item: DownloadItem) { + currentDownloadItem = null + progressListener?.onFinished(item) } } - private var progressListener: QuestDownloadProgressListener? = null + private var progressListener: DownloadProgressListener? = null // state private var isPriorityDownload: Boolean = false @@ -69,7 +72,7 @@ class QuestDownloadService : SingleIntentService(TAG) { else notificationController.show() } - private var currentQuestType: QuestType<*>? = null + private var currentDownloadItem: DownloadItem? = null init { Injector.applicationComponent.inject(this) @@ -95,16 +98,13 @@ class QuestDownloadService : SingleIntentService(TAG) { } val tiles = intent.getSerializableExtra(ARG_TILES_RECT) as TilesRect - val maxQuestTypes = - if (intent.hasExtra(ARG_MAX_QUEST_TYPES)) intent.getIntExtra(ARG_MAX_QUEST_TYPES, 0) - else null val dl = questDownloaderProvider.get() dl.progressListener = progressListenerRelay try { isPriorityDownload = intent.hasExtra(ARG_IS_PRIORITY) isDownloading = true - dl.download(tiles, maxQuestTypes, cancelState) + dl.download(tiles, cancelState) } catch (e: Exception) { Log.e(TAG, "Unable to download quests", e) progressListenerRelay.onError(e) @@ -115,7 +115,7 @@ class QuestDownloadService : SingleIntentService(TAG) { /** Public interface to classes that are bound to this service */ inner class Interface : Binder() { - fun setProgressListener(listener: QuestDownloadProgressListener?) { + fun setProgressListener(listener: DownloadProgressListener?) { progressListener = listener } @@ -123,7 +123,7 @@ class QuestDownloadService : SingleIntentService(TAG) { val isDownloadInProgress: Boolean get() = isDownloading - val currentDownloadingQuestType: QuestType<*>? get() = currentQuestType + val currentDownloadItem: DownloadItem? get() = this@QuestDownloadService.currentDownloadItem var showDownloadNotification: Boolean get() = showNotification @@ -133,15 +133,13 @@ class QuestDownloadService : SingleIntentService(TAG) { companion object { private const val TAG = "QuestDownload" const val ARG_TILES_RECT = "tilesRect" - const val ARG_MAX_QUEST_TYPES = "maxQuestTypes" const val ARG_IS_PRIORITY = "isPriority" const val ARG_CANCEL = "cancel" - fun createIntent(context: Context, tilesRect: TilesRect?, maxQuestTypesToDownload: Int?, isPriority: Boolean): Intent { + fun createIntent(context: Context, tilesRect: TilesRect?,isPriority: Boolean): Intent { val intent = Intent(context, QuestDownloadService::class.java) intent.putExtra(ARG_TILES_RECT, tilesRect) intent.putExtra(ARG_IS_PRIORITY, isPriority) - maxQuestTypesToDownload?.let { intent.putExtra(ARG_MAX_QUEST_TYPES, it) } return intent } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloader.kt index f43e15a298..d64d83dca3 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloader.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/QuestDownloader.kt @@ -2,16 +2,13 @@ package de.westnordost.streetcomplete.data.download import android.util.Log import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.osmapi.map.data.LatLon import de.westnordost.streetcomplete.ApplicationConstants -import de.westnordost.streetcomplete.data.quest.QuestType -import de.westnordost.streetcomplete.data.quest.QuestTypeRegistry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType -import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestDownloader -import de.westnordost.streetcomplete.data.osmnotes.notequests.OsmNoteQuestType +import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osmnotes.OsmNotesDownloader import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesDao -import de.westnordost.streetcomplete.data.osmnotes.NotePositionsSource +import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesType +import de.westnordost.streetcomplete.data.osm.osmquest.* +import de.westnordost.streetcomplete.data.osm.osmquest.OsmApiQuestDownloader import de.westnordost.streetcomplete.data.user.UserStore import de.westnordost.streetcomplete.data.visiblequests.OrderedVisibleQuestTypesProvider import de.westnordost.streetcomplete.util.TilesRect @@ -23,21 +20,18 @@ import kotlin.math.max /** Takes care of downloading all note and osm quests */ class QuestDownloader @Inject constructor( private val osmNotesDownloaderProvider: Provider, - private val osmQuestDownloaderProvider: Provider, + private val osmApiQuestDownloaderProvider: Provider, private val downloadedTilesDao: DownloadedTilesDao, - private val notePositionsSource: NotePositionsSource, - private val questTypeRegistry: QuestTypeRegistry, private val questTypesProvider: OrderedVisibleQuestTypesProvider, private val userStore: UserStore ) { - var progressListener: QuestDownloadProgressListener? = null + var progressListener: DownloadProgressListener? = null - @Synchronized fun download(tiles: TilesRect, maxQuestTypes: Int?, cancelState: AtomicBoolean) { + @Synchronized fun download(tiles: TilesRect, cancelState: AtomicBoolean) { if (cancelState.get()) return progressListener?.onStarted() - val questTypes = getQuestTypesToDownload(tiles, maxQuestTypes) - if (questTypes.isEmpty()) { + if (hasQuestsAlready(tiles)) { progressListener?.onSuccess() progressListener?.onFinished() return @@ -46,23 +40,9 @@ class QuestDownloader @Inject constructor( val bbox = tiles.asBoundingBox(ApplicationConstants.QUEST_TILE_ZOOM) Log.i(TAG, "(${bbox.asLeftBottomRightTopString}) Starting") - Log.i(TAG, "Quest types to download: ${questTypes.joinToString { it.javaClass.simpleName }}") try { - // always first download notes, because note positions are blockers for creating other - // quests - val noteQuestType = getOsmNoteQuestType() - if (questTypes.contains(noteQuestType)) { - downloadNotes(bbox, tiles) - - } - - val notesPositions = notePositionsSource.getAllPositions(bbox).toSet() - - for (questType in questTypes) { - if (cancelState.get()) break - downloadQuestType(bbox, tiles, questType, notesPositions) - } + downloadQuestTypes(tiles, bbox, cancelState) progressListener?.onSuccess() } finally { progressListener?.onFinished() @@ -70,50 +50,43 @@ class QuestDownloader @Inject constructor( } } - private fun getOsmNoteQuestType() = - questTypeRegistry.getByName(OsmNoteQuestType::class.java.simpleName)!! + private fun downloadQuestTypes(tiles: TilesRect, bbox: BoundingBox, cancelState: AtomicBoolean) { + val downloadItem = DownloadItem(R.drawable.ic_search_black_128dp, "Multi download") + progressListener?.onStarted(downloadItem) + + // always first download notes, note positions are blockers for creating other quests + downloadNotes(bbox) + + if (cancelState.get()) return + + downloadOsmElementQuestTypes(bbox) - private fun getQuestTypesToDownload(tiles: TilesRect, maxQuestTypes: Int?): List> { - val result = questTypesProvider.get().toMutableList() + downloadedTilesDao.put(tiles, DownloadedTilesType.QUESTS) + + progressListener?.onFinished(downloadItem) + } + + private fun hasQuestsAlready(tiles: TilesRect): Boolean { val questExpirationTime = ApplicationConstants.REFRESH_QUESTS_AFTER val ignoreOlderThan = max(0, System.currentTimeMillis() - questExpirationTime) - val alreadyDownloadedNames = downloadedTilesDao.get(tiles, ignoreOlderThan) - if (alreadyDownloadedNames.isNotEmpty()) { - val alreadyDownloaded = alreadyDownloadedNames.map { questTypeRegistry.getByName(it) } - result.removeAll(alreadyDownloaded) - Log.i(TAG, "Quest types already in local store: ${alreadyDownloadedNames.joinToString()}") - } - return if (maxQuestTypes != null && maxQuestTypes < result.size) - result.subList(0, maxQuestTypes) - else - result + return downloadedTilesDao.get(tiles, ignoreOlderThan).contains(DownloadedTilesType.QUESTS) } - private fun downloadNotes(bbox: BoundingBox, tiles: TilesRect) { + private fun downloadNotes(bbox: BoundingBox) { val notesDownload = osmNotesDownloaderProvider.get() val userId: Long = userStore.userId.takeIf { it != -1L } ?: return // do not download notes if not logged in because notes shall only be downloaded if logged in - val noteQuestType = getOsmNoteQuestType() - progressListener?.onStarted(noteQuestType) + val maxNotes = 10000 notesDownload.download(bbox, userId, maxNotes) - downloadedTilesDao.put(tiles, OsmNoteQuestType::class.java.simpleName) - progressListener?.onFinished(noteQuestType) } - private fun downloadQuestType(bbox: BoundingBox, tiles: TilesRect, questType: QuestType<*>, notesPositions: Set) { - if (questType is OsmElementQuestType<*>) { - progressListener?.onStarted(questType) - val questDownload = osmQuestDownloaderProvider.get() - if (questDownload.download(questType, bbox, notesPositions)) { - downloadedTilesDao.put(tiles, questType.javaClass.simpleName) - } - progressListener?.onFinished(questType) - } + private fun downloadOsmElementQuestTypes(bbox: BoundingBox) { + val questTypes = questTypesProvider.get().filterIsInstance>() + osmApiQuestDownloaderProvider.get().download(questTypes, bbox) } companion object { private const val TAG = "QuestDownload" } - } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/WifiAutoDownloadStrategy.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/WifiAutoDownloadStrategy.kt index 909f00644a..345b5f43e8 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/WifiAutoDownloadStrategy.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/WifiAutoDownloadStrategy.kt @@ -10,22 +10,14 @@ import de.westnordost.streetcomplete.data.visiblequests.OrderedVisibleQuestTypes /** Download strategy if user is on wifi */ class WifiAutoDownloadStrategy @Inject constructor( visibleQuestsSource: VisibleQuestsSource, - downloadedTilesDao: DownloadedTilesDao, - questTypes: OrderedVisibleQuestTypesProvider -) : AActiveRadiusStrategy(visibleQuestsSource, downloadedTilesDao, questTypes) { + downloadedTilesDao: DownloadedTilesDao +) : AVariableRadiusStrategy(visibleQuestsSource, downloadedTilesDao) { /** Let's assume that if the user is on wifi, he is either at home, at work, in the hotel, at a * café,... in any case, somewhere that would act as a "base" from which he can go on an * excursion. Let's make sure he can, even if there is no or bad internet. - * - * Since download size is almost unlimited, we can be very generous here. - * However, Overpass is as limited as always, so the number of quest types we download is - * limited as before */ + */ - override val questTypeDownloadCount = 5 - override val minQuestsInActiveRadiusPerKm2 = 12 - - // checks if either in 600 or 200m radius, there are enough quests. - override val activeRadii = intArrayOf(600, 200) - override val downloadRadius = 1200 + override val maxDownloadAreaInKm2 = 12.0 // that's a radius of about 2 km + override val desiredQuestCountInVicinity = 1000 } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDao.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDao.kt index 67f48d203c..b02f007d07 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDao.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesDao.kt @@ -4,39 +4,35 @@ import android.database.sqlite.SQLiteOpenHelper import androidx.core.content.contentValuesOf import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesTable.Columns.X import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesTable.Columns.Y -import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesTable.Columns.QUEST_TYPE +import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesTable.Columns.TYPE import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesTable.Columns.DATE import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesTable.NAME import de.westnordost.streetcomplete.ktx.query -import de.westnordost.streetcomplete.ktx.transaction import de.westnordost.streetcomplete.util.Tile import de.westnordost.streetcomplete.util.TilesRect import javax.inject.Inject -/** Keeps info in which areas quests have been downloaded already in a tile grid of zoom level 14 - * (~0.022° per tile -> a few kilometers sidelength) */ +/** Keeps info in which areas things have been downloaded already in a tile grid */ class DownloadedTilesDao @Inject constructor(private val dbHelper: SQLiteOpenHelper) { private val db get() = dbHelper.writableDatabase - /** Persist that the given quest type has been downloaded in every tile in the given tile range */ - fun put(tilesRect: TilesRect, questTypeName: String) { - db.transaction { - val time = System.currentTimeMillis() - for (tile in tilesRect.asTileSequence()) { - val values = contentValuesOf( - X to tile.x, - Y to tile.y, - QUEST_TYPE to questTypeName, - DATE to time - ) - db.replaceOrThrow(NAME, null, values) - } + /** Persist that the given type has been downloaded in every tile in the given tile range */ + fun put(tilesRect: TilesRect, typeName: String) { + val time = System.currentTimeMillis() + for (tile in tilesRect.asTileSequence()) { + val values = contentValuesOf( + X to tile.x, + Y to tile.y, + TYPE to typeName, + DATE to time + ) + db.replaceOrThrow(NAME, null, values) } } - /** Invalidate all quest types within the given tile. (consider them as not-downloaded) */ + /** Invalidate all types within the given tile. (consider them as not-downloaded) */ fun remove(tile: Tile): Int { return db.delete(NAME, "$X = ? AND $Y = ?", arrayOf(tile.x.toString(), tile.y.toString())) } @@ -45,13 +41,13 @@ class DownloadedTilesDao @Inject constructor(private val dbHelper: SQLiteOpenHel db.execSQL("DELETE FROM $NAME") } - /** @return a list of quest type names which have already been downloaded in every tile in the + /** @return a list of type names which have already been downloaded in every tile in the * given tile range */ fun get(tilesRect: TilesRect, ignoreOlderThan: Long): List { val tileCount = tilesRect.size return db.query(NAME, - columns = arrayOf(QUEST_TYPE), + columns = arrayOf(TYPE), selection = "$X BETWEEN ? AND ? AND $Y BETWEEN ? AND ? AND $DATE > ?", selectionArgs = arrayOf( tilesRect.left.toString(), @@ -60,7 +56,7 @@ class DownloadedTilesDao @Inject constructor(private val dbHelper: SQLiteOpenHel tilesRect.bottom.toString(), ignoreOlderThan.toString() ), - groupBy = QUEST_TYPE, + groupBy = TYPE, having = "COUNT(*) >= $tileCount") { it.getString(0) } } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesTable.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesTable.kt index ea53594b7e..ee215fb1e1 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesTable.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesTable.kt @@ -6,7 +6,7 @@ object DownloadedTilesTable { object Columns { const val X = "x" const val Y = "y" - const val QUEST_TYPE = "quest_type" + const val TYPE = "quest_type" const val DATE = "date" } @@ -14,8 +14,8 @@ object DownloadedTilesTable { CREATE TABLE $NAME ( ${Columns.X} int NOT NULL, ${Columns.Y} int NOT NULL, - ${Columns.QUEST_TYPE} varchar(255) NOT NULL, + ${Columns.TYPE} varchar(255) NOT NULL, ${Columns.DATE} int NOT NULL, - CONSTRAINT primary_key PRIMARY KEY (${Columns.X}, ${Columns.Y}, ${Columns.QUEST_TYPE}) + CONSTRAINT primary_key PRIMARY KEY (${Columns.X}, ${Columns.Y}, ${Columns.TYPE}) );""" } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesType.kt b/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesType.kt new file mode 100644 index 0000000000..48ca14d91d --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/download/tiles/DownloadedTilesType.kt @@ -0,0 +1,5 @@ +package de.westnordost.streetcomplete.data.download.tiles + +object DownloadedTilesType { + const val QUESTS = "quests" +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpression.kt b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpression.kt index 4263bd622b..a3bd9ee4b4 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpression.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpression.kt @@ -5,25 +5,28 @@ import de.westnordost.streetcomplete.data.elementfilter.ElementsTypeFilter.NODES import de.westnordost.streetcomplete.data.elementfilter.ElementsTypeFilter.WAYS import de.westnordost.streetcomplete.data.elementfilter.ElementsTypeFilter.RELATIONS import de.westnordost.streetcomplete.data.elementfilter.filters.ElementFilter +import java.util.* /** Represents a parse result of a string in filter syntax, i.e. * "ways with (highway = residential or highway = tertiary) and !name" */ class ElementFilterExpression( - private val elementsTypes: List, + private val elementsTypes: EnumSet, private val elementExprRoot: BooleanExpression? ) { /** returns whether the given element is found through (=matches) this expression */ - fun matches(element: Element) = - when (element.type) { - Element.Type.NODE -> elementsTypes.contains(NODES) - Element.Type.WAY -> elementsTypes.contains(WAYS) - Element.Type.RELATION -> elementsTypes.contains(RELATIONS) - else -> false - } && (elementExprRoot?.matches(element) ?: true) + fun matches(element: Element): Boolean = + includesElementType(element.type) && (elementExprRoot?.matches(element) ?: true) + + fun includesElementType(elementType: Element.Type): Boolean = when (elementType) { + Element.Type.NODE -> elementsTypes.contains(NODES) + Element.Type.WAY -> elementsTypes.contains(WAYS) + Element.Type.RELATION -> elementsTypes.contains(RELATIONS) + else -> false + } /** returns this expression as a Overpass query string */ fun toOverpassQLString(): String = OverpassQueryCreator(elementsTypes, elementExprRoot).create() } /** Enum that specifies which type(s) of elements to retrieve */ -enum class ElementsTypeFilter { NODES, WAYS, RELATIONS } \ No newline at end of file +enum class ElementsTypeFilter { NODES, WAYS, RELATIONS } diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParser.kt b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParser.kt index d896e2ea1b..4457a48d8c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParser.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParser.kt @@ -5,6 +5,8 @@ import de.westnordost.streetcomplete.data.elementfilter.filters.* import de.westnordost.streetcomplete.data.meta.toCheckDate import java.lang.NumberFormatException import java.text.ParseException +import java.util.* +import kotlin.collections.ArrayList import kotlin.math.min /** @@ -14,13 +16,12 @@ import kotlin.math.min * "ways with (highway = residential or highway = tertiary) and !name" (finds all * residential and tertiary roads that have no name) */ -class ElementFiltersParser { - fun parse(input: String): ElementFilterExpression { - // convert all white-spacey things to whitespaces so we do not have to deal with them later - val cursor = StringWithCursor(input.replace("\\s".toRegex(), " ")) - return ElementFilterExpression(cursor.parseElementsDeclaration(), cursor.parseTags()) - } +fun String.toElementFilterExpression(): ElementFilterExpression { + // convert all white-spacey things to whitespaces so we do not have to deal with them later + val cursor = StringWithCursor(replace("\\s".toRegex(), " ")) + + return ElementFilterExpression(cursor.parseElementsDeclaration(), cursor.parseTags()) } private const val WITH = "with" @@ -71,7 +72,7 @@ private val NUMBER_WORD_REGEX = Regex("(?:([0-9]+(?:\\.[0-9]*)?)|(\\.[0-9]+))(?: private fun String.stripQuotes() = replace("^[\"']|[\"']$".toRegex(), "") -private fun StringWithCursor.parseElementsDeclaration(): List { +private fun StringWithCursor.parseElementsDeclaration(): EnumSet { val result = ArrayList() result.add(parseElementDeclaration()) while (nextIsAndAdvance(',')) { @@ -81,7 +82,14 @@ private fun StringWithCursor.parseElementsDeclaration(): List EnumSet.of(result[0]) + 2 -> EnumSet.of(result[0], result[1]) + 3 -> EnumSet.of(result[0], result[1], result[2]) + else -> throw IllegalStateException() + } + } private fun StringWithCursor.parseElementDeclaration(): ElementsTypeFilter { diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtils.kt b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtils.kt index 2223357597..545488b747 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtils.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtils.kt @@ -1,27 +1,5 @@ package de.westnordost.streetcomplete.data.elementfilter -import de.westnordost.osmapi.map.data.BoundingBox -import java.text.NumberFormat -import java.util.* - -const val DEFAULT_MAX_QUESTS = 2000 - -// by default we limit the number of quests created to something that does not cause -// performance problems -fun getQuestPrintStatement() = "out meta geom $DEFAULT_MAX_QUESTS;" - -fun BoundingBox.toGlobalOverpassBBox() = "[bbox:${toOverpassBbox()}];" - -fun BoundingBox.toOverpassBboxFilter() = "(${toOverpassBbox()})" - -private fun BoundingBox.toOverpassBbox(): String { - val df = NumberFormat.getNumberInstance(Locale.US) - df.maximumFractionDigits = 7 - - return df.format(minLatitude) + "," + df.format(minLongitude) + "," + - df.format(maxLatitude) + "," + df.format(maxLongitude) -} - private val QUOTES_NOT_REQUIRED = Regex("[a-zA-Z_][a-zA-Z0-9_]*|-?[0-9]+") fun String.quoteIfNecessary() = diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQueryCreator.kt b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQueryCreator.kt index 3588f4741d..42029e231a 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQueryCreator.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQueryCreator.kt @@ -1,13 +1,13 @@ package de.westnordost.streetcomplete.data.elementfilter import de.westnordost.osmapi.map.data.Element -import de.westnordost.streetcomplete.ktx.containsExactlyInAnyOrder import de.westnordost.streetcomplete.data.elementfilter.ElementsTypeFilter.* import de.westnordost.streetcomplete.data.elementfilter.filters.ElementFilter +import java.util.* /** Create an overpass query from the given element filter expression */ class OverpassQueryCreator( - elementTypes: List, + elementTypes: EnumSet, private val expr: BooleanExpression?) { private val elementTypes = elementTypes.toOqlNames() @@ -37,11 +37,11 @@ class OverpassQueryCreator( } } - private fun List.toOqlNames(): List = when { - containsExactlyInAnyOrder(listOf(NODES, WAYS)) -> listOf("nw") - containsExactlyInAnyOrder(listOf(WAYS, RELATIONS)) -> listOf("wr") - containsExactlyInAnyOrder(listOf(NODES,WAYS, RELATIONS)) -> listOf("nwr") - else -> map { when (it) { + private fun EnumSet.toOqlNames(): List = when { + containsAll(listOf(NODES, WAYS, RELATIONS)) -> listOf("nwr") + containsAll(listOf(NODES, WAYS)) -> listOf("nw") + containsAll(listOf(WAYS, RELATIONS)) -> listOf("wr") + else -> map { when (it!!) { NODES -> "node" WAYS -> "way" RELATIONS -> "rel" diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/filters/RelativeDate.kt b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/filters/RelativeDate.kt index 1260165f71..427ede6c3c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/filters/RelativeDate.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/elementfilter/filters/RelativeDate.kt @@ -10,9 +10,13 @@ interface DateFilter { class RelativeDate(val deltaDays: Float): DateFilter { override val date: Date get() { val cal: Calendar = Calendar.getInstance() - cal.add(Calendar.SECOND, (deltaDays * 24 * 60 * 60).toInt()) + cal.add(Calendar.SECOND, (deltaDays * 24 * 60 * 60 * MULTIPLIER).toInt()) return cal.time } + + companion object { + var MULTIPLIER: Float = 1f + } } class FixedDate(override val date: Date): DateFilter \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreator.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreator.kt index efb6da09cc..256b5aa02e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreator.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreator.kt @@ -1,5 +1,6 @@ package de.westnordost.streetcomplete.data.osm.elementgeometry +import de.westnordost.osmapi.map.MapData import de.westnordost.osmapi.map.data.* import de.westnordost.streetcomplete.ktx.isArea import de.westnordost.streetcomplete.util.centerPointOfPolygon @@ -11,6 +12,33 @@ import kotlin.collections.ArrayList /** Creates an ElementGeometry from an element and a collection of positions. */ class ElementGeometryCreator @Inject constructor() { + /** Create an ElementGeometry from any element, using the given MapData to find the positions + * of the nodes. + * + * @param element the element to create the geometry for + * @param mapData the MapData that contains the elements with the necessary + * @param allowIncomplete whether incomplete relations should return an incomplete + * ElementGeometry (otherwise: null) + * + * @return an ElementGeometry or null if any necessary element to create the geometry is not + * in the given MapData */ + fun create(element: Element, mapData: MapData, allowIncomplete: Boolean = false): ElementGeometry? { + when(element) { + is Node -> { + return create(element) + } + is Way -> { + val positions = mapData.getNodePositions(element) ?: return null + return create(element, positions) + } + is Relation -> { + val positionsByWayId = mapData.getWaysNodePositions(element, allowIncomplete) ?: return null + return create(element, positionsByWayId) + } + else -> return null + } + } + /** Create an ElementPointGeometry for a node. */ fun create(node: Node) = ElementPointGeometry(node.position) @@ -47,6 +75,7 @@ class ElementGeometryCreator @Inject constructor() { * @return an ElementPolygonsGeometry if the relation describes an area or an * ElementPolylinesGeometry if it describes is a linear feature */ fun create(relation: Relation, wayGeometries: Map>): ElementGeometry? { + return if (relation.isArea()) { createMultipolygonGeometry(relation, wayGeometries) } else { @@ -107,17 +136,17 @@ class ElementGeometryCreator @Inject constructor() { private fun getRelationMemberWaysNodePositions( relation: Relation, wayGeometries: Map> ): List> { - return relation.members.filter { it.type == Element.Type.WAY }.mapNotNull { - getValidNodePositions(wayGeometries[it.ref]) - } + return relation.members + .filter { it.type == Element.Type.WAY } + .mapNotNull { getValidNodePositions(wayGeometries[it.ref]) } } private fun getRelationMemberWaysNodePositions( relation: Relation, withRole: String, wayGeometries: Map> ): List> { - return relation.members.filter { it.type == Element.Type.WAY && it.role == withRole }.mapNotNull { - getValidNodePositions(wayGeometries[it.ref]) - } + return relation.members + .filter { it.type == Element.Type.WAY && it.role == withRole } + .mapNotNull { getValidNodePositions(wayGeometries[it.ref]) } } private fun getValidNodePositions(wayGeometry: List?): List? { @@ -204,3 +233,29 @@ private fun MutableList.eliminateDuplicates() { } } } + +private fun MapData.getNodePositions(way: Way): List? { + return way.nodeIds.map { nodeId -> + val node = getNode(nodeId) ?: return null + node.position + } +} + +private fun MapData.getWaysNodePositions(relation: Relation, allowIncomplete: Boolean = false): Map>? { + val wayMembers = relation.members.filter { it.type == Element.Type.WAY } + val result = mutableMapOf>() + for (wayMember in wayMembers) { + val way = getWay(wayMember.ref) + if (way != null) { + val wayPositions = getNodePositions(way) + if (wayPositions != null) { + result[way.id] = wayPositions + } else { + if (!allowIncomplete) return null + } + } else { + if (!allowIncomplete) return null + } + } + return result +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/OsmApiElementGeometryCreator.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/OsmApiElementGeometryCreator.kt deleted file mode 100644 index d390574a01..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/elementgeometry/OsmApiElementGeometryCreator.kt +++ /dev/null @@ -1,80 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.elementgeometry - -import de.westnordost.streetcomplete.data.MapDataApi -import de.westnordost.osmapi.common.errors.OsmNotFoundException - -import de.westnordost.osmapi.map.data.* -import de.westnordost.osmapi.map.handler.DefaultMapDataHandler -import javax.inject.Inject - -/** Creates the geometry for an element by fetching the necessary geometry data from the OSM API */ -class OsmApiElementGeometryCreator @Inject constructor( - private val mapDataApi: MapDataApi, - private val elementCreator: ElementGeometryCreator) { - - fun create(element: Element): ElementGeometry? { - when(element) { - is Node -> { - return elementCreator.create(element) - } - is Way -> { - val positions = getNodePositions(element.id) ?: return null - return elementCreator.create(element, positions) - } - is Relation -> { - val positionsByWayId = getWaysNodePositions(element.id) ?: return null - return elementCreator.create(element, positionsByWayId) - } - else -> return null - } - } - - private fun getNodePositions(wayId: Long): List? { - var way: Way? = null - val nodes = mutableMapOf() - val handler = object : DefaultMapDataHandler() { - override fun handle(n: Node) { nodes[n.id] = n } - override fun handle(w: Way) { way = w } - } - - try { - mapDataApi.getWayComplete(wayId, handler) - } catch (e : OsmNotFoundException) { - return null - } - val wayNodeIds = way?.nodeIds ?: return null - return wayNodeIds.map { nodeId -> - val node = nodes[nodeId] ?: return null - node.position - } - } - - private fun getWaysNodePositions(relationId: Long): Map>? { - val nodes = mutableMapOf() - val ways = mutableMapOf() - var relation: Relation? = null - - val handler = object : DefaultMapDataHandler() { - override fun handle(n: Node) { nodes[n.id] = n } - override fun handle(w: Way) { ways[w.id] = w } - override fun handle(r: Relation) { relation = r } - } - - try { - mapDataApi.getRelationComplete(relationId, handler) - } catch (e : OsmNotFoundException) { - return null - } - - val members = relation?.members ?: return null - val wayMembers = members.filter { it.type == Element.Type.WAY } - return wayMembers.associate { wayMember -> - val way = ways[wayMember.ref] ?: return null - val wayPositions = way.nodeIds.map { nodeId -> - val node = nodes[nodeId] ?: return null - node.position - } - way.id to wayPositions - } - } -} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/mapdata/OverpassMapDataAndGeometryApi.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/mapdata/OverpassMapDataAndGeometryApi.kt deleted file mode 100644 index 62470840bf..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/mapdata/OverpassMapDataAndGeometryApi.kt +++ /dev/null @@ -1,78 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.mapdata - -import android.util.Log -import de.westnordost.streetcomplete.data.OverpassMapDataApi -import de.westnordost.osmapi.map.data.* -import de.westnordost.osmapi.overpass.MapDataWithGeometryHandler -import de.westnordost.osmapi.overpass.OsmTooManyRequestsException -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryCreator -import javax.inject.Inject - -/** Queries data from the Overpass API and handles quota by suspending the thread until it has - * replenished.*/ -class OverpassMapDataAndGeometryApi @Inject constructor( - private val api: OverpassMapDataApi, - private val elementGeometryCreator: ElementGeometryCreator -) { - - /** Get data from Overpass assuming a "out meta geom;" query and handles automatically waiting - * for the request quota to replenish. - * - * @param query Query string. Either Overpass QL or Overpass XML query string - * @param callback map data callback that is fed the map data and geometry - * @return false if it was interrupted while waiting for the quota to be replenished - * - * @throws OsmBadUserInputException if there is an error if the query - */ - fun query(query: String, callback: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - - val handler = object : MapDataWithGeometryHandler { - override fun handle(bounds: BoundingBox) {} - - override fun handle(node: Node) { - callback(node, elementGeometryCreator.create(node)) - } - - override fun handle(way: Way, bounds: BoundingBox, geometry: MutableList) { - callback(way, elementGeometryCreator.create(way, geometry)) - } - - override fun handle( - relation: Relation, - bounds: BoundingBox, - nodeGeometries: MutableMap, - wayGeometries: MutableMap> - ) { - callback(relation, elementGeometryCreator.create(relation, wayGeometries)) - } - } - - try { - api.queryElementsWithGeometry(query, handler) - } catch (e: OsmTooManyRequestsException) { - val status = api.getStatus() - if (status.availableSlots == 0) { - // apparently sometimes Overpass does not tell the client when the next slot is - // available when there is currently no slot available. So let's just wait 60s - // before trying again - // also, rather wait 1s longer than required cause we only get the time in seconds - val nextAvailableSlotIn = status.nextAvailableSlotIn - val waitInSeconds = if (nextAvailableSlotIn != null) nextAvailableSlotIn + 1 else 60 - Log.i(TAG, "Hit Overpass quota. Waiting ${waitInSeconds}s before continuing") - try { - Thread.sleep(waitInSeconds * 1000L) - } catch (ie: InterruptedException) { - Log.d(TAG, "Thread interrupted while waiting for Overpass quota to be replenished") - return false - } - } - return query(query, callback) - } - return true - } - - companion object { - private const val TAG = "OverpassMapDataGeomDao" - } -} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/CachingMapDataWithGeometry.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/CachingMapDataWithGeometry.kt new file mode 100644 index 0000000000..fcdb464c06 --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/CachingMapDataWithGeometry.kt @@ -0,0 +1,55 @@ +package de.westnordost.streetcomplete.data.osm.osmquest + +import de.westnordost.osmapi.map.MapDataWithGeometry +import de.westnordost.osmapi.map.MutableMapData +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryCreator +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry +import javax.inject.Inject + +/** MapDataWithGeometry that lazily creates the element geometry. Will create incomplete (relation) + * geometry */ +class CachingMapDataWithGeometry @Inject constructor( + private val elementGeometryCreator: ElementGeometryCreator +) : MutableMapData(), MapDataWithGeometry { + + private val nodeGeometriesById: MutableMap = mutableMapOf() + private val wayGeometriesById: MutableMap = mutableMapOf() + private val relationGeometriesById: MutableMap = mutableMapOf() + + override fun getNodeGeometry(id: Long): ElementPointGeometry? { + val node = nodesById[id] ?: return null + if (!nodeGeometriesById.containsKey(id)) { + synchronized(nodeGeometriesById) { + if (!nodeGeometriesById.containsKey(id)) { + nodeGeometriesById[id] = elementGeometryCreator.create(node) + } + } + } + return nodeGeometriesById[id] + } + + override fun getWayGeometry(id: Long): ElementGeometry? { + val way = waysById[id] ?: return null + if (!wayGeometriesById.containsKey(id)) { + synchronized(wayGeometriesById) { + if (!wayGeometriesById.containsKey(id)) { + wayGeometriesById[id] = elementGeometryCreator.create(way, this, true) + } + } + } + return wayGeometriesById[id] + } + + override fun getRelationGeometry(id: Long): ElementGeometry? { + val relation = relationsById[id] ?: return null + if (!relationGeometriesById.containsKey(id)) { + synchronized(relationGeometriesById) { + if (!relationGeometriesById.containsKey(id)) { + relationGeometriesById[id] = elementGeometryCreator.create(relation, this, true) + } + } + } + return relationGeometriesById[id] + } +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmApiQuestDownloader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmApiQuestDownloader.kt new file mode 100644 index 0000000000..6334bf993a --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmApiQuestDownloader.kt @@ -0,0 +1,209 @@ +package de.westnordost.streetcomplete.data.osm.osmquest + +import android.util.Log +import de.westnordost.countryboundaries.isInAny +import de.westnordost.countryboundaries.CountryBoundaries +import de.westnordost.countryboundaries.intersects +import de.westnordost.osmapi.common.errors.OsmQueryTooBigException +import de.westnordost.osmapi.map.MapDataWithGeometry +import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.data.Element +import de.westnordost.osmapi.map.data.Element.Type.* +import de.westnordost.osmapi.map.data.LatLon +import de.westnordost.osmapi.map.data.OsmLatLon +import de.westnordost.osmapi.map.getRelationComplete +import de.westnordost.osmapi.map.handler.MapDataHandler +import de.westnordost.osmapi.map.isRelationComplete +import de.westnordost.streetcomplete.data.MapDataApi +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryCreator +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry +import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.data.osmnotes.NotePositionsSource +import de.westnordost.streetcomplete.data.quest.QuestType +import de.westnordost.streetcomplete.util.measuredLength +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import kotlinx.coroutines.runBlocking +import java.util.* +import java.util.concurrent.FutureTask +import javax.inject.Inject +import javax.inject.Provider +import kotlin.collections.ArrayList + +/** Does one API call to get all the map data and generates quests from that. Calls getApplicableElements + * on all the quest types */ +class OsmApiQuestDownloader @Inject constructor( + private val elementDB: MergedElementDao, + private val osmQuestController: OsmQuestController, + private val countryBoundariesFuture: FutureTask, + private val notePositionsSource: NotePositionsSource, + private val mapDataApi: MapDataApi, + private val mapDataWithGeometry: Provider, + private val elementGeometryCreator: ElementGeometryCreator +) : CoroutineScope by CoroutineScope(Dispatchers.Default) { + + fun download(questTypes: List>, bbox: BoundingBox) { + if (questTypes.isEmpty()) return + + var time = System.currentTimeMillis() + + val completeRelationGeometries = mutableMapOf() + + val mapData = mapDataWithGeometry.get() + getMapAndHandleTooBigQuery(bbox, mapData) + // bbox should be the bbox of the complete download + mapData.handle(bbox) + + val quests = ArrayList() + val questElements = HashSet() + + val secondsSpentDownloading = (System.currentTimeMillis() - time) / 1000 + Log.i(TAG,"Downloaded ${mapData.nodes.size} nodes, ${mapData.ways.size} ways and ${mapData.relations.size} relations in ${secondsSpentDownloading}s") + time = System.currentTimeMillis() + + val truncatedBlacklistedPositions = notePositionsSource.getAllPositions(bbox).map { it.truncateTo5Decimals() }.toSet() + + val countryBoundaries = countryBoundariesFuture.get() + + runBlocking { + for (questType in questTypes) { + launch(Dispatchers.Default) { + val questTypeName = questType.getName() + if (!countryBoundaries.intersects(bbox, questType.enabledInCountries)) { + Log.d(TAG, "$questTypeName: Skipped because it is disabled for this country") + } else { + var i = 0 + val questTime = System.currentTimeMillis() + for (element in questType.getApplicableElements(mapData)) { + val geometry = getCompleteGeometry(element.type, element.id, mapData, completeRelationGeometries) + val quest = createQuest(questType, element, geometry, truncatedBlacklistedPositions) ?: continue + + quests.add(quest) + questElements.add(element) + ++i + } + Log.d(TAG, "$questTypeName: Found $i quests in ${System.currentTimeMillis() - questTime}ms") + } + } + } + } + val secondsSpentAnalyzing = (System.currentTimeMillis() - time) / 1000 + + Log.i(TAG,"Created ${quests.size} quests in ${secondsSpentAnalyzing}s") + + time = System.currentTimeMillis() + + // elements must be put into DB first because quests have foreign keys on it + elementDB.putAll(questElements) + + val questTypeNames = questTypes.map { it.getName() } + val replaceResult = osmQuestController.replaceInBBox(quests, bbox, questTypeNames) + + elementDB.deleteUnreferenced() + + for (questType in questTypes) { + questType.cleanMetadata() + } + + val secondsSpentPersisting = (System.currentTimeMillis() - time) / 1000 + + Log.i(TAG,"Persisting ${quests.size} quests in ${secondsSpentPersisting}s") + + Log.i(TAG,"Added ${replaceResult.added} new and removed ${replaceResult.deleted} already resolved quests") + } + + private fun createQuest(questType: OsmElementQuestType<*>, element: Element, geometry: ElementGeometry?, blacklistedPositions: Set): OsmQuest? { + // invalid geometry -> can't show this quest, so skip it + val pos = geometry?.center ?: return null + + // do not create quests whose marker is at/near a blacklisted position + if (blacklistedPositions.contains(pos.truncateTo5Decimals())) return null + + // do not create quests in countries where the quest is not activated + val countries = questType.enabledInCountries + if (!countryBoundariesFuture.get().isInAny(pos, countries)) return null + + // do not create quests that refer to geometry that is too long for a surveyor to be expected to survey + if (geometry is ElementPolylinesGeometry) { + val totalLength = geometry.polylines.sumByDouble { it.measuredLength() } + if (totalLength > MAX_GEOMETRY_LENGTH_IN_METERS) { + return null + } + } + + return OsmQuest(questType, element.type, element.id, geometry) + } + + private fun getCompleteGeometry( + elementType: Element.Type, + elementId: Long, + mapData: MapDataWithGeometry, + cache: MutableMap + ): ElementGeometry? { + return when(elementType) { + NODE -> mapData.getNodeGeometry(elementId) + WAY -> mapData.getWayGeometry(elementId) + // relations are downloaded incomplete from the OSM API, we want the complete geometry here + RELATION -> getCompleteRelationGeometry(elementId, mapData, cache) + } + } + + private fun getCompleteRelationGeometry(id: Long, mapData: MapDataWithGeometry, cache: MutableMap): ElementGeometry? { + if (!cache.containsKey(id)) { + synchronized(cache) { + if (!cache.containsKey(id)) { + cache[id] = createCompleteRelationGeometry(id, mapData) + } + } + } + return cache[id] + } + + private fun createCompleteRelationGeometry(id: Long, mapData: MapDataWithGeometry): ElementGeometry? { + val isComplete = mapData.isRelationComplete(id) + if (isComplete) { + // if the relation is already complete within the given mapData, we can just take it from there + return mapData.getRelationGeometry(id) + } else { + // otherwise we need to query the API first and create it from that data instead + val completeRelationData = mapDataApi.getRelationComplete(id) + val relation = mapData.getRelation(id) ?: return null + return elementGeometryCreator.create(relation, completeRelationData, false) + } + } + + private fun getMapAndHandleTooBigQuery(bounds: BoundingBox, mapDataHandler: MapDataHandler) { + try { + mapDataApi.getMap(bounds, mapDataHandler) + } catch (e : OsmQueryTooBigException) { + for (subBounds in bounds.splitIntoFour()) { + getMapAndHandleTooBigQuery(subBounds, mapDataHandler) + } + } + } + + companion object { + private const val TAG = "QuestDownload" + } +} + +private fun QuestType<*>.getName() = javaClass.simpleName + +// the resulting precision is about ~1 meter (see #1089) +private fun LatLon.truncateTo5Decimals() = OsmLatLon(latitude.truncateTo5Decimals(), longitude.truncateTo5Decimals()) + +private fun Double.truncateTo5Decimals() = (this * 1e5).toInt().toDouble() / 1e5 + +private fun BoundingBox.splitIntoFour(): List { + val center = OsmLatLon((maxLatitude + minLatitude) / 2, (maxLongitude + minLongitude) / 2) + return listOf( + BoundingBox(minLatitude, minLongitude, center.latitude, center.longitude), + BoundingBox(minLatitude, center.longitude, center.latitude, maxLongitude), + BoundingBox(center.latitude, minLongitude, maxLatitude, center.longitude), + BoundingBox(center.latitude, center.longitude, maxLatitude, maxLongitude) + ) +} + +const val MAX_GEOMETRY_LENGTH_IN_METERS = 600 diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementQuestType.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementQuestType.kt index ac9c6f1a54..58e5218c19 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementQuestType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementQuestType.kt @@ -1,11 +1,10 @@ package de.westnordost.streetcomplete.data.osm.osmquest -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.data.quest.QuestType import de.westnordost.streetcomplete.data.quest.AllCountries import de.westnordost.streetcomplete.data.quest.Countries -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder /** Quest type where each quest refers to an OSM element */ @@ -38,14 +37,12 @@ interface OsmElementQuestType : QuestType { override val title: Int get() = getTitle(emptyMap()) - /** Downloads map data for this quest type for the given [bbox] and puts the received data into - * the [handler]. Returns whether the download was successful - */ - fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean + /** return all elements within the given map data that are applicable to this quest type. */ + fun getApplicableElements(mapData: MapDataWithGeometry): Iterable /** returns whether a quest of this quest type could be created out of the given [element]. If the - * element alone does not suffice to find this out (but e.g. an Overpass query would need to be - * made to find this out), this should return null. + * element alone does not suffice to find this out (but f.e. is determined by the data around + * it), this should return null. * * The implications of returning null here is that this quest will never be created directly * as consequence of solving another quest and also after reverting an input, the quest will diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementUpdateController.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementUpdateController.kt new file mode 100644 index 0000000000..352ec11934 --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementUpdateController.kt @@ -0,0 +1,85 @@ +package de.westnordost.streetcomplete.data.osm.osmquest + +import de.westnordost.osmapi.common.errors.OsmNotFoundException +import de.westnordost.osmapi.map.MapData +import de.westnordost.osmapi.map.data.Element +import de.westnordost.osmapi.map.data.Node +import de.westnordost.osmapi.map.data.Relation +import de.westnordost.osmapi.map.data.Way +import de.westnordost.osmapi.map.getRelationComplete +import de.westnordost.osmapi.map.getWayComplete +import de.westnordost.streetcomplete.data.MapDataApi +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryCreator +import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import javax.inject.Inject + +/** When an element has been updated or deleted (from the API), this class takes care of updating + * the element and the data that is dependent on the element - the quests */ +class OsmElementUpdateController @Inject constructor( + private val mapDataApi: MapDataApi, + private val elementGeometryCreator: ElementGeometryCreator, + private val elementDB: MergedElementDao, + private val questGiver: OsmQuestGiver, +){ + + /** The [element] has been updated. Persist that, determine its geometry and update the quests + * based on that element. If [recreateQuestTypes] is not null, always (re)create the given + * quest types on the element without checking for its eligibility */ + fun update(element: Element, recreateQuestTypes: List>?) { + val newGeometry = createGeometry(element) + if (newGeometry != null) { + elementDB.put(element) + + if (recreateQuestTypes == null) { + questGiver.updateQuests(element, newGeometry) + } else { + questGiver.recreateQuests(element, newGeometry, recreateQuestTypes) + } + } else { + // new element has invalid geometry + delete(element.type, element.id) + } + } + + fun delete(elementType: Element.Type, elementId: Long) { + elementDB.delete(elementType, elementId) + // geometry is deleted by the osmQuestController + questGiver.deleteQuests(elementType, elementId) + } + + fun get(elementType: Element.Type, elementId: Long): Element? { + return elementDB.get(elementType, elementId) + } + + fun cleanUp() { + elementDB.deleteUnreferenced() + } + + private fun createGeometry(element: Element): ElementGeometry? { + when(element) { + is Node -> { + return elementGeometryCreator.create(element) + } + is Way -> { + val mapData: MapData + try { + mapData = mapDataApi.getWayComplete(element.id) + } catch (e: OsmNotFoundException) { + return null + } + return elementGeometryCreator.create(element, mapData) + } + is Relation -> { + val mapData: MapData + try { + mapData = mapDataApi.getRelationComplete(element.id) + } catch (e: OsmNotFoundException) { + return null + } + return elementGeometryCreator.create(element, mapData) + } + else -> return null + } + } +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmFilterQuestType.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmFilterQuestType.kt new file mode 100644 index 0000000000..cc577d637c --- /dev/null +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmFilterQuestType.kt @@ -0,0 +1,27 @@ +package de.westnordost.streetcomplete.data.osm.osmquest + +import de.westnordost.osmapi.map.MapDataWithGeometry +import de.westnordost.osmapi.map.data.Element +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression +import de.westnordost.streetcomplete.util.MultiIterable + +/** Quest type that's based on a simple element filter expression */ +abstract class OsmFilterQuestType : OsmElementQuestType { + + val filter by lazy { elementFilter.toElementFilterExpression() } + + protected abstract val elementFilter: String + + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + /* this is a considerate performance improvement over just iterating over the whole MapData + * because for quests that only filter for one (or two) element types, any filter checks + * are completely avoided */ + val iterable = MultiIterable() + if (filter.includesElementType(Element.Type.NODE)) iterable.add(mapData.nodes) + if (filter.includesElementType(Element.Type.WAY)) iterable.add(mapData.ways) + if (filter.includesElementType(Element.Type.RELATION)) iterable.add(mapData.relations) + return iterable.filter { element -> filter.matches(element) } + } + + override fun isApplicableTo(element: Element) = filter.matches(element) +} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuest.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuest.kt index 5f4f68a190..2225efd40f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuest.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuest.kt @@ -18,15 +18,15 @@ import de.westnordost.streetcomplete.util.pointOnPolylineFromStart /** Represents one task for the user to complete/correct the data based on one OSM element */ data class OsmQuest( - override var id: Long?, - override val osmElementQuestType: OsmElementQuestType<*>, // underlying OSM data - override val elementType: Element.Type, - override val elementId: Long, - override var status: QuestStatus, - override var changes: StringMapChanges?, - var changesSource: String?, - override var lastUpdate: Date, - override val geometry: ElementGeometry + override var id: Long?, + override val osmElementQuestType: OsmElementQuestType<*>, // underlying OSM data + override val elementType: Element.Type, + override val elementId: Long, + override var status: QuestStatus, + override var changes: StringMapChanges?, + var changesSource: String?, + override var lastUpdate: Date, + override val geometry: ElementGeometry ) : Quest, UploadableInChangeset, HasElementTagChanges { constructor(type: OsmElementQuestType<*>, elementType: Element.Type, elementId: Long, geometry: ElementGeometry) diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestController.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestController.kt index 7d0063605b..9f087767f4 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestController.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestController.kt @@ -4,6 +4,7 @@ import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.ApplicationConstants import de.westnordost.streetcomplete.data.osm.changes.StringMapChanges +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryEntry import de.westnordost.streetcomplete.data.osm.mapdata.ElementKey @@ -92,14 +93,14 @@ import javax.inject.Singleton /** Replace all quests of the given type in the given bounding box with the given quests, * including their geometry. Called on download of a quest type for a bounding box. */ - fun replaceInBBox(quests: List, bbox: BoundingBox, questType: String): UpdateResult { + fun replaceInBBox(quests: List, bbox: BoundingBox, questTypes: List): UpdateResult { + require(questTypes.isNotEmpty()) { "questTypes must not be empty if not null" } /* All quests in the given bounding box and of the given type should be replaced by the * input list. So, there may be 1. new quests that are added and 2. there may be previous * quests that have been there before but now not anymore, these need to be removed. */ - val previousQuestIdsByElement = dao.getAll( bounds = bbox, - questTypes = listOf(questType) + questTypes = questTypes ).associate { ElementKey(it.elementType, it.elementId) to it.id!! }.toMutableMap() val addedQuests = mutableListOf() @@ -130,7 +131,14 @@ import javax.inject.Singleton /** Add new unanswered quests and remove others for the given element, including their linked * geometry. Called when an OSM element is updated, so the quests that reference that element * need to be updated as well. */ - fun updateForElement(added: List, removedIds: List, elementType: Element.Type, elementId: Long): UpdateResult { + fun updateForElement( + added: List, + removedIds: List, + updatedGeometry: ElementGeometry, + elementType: Element.Type, + elementId: Long + ): UpdateResult { + geometryDao.put(ElementGeometryEntry(elementType, elementId, updatedGeometry)) val e = ElementKey(elementType, elementId) var deletedCount = removeObsolete(removedIds) @@ -229,21 +237,24 @@ import javax.inject.Singleton } - /** Get count of all unanswered quests in given bounding box of given types */ - fun getAllVisibleInBBoxCount(bbox: BoundingBox, questTypes: Collection) : Int = - dao.getCount( + /** Get count of all unanswered quests in given bounding box */ + fun getAllVisibleInBBoxCount(bbox: BoundingBox) : Int { + return dao.getCount( statusIn = listOf(QuestStatus.NEW), - bounds = bbox, - questTypes = questTypes + bounds = bbox ) + } /** Get all unanswered quests in given bounding box of given types */ - fun getAllVisibleInBBox(bbox: BoundingBox, questTypes: Collection): List = - dao.getAll( + fun getAllVisibleInBBox(bbox: BoundingBox, questTypes: Collection): List { + if (questTypes.isEmpty()) return listOf() + return dao.getAll( statusIn = listOf(QuestStatus.NEW), bounds = bbox, questTypes = questTypes ) + } + /** Get single quest by id */ fun get(id: Long): OsmQuest? = dao.get(id) diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDao.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDao.kt index 5b15dbc964..0ad77a3cdb 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDao.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDao.kt @@ -149,7 +149,8 @@ private fun createQuery( add("$ELEMENT_TYPE = ?", element.elementType.name) add("$ELEMENT_ID = ?", element.elementId.toString()) } - if (statusIn != null && statusIn.isNotEmpty()) { + if (statusIn != null) { + require(statusIn.isNotEmpty()) { "statusIn must not be empty if not null" } if (statusIn.size == 1) { add("$QUEST_STATUS = ?", statusIn.single().name) } else { @@ -157,7 +158,8 @@ private fun createQuery( add("$QUEST_STATUS IN ($names)") } } - if (questTypes != null && questTypes.isNotEmpty()) { + if (questTypes != null) { + require(questTypes.isNotEmpty()) { "questTypes must not be empty if not null" } if (questTypes.size == 1) { add("$QUEST_TYPE = ?", questTypes.single()) } else { diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDownloader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDownloader.kt deleted file mode 100644 index 379f75da0e..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDownloader.kt +++ /dev/null @@ -1,126 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.osmquest - -import android.util.Log - -import java.util.Locale -import java.util.concurrent.FutureTask - -import javax.inject.Inject - -import de.westnordost.countryboundaries.CountryBoundaries -import de.westnordost.countryboundaries.intersects -import de.westnordost.countryboundaries.isInAny -import de.westnordost.osmapi.map.data.OsmLatLon -import de.westnordost.streetcomplete.data.quest.QuestType -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao -import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.osmapi.map.data.Element -import de.westnordost.osmapi.map.data.LatLon -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry -import de.westnordost.streetcomplete.util.measuredLength - -/** Takes care of downloading one quest type in a bounding box and persisting the downloaded quests */ -class OsmQuestDownloader @Inject constructor( - private val elementDB: MergedElementDao, - private val osmQuestController: OsmQuestController, - private val countryBoundariesFuture: FutureTask -) { - private val countryBoundaries: CountryBoundaries get() = countryBoundariesFuture.get() - - fun download(questType: OsmElementQuestType<*>, bbox: BoundingBox, blacklistedPositions: Set): Boolean { - val questTypeName = questType.getName() - - val countries = questType.enabledInCountries - if (!countryBoundaries.intersects(bbox, countries)) { - Log.i(TAG, "$questTypeName: Skipped because it is disabled for this country") - return true - } - - val elements = mutableListOf() - val quests = ArrayList() - val truncatedBlacklistedPositions = blacklistedPositions.map { it.truncateTo5Decimals() }.toSet() - - val time = System.currentTimeMillis() - val success = questType.download(bbox) { element, geometry -> - if (mayCreateQuestFrom(questType, element, geometry, truncatedBlacklistedPositions)) { - val quest = OsmQuest(questType, element.type, element.id, geometry!!) - - quests.add(quest) - elements.add(element) - } - } - if (!success) return false - - // elements must be put into DB first because quests have foreign keys on it - elementDB.putAll(elements) - - val replaceResult = osmQuestController.replaceInBBox(quests, bbox, questTypeName) - - // note: this could be done after ALL osm quest types have been downloaded if this - // turns out to be slow if done for every quest type - elementDB.deleteUnreferenced() - questType.cleanMetadata() - - val secondsSpent = (System.currentTimeMillis() - time) / 1000 - Log.i(TAG,"$questTypeName: Added ${replaceResult.added} new and removed ${replaceResult.deleted} already resolved quests (total: ${quests.size}) in ${secondsSpent}s") - - return true - } - - private fun mayCreateQuestFrom( - questType: OsmElementQuestType<*>, element: Element, geometry: ElementGeometry?, - blacklistedPositions: Set - ): Boolean { - val questTypeName = questType.getName() - - // invalid geometry -> can't show this quest, so skip it - if (geometry == null) { - // classified as warning because it might very well be a bug on the geometry creation on our side - Log.w(TAG, "$questTypeName: Not adding a quest because the element ${element.toLogString()} has no valid geometry") - return false - } - val pos = geometry.center - - // do not create quests that refer to geometry that is too long for a surveyor to be expected to survey - if (geometry is ElementPolylinesGeometry) { - val totalLength = geometry.polylines.sumByDouble { it.measuredLength() } - if (totalLength > MAX_GEOMETRY_LENGTH_IN_METERS) { - Log.d(TAG, "$questTypeName: Not adding a quest for ${element.toLogString()} at ${pos.toLogString()} because the geometry is too long") - return false - } - } - - // do not create quests whose marker is at/near a blacklisted position - if (blacklistedPositions.contains(pos.truncateTo5Decimals())) { - Log.d(TAG, "$questTypeName: Not adding a quest for ${element.toLogString()} at ${pos.toLogString()} because there is a note at that position") - return false - } - - // do not create quests in countries where the quest is not activated - val countries = questType.enabledInCountries - if (!countryBoundaries.isInAny(pos, countries)) { - Log.d(TAG, "$questTypeName: Not adding a quest for ${element.toLogString()} at ${pos.toLogString()} because the quest is disabled in this country") - return false - } - - return true - } - - companion object { - private const val TAG = "QuestDownload" - } -} - -const val MAX_GEOMETRY_LENGTH_IN_METERS = 500 - -private fun QuestType<*>.getName() = javaClass.simpleName - -// the resulting precision is about ~1 meter (see #1089) -private fun LatLon.truncateTo5Decimals() = OsmLatLon(latitude.truncateTo5Decimals(), longitude.truncateTo5Decimals()) - -private fun Double.truncateTo5Decimals() = (this * 1e5).toInt().toDouble() / 1e5 - -private fun Element.toLogString() = "${type.name.toLowerCase(Locale.US)} #$id" - -private fun LatLon.toLogString() = "$latitude, $longitude" diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiver.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiver.kt index 6567281835..d7b8a0f647 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiver.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiver.kt @@ -34,7 +34,7 @@ class OsmQuestGiver @Inject constructor( createdQuests.add(quest) createdQuestsLog.add(questType.javaClass.simpleName) } - val updates = osmQuestController.updateForElement(createdQuests, emptyList(), element.type, element.id) + val updates = osmQuestController.updateForElement(createdQuests, emptyList(), geometry, element.type, element.id) Log.d(TAG, "Recreated ${updates.added} quests for ${element.type.name}#${element.id}: ${createdQuestsLog.joinToString()}") } @@ -73,7 +73,7 @@ class OsmQuestGiver @Inject constructor( } } } - val updates = osmQuestController.updateForElement(createdQuests, removedQuestIds, element.type, element.id) + val updates = osmQuestController.updateForElement(createdQuests, removedQuestIds, geometry, element.type, element.id) if (updates.added > 0) { Log.d(TAG, "Created ${updates.added} new quests for ${element.type.name}#${element.id}: ${createdQuestsLog.joinToString()}") diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploader.kt index b9af99d69c..4583928dc5 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploader.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploader.kt @@ -4,9 +4,6 @@ import android.util.Log import de.westnordost.osmapi.map.data.Element import javax.inject.Inject -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.osm.upload.OsmInChangesetsUploader import de.westnordost.streetcomplete.data.user.StatisticsUpdater @@ -15,16 +12,12 @@ import java.util.concurrent.atomic.AtomicBoolean /** Gets all answered osm quests from local DB and uploads them via the OSM API */ class OsmQuestsUploader @Inject constructor( - elementDB: MergedElementDao, - elementGeometryDB: ElementGeometryDao, - changesetManager: OpenQuestChangesetsManager, - questGiver: OsmQuestGiver, - osmApiElementGeometryCreator: OsmApiElementGeometryCreator, - private val osmQuestController: OsmQuestController, - private val singleChangeUploader: SingleOsmElementTagChangesUploader, - private val statisticsUpdater: StatisticsUpdater -) : OsmInChangesetsUploader(elementDB, elementGeometryDB, changesetManager, questGiver, - osmApiElementGeometryCreator) { + changesetManager: OpenQuestChangesetsManager, + elementUpdateController: OsmElementUpdateController, + private val osmQuestController: OsmQuestController, + private val singleChangeUploader: SingleOsmElementTagChangesUploader, + private val statisticsUpdater: StatisticsUpdater +) : OsmInChangesetsUploader(changesetManager, elementUpdateController) { @Synchronized override fun upload(cancelled: AtomicBoolean) { Log.i(TAG, "Applying quest changes") diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/SimpleOverpassQuestType.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/SimpleOverpassQuestType.kt deleted file mode 100644 index 6273671a9e..0000000000 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/SimpleOverpassQuestType.kt +++ /dev/null @@ -1,29 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.osmquest - -import de.westnordost.osmapi.map.data.Element -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox - -/** Quest type that simply makes a certain overpass query using tag filters and creates quests for - * every element received */ -abstract class SimpleOverpassQuestType( - private val overpassApi: OverpassMapDataAndGeometryApi -) : OsmElementQuestType { - - private val filter by lazy { ElementFiltersParser().parse(tagFilters) } - - abstract val tagFilters: String - - fun getOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + filter.toOverpassQLString() + getQuestPrintStatement() - - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox), handler) - } - - override fun isApplicableTo(element: Element) = filter.matches(element) -} diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploader.kt index cfd19364df..482f486bb8 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploader.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploader.kt @@ -2,12 +2,9 @@ package de.westnordost.streetcomplete.data.osm.osmquest.undo import android.util.Log import de.westnordost.osmapi.map.data.Element -import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestGiver -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao import javax.inject.Inject -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementUpdateController import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.osm.upload.OsmInChangesetsUploader import de.westnordost.streetcomplete.data.osm.osmquest.SingleOsmElementTagChangesUploader @@ -17,16 +14,12 @@ import java.util.concurrent.atomic.AtomicBoolean /** Gets all undo osm quests from local DB and uploads them via the OSM API */ class UndoOsmQuestsUploader @Inject constructor( - elementDB: MergedElementDao, - elementGeometryDB: ElementGeometryDao, - changesetManager: OpenQuestChangesetsManager, - questGiver: OsmQuestGiver, - osmApiElementGeometryCreator: OsmApiElementGeometryCreator, - private val undoQuestDB: UndoOsmQuestDao, - private val singleChangeUploader: SingleOsmElementTagChangesUploader, - private val statisticsUpdater: StatisticsUpdater -) : OsmInChangesetsUploader(elementDB, elementGeometryDB, changesetManager, questGiver, - osmApiElementGeometryCreator) { + changesetManager: OpenQuestChangesetsManager, + elementUpdateController: OsmElementUpdateController, + private val undoQuestDB: UndoOsmQuestDao, + private val singleChangeUploader: SingleOsmElementTagChangesUploader, + private val statisticsUpdater: StatisticsUpdater +) : OsmInChangesetsUploader(changesetManager, elementUpdateController) { @Synchronized override fun upload(cancelled: AtomicBoolean) { Log.i(TAG, "Undoing quest changes") diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploader.kt index accbb565e1..781f6d03a0 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploader.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploader.kt @@ -3,11 +3,7 @@ package de.westnordost.streetcomplete.data.osm.splitway import android.util.Log import de.westnordost.osmapi.map.data.Element import de.westnordost.osmapi.map.data.Way -import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestGiver -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryEntry -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementUpdateController import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.osm.upload.OsmInChangesetsUploader import de.westnordost.streetcomplete.data.user.StatisticsUpdater @@ -16,16 +12,12 @@ import javax.inject.Inject /** Gets all split ways from local DB and uploads them via the OSM API */ class SplitWaysUploader @Inject constructor( - private val elementDB: MergedElementDao, - private val elementGeometryDB: ElementGeometryDao, - changesetManager: OpenQuestChangesetsManager, - private val questGiver: OsmQuestGiver, - private val osmApiElementGeometryCreator: OsmApiElementGeometryCreator, - private val splitWayDB: OsmQuestSplitWayDao, - private val splitSingleOsmWayUploader: SplitSingleWayUploader, - private val statisticsUpdater: StatisticsUpdater -) : OsmInChangesetsUploader(elementDB, elementGeometryDB, changesetManager, - questGiver, osmApiElementGeometryCreator) { + changesetManager: OpenQuestChangesetsManager, + private val elementUpdateController: OsmElementUpdateController, + private val splitWayDB: OsmQuestSplitWayDao, + private val splitSingleOsmWayUploader: SplitSingleWayUploader, + private val statisticsUpdater: StatisticsUpdater +) : OsmInChangesetsUploader(changesetManager, elementUpdateController) { @Synchronized override fun upload(cancelled: AtomicBoolean) { Log.i(TAG, "Splitting ways") @@ -38,17 +30,11 @@ class SplitWaysUploader @Inject constructor( return splitSingleOsmWayUploader.upload(changesetId, element as Way, quest.splits) } - override fun updateElement(newElement: Element, quest: OsmQuestSplitWay) { - val geometry = osmApiElementGeometryCreator.create(newElement) - if (geometry != null) { - elementGeometryDB.put(ElementGeometryEntry(newElement.type, newElement.id, geometry)) - elementDB.put(newElement) - questGiver.recreateQuests(newElement, geometry, quest.questTypesOnWay) - } else { - // new element has invalid geometry - elementDB.delete(newElement.type, newElement.id) - questGiver.deleteQuests(newElement.type, newElement.id) - } + override fun updateElement(element: Element, quest: OsmQuestSplitWay) { + /* We override this because in case of a split, the two (or more) sections of the way should + * actually get the same quests as the original way, there is no need to again check for + * the eligibility of the element for each quest which would be done normally */ + elementUpdateController.update(element, quest.questTypesOnWay) } override fun onUploadSuccessful(quest: OsmQuestSplitWay) { diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osm/upload/OsmInChangesetsUploader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osm/upload/OsmInChangesetsUploader.kt index 2e35b4f881..31d5251544 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osm/upload/OsmInChangesetsUploader.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osm/upload/OsmInChangesetsUploader.kt @@ -4,11 +4,7 @@ import androidx.annotation.CallSuper import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.data.quest.QuestType import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType -import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestGiver -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryEntry -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementUpdateController import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.upload.OnUploadedChangeListener import de.westnordost.streetcomplete.data.upload.Uploader @@ -18,12 +14,9 @@ import java.util.concurrent.atomic.AtomicBoolean /** Base class for all uploaders which upload OSM data. They all have in common that they handle * OSM data (of course), and that the data is uploaded in changesets. */ abstract class OsmInChangesetsUploader( - private val elementDB: MergedElementDao, - private val elementGeometryDB: ElementGeometryDao, - private val changesetManager: OpenQuestChangesetsManager, - private val questGiver: OsmQuestGiver, - private val osmApiElementGeometryCreator: OsmApiElementGeometryCreator - ): Uploader { + private val changesetManager: OpenQuestChangesetsManager, + private val elementUpdateController: OsmElementUpdateController +): Uploader { override var uploadedChangeListener: OnUploadedChangeListener? = null @@ -44,7 +37,7 @@ abstract class OsmInChangesetsUploader( onUploadSuccessful(quest) uploadedChangeListener?.onUploaded(quest.osmElementQuestType.name, quest.position) } catch (e: ElementIncompatibleException) { - deleteElement(quest.elementType, quest.elementId) + elementUpdateController.delete(quest.elementType, quest.elementId) onUploadFailed(quest, e) uploadedChangeListener?.onDiscarded(quest.osmElementQuestType.name, quest.position) } catch (e: ElementConflictException) { @@ -55,8 +48,12 @@ abstract class OsmInChangesetsUploader( cleanUp(uploadedQuestTypes) } + protected open fun updateElement(element: Element, quest: T) { + elementUpdateController.update(element, null) + } + private fun uploadSingle(quest: T) : List { - val element = elementDB.get(quest.elementType, quest.elementId) + val element = elementUpdateController.get(quest.elementType, quest.elementId) ?: throw ElementDeletedException("Element deleted") return try { @@ -68,31 +65,8 @@ abstract class OsmInChangesetsUploader( } } - /* TODO REFACTOR: It shouldn't be the duty of OsmInChangesetsUploader to do (or delegate) the - * work necessary that entails when updating an element (grant/remove quests, - * update element geometry). Instead, there should be an observer on the - * ElementDao (or a controller in front of it) that takes care of that. - * - * This will remove the dependencies to elementGeometryDB, questGiver etc */ - protected open fun updateElement(newElement: Element, quest: T) { - val geometry = osmApiElementGeometryCreator.create(newElement) - if (geometry != null) { - elementGeometryDB.put(ElementGeometryEntry(newElement.type, newElement.id, geometry)) - elementDB.put(newElement) - questGiver.updateQuests(newElement, geometry) - } else { - // new element has invalid geometry - deleteElement(newElement.type, newElement.id) - } - } - - private fun deleteElement(elementType: Element.Type, elementId: Long) { - elementDB.delete(elementType, elementId) - questGiver.deleteQuests(elementType, elementId) - } - @CallSuper protected open fun cleanUp(questTypes: Set>) { - elementDB.deleteUnreferenced() + elementUpdateController.cleanUp() // must be after unreferenced elements have been deleted for (questType in questTypes) { questType.cleanMetadata() diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/osmnotes/notequests/OsmNoteQuestDao.kt b/app/src/main/java/de/westnordost/streetcomplete/data/osmnotes/notequests/OsmNoteQuestDao.kt index c763982afb..8074d3d7ac 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/osmnotes/notequests/OsmNoteQuestDao.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/osmnotes/notequests/OsmNoteQuestDao.kt @@ -137,7 +137,8 @@ private fun createQuery( bounds: BoundingBox? = null, changedBefore: Long? = null ) = WhereSelectionBuilder().apply { - if (statusIn != null && statusIn.isNotEmpty()) { + if (statusIn != null) { + require(statusIn.isNotEmpty()) { "statusIn must not be empty if not null" } if (statusIn.size == 1) { add("$QUEST_STATUS = ?", statusIn.single().name) } else { diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestAutoSyncer.kt b/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestAutoSyncer.kt index bb6b937763..e549b1e873 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestAutoSyncer.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestAutoSyncer.kt @@ -33,9 +33,8 @@ import javax.inject.Singleton private val mobileDataDownloadStrategy: MobileDataAutoDownloadStrategy, private val wifiDownloadStrategy: WifiAutoDownloadStrategy, private val context: Context, - private val visibleQuestsSource: VisibleQuestsSource, private val unsyncedChangesCountSource: UnsyncedChangesCountSource, - private val downloadProgressSource: QuestDownloadProgressSource, + private val downloadProgressSource: DownloadProgressSource, private val prefs: SharedPreferences, private val userController: UserController ) : LifecycleObserver, CoroutineScope by CoroutineScope(Dispatchers.Default) { @@ -45,13 +44,6 @@ import javax.inject.Singleton private var isConnected: Boolean = false private var isWifi: Boolean = false - // amount of visible quests is reduced -> check if re-downloading makes sense now - private val visibleQuestListener = object : VisibleQuestListener { - override fun onUpdatedVisibleQuests(added: Collection, removed: Collection, group: QuestGroup) { - if (removed.isNotEmpty()) { triggerAutoDownload() } - } - } - // new location is known -> check if downloading makes sense now private val locationManager = FineLocationManager(context.getSystemService()!!) { location -> pos = OsmLatLon(location.latitude, location.longitude) @@ -78,7 +70,7 @@ import javax.inject.Singleton } // on download finished, should recheck conditions for download - private val downloadProgressListener = object : QuestDownloadProgressListener { + private val downloadProgressListener = object : DownloadProgressListener { override fun onSuccess() { triggerAutoDownload() } @@ -93,9 +85,8 @@ import javax.inject.Singleton /* ---------------------------------------- Lifecycle --------------------------------------- */ init { - visibleQuestsSource.addListener(visibleQuestListener) unsyncedChangesCountSource.addListener(unsyncedChangesListener) - downloadProgressSource.addQuestDownloadProgressListener(downloadProgressListener) + downloadProgressSource.addDownloadProgressListener(downloadProgressListener) } @OnLifecycleEvent(Lifecycle.Event.ON_RESUME) fun onResume() { @@ -113,9 +104,8 @@ import javax.inject.Singleton } @OnLifecycleEvent(Lifecycle.Event.ON_DESTROY) fun onDestroy() { - visibleQuestsSource.removeListener(visibleQuestListener) unsyncedChangesCountSource.removeListener(unsyncedChangesListener) - downloadProgressSource.removeQuestDownloadProgressListener(downloadProgressListener) + downloadProgressSource.removeDownloadProgressListener(downloadProgressListener) coroutineContext.cancel() } @@ -131,7 +121,6 @@ import javax.inject.Singleton /* ------------------------------------------------------------------------------------------ */ fun triggerAutoDownload() { - if (!isAllowedByPreference) return val pos = pos ?: return if (!isConnected) return if (questDownloadController.isDownloadInProgress) return @@ -140,12 +129,10 @@ import javax.inject.Singleton launch { val downloadStrategy = if (isWifi) wifiDownloadStrategy else mobileDataDownloadStrategy - if (downloadStrategy.mayDownloadHere(pos)) { + val downloadBoundingBox = downloadStrategy.getDownloadBoundingBox(pos) + if (downloadBoundingBox != null) { try { - questDownloadController.download( - downloadStrategy.getDownloadBoundingBox(pos), - downloadStrategy.questTypeDownloadCount - ) + questDownloadController.download(downloadBoundingBox) } catch (e: IllegalStateException) { // The Android 9 bug described here should not result in a hard crash of the app // https://stackoverflow.com/questions/52013545/android-9-0-not-allowed-to-start-service-app-is-in-background-after-onresume diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestStatus.kt b/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestStatus.kt index a2cfcccea9..2bcf8d1552 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestStatus.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/quest/QuestStatus.kt @@ -10,9 +10,7 @@ enum class QuestStatus { /** the system (decided that it) doesn't show the quest. They may become visible again (-> NEW) */ INVISIBLE, /** the quest has been uploaded (either solved or dropped through conflict). The app needs to - * remember its solved quests for some time before deleting them because the source the app - * is pulling it's data for creating quests from (usually Overpass) lags behind the database - * where the app is uploading its changes to. + * remember its solved quests for some time before deleting them so that they can be reverted * Note quests are generally closed after upload, they are never deleted */ CLOSED, /** the quest has been closed and after that the user chose to revert (aka undo) it. This state diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSource.kt b/app/src/main/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSource.kt index 3fcea7b8c4..aac7b5ffe6 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSource.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSource.kt @@ -68,15 +68,15 @@ import javax.inject.Singleton } - /** Get count of all unanswered quests in given bounding box of given types */ - fun getAllVisibleCount(bbox: BoundingBox, questTypes: Collection): Int { - if (questTypes.isEmpty()) return 0 - return osmQuestController.getAllVisibleInBBoxCount(bbox, questTypes) + + /** Get count of all unanswered quests in given bounding box */ + fun getAllVisibleCount(bbox: BoundingBox): Int { + return osmQuestController.getAllVisibleInBBoxCount(bbox) + osmNoteQuestController.getAllVisibleInBBoxCount(bbox) } /** Retrieve all visible (=new) quests in the given bounding box from local database */ fun getAllVisible(bbox: BoundingBox, questTypes: Collection): List { + if (questTypes.isEmpty()) return listOf() val osmQuests = osmQuestController.getAllVisibleInBBox(bbox, questTypes) val osmNoteQuests = osmNoteQuestController.getAllVisibleInBBox(bbox) diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/user/StatisticsDownloader.kt b/app/src/main/java/de/westnordost/streetcomplete/data/user/StatisticsDownloader.kt index 1f508e05bb..b094ff8114 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/user/StatisticsDownloader.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/user/StatisticsDownloader.kt @@ -1,16 +1,17 @@ package de.westnordost.streetcomplete.data.user -import de.westnordost.osmapi.common.Iso8601CompatibleDateFormat import de.westnordost.streetcomplete.ApplicationConstants import org.json.JSONObject import java.io.IOException import java.net.HttpURLConnection import java.net.URL +import java.text.SimpleDateFormat +import java.util.* /** Downloads statistics from the backend */ class StatisticsDownloader(private val baseUrl: String) { - private val lastActivityDateFormat = Iso8601CompatibleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ") + private val lastActivityDateFormat = SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ", Locale.US) fun download(osmUserId: Long): Statistics { (URL("$baseUrl?user_id=$osmUserId").openConnection() as HttpURLConnection).run { diff --git a/app/src/main/java/de/westnordost/streetcomplete/data/user/UserStore.kt b/app/src/main/java/de/westnordost/streetcomplete/data/user/UserStore.kt index 822649b9a5..0f6bfce70f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/data/user/UserStore.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/data/user/UserStore.kt @@ -2,9 +2,9 @@ package de.westnordost.streetcomplete.data.user import android.content.SharedPreferences import androidx.core.content.edit -import de.westnordost.osmapi.common.Iso8601CompatibleDateFormat import de.westnordost.osmapi.user.UserDetails import de.westnordost.streetcomplete.Prefs +import java.text.SimpleDateFormat import java.util.* import java.util.concurrent.CopyOnWriteArrayList import javax.inject.Inject @@ -21,7 +21,7 @@ import javax.inject.Singleton } private val listeners: MutableList = CopyOnWriteArrayList() - private val dateFormat = Iso8601CompatibleDateFormat("yyyy-MM-dd HH:mm:ss z") + private val dateFormat = SimpleDateFormat("yyyy-MM-dd HH:mm:ss z", Locale.US) val userId: Long get() = prefs.getLong(Prefs.OSM_USER_ID, -1) val userName: String? get() = prefs.getString(Prefs.OSM_USER_NAME, null) diff --git a/app/src/main/java/de/westnordost/streetcomplete/ktx/Collections.kt b/app/src/main/java/de/westnordost/streetcomplete/ktx/Collections.kt index 054f1c10a3..5992d36232 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/ktx/Collections.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/ktx/Collections.kt @@ -1,5 +1,7 @@ package de.westnordost.streetcomplete.ktx +import de.westnordost.osmapi.map.data.LatLon + /** Return the first and last element of this list. If it contains only one element, just that one */ fun List.firstAndLast() = if (size == 1) listOf(first()) else listOf(first(), last()) @@ -32,8 +34,8 @@ inline fun List.findNext(index: Int, predicate: (T) -> Boolean): T? { return null } -/** Iterate through the given list in pairs (advancing each one item, not two) */ -inline fun Iterable.forEachPair(predicate: (first: T, second: T) -> Unit) { +/** Iterate through the given list of points in pairs, so [predicate] is called for every line */ +inline fun Iterable.forEachLine(predicate: (first: LatLon, second: LatLon) -> Unit) { val it = iterator() if (!it.hasNext()) return var item1 = it.next() diff --git a/app/src/main/java/de/westnordost/streetcomplete/ktx/Element.kt b/app/src/main/java/de/westnordost/streetcomplete/ktx/Element.kt index 6f28b47d5f..f0fc70d796 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/ktx/Element.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/ktx/Element.kt @@ -1,7 +1,7 @@ package de.westnordost.streetcomplete.ktx import de.westnordost.osmapi.map.data.* -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import java.util.ArrayList fun Element.copy(newId: Long = id, newVersion: Int = version): Element { @@ -24,7 +24,7 @@ fun Element.isArea(): Boolean { } } -private val IS_AREA_EXPR = ElementFiltersParser().parse(""" +private val IS_AREA_EXPR = """ ways with area = yes or area != no and ( aeroway or amenity @@ -49,4 +49,4 @@ private val IS_AREA_EXPR = ElementFiltersParser().parse(""" or cemetery ~ sector|grave or natural ~ wood|scrub|heath|moor|grassland|fell|bare_rock|scree|shingle|sand|mud|water|wetland|glacier|beach|rock|sinkhole or man_made ~ beacon|bridge|campanile|dolphin|lighthouse|obelisk|observatory|tower|bunker_silo|chimney|gasometer|kiln|mineshaft|petroleum_well|silo|storage_tank|watermill|windmill|works|communications_tower|monitoring_station|street_cabinet|pumping_station|reservoir_covered|wastewater_plant|water_tank|water_tower|water_well|water_works - )""") + )""".toElementFilterExpression() diff --git a/app/src/main/java/de/westnordost/streetcomplete/map/MainFragment.kt b/app/src/main/java/de/westnordost/streetcomplete/map/MainFragment.kt index e5e46f79f0..3269254f8f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/map/MainFragment.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/map/MainFragment.kt @@ -71,6 +71,7 @@ import javax.inject.Inject import kotlin.math.PI import kotlin.math.cos import kotlin.math.sin +import kotlin.math.sqrt /** Contains the quests map and the controls for it. */ class MainFragment : Fragment(R.layout.fragment_main), @@ -612,7 +613,7 @@ class MainFragment : Fragment(R.layout.fragment_main), context?.toast(R.string.cannot_find_bbox_or_reduce_tilt, Toast.LENGTH_LONG) } else { val enclosingBBox = displayArea.asBoundingBoxOfEnclosingTiles(ApplicationConstants.QUEST_TILE_ZOOM) - val areaInSqKm = enclosingBBox.area(EARTH_RADIUS) / 1000000 + val areaInSqKm = enclosingBBox.area() / 1000000 if (areaInSqKm > ApplicationConstants.MAX_DOWNLOADABLE_AREA_IN_SQKM) { context?.toast(R.string.download_area_too_big, Toast.LENGTH_LONG) } else { @@ -635,18 +636,16 @@ class MainFragment : Fragment(R.layout.fragment_main), private fun downloadAreaConfirmed(bbox: BoundingBox) { var bbox = bbox - val areaInSqKm = bbox.area(EARTH_RADIUS) / 1000000 + val areaInSqKm = bbox.area() / 1000000 // below a certain threshold, it does not make sense to download, so let's enlarge it if (areaInSqKm < ApplicationConstants.MIN_DOWNLOADABLE_AREA_IN_SQKM) { val cameraPosition = mapFragment?.cameraPosition if (cameraPosition != null) { - bbox = cameraPosition.position.enclosingBoundingBox( - ApplicationConstants.MIN_DOWNLOADABLE_RADIUS_IN_METERS, - EARTH_RADIUS - ) + val radius = sqrt( 1000000 * ApplicationConstants.MIN_DOWNLOADABLE_AREA_IN_SQKM / PI) + bbox = cameraPosition.position.enclosingBoundingBox(radius) } } - questDownloadController.download(bbox, ApplicationConstants.MANUAL_DOWNLOAD_QUEST_TYPE_COUNT, true) + questDownloadController.download(bbox, true) } // ---------------------------------- Location Pointer Pin --------------------------------- */ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/QuestModule.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/QuestModule.kt index 07acc577f0..48591d7785 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/QuestModule.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/QuestModule.kt @@ -3,7 +3,6 @@ package de.westnordost.streetcomplete.quests import dagger.Module import dagger.Provides import de.westnordost.osmfeatures.FeatureDictionary -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.osmnotes.notequests.OsmNoteQuestType import de.westnordost.streetcomplete.data.quest.QuestTypeRegistry import de.westnordost.streetcomplete.quests.accepts_cash.AddAcceptsCash @@ -91,18 +90,14 @@ import de.westnordost.streetcomplete.quests.traffic_signals_vibrate.AddTrafficSi import de.westnordost.streetcomplete.quests.traffic_signals_sound.AddTrafficSignalsSound import de.westnordost.streetcomplete.quests.way_lit.AddWayLit import de.westnordost.streetcomplete.quests.wheelchair_access.* -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore import java.util.concurrent.FutureTask import javax.inject.Singleton -@Module -object QuestModule +@Module object QuestModule { @Provides @Singleton fun questTypeRegistry( osmNoteQuestType: OsmNoteQuestType, - o: OverpassMapDataAndGeometryApi, - r: ResurveyIntervalsStore, roadNameSuggestionsDao: RoadNameSuggestionsDao, trafficFlowSegmentsApi: TrafficFlowSegmentsApi, trafficFlowDao: WayTrafficFlowDao, featureDictionaryFuture: FutureTask @@ -112,108 +107,108 @@ object QuestModule osmNoteQuestType, // ↓ 2. important data that is used by many data consumers - AddRoadName(o, roadNameSuggestionsDao), - AddPlaceName(o, featureDictionaryFuture), - AddOneway(o), - AddSuspectedOneway(o, trafficFlowSegmentsApi, trafficFlowDao), - AddBusStopName(o), - AddBusStopRef(o), - AddIsBuildingUnderground(o), //to avoid asking AddHousenumber and other for underground buildings - AddHousenumber(o), - AddAddressStreet(o, roadNameSuggestionsDao), - MarkCompletedHighwayConstruction(o, r), - AddReligionToPlaceOfWorship(o), // icons on maps are different - OSM Carto, mapy.cz, OsmAnd, Sputnik etc - AddParkingAccess(o), //OSM Carto, mapy.cz, OSMand, Sputnik etc + AddRoadName(roadNameSuggestionsDao), + AddPlaceName(featureDictionaryFuture), + AddOneway(), + AddSuspectedOneway(trafficFlowSegmentsApi, trafficFlowDao), + AddCycleway(), // for any cyclist routers (and cyclist maps) + AddSidewalk(), // for any pedestrian routers + AddBusStopName(), + AddBusStopRef(), + AddIsBuildingUnderground(), //to avoid asking AddHousenumber and other for underground buildings + AddHousenumber(), + AddAddressStreet(roadNameSuggestionsDao), + MarkCompletedHighwayConstruction(), + AddReligionToPlaceOfWorship(), // icons on maps are different - OSM Carto, mapy.cz, OsmAnd, Sputnik etc + AddParkingAccess(), //OSM Carto, mapy.cz, OSMand, Sputnik etc // ↓ 3. useful data that is used by some data consumers - AddRecyclingType(o), - AddRecyclingContainerMaterials(o, r), - AddSport(o), - AddRoadSurface(o, r), // used by BRouter, OsmAnd, OSRM, graphhopper, HOT map style... - AddMaxSpeed(o), // should best be after road surface because it excludes unpaved roads - AddMaxHeight(o), // OSRM and other routing engines - AddRailwayCrossingBarrier(o, r), // useful for routing - AddPostboxCollectionTimes(o, r), - AddOpeningHours(o, featureDictionaryFuture, r), - AddBikeParkingCapacity(o, r), // used by cycle map layer on osm.org, OsmAnd - AddOrchardProduce(o), - AddBuildingType(o), // because housenumber, building levels etc. depend on it - AddCycleway(o,r), // SLOW QUERY - AddSidewalk(o), // SLOW QUERY - AddProhibitedForPedestrians(o), // uses info from AddSidewalk quest, should be after it - AddCrossingType(o, r), - AddCrossingIsland(o), - AddBuildingLevels(o), - AddBusStopShelter(o, r), // at least OsmAnd - AddVegetarian(o, r), - AddVegan(o, r), - AddInternetAccess(o, r), // used by OsmAnd - AddParkingFee(o, r), // used by OsmAnd - AddMotorcycleParkingCapacity(o, r), - AddPathSurface(o, r), // used by OSM Carto, BRouter, OsmAnd, OSRM, graphhopper... - AddTracktype(o, r), // widely used in map rendering - OSM Carto, OsmAnd... - AddMaxWeight(o), // used by OSRM and other routing engines - AddForestLeafType(o), // used by OSM Carto - AddBikeParkingType(o), // used by OsmAnd - AddStepsRamp(o, r), - AddWheelchairAccessToilets(o, r), // used by wheelmap, OsmAnd, MAPS.ME - AddPlaygroundAccess(o), //late as in many areas all needed access=private is already mapped - AddWheelchairAccessBusiness(o, featureDictionaryFuture), // used by wheelmap, OsmAnd, MAPS.ME - AddToiletAvailability(o), //OSM Carto, shown in OsmAnd descriptions - AddFerryAccessPedestrian(o), - AddFerryAccessMotorVehicle(o), - AddAcceptsCash(o, featureDictionaryFuture), + AddRecyclingType(), + AddRecyclingContainerMaterials(), + AddSport(), + AddRoadSurface(), // used by BRouter, OsmAnd, OSRM, graphhopper, HOT map style... + AddMaxSpeed(), // should best be after road surface because it excludes unpaved roads + AddMaxHeight(), // OSRM and other routing engines + AddRailwayCrossingBarrier(), // useful for routing + AddPostboxCollectionTimes(), + AddOpeningHours(featureDictionaryFuture), + AddBikeParkingCapacity(), // used by cycle map layer on osm.org, OsmAnd + AddOrchardProduce(), + AddBuildingType(), // because housenumber, building levels etc. depend on it + AddProhibitedForPedestrians(), // uses info from AddSidewalk quest, should be after it + AddCrossingType(), + AddCrossingIsland(), + AddBuildingLevels(), + AddBusStopShelter(), // at least OsmAnd + AddVegetarian(), + AddVegan(), + AddInternetAccess(), // used by OsmAnd + AddParkingFee(), // used by OsmAnd + AddMotorcycleParkingCapacity(), + AddPathSurface(), // used by OSM Carto, BRouter, OsmAnd, OSRM, graphhopper... + AddTracktype(), // widely used in map rendering - OSM Carto, OsmAnd... + AddMaxWeight(), // used by OSRM and other routing engines + AddForestLeafType(), // used by OSM Carto + AddBikeParkingType(), // used by OsmAnd + AddStepsRamp(), + AddWheelchairAccessToilets(), // used by wheelmap, OsmAnd, MAPS.ME + AddPlaygroundAccess(), //late as in many areas all needed access=private is already mapped + AddWheelchairAccessBusiness(featureDictionaryFuture), // used by wheelmap, OsmAnd, MAPS.ME + AddToiletAvailability(), //OSM Carto, shown in OsmAnd descriptions + AddFerryAccessPedestrian(), + AddFerryAccessMotorVehicle(), + AddAcceptsCash(featureDictionaryFuture), // ↓ 4. definitely shown as errors in QA tools // ↓ 5. may be shown as missing in QA tools - DetermineRecyclingGlass(o), // because most recycling:glass=yes is a tagging mistake + DetermineRecyclingGlass(), // because most recycling:glass=yes is a tagging mistake // ↓ 6. may be shown as possibly missing in QA tools // ↓ 7. data useful for only a specific use case - AddWayLit(o, r), // used by OsmAnd if "Street lighting" is enabled. (Configure map, Map rendering, Details) - AddToiletsFee(o), // used by OsmAnd in the object description - AddBabyChangingTable(o), // used by OsmAnd in the object description - AddBikeParkingCover(o), // used by OsmAnd in the object description - AddTactilePavingCrosswalk(o, r), // Paving can be completed while waiting to cross - AddTrafficSignalsSound(o, r), // Sound needs to be done as or after you're crossing - AddTrafficSignalsVibration(o, r), - AddRoofShape(o), - AddWheelchairAccessPublicTransport(o, r), - AddWheelchairAccessOutside(o, r), - AddTactilePavingBusStop(o, r), - AddBridgeStructure(o), - AddReligionToWaysideShrine(o), - AddCyclewaySegregation(o, r), - MarkCompletedBuildingConstruction(o, r), - AddGeneralFee(o), - AddSelfServiceLaundry(o), - AddStepsIncline(o), // can be gathered while walking perpendicular to the way e.g. the other side of the road or when running/cycling past - AddHandrail(o, r), // for accessibility of pedestrian routing, can be gathered when walking past - AddStepCount(o), // can only be gathered when walking along this way, also needs the most effort and least useful - AddInformationToTourism(o), - AddAtmOperator(o), - AddChargingStationOperator(o), - AddClothingBinOperator(o), + AddWayLit(), // used by OsmAnd if "Street lighting" is enabled. (Configure map, Map rendering, Details) + AddToiletsFee(), // used by OsmAnd in the object description + AddBabyChangingTable(), // used by OsmAnd in the object description + AddBikeParkingCover(), // used by OsmAnd in the object description + AddTactilePavingCrosswalk(), // Paving can be completed while waiting to cross + AddTrafficSignalsSound(), // Sound needs to be done as or after you're crossing + AddTrafficSignalsVibration(), + AddRoofShape(), + AddWheelchairAccessPublicTransport(), + AddWheelchairAccessOutside(), + AddTactilePavingBusStop(), + AddBridgeStructure(), + AddReligionToWaysideShrine(), + AddCyclewaySegregation(), + MarkCompletedBuildingConstruction(), + AddGeneralFee(), + AddSelfServiceLaundry(), + AddStepsIncline(), // can be gathered while walking perpendicular to the way e.g. the other side of the road or when running/cycling past + AddHandrail(), // for accessibility of pedestrian routing, can be gathered when walking past + AddStepCount(), // can only be gathered when walking along this way, also needs the most effort and least useful + AddInformationToTourism(), + AddAtmOperator(), + AddChargingStationOperator(), + AddClothingBinOperator(), // ↓ 8. defined in the wiki, but not really used by anyone yet. Just collected for // the sake of mapping it in case it makes sense later - AddIsDefibrillatorIndoor(o), - AddSummitRegister(o, r), - AddCyclewayPartSurface(o, r), - AddFootwayPartSurface(o, r), - AddMotorcycleParkingCover(o), - AddFireHydrantType(o), - AddParkingType(o), - AddPostboxRef(o), - AddWheelchairAccessToiletsPart(o, r), - AddBoardType(o), - AddPowerPolesMaterial(o), - AddCarWashType(o), - AddBenchStatusOnBusStop(o, r), - AddBenchBackrest(o), - AddTrafficSignalsButton(o) + AddIsDefibrillatorIndoor(), + AddSummitRegister(), + AddCyclewayPartSurface(), + AddFootwayPartSurface(), + AddMotorcycleParkingCover(), + AddFireHydrantType(), + AddParkingType(), + AddPostboxRef(), + AddWheelchairAccessToiletsPart(), + AddBoardType(), + AddPowerPolesMaterial(), + AddCarWashType(), + AddBenchStatusOnBusStop(), + AddBenchBackrest(), + AddTrafficSignalsButton() )) @Provides @Singleton fun osmNoteQuestType(): OsmNoteQuestType = OsmNoteQuestType() diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/SplitWayFragment.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/SplitWayFragment.kt index bedfd5ee43..c8e0ee24ea 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/SplitWayFragment.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/SplitWayFragment.kt @@ -246,14 +246,14 @@ class SplitWayFragment : Fragment(R.layout.fragment_split_way), private fun createSplitsForLines(clickPosition: LatLon, clickAreaSizeInMeters: Double): Set { val result = mutableSetOf() - positions.forEachPair { first, second -> + positions.forEachLine { first, second -> val crossTrackDistance = abs(clickPosition.crossTrackDistanceTo(first, second)) if (clickAreaSizeInMeters > crossTrackDistance) { val alongTrackDistance = clickPosition.alongTrackDistanceTo(first, second) val distance = first.distanceTo(second) if (distance > alongTrackDistance && alongTrackDistance > 0) { val delta = alongTrackDistance / distance - result.add(SplitAtLinePosition(first, second, delta)) + result.add(SplitAtLinePosition(OsmLatLon(first), OsmLatLon(second), delta)) } } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/accepts_cash/AddAcceptsCash.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/accepts_cash/AddAcceptsCash.kt index 74979a5f87..b76d742d28 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/accepts_cash/AddAcceptsCash.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/accepts_cash/AddAcceptsCash.kt @@ -3,59 +3,56 @@ package de.westnordost.streetcomplete.quests.accepts_cash import de.westnordost.osmfeatures.FeatureDictionary import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.quest.NoCountriesExcept -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment import java.util.concurrent.FutureTask class AddAcceptsCash( - o: OverpassMapDataAndGeometryApi, private val featureDictionaryFuture: FutureTask -) : SimpleOverpassQuestType(o) { - private val amenity = listOf( - "bar", "cafe", "fast_food", "food_court", "ice_cream", "pub", "biergarten", - "restaurant", "cinema", "nightclub", "planetarium", "theatre", "marketplace", - "internet_cafe", "car_wash", "fuel", "pharmacy", "telephone", "vending_machine" - ) - private val tourismWithImpliedFees = listOf( - "zoo", "aquarium", "theme_park", "hotel", "hostel", "motel", "guest_house", - "apartment", "camp_site" - ) - private val tourismWithoutImpliedFees = listOf( - "attraction", "museum", "gallery" - ) - private val leisure = listOf( - "adult_gaming_centre", "amusement_arcade", "bowling_alley", "escape_game", "miniature_golf", - "sauna", "trampoline_park", "tanning_salon" - ) - private val craft = listOf( - "carpenter", "shoemaker", "tailor", "photographer", "dressmaker", - "electronics_repair", "key_cutter", "stonemason" - ) +) : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter: String get() { + val amenities = listOf( + "bar", "cafe", "fast_food", "food_court", "ice_cream", "pub", "biergarten", + "restaurant", "cinema", "nightclub", "planetarium", "theatre", "marketplace", + "internet_cafe", "car_wash", "fuel", "pharmacy", "telephone", "vending_machine" + ) + val tourismsWithImpliedFees = listOf( + "zoo", "aquarium", "theme_park", "hotel", "hostel", "motel", "guest_house", + "apartment", "camp_site" + ) + val tourismsWithoutImpliedFees = listOf( + "attraction", "museum", "gallery" + ) + val leisures = listOf( + "adult_gaming_centre", "amusement_arcade", "bowling_alley", "escape_game", "miniature_golf", + "sauna", "trampoline_park", "tanning_salon" + ) + val crafts = listOf( + "carpenter", "shoemaker", "tailor", "photographer", "dressmaker", + "electronics_repair", "key_cutter", "stonemason" + ) + return """ nodes, ways, relations with ( (shop and shop !~ no|vacant|mall) - or amenity ~ ${amenity.joinToString("|")} - or leisure ~ ${leisure.joinToString("|")} - or craft ~ ${craft.joinToString("|")} - or tourism ~ ${tourismWithImpliedFees.joinToString("|")} - or tourism ~ ${tourismWithoutImpliedFees.joinToString("|")} and fee = yes + or amenity ~ ${amenities.joinToString("|")} + or leisure ~ ${leisures.joinToString("|")} + or craft ~ ${crafts.joinToString("|")} + or tourism ~ ${tourismsWithImpliedFees.joinToString("|")} + or tourism ~ ${tourismsWithoutImpliedFees.joinToString("|")} and fee = yes ) and name and !payment:cash and !payment:coins and !payment:notes - """ + """} + override val commitMessage = "Add whether this place accepts cash as payment" override val defaultDisabledMessage = R.string.default_disabled_msg_go_inside override val wikiLink = "Key:payment" override val icon = R.drawable.ic_quest_cash - override val enabledInCountries = NoCountriesExcept( - // Europe - "SE" - ) + override val enabledInCountries = NoCountriesExcept("SE") override fun getTitle(tags: Map) = if (hasFeatureName(tags) && !tags.containsKey("brand")) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/address/AddAddressStreet.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/address/AddAddressStreet.kt index e6b7367b9d..9f1bdd3844 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/address/AddAddressStreet.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/address/AddAddressStreet.kt @@ -1,24 +1,35 @@ package de.westnordost.streetcomplete.quests.address -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element +import de.westnordost.osmapi.map.data.Relation import de.westnordost.streetcomplete.R +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.meta.ALL_ROADS import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry import de.westnordost.streetcomplete.data.quest.AllCountriesExcept -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.quests.road_name.data.RoadNameSuggestionEntry import de.westnordost.streetcomplete.quests.road_name.data.RoadNameSuggestionsDao -import de.westnordost.streetcomplete.quests.road_name.data.putRoadNameSuggestion +import de.westnordost.streetcomplete.quests.road_name.data.toRoadNameByLanguage class AddAddressStreet( - private val overpassApi: OverpassMapDataAndGeometryApi, private val roadNameSuggestionsDao: RoadNameSuggestionsDao ) : OsmElementQuestType { + private val filter by lazy { """ + nodes, ways, relations with + addr:housenumber and !addr:street and !addr:place and !addr:block_number + or addr:streetnumber and !addr:street + """.toElementFilterExpression() } + + private val roadsWithNamesFilter by lazy { """ + ways with + highway ~ ${ALL_ROADS.joinToString("|")} + and name + """.toElementFilterExpression()} + override val commitMessage = "Add street/place names to address" override val icon = R.drawable.ic_quest_housenumber_street // In Japan, housenumbers usually have block numbers, not streets @@ -31,34 +42,36 @@ class AddAddressStreet( return if (housenumber != null) arrayOf(housenumber) else arrayOf() } - override fun createForm() = AddAddressStreetForm() + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val associatedStreetRelations = mapData.relations.filter { + val type = it.tags?.get("type") + type == "associatedStreet" || type == "street" + } - /* cannot be determined offline because the quest kinda needs the street name suggestions - to work conveniently (see #1856) */ - override fun isApplicableTo(element: Element): Boolean? = null + val addressesWithoutStreet = mapData.filter { address -> + filter.matches(address) && + associatedStreetRelations.none { it.contains(address.type, address.id) } + } - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - if (!overpassApi.query(getStreetNameSuggestionsOverpassQuery(bbox), roadNameSuggestionsDao::putRoadNameSuggestion)) return false - if (!overpassApi.query(getOverpassQuery(bbox), handler)) return false - return true + if (addressesWithoutStreet.isNotEmpty()) { + val roadsWithNames = mapData.ways + .filter { roadsWithNamesFilter.matches(it) } + .mapNotNull { + val geometry = mapData.getWayGeometry(it.id) as? ElementPolylinesGeometry + val roadNamesByLanguage = it.tags?.toRoadNameByLanguage() + if (geometry != null && roadNamesByLanguage != null) { + RoadNameSuggestionEntry(it.id, roadNamesByLanguage, geometry.polylines.first()) + } else null + } + roadNameSuggestionsDao.putRoads(roadsWithNames) + } + return addressesWithoutStreet } - private fun getOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + """ - relation["type"="associatedStreet"]; > -> .inStreetRelation; - $ADDRESSES_WITHOUT_STREETS -> .missing_data; - (.missing_data; - .inStreetRelation;); - """.trimIndent() + getQuestPrintStatement() + override fun createForm() = AddAddressStreetForm() - /** return overpass query string to get roads with names around addresses without streets - * */ - private fun getStreetNameSuggestionsOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + """ - $ADDRESSES_WITHOUT_STREETS -> .address_missing_street; - $ROADS_WITH_NAMES -> .named_roads; - way.named_roads( - around.address_missing_street: $MAX_DIST_FOR_ROAD_NAME_SUGGESTION); - out body geom;""".trimIndent() + /* cannot be determined because of the associated street relations */ + override fun isApplicableTo(element: Element): Boolean? = null override fun applyAnswerTo(answer: AddressStreetAnswer, changes: StringMapChangesBuilder) { val key = when(answer) { @@ -68,15 +81,11 @@ class AddAddressStreet( changes.add(key, answer.name) } - companion object { - const val MAX_DIST_FOR_ROAD_NAME_SUGGESTION = 100.0 - - private val ADDRESSES_WITHOUT_STREETS = """ - ( - nwr["addr:housenumber"][!"addr:street"][!"addr:place"][!"addr:block_number"]; - nwr["addr:streetnumber"][!"addr:street"]; - )""".trimIndent() - private val ROADS_WITH_NAMES = - "way[highway ~ \"^(${ALL_ROADS.joinToString("|")})$\"][name]" + override fun cleanMetadata() { + roadNameSuggestionsDao.cleanUp() } } + +private fun Relation.contains(elementType: Element.Type, elementId: Long) : Boolean { + return members.any { it.type == elementType && it.ref == elementId } +} \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/atm_operator/AddAtmOperator.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/atm_operator/AddAtmOperator.kt index 6d06d207bb..415884e00e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/atm_operator/AddAtmOperator.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/atm_operator/AddAtmOperator.kt @@ -2,12 +2,11 @@ package de.westnordost.streetcomplete.quests.atm_operator import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType -class AddAtmOperator(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddAtmOperator : OsmFilterQuestType() { - override val tagFilters = "nodes with amenity = atm and !operator and !name and !brand" + override val elementFilter = "nodes with amenity = atm and !operator and !name and !brand" override val commitMessage = "Add ATM operator" override val wikiLink = "Tag:amenity=atm" override val icon = R.drawable.ic_quest_money diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/baby_changing_table/AddBabyChangingTable.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/baby_changing_table/AddBabyChangingTable.kt index 3cfd4ab3f6..5e987d2b75 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/baby_changing_table/AddBabyChangingTable.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/baby_changing_table/AddBabyChangingTable.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.baby_changing_table import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddBabyChangingTable(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBabyChangingTable : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with ( ( diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bench_backrest/AddBenchBackrest.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bench_backrest/AddBenchBackrest.kt index cc0ea6547e..19adf40e9f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bench_backrest/AddBenchBackrest.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bench_backrest/AddBenchBackrest.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.bench_backrest import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.bench_backrest.BenchBackrestAnswer.* -class AddBenchBackrest(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBenchBackrest : OsmFilterQuestType() { - override val tagFilters = "nodes with amenity = bench and !backrest" + override val elementFilter = "nodes with amenity = bench and !backrest" override val commitMessage = "Add backrest information to benches" override val wikiLink = "Tag:amenity=bench" override val icon = R.drawable.ic_quest_bench_poi diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_capacity/AddBikeParkingCapacity.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_capacity/AddBikeParkingCapacity.kt index e9995fc299..8eed319adb 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_capacity/AddBikeParkingCapacity.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_capacity/AddBikeParkingCapacity.kt @@ -2,20 +2,17 @@ package de.westnordost.streetcomplete.quests.bike_parking_capacity import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddBikeParkingCapacity(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddBikeParkingCapacity : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with amenity = bicycle_parking and access !~ private|no and ( !capacity - or bicycle_parking ~ stands|wall_loops and capacity older today -${r * 4} years + or bicycle_parking ~ stands|wall_loops and capacity older today -4 years ) """ /* Bike capacity may change more often for stands and wheelbenders as adding or diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_cover/AddBikeParkingCover.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_cover/AddBikeParkingCover.kt index bc4299e615..b3cf867791 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_cover/AddBikeParkingCover.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_cover/AddBikeParkingCover.kt @@ -1,14 +1,14 @@ package de.westnordost.streetcomplete.quests.bike_parking_cover import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddBikeParkingCover(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { - override val tagFilters = """ +class AddBikeParkingCover : OsmFilterQuestType() { + + override val elementFilter = """ nodes, ways with amenity = bicycle_parking and access !~ private|no diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_type/AddBikeParkingType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_type/AddBikeParkingType.kt index 80def2eb92..bef97a925e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_type/AddBikeParkingType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bike_parking_type/AddBikeParkingType.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.bike_parking_type import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddBikeParkingType(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBikeParkingType : OsmFilterQuestType() { - override val tagFilters = "nodes, ways with amenity = bicycle_parking and access !~ private|no and !bicycle_parking" + override val elementFilter = "nodes, ways with amenity = bicycle_parking and access !~ private|no and !bicycle_parking" override val commitMessage = "Add bicycle parking type" override val wikiLink = "Key:bicycle_parking" override val icon = R.drawable.ic_quest_bicycle_parking diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCycleway.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCycleway.kt index 7b6bd70497..46f5321e94 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCycleway.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCycleway.kt @@ -1,30 +1,25 @@ package de.westnordost.streetcomplete.quests.bikeway -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.elementfilter.filters.RelativeDate import de.westnordost.streetcomplete.data.elementfilter.filters.TagOlderThan +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.meta.ANYTHING_UNPAVED -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder import de.westnordost.streetcomplete.data.quest.NoCountriesExcept -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox import de.westnordost.streetcomplete.data.meta.deleteCheckDatesForKey import de.westnordost.streetcomplete.data.meta.updateCheckDateForKey import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.ktx.containsAny import de.westnordost.streetcomplete.quests.bikeway.Cycleway.* -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore +import de.westnordost.streetcomplete.util.isNear -class AddCycleway( - private val overpassApi: OverpassMapDataAndGeometryApi, - private val r: ResurveyIntervalsStore -) : OsmElementQuestType { +class AddCycleway : OsmElementQuestType { override val commitMessage = "Add whether there are cycleways" override val wikiLink = "Key:cycleway" @@ -68,81 +63,56 @@ class AddCycleway( R.string.quest_cycleway_title2 } - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox), handler) - } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val eligibleRoads = mapData.ways.filter { roadsFilter.matches(it) } - private fun getOverpassQuery(bbox: BoundingBox): String { - val minDistToCycleways = 15 //m + /* we want to return two sets of roads: Those that do not have any cycleway tags, and those + * that do but have been checked more than 4 years ago. */ - val anythingUnpaved = ANYTHING_UNPAVED.joinToString("|") - val olderThan4Years = olderThan(4).toOverpassQLString() - val handledCycleways = KNOWN_CYCLEWAY_VALUES.joinToString("|") - val handledCyclewayLanes = KNOWN_CYCLEWAY_LANE_VALUES.joinToString("|") + /* For the first, the roadsWithMissingCycleway filter is not enough. In OSM, cycleways may be + * mapped as separate ways as well and it is not guaranteed that in this case, + * cycleway = separate or something is always tagged on the main road then. So, all roads + * should be excluded whose center is within of ~15 meters of a cycleway, to be on the safe + * side. */ - /* Excluded is - - anything explicitly tagged as no bicycles or having to use separately mapped sidepath - - if not already tagged with a cycleway: streets with low speed or that are not paved, as - they are very unlikely to have cycleway infrastructure - - if not already tagged, roads that are close (15m) to foot or cycleways (see #718) - - if already tagged, if not older than 8 years or if the cycleway tag uses some unknown value - */ - return bbox.toGlobalOverpassBBox() + """ - way - [highway ~ '^(primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential|service)$'] - [motorroad != yes] - [bicycle_road != yes][cyclestreet != yes] - [area != yes] - [bicycle != no][bicycle != designated] - [access !~ '^(private|no)$'] - [bicycle != use_sidepath] - ['bicycle:backward' != use_sidepath]['bicycle:forward' != use_sidepath] - -> .streets; - - way.streets - [highway ~ '^(primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified)$'] - [!cycleway] - [!'cycleway:left'][!'cycleway:right'][!'cycleway:both'] - [!'sidewalk:bicycle'] - [!'sidewalk:both:bicycle'][!'sidewalk:left:bicycle'][!'sidewalk:right:bicycle'] - [maxspeed !~ '^(20|15|10|8|7|6|5|10 mph|5 mph|walk)$'] - [surface !~ '^($anythingUnpaved)$'] - -> .untagged; - - way[highway ~ '^(path|footway|cycleway)$'](around.streets: $minDistToCycleways) -> .cycleways; - way.untagged(around.cycleways: $minDistToCycleways) -> .untagged_near_cycleways; - - way.streets - [~'^(cycleway(:(left|right|both))?)$' ~ '.*'] - $olderThan4Years - -> .old; - - (""" + - KNOWN_CYCLEWAY_KEYS.joinToString("") { "way.old['$it']['$it' !~ '^($handledCycleways)$'];\n" } + - KNOWN_CYCLEWAY_LANES_KEYS.joinToString("") { "way.old['$it']['$it' !~ '^($handledCyclewayLanes)$'];\n" } + - """) -> .old_with_unknown_tags; - - ( - (.untagged; - .untagged_near_cycleways;); - (.old; - .old_with_unknown_tags;); - ); - - ${getQuestPrintStatement()} - """.trimIndent() + val roadsWithMissingCycleway = eligibleRoads.filter { untaggedRoadsFilter.matches(it) }.toMutableList() + + if (roadsWithMissingCycleway.isNotEmpty()) { + + val maybeSeparatelyMappedCyclewayGeometries = mapData.ways + .filter { maybeSeparatelyMappedCyclewaysFilter.matches(it) } + .mapNotNull { mapData.getWayGeometry(it.id) as? ElementPolylinesGeometry } + + val minDistToWays = 15.0 //m + + // filter out roads with missing sidewalks that are near footways + roadsWithMissingCycleway.removeAll { road -> + val roadGeometry = mapData.getWayGeometry(road.id) as? ElementPolylinesGeometry + roadGeometry?.isNear(minDistToWays, maybeSeparatelyMappedCyclewayGeometries) ?: true + } + } + + /* For the second, nothing special. Filter out ways that have been checked less then 4 + * years ago or have no known cycleway tags */ + + val oldRoadsWithKnownCycleways = eligibleRoads.filter { + OLDER_THAN_4_YEARS.matches(it) && it.hasOnlyKnownCyclewayTags() + } + + return roadsWithMissingCycleway + oldRoadsWithKnownCycleways } + override fun isApplicableTo(element: Element): Boolean? { val tags = element.tags ?: return false // can't determine for yet untagged roads by the tags alone because we need info about // surrounding geometry, but for already tagged ones, we can! if (!tags.keys.containsAny(KNOWN_CYCLEWAY_KEYS)) return null - return olderThan(4).matches(element) - && tags.filterKeys { it in KNOWN_CYCLEWAY_KEYS }.values.all { it in KNOWN_CYCLEWAY_VALUES } - && tags.filterKeys { it in KNOWN_CYCLEWAY_LANES_KEYS }.values.all { it in KNOWN_CYCLEWAY_LANE_VALUES } - } - private fun olderThan(years: Int) = - TagOlderThan("cycleway", RelativeDate(-(r * 365 * years).toFloat())) + return roadsFilter.matches(element) && + OLDER_THAN_4_YEARS.matches(element) && + element.hasOnlyKnownCyclewayTags() + } override fun createForm() = AddCyclewayForm() @@ -302,14 +272,61 @@ class AddCycleway( } companion object { - private val KNOWN_CYCLEWAY_KEYS = listOf( + + /* Excluded is + - anything explicitly tagged as no bicycles or having to use separately mapped sidepath + - if not already tagged with a cycleway: streets with low speed or that are not paved, as + they are very unlikely to have cycleway infrastructure + - if not already tagged, roads that are close (15m) to foot or cycleways (see #718) + - if already tagged, if not older than 8 years or if the cycleway tag uses some unknown value + */ + + // streets what may have cycleway tagging + private val roadsFilter by lazy { """ + ways with + highway ~ primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential|service + and area != yes + and motorroad != yes + and bicycle_road != yes + and cyclestreet != yes + and bicycle != no + and bicycle != designated + and access !~ private|no + and bicycle != use_sidepath + and bicycle:backward != use_sidepath + and bicycle:forward != use_sidepath + """.toElementFilterExpression() } + + // streets that do not have cycleway tagging yet + private val untaggedRoadsFilter by lazy { """ + ways with + highway ~ primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential + and !cycleway + and !cycleway:left + and !cycleway:right + and !cycleway:both + and !sidewalk:bicycle + and !sidewalk:left:bicycle + and !sidewalk:right:bicycle + and !sidewalk:both:bicycle + and (!maxspeed or maxspeed > 20 or maxspeed !~ "10 mph|5 mph|walk") + and surface !~ ${ANYTHING_UNPAVED.joinToString("|")} + """.toElementFilterExpression() } + + private val maybeSeparatelyMappedCyclewaysFilter by lazy { """ + ways with highway ~ path|footway|cycleway + """.toElementFilterExpression() } + + private val OLDER_THAN_4_YEARS = TagOlderThan("cycleway", RelativeDate(-(365 * 4).toFloat())) + + private val KNOWN_CYCLEWAY_KEYS = setOf( "cycleway", "cycleway:left", "cycleway:right", "cycleway:both" ) - private val KNOWN_CYCLEWAY_LANES_KEYS = listOf( + private val KNOWN_CYCLEWAY_LANES_KEYS = setOf( "cycleway:lane", "cycleway:left:lane", "cycleway:right:lane", "cycleway:both:lane" ) - private val KNOWN_CYCLEWAY_VALUES = listOf( + private val KNOWN_CYCLEWAY_VALUES = setOf( "lane", "track", "shared_lane", @@ -346,6 +363,22 @@ class AddCycleway( "mandatory", "exclusive_lane", // same as exclusive. Exclusive lanes are mandatory for bicyclists "soft_lane", "advisory_lane", "dashed" // synonym for advisory lane ) + + private fun Element.hasOnlyKnownCyclewayTags(): Boolean { + val tags = tags ?: return false + + val cyclewayTags = tags.filterKeys { it in KNOWN_CYCLEWAY_KEYS } + // has no cycleway tagging + if (cyclewayTags.isEmpty()) return false + // any cycleway tagging is not known + if (cyclewayTags.values.any { it !in KNOWN_CYCLEWAY_VALUES }) return false + + // any cycleway lane tagging is not known + val cycleLaneTags = tags.filterKeys { it in KNOWN_CYCLEWAY_LANES_KEYS } + if (cycleLaneTags.values.any { it !in KNOWN_CYCLEWAY_LANE_VALUES }) return false + + return true + } } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCyclewayForm.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCyclewayForm.kt index cc5c7a354d..aac643727d 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCyclewayForm.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bikeway/AddCyclewayForm.kt @@ -8,7 +8,7 @@ import java.util.Collections import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.quests.AbstractQuestFormAnswerFragment import de.westnordost.streetcomplete.quests.OtherAnswer import de.westnordost.streetcomplete.quests.StreetSideRotater @@ -32,12 +32,12 @@ class AddCyclewayForm : AbstractQuestFormAnswerFragment() { return result } - private val likelyNoBicycleContraflow = ElementFiltersParser().parse(""" + private val likelyNoBicycleContraflow = """ ways with oneway:bicycle != no and ( oneway ~ yes|-1 and highway ~ primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified or junction = roundabout ) - """) + """.toElementFilterExpression() private var streetSideRotater: StreetSideRotater? = null diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/board_type/AddBoardType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/board_type/AddBoardType.kt index 2d509689d7..d4d1239d51 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/board_type/AddBoardType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/board_type/AddBoardType.kt @@ -2,12 +2,11 @@ package de.westnordost.streetcomplete.quests.board_type import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType -class AddBoardType(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBoardType : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with information = board and access !~ private|no and (!board_type or board_type ~ yes|board) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bridge_structure/AddBridgeStructure.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bridge_structure/AddBridgeStructure.kt index 9b11e68a9b..a89604e0b6 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bridge_structure/AddBridgeStructure.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bridge_structure/AddBridgeStructure.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.bridge_structure import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddBridgeStructure(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBridgeStructure : OsmFilterQuestType() { - override val tagFilters = "ways with man_made = bridge and !bridge:structure and !bridge:movable" + override val elementFilter = "ways with man_made = bridge and !bridge:structure and !bridge:movable" override val icon = R.drawable.ic_quest_bridge override val commitMessage = "Add bridge structures" override val wikiLink = "Key:bridge:structure" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/building_levels/AddBuildingLevels.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/building_levels/AddBuildingLevels.kt index b79d50a1e1..a738e9024b 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/building_levels/AddBuildingLevels.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/building_levels/AddBuildingLevels.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.building_levels import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddBuildingLevels(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBuildingLevels : OsmFilterQuestType() { // building:height is undocumented, but used the same way as height and currently over 50k times - override val tagFilters = """ + override val elementFilter = """ ways, relations with building ~ ${BUILDINGS_WITH_LEVELS.joinToString("|")} and !building:levels and !height and !building:height and !man_made and location != underground and ruins != yes diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/building_type/AddBuildingType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/building_type/AddBuildingType.kt index 9578c61c8d..1cf55f8879 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/building_type/AddBuildingType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/building_type/AddBuildingType.kt @@ -1,16 +1,15 @@ package de.westnordost.streetcomplete.quests.building_type import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddBuildingType (o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBuildingType : OsmFilterQuestType() { // in the case of man_made, historic, military and power, these tags already contain // information about the purpose of the building, so no need to force asking it // same goes (more or less) for tourism, amenity, leisure. See #1854, #1891 - override val tagFilters = """ + override val elementFilter = """ ways, relations with building = yes and !man_made and !historic diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/building_underground/AddIsBuildingUnderground.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/building_underground/AddIsBuildingUnderground.kt index 4db101acf0..7540003b54 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/building_underground/AddIsBuildingUnderground.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/building_underground/AddIsBuildingUnderground.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.building_underground import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddIsBuildingUnderground(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddIsBuildingUnderground : OsmFilterQuestType() { - override val tagFilters = "ways, relations with building and !location and layer~-[0-9]+" + override val elementFilter = "ways, relations with building and !location and layer~-[0-9]+" override val commitMessage = "Determine whatever building is fully underground" override val wikiLink = "Key:location" override val icon = R.drawable.ic_quest_building_underground diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_bench/AddBenchStatusOnBusStop.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_bench/AddBenchStatusOnBusStop.kt index 308bb034fe..5b8c99380d 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_bench/AddBenchStatusOnBusStop.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_bench/AddBenchStatusOnBusStop.kt @@ -1,16 +1,14 @@ package de.westnordost.streetcomplete.quests.bus_stop_bench import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddBenchStatusOnBusStop(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) : SimpleOverpassQuestType(o) { +class AddBenchStatusOnBusStop : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with ( (public_transport = platform and ~bus|trolleybus|tram ~ yes) @@ -18,7 +16,7 @@ class AddBenchStatusOnBusStop(o: OverpassMapDataAndGeometryApi, r: ResurveyInter (highway = bus_stop and public_transport != stop_position) ) and physically_present != no and naptan:BusStopType != HAR - and (!bench or bench older today -${r * 4} years) + and (!bench or bench older today -4 years) """ override val commitMessage = "Add whether a bus stop has a bench" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopName.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopName.kt index ebbe452112..092b0b6c85 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopName.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopName.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.bus_stop_name import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.quest.AllCountriesExcept -class AddBusStopName(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBusStopName : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with ( (public_transport = platform and ~bus|trolleybus|tram ~ yes) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_ref/AddBusStopRef.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_ref/AddBusStopRef.kt index 87b97d4107..c2a28fb7e2 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_ref/AddBusStopRef.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_ref/AddBusStopRef.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.bus_stop_ref import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.quest.NoCountriesExcept -class AddBusStopRef(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddBusStopRef : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with ( (public_transport = platform and ~bus|trolleybus|tram ~ yes) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_shelter/AddBusStopShelter.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_shelter/AddBusStopShelter.kt index 0610d166a1..e95c42f68a 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_shelter/AddBusStopShelter.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/bus_stop_shelter/AddBusStopShelter.kt @@ -2,16 +2,13 @@ package de.westnordost.streetcomplete.quests.bus_stop_shelter import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.bus_stop_shelter.BusStopShelterAnswer.* -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddBusStopShelter(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddBusStopShelter : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with ( (public_transport = platform and ~bus|trolleybus|tram ~ yes) @@ -19,7 +16,7 @@ class AddBusStopShelter(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsSt (highway = bus_stop and public_transport != stop_position) ) and physically_present != no and naptan:BusStopType != HAR - and !covered and (!shelter or shelter older today -${r * 4} years) + and !covered and (!shelter or shelter older today -4 years) """ /* Not asking again if it is covered because it means the stop itself is under a large building or roof building so this won't usually change */ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashType.kt index d83f0ab6bf..f5d525da49 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashType.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.car_wash_type import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.car_wash_type.CarWashType.* -class AddCarWashType(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType>(o) { +class AddCarWashType : OsmFilterQuestType>() { - override val tagFilters = "nodes, ways with amenity = car_wash and !automated and !self_service" + override val elementFilter = "nodes, ways with amenity = car_wash and !automated and !self_service" override val commitMessage = "Add car wash type" override val wikiLink = "Tag:amenity=car_wash" override val icon = R.drawable.ic_quest_car_wash diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/charging_station_operator/AddChargingStationOperator.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/charging_station_operator/AddChargingStationOperator.kt index e40e395c2d..2976f727d6 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/charging_station_operator/AddChargingStationOperator.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/charging_station_operator/AddChargingStationOperator.kt @@ -2,12 +2,12 @@ package de.westnordost.streetcomplete.quests.charging_station_operator import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType -class AddChargingStationOperator(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { - override val tagFilters = "nodes with amenity = charging_station and !operator and !name and !brand" +class AddChargingStationOperator : OsmFilterQuestType() { + + override val elementFilter = "nodes with amenity = charging_station and !operator and !name and !brand" override val commitMessage = "Add charging station operator" override val wikiLink = "Tag:amenity=charging_station" override val icon = R.drawable.ic_quest_car_charger diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperator.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperator.kt index 498d607e78..93f1a2e931 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperator.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperator.kt @@ -1,42 +1,29 @@ package de.westnordost.streetcomplete.quests.clothing_bin_operator -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType -class AddClothingBinOperator(private val overpassApi: OverpassMapDataAndGeometryApi) - : OsmElementQuestType { +class AddClothingBinOperator : OsmElementQuestType { /* not the complete filter, see below: we want to filter out additionally all elements that - contain any recycling:* = yes that is not shoes or clothes but this can neither be expressed - in the elements filter syntax nor overpass QL */ - private val filter by lazy { ElementFiltersParser().parse(""" + contain any recycling:* = yes that is not shoes or clothes but this can not be expressed + in the elements filter syntax */ + private val filter by lazy { """ nodes with amenity = recycling and recycling_type = container and recycling:clothes = yes and !operator - """)} + """.toElementFilterExpression() } override val commitMessage = "Add clothing bin operator" override val wikiLink = "Tag:amenity=recycling" override val icon = R.drawable.ic_quest_recycling_clothes - fun getOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + filter.toOverpassQLString() + getQuestPrintStatement() - - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox)) { element, geometry -> - if (element.tags?.hasNoOtherRecyclingTags() == true) { - handler(element, geometry) - } - } - } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable = + mapData.nodes.filter { filter.matches(it) && it.tags.hasNoOtherRecyclingTags() } override fun isApplicableTo(element: Element): Boolean = filter.matches(element) && element.tags.hasNoOtherRecyclingTags() @@ -45,8 +32,8 @@ class AddClothingBinOperator(private val overpassApi: OverpassMapDataAndGeometry return entries.find { it.key.startsWith("recycling:") it.key != "recycling:shoes" && - it.key != "recycling:clothes" && - it.value == "yes" + it.key != "recycling:clothes" && + it.value == "yes" } == null } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedBuildingConstruction.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedBuildingConstruction.kt index 3d97006126..cf9f5b7373 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedBuildingConstruction.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedBuildingConstruction.kt @@ -4,19 +4,16 @@ import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.SURVEY_MARK_KEY import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore import java.util.* -class MarkCompletedBuildingConstruction(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class MarkCompletedBuildingConstruction : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with building = construction and (!opening_date or opening_date < today) - and older today -${r * 6} months + and older today -6 months """ override val commitMessage = "Determine whether construction is now completed" override val wikiLink = "Tag:building=construction" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedHighwayConstruction.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedHighwayConstruction.kt index f4c08251c8..fcac0ee177 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedHighwayConstruction.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/construction/MarkCompletedHighwayConstruction.kt @@ -5,19 +5,16 @@ import de.westnordost.streetcomplete.data.meta.ALL_ROADS import de.westnordost.streetcomplete.data.meta.SURVEY_MARK_KEY import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore import java.util.* -class MarkCompletedHighwayConstruction(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class MarkCompletedHighwayConstruction : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway = construction and (!opening_date or opening_date < today) - and older today -${r * 2} weeks + and older today -2 weeks """ override val commitMessage = "Determine whether construction is now completed" override val wikiLink = "Tag:highway=construction" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIsland.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIsland.kt index 3a4a80473d..2772886e3f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIsland.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIsland.kt @@ -1,19 +1,30 @@ package de.westnordost.streetcomplete.quests.crossing_island -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddCrossingIsland(private val overpass: OverpassMapDataAndGeometryApi) - : OsmElementQuestType { +class AddCrossingIsland : OsmElementQuestType { + + private val crossingFilter by lazy { """ + nodes with + highway = crossing + and crossing + and crossing != island + and !crossing:island + """.toElementFilterExpression()} + + private val excludedWaysFilter by lazy { """ + ways with + highway and access ~ private|no + or highway and oneway and oneway != no + or highway ~ path|footway|cycleway|pedestrian + """.toElementFilterExpression()} override val commitMessage = "Add whether pedestrian crossing has an island" override val wikiLink = "Key:crossing:island" @@ -21,24 +32,17 @@ class AddCrossingIsland(private val overpass: OverpassMapDataAndGeometryApi) override fun getTitle(tags: Map) = R.string.quest_pedestrian_crossing_island - override fun isApplicableTo(element: Element): Boolean? = null + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val excludedWayNodeIds = mutableSetOf() + mapData.ways + .filter { excludedWaysFilter.matches(it) } + .flatMapTo(excludedWayNodeIds) { it.nodeIds } - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpass.query(getOverpassQuery(bbox), handler) + return mapData.nodes + .filter { crossingFilter.matches(it) && it.id !in excludedWayNodeIds } } - private fun getOverpassQuery(bbox: BoundingBox): String = - bbox.toGlobalOverpassBBox() + """ - node[highway = crossing][crossing][crossing != island][!"crossing:island"] -> .crossings; - .crossings < -> .crossedWays; - ( - way.crossedWays[highway][access ~ "^(private|no)$"]; - way.crossedWays[highway][oneway][oneway != no]; - way.crossedWays[highway ~ "^(path|footway|cycleway|pedestrian)$"]; - ) -> .excludedWays; - - ( .crossings; - node(w.excludedWays); ); - """.trimIndent() + "\n" + getQuestPrintStatement() + override fun isApplicableTo(element: Element): Boolean? = null override fun createForm() = YesNoQuestAnswerFragment() diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_type/AddCrossingType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_type/AddCrossingType.kt index ba6010f56d..aaf6c1a6ab 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_type/AddCrossingType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/crossing_type/AddCrossingType.kt @@ -3,15 +3,12 @@ package de.westnordost.streetcomplete.quests.crossing_type import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateCheckDateForKey import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddCrossingType(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddCrossingType : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with highway = crossing and foot != no and ( @@ -19,7 +16,7 @@ class AddCrossingType(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStor or crossing ~ island|unknown|yes or ( crossing ~ traffic_signals|uncontrolled|zebra|marked|unmarked - and crossing older today -${r * 8} years + and crossing older today -8 years ) ) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/defibrillator/AddIsDefibrillatorIndoor.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/defibrillator/AddIsDefibrillatorIndoor.kt index a4c49e05f0..a201c6ae33 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/defibrillator/AddIsDefibrillatorIndoor.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/defibrillator/AddIsDefibrillatorIndoor.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.defibrillator import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddIsDefibrillatorIndoor(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddIsDefibrillatorIndoor : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with emergency=defibrillator and access !~ private|no diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegan.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegan.kt index a4435187d4..853a5defa8 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegan.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegan.kt @@ -2,15 +2,12 @@ package de.westnordost.streetcomplete.quests.diet_type import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddVegan(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddVegan : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with ( amenity ~ restaurant|cafe|fast_food and diet:vegetarian ~ yes|only @@ -18,7 +15,7 @@ class AddVegan(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) ) and name and ( !diet:vegan - or diet:vegan != only and diet:vegan older today -${r * 2} years + or diet:vegan != only and diet:vegan older today -2 years ) """ override val commitMessage = "Add vegan diet type" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegetarian.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegetarian.kt index c626eb1b54..3c6a164868 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegetarian.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/diet_type/AddVegetarian.kt @@ -2,19 +2,16 @@ package de.westnordost.streetcomplete.quests.diet_type import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddVegetarian(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddVegetarian : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with amenity ~ restaurant|cafe|fast_food and name and ( !diet:vegetarian - or diet:vegetarian != only and diet:vegetarian older today -${r * 2} years + or diet:vegetarian != only and diet:vegetarian older today -2 years ) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessMotorVehicle.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessMotorVehicle.kt index a64a48cdf9..6e27e9a78f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessMotorVehicle.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessMotorVehicle.kt @@ -1,17 +1,14 @@ package de.westnordost.streetcomplete.quests.ferry import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment +class AddFerryAccessMotorVehicle : OsmFilterQuestType() { - -class AddFerryAccessMotorVehicle(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { - - override val tagFilters = "ways, relations with route = ferry and !motor_vehicle" + override val elementFilter = "ways, relations with route = ferry and !motor_vehicle" override val commitMessage = "Specify ferry access for motor vehicles" override val wikiLink = "Tag:route=ferry" override val icon = R.drawable.ic_quest_ferry diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessPedestrian.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessPedestrian.kt index 4acc268c09..cb555e5109 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessPedestrian.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/ferry/AddFerryAccessPedestrian.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.ferry import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddFerryAccessPedestrian(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddFerryAccessPedestrian : OsmFilterQuestType() { - override val tagFilters = "ways, relations with route = ferry and !foot" + override val elementFilter = "ways, relations with route = ferry and !foot" override val commitMessage = "Specify ferry access for pedestrians" override val wikiLink = "Tag:route=ferry" override val icon = R.drawable.ic_quest_ferry_pedestrian diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/fire_hydrant/AddFireHydrantType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/fire_hydrant/AddFireHydrantType.kt index a88fbfa3fe..e462ae1076 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/fire_hydrant/AddFireHydrantType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/fire_hydrant/AddFireHydrantType.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.fire_hydrant import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddFireHydrantType(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddFireHydrantType : OsmFilterQuestType() { - override val tagFilters = "nodes with emergency = fire_hydrant and !fire_hydrant:type" + override val elementFilter = "nodes with emergency = fire_hydrant and !fire_hydrant:type" override val commitMessage = "Add fire hydrant type" override val wikiLink = "Tag:emergency=fire_hydrant" override val icon = R.drawable.ic_quest_fire_hydrant diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/foot/AddProhibitedForPedestrians.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/foot/AddProhibitedForPedestrians.kt index 3148943875..9c2481c0cd 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/foot/AddProhibitedForPedestrians.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/foot/AddProhibitedForPedestrians.kt @@ -2,14 +2,13 @@ package de.westnordost.streetcomplete.quests.foot import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ANYTHING_PAVED -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.foot.ProhibitedForPedestriansAnswer.* -class AddProhibitedForPedestrians(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddProhibitedForPedestrians : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with ( ~'sidewalk(:both)?' ~ none|no or (sidewalk:left ~ none|no and sidewalk:right ~ none|no) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/general_fee/AddGeneralFee.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/general_fee/AddGeneralFee.kt index 2748c80b2f..110d24bb50 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/general_fee/AddGeneralFee.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/general_fee/AddGeneralFee.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.general_fee import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddGeneralFee(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddGeneralFee : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with (tourism = museum or leisure = beach_resort or tourism = gallery) and access !~ private|no diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/handrail/AddHandrail.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/handrail/AddHandrail.kt index 3eb572c2d6..78c61f2efa 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/handrail/AddHandrail.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/handrail/AddHandrail.kt @@ -2,26 +2,23 @@ package de.westnordost.streetcomplete.quests.handrail import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddHandrail(overpassApi: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(overpassApi) { +class AddHandrail : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway = steps and (!indoor or indoor = no) and access !~ private|no and (!conveying or conveying = no) and ( !handrail and !handrail:center and !handrail:left and !handrail:right - or handrail = no and handrail older today -${r * 4} years - or handrail older today -${r * 8} years - or older today -${r * 8} years + or handrail = no and handrail older today -4 years + or handrail older today -8 years + or older today -8 years ) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumber.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumber.kt index d7ec259f34..94bab0d8d8 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumber.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumber.kt @@ -1,26 +1,19 @@ package de.westnordost.streetcomplete.quests.housenumber -import android.util.Log +import de.westnordost.osmapi.map.MapDataWithGeometry +import de.westnordost.osmapi.map.data.* -import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.osmapi.map.data.Element -import de.westnordost.osmapi.map.data.LatLon import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolygonsGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.quest.AllCountriesExcept -import de.westnordost.streetcomplete.data.elementfilter.DEFAULT_MAX_QUESTS -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox -import de.westnordost.streetcomplete.data.elementfilter.toOverpassBboxFilter -import de.westnordost.streetcomplete.util.FlattenIterable +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.util.LatLonRaster -import de.westnordost.streetcomplete.util.enclosingBoundingBox +import de.westnordost.streetcomplete.util.isCompletelyInside import de.westnordost.streetcomplete.util.isInMultipolygon -class AddHousenumber(private val overpass: OverpassMapDataAndGeometryApi) : OsmElementQuestType { +class AddHousenumber : OsmElementQuestType { override val commitMessage = "Add housenumbers" override val wikiLink = "Key:addr" @@ -38,111 +31,90 @@ class AddHousenumber(private val overpass: OverpassMapDataAndGeometryApi) : OsmE override fun getTitle(tags: Map) = R.string.quest_address_title - override fun isApplicableTo(element: Element): Boolean? = null + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val bbox = mapData.boundingBox ?: return listOf() - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - var ms = System.currentTimeMillis() + val addressNodesById = mapData.nodes.filter { nodesWithAddressFilter.matches(it) }.associateBy { it.id } + val addressNodeIds = addressNodesById.keys - val buildings = downloadBuildingsWithoutAddresses(bbox) ?: return false - // empty result: We are done - if (buildings.isEmpty()) return true + /** filter: only buildings with no address that usually should have an address + * ...that do not have an address node on their outline */ - val addrAreas = downloadAreasWithAddresses(bbox) ?: return false + val buildings = mapData.filter { + buildingsWithMissingAddressFilter.matches(it) + && !it.containsAnyNode(addressNodeIds, mapData) + }.toMutableList() - val extendedBbox = getBoundingBoxThatIncludes(buildings.values) + if (buildings.isEmpty()) return listOf() - val addrPositions = downloadFreeFloatingPositionsWithAddresses(extendedBbox) ?: return false + /** exclude buildings which are included in relations that have an address */ - Log.d("AddHousenumber", "Downloaded ${buildings.size} buildings with no address, " + - "${addrAreas.size} areas with address and ${addrPositions.size} address nodes " + - "in ${System.currentTimeMillis() - ms}ms" - ) - ms = System.currentTimeMillis() + val relationsWithAddress = mapData.relations.filter { nonMultipolygonRelationsWithAddressFilter.matches(it) } - val buildingPositions = LatLonRaster(extendedBbox, 0.0005) - for (buildingCenter in buildings.keys) { - buildingPositions.insert(buildingCenter) + buildings.removeAll { building -> + relationsWithAddress.any { it.containsWay(building.id) } } - // exclude buildings that are contained in an area with a housenumber - for (addrArea in addrAreas) { - for (buildingPos in buildingPositions.getAll(addrArea.getBounds())) { - if (buildingPos.isInMultipolygon(addrArea.polygons)) { - buildings.remove(buildingPos) - } - } - } + if (buildings.isEmpty()) return listOf() - var createdQuests = 0 - // only buildings with no housenumber-nodes inside them - for (building in buildings.values) { - // even though we could continue here, limit the max amount of quests created to the - // default maximum to avoid performance problems - if (createdQuests++ >= DEFAULT_MAX_QUESTS) break - - val addrContainedInBuilding = getPositionContainedInBuilding(building.geometry, addrPositions) - if (addrContainedInBuilding != null) { - addrPositions.remove(addrContainedInBuilding) - continue - } + /** exclude buildings that intersect with the bounding box because it is not possible to + ascertain for these if there is an address node within the building - it could be outside + the bounding box */ + + val buildingGeometriesById = buildings.associate { + it.id to mapData.getGeometry(it.type, it.id) as? ElementPolygonsGeometry + } - handler(building.element, building.geometry) + buildings.removeAll { building -> + val buildingBounds = buildingGeometriesById[building.id]?.getBounds() + (buildingBounds == null || !buildingBounds.isCompletelyInside(bbox)) } - Log.d("AddHousenumber", "Processing data took ${System.currentTimeMillis() - ms}ms") + if (buildings.isEmpty()) return listOf() - return true - } + /** exclude buildings that contain an address node somewhere within their area */ - private fun downloadBuildingsWithoutAddresses(bbox: BoundingBox): MutableMap? { - val buildingsByCenterPoint = mutableMapOf() - val query = getBuildingsWithoutAddressesOverpassQuery(bbox) - val success = overpass.query(query) { element, geometry -> - if (geometry is ElementPolygonsGeometry) { - buildingsByCenterPoint[geometry.center] = ElementWithArea(element, geometry) - } + val addressPositions = LatLonRaster(bbox, 0.0005) + for (node in addressNodesById.values) { + addressPositions.insert(node.position) } - return if (success) buildingsByCenterPoint else null - } - private fun downloadFreeFloatingPositionsWithAddresses(bbox: BoundingBox): LatLonRaster? { - val grid = LatLonRaster(bbox, 0.0005) - val query = getFreeFloatingAddressesOverpassQuery(bbox) - val success = overpass.query(query) { _, geometry -> - if (geometry != null) grid.insert(geometry.center) + buildings.removeAll { building -> + val buildingGeometry = buildingGeometriesById[building.id] + if (buildingGeometry != null) { + val nearbyAddresses = addressPositions.getAll(buildingGeometry.getBounds()) + nearbyAddresses.any { it.isInMultipolygon(buildingGeometry.polygons) } + } else true } - return if (success) grid else null - } - private fun downloadAreasWithAddresses(bbox: BoundingBox): List? { - val areas = mutableListOf() - val query = getNonBuildingAreasWithAddressesOverpassQuery(bbox) - val success = overpass.query(query) { _, geometry -> - if (geometry is ElementPolygonsGeometry) areas.add(geometry) - } - return if (success) areas else null - } + if (buildings.isEmpty()) return listOf() + + /** exclude buildings that are contained in an area that has an address */ - private fun getPositionContainedInBuilding(building: ElementPolygonsGeometry, positions: LatLonRaster): LatLon? { - for (pos in positions.getAll(building.getBounds())) { - if (pos.isInMultipolygon(building.polygons)) return pos + val areasWithAddresses = mapData + .filter { nonBuildingAreasWithAddressFilter.matches(it) } + .mapNotNull { mapData.getGeometry(it.type, it.id) as? ElementPolygonsGeometry } + + val buildingsByCenterPosition: Map = buildings.associateBy { buildingGeometriesById[it.id]?.center } + + val buildingPositions = LatLonRaster(bbox, 0.0005) + for (buildingCenterPosition in buildingsByCenterPosition.keys) { + if (buildingCenterPosition != null) buildingPositions.insert(buildingCenterPosition) } - return null - } - private fun getBoundingBoxThatIncludes(buildings: Iterable): BoundingBox { - // see #885: The area in which the app should search for address nodes (and areas) must be - // adjusted to the bounding box of all the buildings found. The found buildings may in parts - // not be within the specified bounding box. But in exactly that part, there may be an - // address + for (areaWithAddress in areasWithAddresses) { + val nearbyBuildings = buildingPositions.getAll(areaWithAddress.getBounds()) + val buildingPositionsInArea = nearbyBuildings.filter { it.isInMultipolygon(areaWithAddress.polygons) } + val buildingsInArea = buildingPositionsInArea.mapNotNull { buildingsByCenterPosition[it] } - val allThePoints = FlattenIterable(LatLon::class.java) - for (building in buildings) { - allThePoints.add(building.geometry.polygons) + buildings.removeAll(buildingsInArea) } - return allThePoints.enclosingBoundingBox() + + return buildings } + override fun isApplicableTo(element: Element): Boolean? = null + override fun createForm() = AddHousenumberForm() override fun applyAnswerTo(answer: HousenumberAnswer, changes: StringMapChangesBuilder) { @@ -161,66 +133,63 @@ class AddHousenumber(private val overpass: OverpassMapDataAndGeometryApi) : OsmE } is HouseAndBlockNumber -> { changes.add("addr:housenumber", answer.houseNumber) - changes.add("addr:block_number", answer.blockNumber) + changes.addOrModify("addr:block_number", answer.blockNumber) } } } - -} - -/** Query that returns all areas that are not buildings but have addresses */ -private fun getNonBuildingAreasWithAddressesOverpassQuery(bbox: BoundingBox): String { - val globalBbox = bbox.toGlobalOverpassBBox() - return """ - $globalBbox - wr[!building] $ANY_ADDRESS_FILTER; - out geom; - """.trimIndent() -} - -/** Query that returns all buildings that don't have an address node on their outline, nor are - * part of a relation that has a housenumber */ -private fun getBuildingsWithoutAddressesOverpassQuery(bbox: BoundingBox): String { - val bboxFilter = bbox.toOverpassBboxFilter() - return """ - ( - way$BUILDINGS_WITHOUT_ADDRESS_FILTER$bboxFilter; - rel$BUILDINGS_WITHOUT_ADDRESS_FILTER$bboxFilter; - ) -> .buildings; - - .buildings << -> .relations_containing_buildings; - rel.relations_containing_buildings$ANY_ADDRESS_FILTER; >> -> .elements_contained_in_relations_with_addr; - - .buildings > -> .building_nodes; - node.building_nodes$ANY_ADDRESS_FILTER; < -> .buildings_with_addr_nodes; - - ((.buildings; - .buildings_with_addr_nodes;); - .elements_contained_in_relations_with_addr; ); - out meta geom; - """.trimIndent() -} - -/** Query that returns all address nodes that are not part of any building outline */ -private fun getFreeFloatingAddressesOverpassQuery(bbox: BoundingBox): String { - val globalBbox = bbox.toGlobalOverpassBBox() - return """ - $globalBbox - ( - node$ANY_ADDRESS_FILTER; - - (wr[building];>;); - ); - out skel; - """.trimIndent() } -private data class ElementWithArea(val element: Element, val geometry: ElementPolygonsGeometry) - -private const val ANY_ADDRESS_FILTER = - "[~'^addr:(housenumber|housename|conscriptionnumber|streetnumber)$'~'.']" +private val nonBuildingAreasWithAddressFilter by lazy { """ + ways, relations with + !building and ~"addr:(housenumber|housename|conscriptionnumber|streetnumber)" + """.toElementFilterExpression()} + +private val nonMultipolygonRelationsWithAddressFilter by lazy { """ + relations with + type != multipolygon + and ~"addr:(housenumber|housename|conscriptionnumber|streetnumber)" + """.toElementFilterExpression()} + +private val nodesWithAddressFilter by lazy { """ + nodes with ~"addr:(housenumber|housename|conscriptionnumber|streetnumber)" + """.toElementFilterExpression()} + +private val buildingsWithMissingAddressFilter by lazy { """ + ways, relations with + building ~ ${buildingTypesThatShouldHaveAddresses.joinToString("|")} + and location != underground + and ruins != yes + and abandoned != yes + and !addr:housenumber + and !addr:housename + and !addr:conscriptionnumber + and !addr:streetnumber + and !noaddress + and !nohousenumber + """.toElementFilterExpression()} + +private val buildingTypesThatShouldHaveAddresses = listOf( + "house", "residential", "apartments", "detached", "terrace", "dormitory", "semi", + "semidetached_house", "farm", "school", "civic", "college", "university", "public", "hospital", + "kindergarten", "train_station", "hotel", "retail", "commercial" +) + +private fun Element.containsAnyNode(nodeIds: Set, mapData: MapDataWithGeometry): Boolean = + when (this) { + is Way -> this.nodeIds.any { it in nodeIds } + is Relation -> containsAnyNode(nodeIds, mapData) + else -> false + } -private const val NO_ADDRESS_FILTER = - "[!'addr:housenumber'][!'addr:housename'][!'addr:conscriptionnumber'][!'addr:streetnumber'][!noaddress][!nohousenumber]" +/** return whether any way contained in this relation contains any of the nodes with the given ids */ +private fun Relation.containsAnyNode(nodeIds: Set, mapData: MapDataWithGeometry): Boolean = + members + .filter { it.type == Element.Type.WAY } + .any { member -> + val way = mapData.getWay(member.ref) + way?.nodeIds?.any { it in nodeIds } ?: false + } -private const val BUILDINGS_WITHOUT_ADDRESS_FILTER = - "['building'~'^(house|residential|apartments|detached|terrace|dormitory|semi|semidetached_house|farm|" + - "school|civic|college|university|public|hospital|kindergarten|train_station|hotel|" + - "retail|commercial)$'][location!=underground][ruins!=yes]" + NO_ADDRESS_FILTER +/** return whether any of the ways with the given ids are contained in this relation */ +private fun Relation.containsWay(wayId: Long): Boolean = + members.any { it.type == Element.Type.WAY && wayId == it.ref } \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/internet_access/AddInternetAccess.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/internet_access/AddInternetAccess.kt index dd92aa4fbd..bb7b63e1c3 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/internet_access/AddInternetAccess.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/internet_access/AddInternetAccess.kt @@ -2,15 +2,12 @@ package de.westnordost.streetcomplete.quests.internet_access import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddInternetAccess(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddInternetAccess : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with ( amenity ~ library|community_centre|youth_centre @@ -20,7 +17,7 @@ class AddInternetAccess(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsSt and ( !internet_access or internet_access = yes - or internet_access older today -${r * 2} years + or internet_access older today -2 years ) """ /* Asked less often than for example opening hours because this quest is only asked for diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/leaf_detail/AddForestLeafType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/leaf_detail/AddForestLeafType.kt index ad69616169..4e9a2bf258 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/leaf_detail/AddForestLeafType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/leaf_detail/AddForestLeafType.kt @@ -1,34 +1,40 @@ package de.westnordost.streetcomplete.quests.leaf_detail -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolygonsGeometry +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.util.measuredMultiPolygonArea + +class AddForestLeafType : OsmElementQuestType { + private val areaFilter by lazy { """ + ways, relations with landuse = forest or natural = wood and !leaf_type + """.toElementFilterExpression()} + + private val wayFilter by lazy { """ + ways with natural = tree_row and !leaf_type + """.toElementFilterExpression()} -class AddForestLeafType(private val overpassApi: OverpassMapDataAndGeometryApi) : OsmElementQuestType { override val commitMessage = "Add leaf type" override val wikiLink = "Key:leaf_type" override val icon = R.drawable.ic_quest_leaf override val isSplitWayEnabled = true - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox), handler) + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val forests = mapData + .filter { areaFilter.matches(it) } + .filter { + val geometry = mapData.getGeometry(it.type, it.id) as? ElementPolygonsGeometry + val area = geometry?.polygons?.measuredMultiPolygonArea() ?: 0.0 + area > 0.0 && area < 10000 + } + val treeRows = mapData.filter { wayFilter.matches(it) } + return forests + treeRows } - private fun getOverpassQuery(bbox: BoundingBox) = """ - ${bbox.toGlobalOverpassBBox()} - ( - wr[landuse = forest][!leaf_type](if: length()<700.0); - wr[natural = wood][!leaf_type](if: length()<700.0); - way[natural = tree_row][!leaf_type](if: length()<700.0); - ); - ${getQuestPrintStatement()}""".trimIndent() - override fun isApplicableTo(element: Element):Boolean? = null override fun getTitle(tags: Map) = R.string.quest_leafType_title diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/max_height/AddMaxHeight.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/max_height/AddMaxHeight.kt index 1fc3fc8093..728062b744 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/max_height/AddMaxHeight.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/max_height/AddMaxHeight.kt @@ -1,28 +1,24 @@ package de.westnordost.streetcomplete.quests.max_height -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType -class AddMaxHeight(private val overpassApi: OverpassMapDataAndGeometryApi) : OsmElementQuestType { +class AddMaxHeight : OsmElementQuestType { - private val nodeFilter by lazy { ElementFiltersParser().parse(""" + private val nodeFilter by lazy { """ nodes with ( barrier = height_restrictor or amenity = parking_entrance and parking ~ underground|multi-storey ) and !maxheight and !maxheight:physical - """)} + """.toElementFilterExpression()} - private val wayFilter by lazy { ElementFiltersParser().parse(""" + private val wayFilter by lazy { """ ways with ( highway ~ motorway|motorway_link|trunk|trunk_link|primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential|living_street|track|road @@ -30,7 +26,7 @@ class AddMaxHeight(private val overpassApi: OverpassMapDataAndGeometryApi) : Osm ) and (covered = yes or tunnel ~ yes|building_passage|avalanche_protector) and !maxheight and !maxheight:physical - """)} + """.toElementFilterExpression()} override val commitMessage = "Add maximum heights" override val wikiLink = "Key:maxheight" @@ -49,20 +45,12 @@ class AddMaxHeight(private val overpassApi: OverpassMapDataAndGeometryApi) : Osm } } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable = + mapData.nodes.filter { nodeFilter.matches(it) } + mapData.ways.filter { wayFilter.matches(it) } + override fun isApplicableTo(element: Element) = nodeFilter.matches(element) || wayFilter.matches(element) - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getNodeOverpassQuery(bbox), handler) - && overpassApi.query(getWayOverpassQuery(bbox), handler) - } - - private fun getNodeOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + nodeFilter.toOverpassQLString() + "\n" + getQuestPrintStatement() - - private fun getWayOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + wayFilter.toOverpassQLString() + "\n" + getQuestPrintStatement() - override fun createForm() = AddMaxHeightForm() override fun applyAnswerTo(answer: MaxHeightAnswer, changes: StringMapChangesBuilder) { diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/max_speed/AddMaxSpeed.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/max_speed/AddMaxSpeed.kt index 51cafa35af..63314d6136 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/max_speed/AddMaxSpeed.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/max_speed/AddMaxSpeed.kt @@ -2,14 +2,13 @@ package de.westnordost.streetcomplete.quests.max_speed import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ANYTHING_UNPAVED -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.quest.AllCountriesExcept -class AddMaxSpeed(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddMaxSpeed : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway ~ motorway|trunk|primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential and !maxspeed and !maxspeed:advisory and !maxspeed:forward and !maxspeed:backward and !source:maxspeed and !zone:maxspeed and !maxspeed:type and !zone:traffic diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/max_weight/AddMaxWeight.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/max_weight/AddMaxWeight.kt index 64c176b36b..45000b52d4 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/max_weight/AddMaxWeight.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/max_weight/AddMaxWeight.kt @@ -1,19 +1,18 @@ package de.westnordost.streetcomplete.quests.max_weight import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.max_weight.MaxWeightSign.* -class AddMaxWeight(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddMaxWeight : OsmFilterQuestType() { override val commitMessage = "Add maximum allowed weight" override val wikiLink = "Key:maxweight" override val icon = R.drawable.ic_quest_max_weight override val hasMarkersAtEnds = true - override val tagFilters = """ + override val elementFilter = """ ways with highway ~ trunk|trunk_link|primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential|living_street|service and service != driveway and !maxweight and maxweight:signed != no diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_capacity/AddMotorcycleParkingCapacity.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_capacity/AddMotorcycleParkingCapacity.kt index 03871837f9..c91c832d4f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_capacity/AddMotorcycleParkingCapacity.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_capacity/AddMotorcycleParkingCapacity.kt @@ -2,18 +2,15 @@ package de.westnordost.streetcomplete.quests.motorcycle_parking_capacity import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddMotorcycleParkingCapacity(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddMotorcycleParkingCapacity : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with amenity = motorcycle_parking and access !~ private|no - and (!capacity or capacity older today -${r * 4} years) + and (!capacity or capacity older today -4 years) """ override val commitMessage = "Add motorcycle parking capacities" override val wikiLink = "Tag:amenity=motorcycle_parking" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_cover/AddMotorcycleParkingCover.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_cover/AddMotorcycleParkingCover.kt index 485474d1b2..21a8834efc 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_cover/AddMotorcycleParkingCover.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/motorcycle_parking_cover/AddMotorcycleParkingCover.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.motorcycle_parking_cover import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddMotorcycleParkingCover(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddMotorcycleParkingCover : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with amenity = motorcycle_parking and access !~ private|no and !covered diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/oneway/AddOneway.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/oneway/AddOneway.kt index 2a114c93fd..af603c2123 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/oneway/AddOneway.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/oneway/AddOneway.kt @@ -1,38 +1,32 @@ package de.westnordost.streetcomplete.quests.oneway -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.osmapi.map.data.Way import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.meta.ALL_ROADS import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.quests.bikeway.createCyclewaySides import de.westnordost.streetcomplete.quests.bikeway.estimatedWidth +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.quests.oneway.OnewayAnswer.* import de.westnordost.streetcomplete.quests.parking_lanes.* -class AddOneway( - private val overpassMapDataApi: OverpassMapDataAndGeometryApi -) : OsmElementQuestType { +class AddOneway : OsmElementQuestType { /** find all roads */ - private val allRoadsFilter by lazy { ElementFiltersParser().parse(""" + private val allRoadsFilter by lazy { """ ways with highway ~ ${ALL_ROADS.joinToString("|")} and area != yes - """) } + """.toElementFilterExpression() } /** find only those roads eligible for asking for oneway */ - private val tagFilter by lazy { ElementFiltersParser().parse(""" + private val elementFilter by lazy { """ ways with highway ~ living_street|residential|service|tertiary|unclassified and !oneway and area != yes and junction != roundabout and (access !~ private|no or (foot and foot !~ private|no)) and lanes <= 1 and width - """) } + """.toElementFilterExpression() } override val commitMessage = "Add whether this road is a one-way road because it is quite slim" override val wikiLink = "Key:oneway" @@ -42,49 +36,41 @@ class AddOneway( override fun getTitle(tags: Map) = R.string.quest_oneway2_title - override fun isApplicableTo(element: Element): Boolean? = null - /* return null because we also want to have a look at the surrounding geometry to filter out - * (some) dead ends */ - - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - val query = bbox.toGlobalOverpassBBox() + "\n" + allRoadsFilter.toOverpassQLString() + getQuestPrintStatement() + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val allRoads = mapData.ways.filter { allRoadsFilter.matches(it) && it.nodeIds.size >= 2 } val connectionCountByNodeIds = mutableMapOf() - val onewayCandidates = mutableListOf>() - val result = overpassMapDataApi.query(query) { element, geometry -> - if (element is Way && element.nodeIds.size >= 2) { - for (nodeId in element.nodeIds) { - val prevCount = connectionCountByNodeIds[nodeId] ?: 0 - connectionCountByNodeIds[nodeId] = prevCount + 1 - } + val onewayCandidates = mutableListOf() - if (element.tags != null && tagFilter.matches(element)) { - // check if the width of the road minus the space consumed by parking lanes is quite narrow - val width = element.tags["width"]?.toFloatOrNull() - val isNarrow = width != null && width <= estimatedWidthConsumedByOtherThings(element.tags) + 4f - if (isNarrow) { - onewayCandidates.add(element to geometry) - } + for (road in allRoads) { + for (nodeId in road.nodeIds) { + val prevCount = connectionCountByNodeIds[nodeId] ?: 0 + connectionCountByNodeIds[nodeId] = prevCount + 1 + } + if (elementFilter.matches(road)) { + // check if the width of the road minus the space consumed by other stuff is quite narrow + val width = road.tags["width"]?.toFloatOrNull() + val isNarrow = width != null && width <= estimatedWidthConsumedByOtherThings(road.tags) + 4f + if (isNarrow) { + onewayCandidates.add(road) } } } - if (!result) return false - for ((way, geometry) in onewayCandidates) { + return onewayCandidates.filter { /* ways that are simply at the border of the download bounding box are treated as if they are dead ends. This is fine though, because it only leads to this quest not showing up for those streets (which is better than the other way round) */ - val hasConnectionOnBothEnds = - (connectionCountByNodeIds[way.nodeIds.first()] ?: 0) > 1 && - (connectionCountByNodeIds[way.nodeIds.last()] ?: 0) > 1 - - if (hasConnectionOnBothEnds) { - handler(way, geometry) - } + // check if the way has connections to other roads at both ends + (connectionCountByNodeIds[it.nodeIds.first()] ?: 0) > 1 && + (connectionCountByNodeIds[it.nodeIds.last()] ?: 0) > 1 } - return true } + override fun isApplicableTo(element: Element): Boolean? = null + /* return null because we also want to have a look at the surrounding geometry to filter out + * (some) dead ends */ + private fun estimatedWidthConsumedByOtherThings(tags: Map): Float { return estimateWidthConsumedByParkingLanes(tags) + estimateWidthConsumedByCycleLanes(tags) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/oneway_suspects/AddSuspectedOneway.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/oneway_suspects/AddSuspectedOneway.kt index 9fd50b51c0..ae7e43b3f1 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/oneway_suspects/AddSuspectedOneway.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/oneway_suspects/AddSuspectedOneway.kt @@ -1,33 +1,36 @@ package de.westnordost.streetcomplete.quests.oneway_suspects import android.util.Log +import de.westnordost.osmapi.map.MapDataWithGeometry -import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.osmapi.map.data.Element import de.westnordost.osmapi.map.data.LatLon -import de.westnordost.osmapi.map.data.Way import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.quests.oneway_suspects.data.TrafficFlowSegment import de.westnordost.streetcomplete.quests.oneway_suspects.data.TrafficFlowSegmentsApi import de.westnordost.streetcomplete.quests.oneway_suspects.data.WayTrafficFlowDao +import kotlin.math.hypot class AddSuspectedOneway( - private val overpassMapDataApi: OverpassMapDataAndGeometryApi, private val trafficFlowSegmentsApi: TrafficFlowSegmentsApi, private val db: WayTrafficFlowDao ) : OsmElementQuestType { - private val tagFilters = """ - ways with highway ~ trunk|trunk_link|primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential|living_street|pedestrian|track|road - and !oneway and junction != roundabout and area != yes - and (access !~ private|no or (foot and foot !~ private|no)) - """ + private val filter by lazy { """ + ways with + highway ~ trunk|trunk_link|primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential|living_street|pedestrian|track|road + and !oneway + and junction != roundabout + and area != yes + and ( + access !~ private|no + or (foot and foot !~ private|no) + ) + """.toElementFilterExpression() } override val commitMessage = "Add whether roads are one-way roads as they were marked as likely oneway by improveosm.org" @@ -36,55 +39,59 @@ class AddSuspectedOneway( override val hasMarkersAtEnds = true override val isSplitWayEnabled = true - private val filter by lazy { ElementFiltersParser().parse(tagFilters) } - override fun getTitle(tags: Map) = R.string.quest_oneway_title - override fun isApplicableTo(element: Element) = - filter.matches(element) && db.isForward(element.id) != null + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val bbox = mapData.boundingBox ?: return emptyList() - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { val trafficDirectionMap: Map> try { trafficDirectionMap = trafficFlowSegmentsApi.get(bbox) } catch (e: Exception) { - Log.e("AddOneway", "Unable to download traffic metadata", e) - return false + Log.e("AddSuspectedOneway", "Unable to download traffic metadata", e) + return emptyList() } - if (trafficDirectionMap.isEmpty()) return true - - val query = "way(id:${trafficDirectionMap.keys.joinToString(",")}); out meta geom;" - overpassMapDataApi.query(query) { element, geometry -> - fun checkValidAndHandle(element: Element, geometry: ElementGeometry?) { - if (geometry == null) return - if (geometry !is ElementPolylinesGeometry) return - // filter the data as ImproveOSM data may be outdated or catching too much - if (!filter.matches(element)) return - - val way = element as? Way ?: return - val segments = trafficDirectionMap[way.id] ?: return - /* exclude rings because the driving direction can then not be determined reliably - from the improveosm data and the quest should stay simple, i.e not require the - user to input it in those cases. Additionally, whether a ring-road is a oneway or - not is less valuable information (for routing) and many times such a ring will - actually be a roundabout. Oneway information on roundabouts is superfluous. - See #1320 */ - if (way.nodeIds.last() == way.nodeIds.first()) return - /* only create quest if direction can be clearly determined and is the same - direction for all segments belonging to one OSM way (because StreetComplete - cannot split ways up) */ - val isForward = isForward(geometry.polylines.first(), segments) ?: return - - db.put(way.id, isForward) - handler(element, geometry) - } - checkValidAndHandle(element, geometry) + val suspectedOnewayWayIds = trafficDirectionMap.keys + + val onewayCandidates = mapData.ways.filter { + // so, only the ways suspected by improveOSM to be oneways + it.id in suspectedOnewayWayIds && + // but also filter the data as ImproveOSM data may be outdated or catching too much + filter.matches(it) && + /* also exclude rings because the driving direction can then not be determined reliably + from the improveosm data and the quest should stay simple, i.e not require the + user to input it in those cases. Additionally, whether a ring-road is a oneway or + not is less valuable information (for routing) and many times such a ring will + actually be a roundabout. Oneway information on roundabouts is superfluous. + See #1320 */ + it.nodeIds.first() != it.nodeIds.last() + } + + // rehash traffic direction data into simple "way id -> forward/backward" data and persist + val onewayDirectionMap = onewayCandidates.associate { way -> + val segments = trafficDirectionMap[way.id] + val geometry = mapData.getWayGeometry(way.id) as? ElementPolylinesGeometry + val isForward = + if (segments != null && geometry != null) + isForward(geometry.polylines.first(), segments) + else null + + way.id to isForward + } + + for ((wayId, isForward) in onewayDirectionMap) { + if (isForward != null) db.put(wayId, isForward) } - return true + /* only create quest if direction could be clearly determined (isForward != null) and is the + same direction for all segments belonging to one OSM way */ + return onewayCandidates.filter { onewayDirectionMap[it.id] != null } } + override fun isApplicableTo(element: Element) = + filter.matches(element) && db.isForward(element.id) != null + /** returns true if all given [trafficFlowSegments] point forward in relation to the * direction of the OSM [way] and false if they all point backward. * @@ -113,9 +120,8 @@ class AddSuspectedOneway( private fun findClosestPositionIndexOf(positions: List, latlon: LatLon): Int { var shortestDistance = 1.0 var result = -1 - var index = 0 - for (pos in positions) { - val distance = Math.hypot( + for ((index, pos) in positions.withIndex()) { + val distance = hypot( pos.longitude - latlon.longitude, pos.latitude - latlon.latitude ) @@ -123,7 +129,6 @@ class AddSuspectedOneway( shortestDistance = distance result = index } - index++ } return result diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHours.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHours.kt index 1b8bcc0c4e..7a03198b0e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHours.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHours.kt @@ -1,32 +1,25 @@ package de.westnordost.streetcomplete.quests.opening_hours -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.osmfeatures.FeatureDictionary import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.meta.updateWithCheckDate +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.ktx.containsAny import de.westnordost.streetcomplete.quests.opening_hours.parser.toOpeningHoursRows import de.westnordost.streetcomplete.quests.opening_hours.parser.toOpeningHoursRules -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore import java.util.concurrent.FutureTask class AddOpeningHours ( - private val overpassApi: OverpassMapDataAndGeometryApi, - private val featureDictionaryFuture: FutureTask, - private val r: ResurveyIntervalsStore + private val featureDictionaryFuture: FutureTask ) : OsmElementQuestType { /* See also AddWheelchairAccessBusiness and AddPlaceName, which has a similar list and is/should be ordered in the same way for better overview */ - private val filter by lazy { ElementFiltersParser().parse(""" + private val filter by lazy { (""" nodes, ways, relations with ( ( @@ -89,13 +82,12 @@ class AddOpeningHours ( ) and !opening_hours ) - or opening_hours older today -${r * 1} years + or opening_hours older today -1 years ) and (access !~ private|no) and (name or brand or noname = yes) and opening_hours:signed != no - """.trimIndent() - )} + """.trimIndent()).toElementFilterExpression() } override val commitMessage = "Add opening hours" override val wikiLink = "Key:opening_hours" @@ -132,11 +124,8 @@ class AddOpeningHours ( } } - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox)) { element, geometry -> - if (isApplicableTo(element)) handler(element, geometry) - } - } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable = + mapData.filter { isApplicableTo(it) } override fun isApplicableTo(element: Element) : Boolean { if (!filter.matches(element)) return false @@ -174,9 +163,6 @@ class AddOpeningHours ( } } - private fun getOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + filter.toOverpassQLString() + getQuestPrintStatement() - private fun hasName(tags: Map?) = hasProperName(tags) || hasFeatureName(tags) private fun hasProperName(tags: Map?): Boolean = diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/orchard_produce/AddOrchardProduce.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/orchard_produce/AddOrchardProduce.kt index a4f5881f3f..c8a367069a 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/orchard_produce/AddOrchardProduce.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/orchard_produce/AddOrchardProduce.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.orchard_produce import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddOrchardProduce(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType>(o) { +class AddOrchardProduce : OsmFilterQuestType>() { - override val tagFilters = """ + override val elementFilter = """ ways, relations with landuse = orchard and !trees and !produce and !crop and orchard != meadow_orchard diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/parking_access/AddParkingAccess.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/parking_access/AddParkingAccess.kt index a2131f4717..da3cca0e93 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/parking_access/AddParkingAccess.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/parking_access/AddParkingAccess.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.parking_access import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddParkingAccess(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddParkingAccess : OsmFilterQuestType() { - override val tagFilters = "nodes, ways, relations with amenity=parking and (!access or access=unknown)" + override val elementFilter = "nodes, ways, relations with amenity=parking and (!access or access=unknown)" override val commitMessage = "Add type of parking access" override val wikiLink = "Tag:amenity=parking" override val icon = R.drawable.ic_quest_parking_access diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/parking_fee/AddParkingFee.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/parking_fee/AddParkingFee.kt index 6859121894..06e35af67e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/parking_fee/AddParkingFee.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/parking_fee/AddParkingFee.kt @@ -2,20 +2,17 @@ package de.westnordost.streetcomplete.quests.parking_fee import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddParkingFee(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddParkingFee : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with amenity = parking and access ~ yes|customers|public and ( !fee and !fee:conditional - or fee older today -${r * 8} years + or fee older today -8 years ) """ override val commitMessage = "Add whether there is a parking fee" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/parking_type/AddParkingType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/parking_type/AddParkingType.kt index b3543e07c2..2f9d31a7c2 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/parking_type/AddParkingType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/parking_type/AddParkingType.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.parking_type import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddParkingType(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddParkingType : OsmFilterQuestType() { - override val tagFilters = "nodes, ways, relations with amenity = parking and !parking" + override val elementFilter = "nodes, ways, relations with amenity = parking and !parking" override val commitMessage = "Add parking type" override val wikiLink = "Tag:amenity=parking" override val icon = R.drawable.ic_quest_parking diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/place_name/AddPlaceName.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/place_name/AddPlaceName.kt index bab0a2fe28..5c2af71dbe 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/place_name/AddPlaceName.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/place_name/AddPlaceName.kt @@ -1,24 +1,19 @@ package de.westnordost.streetcomplete.quests.place_name -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.osmfeatures.FeatureDictionary import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import java.util.concurrent.FutureTask class AddPlaceName( - private val overpassApi: OverpassMapDataAndGeometryApi, private val featureDictionaryFuture: FutureTask ) : OsmElementQuestType { - private val filter by lazy { ElementFiltersParser().parse(""" + private val filter by lazy { (""" nodes, ways, relations with ( shop and shop !~ no|vacant @@ -96,8 +91,7 @@ class AddPlaceName( ).map { it.key + " ~ " + it.value.joinToString("|") }.joinToString("\n or ") + "\n" + """ ) and !name and !brand and noname != yes and name:signed != no - """.trimIndent() - )} + """.trimIndent()).toElementFilterExpression() } override val commitMessage = "Determine place names" override val wikiLink = "Key:name" @@ -108,12 +102,8 @@ class AddPlaceName( override fun getTitleArgs(tags: Map, featureName: Lazy) = featureName.value?.let { arrayOf(it) } ?: arrayOf() - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox)) { element, geometry -> - // only show places without names as quests for which a feature name is available - if (hasFeatureName(element.tags)) handler(element, geometry) - } - } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable = + mapData.filter { isApplicableTo(it) } override fun isApplicableTo(element: Element) = filter.matches(element) && hasFeatureName(element.tags) @@ -127,9 +117,6 @@ class AddPlaceName( } } - private fun getOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + filter.toOverpassQLString() + getQuestPrintStatement() - private fun hasFeatureName(tags: Map?): Boolean = tags?.let { featureDictionaryFuture.get().byTags(it).find().isNotEmpty() } ?: false } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/playground_access/AddPlaygroundAccess.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/playground_access/AddPlaygroundAccess.kt index ca4e602054..5e76e9050c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/playground_access/AddPlaygroundAccess.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/playground_access/AddPlaygroundAccess.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.playground_access import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddPlaygroundAccess(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddPlaygroundAccess : OsmFilterQuestType() { - override val tagFilters = "nodes, ways, relations with leisure = playground and (!access or access = unknown)" + override val elementFilter = "nodes, ways, relations with leisure = playground and (!access or access = unknown)" override val commitMessage = "Add playground access" override val wikiLink = "Tag:leisure=playground" override val icon = R.drawable.ic_quest_playground diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_collection_times/AddPostboxCollectionTimes.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_collection_times/AddPostboxCollectionTimes.kt index 0e91a924d5..9da87ace9e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_collection_times/AddPostboxCollectionTimes.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_collection_times/AddPostboxCollectionTimes.kt @@ -2,21 +2,18 @@ package de.westnordost.streetcomplete.quests.postbox_collection_times import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.containsAny import de.westnordost.streetcomplete.data.quest.NoCountriesExcept -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddPostboxCollectionTimes(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddPostboxCollectionTimes : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with amenity = post_box and access !~ private|no and collection_times:signed != no - and (!collection_times or collection_times older today -${r * 2} years) + and (!collection_times or collection_times older today -2 years) """ /* Don't ask again for postboxes without signed collection times. This is very unlikely to diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_ref/AddPostboxRef.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_ref/AddPostboxRef.kt index fd4b559880..f550ac3fe1 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_ref/AddPostboxRef.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/postbox_ref/AddPostboxRef.kt @@ -2,14 +2,13 @@ package de.westnordost.streetcomplete.quests.postbox_ref import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.quest.NoCountriesExcept -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.containsAny -class AddPostboxRef(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddPostboxRef : OsmFilterQuestType() { - override val tagFilters = "nodes with amenity = post_box and !ref and !ref:signed" + override val elementFilter = "nodes with amenity = post_box and !ref and !ref:signed" override val icon = R.drawable.ic_quest_mail_ref override val commitMessage = "Add postbox refs" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/powerpoles_material/AddPowerPolesMaterial.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/powerpoles_material/AddPowerPolesMaterial.kt index 3e2042a4ad..306151eeaf 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/powerpoles_material/AddPowerPolesMaterial.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/powerpoles_material/AddPowerPolesMaterial.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.powerpoles_material import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddPowerPolesMaterial(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddPowerPolesMaterial : OsmFilterQuestType() { - override val tagFilters = "nodes with power = pole and !material" + override val elementFilter = "nodes with power = pole and !material" override val commitMessage = "Add powerpoles material type" override val wikiLink = "Tag:power=pole" override val icon = R.drawable.ic_quest_power diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrier.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrier.kt index 1f2d50c57a..a1bfd84f9e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrier.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrier.kt @@ -1,23 +1,26 @@ package de.westnordost.streetcomplete.quests.railway_crossing -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.elementfilter.filters.RelativeDate -import de.westnordost.streetcomplete.data.elementfilter.filters.TagOlderThan -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType + +class AddRailwayCrossingBarrier : OsmElementQuestType { -class AddRailwayCrossingBarrier( - private val overpassMapDataApi: OverpassMapDataAndGeometryApi, - private val r: ResurveyIntervalsStore -) : OsmElementQuestType { + private val crossingFilter by lazy { """ + nodes with + railway = level_crossing + and (!crossing:barrier or crossing:barrier older today -8 years) + """.toElementFilterExpression() } + + private val excludedWaysFilter by lazy { """ + ways with + highway and access ~ private|no + or railway ~ tram|abandoned + """.toElementFilterExpression() } override val commitMessage = "Add type of barrier for railway crossing" override val wikiLink = "Key:crossing:barrier" @@ -25,36 +28,21 @@ class AddRailwayCrossingBarrier( override fun getTitle(tags: Map) = R.string.quest_railway_crossing_barrier_title2 - override fun createForm() = AddRailwayCrossingBarrierForm() + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val excludedWayNodeIds = mutableSetOf() + mapData.ways + .filter { excludedWaysFilter.matches(it) } + .flatMapTo(excludedWayNodeIds) { it.nodeIds } - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassMapDataApi.query(getOverpassQuery(bbox), handler) + return mapData.nodes + .filter { crossingFilter.matches(it) && it.id !in excludedWayNodeIds } } override fun isApplicableTo(element: Element): Boolean? = null + override fun createForm() = AddRailwayCrossingBarrierForm() + override fun applyAnswerTo(answer: String, changes: StringMapChangesBuilder) { changes.updateWithCheckDate("crossing:barrier", answer) } - - private fun getOverpassQuery(bbox: BoundingBox) = """ - ${bbox.toGlobalOverpassBBox()} - - way[highway][access ~ '^(private|no)$']; node(w) -> .private_road_nodes; - way[railway ~ '^(tram|abandoned)$']; node(w) -> .excluded_railways_nodes; - (.private_road_nodes; .excluded_railways_nodes;) -> .excluded; - - node[railway = level_crossing] -> .crossings; - - node.crossings[!'crossing:barrier'] -> .crossings_with_unknown_barrier; - node.crossings${olderThan(8).toOverpassQLString()} -> .crossings_with_old_barrier; - - ((.crossings_with_unknown_barrier; .crossings_with_old_barrier;); - .excluded;); - - ${getQuestPrintStatement()} - """.trimIndent() - - private fun olderThan(years: Int) = - TagOlderThan("crossing:barrier", RelativeDate(-(r * 365 * years).toFloat())) - } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/recycling/AddRecyclingType.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/recycling/AddRecyclingType.kt index ab89977241..03ea4cf21c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/recycling/AddRecyclingType.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/recycling/AddRecyclingType.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.recycling import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.quests.recycling.RecyclingType.* -class AddRecyclingType(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddRecyclingType : OsmFilterQuestType() { - override val tagFilters = "nodes, ways, relations with amenity = recycling and !recycling_type" + override val elementFilter = "nodes, ways, relations with amenity = recycling and !recycling_type" override val commitMessage = "Add recycling type to recycling amenity" override val wikiLink = "Key:recycling_type" override val icon = R.drawable.ic_quest_recycling diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_glass/DetermineRecyclingGlass.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_glass/DetermineRecyclingGlass.kt index 0e3c6dc053..9fac65c5ef 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_glass/DetermineRecyclingGlass.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_glass/DetermineRecyclingGlass.kt @@ -1,16 +1,14 @@ package de.westnordost.streetcomplete.quests.recycling_glass import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.quest.AllCountriesExcept import de.westnordost.streetcomplete.quests.recycling_glass.RecyclingGlass.* -class DetermineRecyclingGlass(overpassApi: OverpassMapDataAndGeometryApi) : - SimpleOverpassQuestType(overpassApi) { +class DetermineRecyclingGlass : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with amenity = recycling and recycling_type = container and recycling:glass = yes and !recycling:glass_bottles """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_material/AddRecyclingContainerMaterials.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_material/AddRecyclingContainerMaterials.kt index 9895153ab2..e5f64387bd 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_material/AddRecyclingContainerMaterials.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/recycling_material/AddRecyclingContainerMaterials.kt @@ -1,67 +1,64 @@ package de.westnordost.streetcomplete.quests.recycling_material -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.elementfilter.filters.RelativeDate import de.westnordost.streetcomplete.data.elementfilter.filters.TagOlderThan -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox import de.westnordost.streetcomplete.data.meta.deleteCheckDatesForKey import de.westnordost.streetcomplete.data.meta.updateCheckDateForKey import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.util.LatLonRaster +import de.westnordost.streetcomplete.util.distanceTo +import de.westnordost.streetcomplete.util.enclosingBoundingBox -class AddRecyclingContainerMaterials( - private val overpassApi: OverpassMapDataAndGeometryApi, - private val r: ResurveyIntervalsStore -) : OsmElementQuestType { +class AddRecyclingContainerMaterials : OsmElementQuestType { + + private val filter by lazy { """ + nodes with + amenity = recycling and recycling_type = container + """.toElementFilterExpression() } override val commitMessage = "Add recycled materials to container" override val wikiLink = "Key:recycling" override val icon = R.drawable.ic_quest_recycling_container - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox), handler) - } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val bbox = mapData.boundingBox ?: return emptyList() - private val allKnownMaterials = RecyclingMaterial.values().map { "recycling:" + it.value } - - /* Return recycling containers that do either not have any recycling:* tag yet, or if they do, - * haven't been touched for 2 years and are exclusively recycling types selectable in - * StreetComplete. - * Also, exclude recycling containers right next to another because the user can't know if - * certain materials are already recycled in that other container */ - private fun getOverpassQuery(bbox: BoundingBox) = """ - ${bbox.toGlobalOverpassBBox()} - node[amenity = recycling][recycling_type = container] -> .all; - - node.all[~"^recycling:.*$" ~ "yes"] -> .with_recycling; - (.all; - .with_recycling;) -> .without_recycling; - - node.with_recycling[~"^(${allKnownMaterials.joinToString("|")})$" ~ "yes" ] -> .with_known_recycling; - (.with_recycling; - .with_known_recycling;) -> .with_unknown_recycling; - - node.all${olderThan(2).toOverpassQLString()} -> .old; - - (.without_recycling; (.old; - .with_unknown_recycling;);) -> .recyclings; - - foreach .recyclings ( - node[amenity = recycling][recycling_type = container](around: 20); - node._(if:count(nodes) == 1); - out meta geom; - ); - """.trimIndent() + val olderThan2Years = TagOlderThan("recycling", RelativeDate(-(365 * 2).toFloat())) + + val containers = mapData.nodes.filter { filter.matches(it) } + + /* Only recycling containers that do either not have any recycling:* tag yet or + * haven't been touched for 2 years and are exclusively recycling types selectable in + * StreetComplete. */ + val eligibleContainers = containers.filter { + !it.hasAnyRecyclingMaterials() || + (olderThan2Years.matches(it) && !it.hasUnknownRecyclingMaterials()) + } + /* Also, exclude recycling containers right next to another because the user can't know if + * certain materials are already recycled in that other container */ + val containerPositions = LatLonRaster(bbox, 0.0005) + for (container in containers) { + containerPositions.insert(container.position) + } + + val minDistance = 20.0 + return eligibleContainers.filter { container -> + val nearbyBounds = container.position.enclosingBoundingBox(minDistance) + val nearbyContainerPositions = containerPositions.getAll(nearbyBounds) + // only finds one position = only found self -> no other container is near + nearbyContainerPositions.count { container.position.distanceTo(it) <= minDistance } == 1 + } + } // can't determine by tags alone because we need info about geometry surroundings override fun isApplicableTo(element: Element): Boolean? = null - private fun olderThan(years: Int) = - TagOlderThan("recycling", RelativeDate(-(r * 365 * years).toFloat())) - override fun getTitle(tags: Map) = R.string.quest_recycling_materials_title override fun createForm() = AddRecyclingContainerMaterialsForm() @@ -137,3 +134,15 @@ class AddRecyclingContainerMaterials( changes.deleteCheckDatesForKey("recycling") } } + +private val allKnownMaterials = RecyclingMaterial.values().map { "recycling:" + it.value } + +private fun Element.hasAnyRecyclingMaterials(): Boolean = + tags?.any { it.key.startsWith("recycling:") && it.value == "yes" } ?: false + +private fun Element.hasUnknownRecyclingMaterials(): Boolean = + tags?.any { + it.key.startsWith("recycling:") && + it.key !in allKnownMaterials && + it.value == "yes" + } ?: true \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToPlaceOfWorship.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToPlaceOfWorship.kt index 546120bddd..36aa7598c4 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToPlaceOfWorship.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToPlaceOfWorship.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.religion import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddReligionToPlaceOfWorship(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddReligionToPlaceOfWorship : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with ( amenity = place_of_worship diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToWaysideShrine.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToWaysideShrine.kt index 48df601b3c..141b48b88c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToWaysideShrine.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/religion/AddReligionToWaysideShrine.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.religion import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddReligionToWaysideShrine(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddReligionToWaysideShrine : OsmFilterQuestType() { - override val tagFilters = + override val elementFilter = "nodes, ways, relations with historic = wayside_shrine and !religion and (access !~ private|no)" override val commitMessage = "Add religion for wayside shrine" override val wikiLink = "Key:religion" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/localized_name/AddRoadName.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/AddRoadName.kt similarity index 51% rename from app/src/main/java/de/westnordost/streetcomplete/quests/localized_name/AddRoadName.kt rename to app/src/main/java/de/westnordost/streetcomplete/quests/road_name/AddRoadName.kt index f8958c615b..5415496039 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/localized_name/AddRoadName.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/AddRoadName.kt @@ -1,26 +1,43 @@ package de.westnordost.streetcomplete.quests.road_name -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ALL_ROADS -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.quest.AllCountriesExcept -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.elementfilter.ElementFiltersParser -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.quests.LocalizedName +import de.westnordost.streetcomplete.quests.road_name.data.RoadNameSuggestionEntry import de.westnordost.streetcomplete.quests.road_name.data.RoadNameSuggestionsDao -import de.westnordost.streetcomplete.quests.road_name.data.putRoadNameSuggestion +import de.westnordost.streetcomplete.quests.road_name.data.toRoadNameByLanguage class AddRoadName( - private val overpassApi: OverpassMapDataAndGeometryApi, private val roadNameSuggestionsDao: RoadNameSuggestionsDao ) : OsmElementQuestType { + private val filter by lazy { """ + ways with + highway ~ primary|secondary|tertiary|unclassified|residential|living_street|pedestrian + and !name + and !ref + and noname != yes + and !junction + and area != yes + and ( + access !~ private|no + or foot and foot !~ private|no + ) + """.toElementFilterExpression() } + + private val roadsWithNamesFilter by lazy { """ + ways with + highway ~ ${ALL_ROADS.joinToString("|")} + and name + """.toElementFilterExpression() } + override val enabledInCountries = AllCountriesExcept("JP") override val commitMessage = "Determine road names and types" override val wikiLink = "Key:name" @@ -34,35 +51,25 @@ class AddRoadName( else R.string.quest_streetName_title - override fun isApplicableTo(element: Element) = ROADS_WITHOUT_NAMES_TFE.matches(element) + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val roadsWithoutNames = mapData.ways.filter { filter.matches(it) } - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - if (!overpassApi.query(getOverpassQuery(bbox), handler)) return false - if (!overpassApi.query(getStreetNameSuggestionsOverpassQuery(bbox), roadNameSuggestionsDao::putRoadNameSuggestion)) return false - return true + if (roadsWithoutNames.isNotEmpty()) { + val roadsWithNames = mapData.ways + .filter { roadsWithNamesFilter.matches(it) } + .mapNotNull { + val geometry = mapData.getWayGeometry(it.id) as? ElementPolylinesGeometry + val roadNamesByLanguage = it.tags?.toRoadNameByLanguage() + if (geometry != null && roadNamesByLanguage != null) { + RoadNameSuggestionEntry(it.id, roadNamesByLanguage, geometry.polylines.first()) + } else null + } + roadNameSuggestionsDao.putRoads(roadsWithNames) + } + return roadsWithoutNames } - /** returns overpass query string for creating the quests */ - private fun getOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + - ROADS_WITHOUT_NAMES + "->.unnamed;\n" + - "(\n" + - " way.unnamed['access' !~ '^(private|no)$'];\n" + - " way.unnamed['foot']['foot' !~ '^(private|no)$'];\n" + - "); " + - getQuestPrintStatement() - - /** return overpass query string to get roads with names near roads that don't have names - * private roads are not filtered out here, partially to reduce complexity but also - * because the road may have a private segment that is named already or is close to a road - * with a useful name - * */ - private fun getStreetNameSuggestionsOverpassQuery(bbox: BoundingBox) = - bbox.toGlobalOverpassBBox() + "\n" + """ - $ROADS_WITHOUT_NAMES -> .without_names; - $ROADS_WITH_NAMES -> .with_names; - way.with_names(around.without_names: $MAX_DIST_FOR_ROAD_NAME_SUGGESTION ); - out body geom;""".trimIndent() + override fun isApplicableTo(element: Element) = filter.matches(element) override fun createForm() = AddRoadNameForm() @@ -103,22 +110,8 @@ class AddRoadName( roadNameSuggestionsDao.putRoad( answer.wayId, roadNameByLanguage, answer.wayGeometry) } - companion object { - const val MAX_DIST_FOR_ROAD_NAME_SUGGESTION = 30.0 //m - - // to avoid spam, only ask for names on a limited set of roads - private const val NAMEABLE_ROADS = - "primary|secondary|tertiary|unclassified|residential|living_street|pedestrian" - private const val ROADS_WITHOUT_NAMES = - "way[highway ~ \"^($NAMEABLE_ROADS)$\"][!name][!ref][noname != yes][!junction][area != yes]" - // this must be the same as above but in tag filter expression syntax - private val ROADS_WITHOUT_NAMES_TFE by lazy { ElementFiltersParser().parse( - "ways with highway ~ $NAMEABLE_ROADS and !name and !ref and noname != yes and !junction and area != yes" - )} - - // but we can find name suggestions on any type of already-named road - private val ROADS_WITH_NAMES = - "way[highway ~ \"^(${ALL_ROADS.joinToString("|")})$\"][name]" + override fun cleanMetadata() { + roadNameSuggestionsDao.cleanUp() } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/localized_name/AddRoadNameForm.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameForm.kt similarity index 97% rename from app/src/main/java/de/westnordost/streetcomplete/quests/localized_name/AddRoadNameForm.kt rename to app/src/main/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameForm.kt index 6dd5226837..c817201709 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/localized_name/AddRoadNameForm.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameForm.kt @@ -52,7 +52,7 @@ class AddRoadNameForm : AAddLocalizedNameForm() { } return roadNameSuggestionsDao.getNames( listOf(polyline.first(), polyline.last()), - AddRoadName.MAX_DIST_FOR_ROAD_NAME_SUGGESTION + MAX_DIST_FOR_ROAD_NAME_SUGGESTION ) } @@ -145,4 +145,8 @@ class AddRoadNameForm : AAddLocalizedNameForm() { .setNegativeButton(R.string.quest_generic_confirmation_no, null) .show() } + + companion object { + const val MAX_DIST_FOR_ROAD_NAME_SUGGESTION = 30.0 //m + } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNameSuggestionsDao.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNameSuggestionsDao.kt index cae311fb79..3402198d57 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNameSuggestionsDao.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNameSuggestionsDao.kt @@ -7,6 +7,8 @@ import de.westnordost.osmapi.map.data.Element import javax.inject.Inject import de.westnordost.osmapi.map.data.LatLon +import de.westnordost.osmapi.map.data.Way +import de.westnordost.streetcomplete.ApplicationConstants import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry import de.westnordost.streetcomplete.ktx.getBlob @@ -21,9 +23,14 @@ import de.westnordost.streetcomplete.quests.road_name.data.RoadNamesTable.Column import de.westnordost.streetcomplete.quests.road_name.data.RoadNamesTable.NAME import de.westnordost.streetcomplete.util.Serializer import de.westnordost.streetcomplete.ktx.toObject +import de.westnordost.streetcomplete.ktx.transaction +import de.westnordost.streetcomplete.quests.road_name.data.RoadNamesTable.Columns.LAST_UPDATE import de.westnordost.streetcomplete.util.distanceToArcs import de.westnordost.streetcomplete.util.enclosingBoundingBox +import java.util.* import java.util.regex.Pattern +import kotlin.collections.ArrayList +import kotlin.collections.HashMap class RoadNameSuggestionsDao @Inject constructor( private val dbHelper: SQLiteOpenHelper, @@ -40,11 +47,20 @@ class RoadNameSuggestionsDao @Inject constructor( MIN_LATITUDE to bbox.minLatitude, MIN_LONGITUDE to bbox.minLongitude, MAX_LATITUDE to bbox.maxLatitude, - MAX_LONGITUDE to bbox.maxLongitude + MAX_LONGITUDE to bbox.maxLongitude, + LAST_UPDATE to Date().time ) db.replaceOrThrow(NAME, null, v) } + fun putRoads(roads: Iterable) { + db.transaction { + for (road in roads) { + putRoad(road.wayId, road.namesByLanguage, road.geometry) + } + } + } + /** returns something like [{"": "17th Street", "de": "17. Straße", "en": "17th Street", "international": "17 🛣 ️" }, ...] */ fun getNames(points: List, maxDistance: Double): List> { if (points.isEmpty()) return emptyList() @@ -93,19 +109,16 @@ class RoadNameSuggestionsDao @Inject constructor( // return only the road names, sorted by distance ascending return distancesByRoad.entries.sortedBy { it.value }.map { it.key } } -} -fun RoadNameSuggestionsDao.putRoadNameSuggestion(element: Element, geometry: ElementGeometry?) { - if (element.type != Element.Type.WAY) return - if (geometry !is ElementPolylinesGeometry) return - val namesByLanguage = element.tags?.toRoadNameByLanguage() ?: return - - putRoad(element.id, namesByLanguage, geometry.polylines.first()) + fun cleanUp() { + val oldNameSuggestionsTimestamp = System.currentTimeMillis() - ApplicationConstants.DELETE_UNSOLVED_QUESTS_AFTER + db.delete(NAME, "$LAST_UPDATE < ?", arrayOf(oldNameSuggestionsTimestamp.toString())) + } } /** OSM tags (i.e. name:de=Bäckergang) to map of language code -> name (i.e. de=Bäckergang) * int_name becomes "international" */ -private fun Map.toRoadNameByLanguage(): Map? { +fun Map.toRoadNameByLanguage(): Map? { val result = mutableMapOf() val namePattern = Pattern.compile("name(:(.*))?") for ((key, value) in this) { @@ -118,4 +131,10 @@ private fun Map.toRoadNameByLanguage(): Map? { } } return if (result.isEmpty()) null else result -} \ No newline at end of file +} + +data class RoadNameSuggestionEntry( + val wayId: Long, + val namesByLanguage: Map, + val geometry: List +) \ No newline at end of file diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNamesTable.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNamesTable.kt index 27d9a2194d..88f14c5a8e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNamesTable.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/road_name/data/RoadNamesTable.kt @@ -11,6 +11,7 @@ object RoadNamesTable { const val MIN_LONGITUDE = "min_longitude" const val MAX_LATITUDE = "max_latitude" const val MAX_LONGITUDE = "max_longitude" + const val LAST_UPDATE = "last_update" } const val CREATE = """ @@ -21,6 +22,7 @@ object RoadNamesTable { ${Columns.MIN_LATITUDE} double NOT NULL, ${Columns.MIN_LONGITUDE} double NOT NULL, ${Columns.MAX_LATITUDE} double NOT NULL, - ${Columns.MAX_LONGITUDE} double NOT NULL + ${Columns.MAX_LONGITUDE} double NOT NULL, + ${Columns.LAST_UPDATE} int NOT NULL );""" } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/roof_shape/AddRoofShape.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/roof_shape/AddRoofShape.kt index 853d28c75c..ca027294e5 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/roof_shape/AddRoofShape.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/roof_shape/AddRoofShape.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.roof_shape import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddRoofShape(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddRoofShape : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways, relations with roof:levels and roof:levels != 0 and !roof:shape and !3dr:type and !3dr:roof """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/segregated/AddCyclewaySegregation.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/segregated/AddCyclewaySegregation.kt index 7f74d0df8d..9fbb92527b 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/segregated/AddCyclewaySegregation.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/segregated/AddCyclewaySegregation.kt @@ -3,16 +3,13 @@ package de.westnordost.streetcomplete.quests.segregated import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ANYTHING_PAVED import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddCyclewaySegregation(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddCyclewaySegregation : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with ( (highway = path and bicycle = designated and foot = designated) @@ -21,7 +18,7 @@ class AddCyclewaySegregation(o: OverpassMapDataAndGeometryApi, r: ResurveyInterv ) and surface ~ ${ANYTHING_PAVED.joinToString("|")} and area != yes - and (!segregated or segregated older today -${r * 8} years) + and (!segregated or segregated older today -8 years) """ override val commitMessage = "Add segregated status for combined footway with cycleway" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/self_service/AddSelfServiceLaundry.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/self_service/AddSelfServiceLaundry.kt index 4ddc517fce..03b25fbe04 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/self_service/AddSelfServiceLaundry.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/self_service/AddSelfServiceLaundry.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.self_service import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddSelfServiceLaundry(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddSelfServiceLaundry : OsmFilterQuestType() { - override val tagFilters = "nodes, ways with shop = laundry and !self_service" + override val elementFilter = "nodes, ways with shop = laundry and !self_service" override val commitMessage = "Add self service info" override val wikiLink = "Tag:shop=laundry" override val icon = R.drawable.ic_quest_laundry diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/sidewalk/AddSidewalk.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/sidewalk/AddSidewalk.kt index 46c4d1066a..76fa802239 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/sidewalk/AddSidewalk.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/sidewalk/AddSidewalk.kt @@ -1,17 +1,39 @@ package de.westnordost.streetcomplete.quests.sidewalk -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.meta.ANYTHING_UNPAVED -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.util.isNear + +class AddSidewalk : OsmElementQuestType { -class AddSidewalk(private val overpassApi: OverpassMapDataAndGeometryApi) : OsmElementQuestType { + /* the filter additionally filters out ways that are unlikely to have sidewalks: + * unpaved roads, roads with very low speed limits and roads that are probably not developed + * enough to have pavement (that are not lit). + * Also, anything explicitly tagged as no pedestrians or explicitly tagged that the sidewalk + * is mapped as a separate way + * */ + private val filter by lazy { """ + ways with + highway ~ primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential + and area != yes + and motorroad != yes + and !sidewalk and !sidewalk:left and !sidewalk:right and !sidewalk:both + and (!maxspeed or maxspeed > 8 or maxspeed !~ "5 mph|walk") + and surface !~ ${ANYTHING_UNPAVED.joinToString("|")} + and lit = yes + and foot != no and access !~ private|no + and foot != use_sidepath + """.toElementFilterExpression() } + + private val maybeSeparatelyMappedSidewalksFilter by lazy { """ + ways with highway ~ path|footway|cycleway + """.toElementFilterExpression() } override val commitMessage = "Add whether there are sidewalks" override val wikiLink = "Key:sidewalk" @@ -20,42 +42,31 @@ class AddSidewalk(private val overpassApi: OverpassMapDataAndGeometryApi) : OsmE override fun getTitle(tags: Map) = R.string.quest_sidewalk_title - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassApi.query(getOverpassQuery(bbox), handler) - } + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val roadsWithMissingSidewalks = mapData.ways.filter { filter.matches(it) } + if (roadsWithMissingSidewalks.isEmpty()) return emptyList() - /** returns overpass query string to get streets without sidewalk info not near separately mapped - * sidewalks (and other paths) - */ - private fun getOverpassQuery(bbox: BoundingBox): String { - val minDistToWays = 15 //m + /* Unfortunately, the filter above is not enough. In OSM, sidewalks may be mapped as + * separate ways as well and it is not guaranteed that in this case, sidewalk = separate + * (or foot = use_sidepath) is always tagged on the main road then. So, all roads should + * be excluded whose center is within of ~15 meters of a footway, to be on the safe side. */ - // note: this query is very similar to the query in AddCycleway - return bbox.toGlobalOverpassBBox() + "\n" + - "way[highway ~ '^(primary|primary_link|secondary|secondary_link|tertiary|tertiary_link|unclassified|residential)$']" + - "[area != yes]" + - // not any motorroads - "[motorroad != yes]" + - // only without sidewalk tags - "[!sidewalk][!'sidewalk:left'][!'sidewalk:right'][!'sidewalk:both']" + - // not any with very low speed limit because they not very likely to have sidewalks - "[maxspeed !~ '^(8|7|6|5|5 mph|walk)$']" + - // not any unpaved because of the same reason - "[surface !~ '^(" + ANYTHING_UNPAVED.joinToString("|") + ")$']" + - "[lit = yes]" + - // not any explicitly tagged as no pedestrians - "[foot != no]" + - "[access !~ '^(private|no)$']" + - // some roads may be farther than minDistToWays from ways, not tagged with - // footway=separate/sidepath but may have a hint that there is a separately tagged - // sidewalk - "[foot != use_sidepath]" + - " -> .streets;\n" + - "way[highway ~ '^(path|footway|cycleway)$'](around.streets: " + minDistToWays + ")" + - " -> .ways;\n" + - "way.streets(around.ways: " + minDistToWays + ") -> .streets_near_ways;\n" + - "(.streets; - .streets_near_ways;);\n" + - getQuestPrintStatement() + val maybeSeparatelyMappedSidewalkGeometries = mapData.ways + .filter { maybeSeparatelyMappedSidewalksFilter.matches(it) } + .mapNotNull { mapData.getWayGeometry(it.id) as? ElementPolylinesGeometry } + if (maybeSeparatelyMappedSidewalkGeometries.isEmpty()) return roadsWithMissingSidewalks + + val minDistToWays = 15.0 //m + + // filter out roads with missing sidewalks that are near footways + return roadsWithMissingSidewalks.filter { road -> + val roadGeometry = mapData.getWayGeometry(road.id) as? ElementPolylinesGeometry + if (roadGeometry != null) { + !roadGeometry.isNear(minDistToWays, maybeSeparatelyMappedSidewalkGeometries) + } else { + false + } + } } override fun isApplicableTo(element: Element): Boolean? = null @@ -76,4 +87,4 @@ class AddSidewalk(private val overpassApi: OverpassMapDataAndGeometryApi) : OsmE else -> "none" } } - } +} diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/sport/AddSport.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/sport/AddSport.kt index 6a6d5c30a0..11645744da 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/sport/AddSport.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/sport/AddSport.kt @@ -1,11 +1,10 @@ package de.westnordost.streetcomplete.quests.sport import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddSport(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType>(o) { +class AddSport : OsmFilterQuestType>() { private val ambiguousSportValues = listOf( "team_handball", // -> not really ambiguous but same as handball @@ -14,7 +13,7 @@ class AddSport(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType american_football, soccer or other *_football ) - override val tagFilters = """ + override val elementFilter = """ nodes, ways with leisure = pitch and (!sport or sport ~ ${ambiguousSportValues.joinToString("|")} ) and (access !~ private|no) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/step_count/AddStepCount.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/step_count/AddStepCount.kt index e079c09127..b822a732dd 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/step_count/AddStepCount.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/step_count/AddStepCount.kt @@ -1,14 +1,12 @@ package de.westnordost.streetcomplete.quests.step_count import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddStepCount(overpassApi: OverpassMapDataAndGeometryApi) - : SimpleOverpassQuestType(overpassApi) { +class AddStepCount : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway = steps and (!indoor or indoor = no) and access !~ private|no diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/steps_incline/AddStepsIncline.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/steps_incline/AddStepsIncline.kt index 0331c2c227..ff27461dda 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/steps_incline/AddStepsIncline.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/steps_incline/AddStepsIncline.kt @@ -2,13 +2,12 @@ package de.westnordost.streetcomplete.quests.steps_incline import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.quests.steps_incline.StepsIncline.* -class AddStepsIncline(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddStepsIncline : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway = steps and (!indoor or indoor = no) and area != yes diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRamp.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRamp.kt index 717f313d19..4363478bad 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRamp.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRamp.kt @@ -3,15 +3,12 @@ package de.westnordost.streetcomplete.quests.steps_ramp import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.ktx.toYesNo -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddStepsRamp(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddStepsRamp : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway = steps and (!indoor or indoor = no) and access !~ private|no @@ -20,8 +17,8 @@ class AddStepsRamp(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) and ( !ramp or (ramp = yes and !ramp:stroller and !ramp:bicycle and !ramp:wheelchair) - or ramp = no and ramp older today -${r * 4} years - or ramp older today -${r * 8} years + or ramp = no and ramp older today -4 years + or ramp older today -8 years ) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/summit_register/AddSummitRegister.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/summit_register/AddSummitRegister.kt index 94b5a77f79..6cb5001025 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/summit_register/AddSummitRegister.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/summit_register/AddSummitRegister.kt @@ -1,26 +1,25 @@ package de.westnordost.streetcomplete.quests.summit_register -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.elementfilter.filters.RelativeDate -import de.westnordost.streetcomplete.data.elementfilter.filters.TagOlderThan -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox import de.westnordost.streetcomplete.data.meta.updateWithCheckDate +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.quest.NoCountriesExcept import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore +import de.westnordost.streetcomplete.util.distanceToArcs -class AddSummitRegister( - private val overpassMapDataApi: OverpassMapDataAndGeometryApi, - private val r: ResurveyIntervalsStore -) : OsmElementQuestType { +class AddSummitRegister : OsmElementQuestType { + + private val filter by lazy { """ + nodes with + natural = peak and name and + (!summit:register or summit:register older today -4 years) + """.toElementFilterExpression() } override val commitMessage = "Add whether summit register is present" override val wikiLink = "Key:summit:register" @@ -39,33 +38,30 @@ class AddSummitRegister( override fun getTitle(tags: Map) = R.string.quest_summit_register_title - override fun createForm() = YesNoQuestAnswerFragment() - - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassMapDataApi.query(getOverpassQuery(bbox), handler) + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val peaks = mapData.nodes.filter { filter.matches(it) } + if (peaks.isEmpty()) return emptyList() + + val hikingRoutes = mapData.relations + .filter { it.tags?.get("route") == "hiking" } + .mapNotNull { mapData.getRelationGeometry(it.id) as? ElementPolylinesGeometry } + if (hikingRoutes.isEmpty()) return emptyList() + + // yes, this is very inefficient, however, peaks are very rare + return peaks.filter { peak -> + hikingRoutes.any { hikingRoute -> + hikingRoute.polylines.any { ways -> + peak.position.distanceToArcs(ways) <= 10 + } + } + } } override fun isApplicableTo(element: Element): Boolean? = null + override fun createForm() = YesNoQuestAnswerFragment() + override fun applyAnswerTo(answer: Boolean, changes: StringMapChangesBuilder) { changes.updateWithCheckDate("summit:register", answer.toYesNo()) } - - private fun getOverpassQuery(bbox: BoundingBox) = """ - ${bbox.toGlobalOverpassBBox()} - - ( - relation["route"="hiking"]; - )->.hiking; - node(around.hiking:10)[natural=peak][!"summit:register"][name] -> .summits_with_unknown_status; - node(around.hiking:10)["summit:register"][name]${olderThan(4).toOverpassQLString()} -> .summits_with_old_status; - - (.summits_with_unknown_status; .summits_with_old_status;); - - ${getQuestPrintStatement()} - """.trimIndent() - - private fun olderThan(years: Int) = - TagOlderThan("summit:register", RelativeDate(-(r * 365 * years).toFloat())) - } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddCyclewayPartSurface.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddCyclewayPartSurface.kt index 525ded4f70..9c8dd9166f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddCyclewayPartSurface.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddCyclewayPartSurface.kt @@ -2,32 +2,27 @@ package de.westnordost.streetcomplete.quests.surface import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddCyclewayPartSurface(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddCyclewayPartSurface : OsmFilterQuestType() { - override val tagFilters = """ - ways with - ( + override val elementFilter = """ + ways with ( highway = cycleway or (highway ~ path|footway and bicycle != no) or (highway = bridleway and bicycle ~ designated|yes) ) and segregated = yes and ( - !cycleway:surface or - cycleway:surface older today -${r * 8} years - or - ( - cycleway:surface ~ paved|unpaved - and !cycleway:surface:note - and !note:cycleway:surface - ) - ) + !cycleway:surface + or cycleway:surface older today -8 years + or ( + cycleway:surface ~ paved|unpaved + and !cycleway:surface:note + and !note:cycleway:surface + ) + ) """ override val commitMessage = "Add path surfaces" override val wikiLink = "Key:surface" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddFootwayPartSurface.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddFootwayPartSurface.kt index 87ce235423..fc60cf2205 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddFootwayPartSurface.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddFootwayPartSurface.kt @@ -2,31 +2,26 @@ package de.westnordost.streetcomplete.quests.surface import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddFootwayPartSurface(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddFootwayPartSurface : OsmFilterQuestType() { - override val tagFilters = """ - ways with - ( + override val elementFilter = """ + ways with ( highway = footway or (highway ~ path|cycleway|bridleway and foot != no) ) and segregated = yes and ( - !footway:surface or - footway:surface older today -${r * 8} years - or - ( - footway:surface ~ paved|unpaved - and !footway:surface:note - and !note:footway:surface - ) - ) + !footway:surface + or footway:surface older today -8 years + or ( + footway:surface ~ paved|unpaved + and !footway:surface:note + and !note:footway:surface + ) + ) """ override val commitMessage = "Add path surfaces" override val wikiLink = "Key:surface" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddPathSurface.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddPathSurface.kt index 505b7cbda7..92c8348de3 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddPathSurface.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddPathSurface.kt @@ -3,29 +3,26 @@ package de.westnordost.streetcomplete.quests.surface import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ANYTHING_UNPAVED import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddPathSurface(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddPathSurface : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway ~ path|footway|cycleway|bridleway|steps and segregated != yes and access !~ private|no - and (!conveying or conveying = no) and (!indoor or indoor = no) + and (!conveying or conveying = no) + and (!indoor or indoor = no) and ( - !surface - or surface ~ ${ANYTHING_UNPAVED.joinToString("|")} and surface older today -${r * 4} years - or surface older today -${r * 8} years - or - ( - surface ~ paved|unpaved - and !surface:note - and !note:surface - ) + !surface + or surface ~ ${ANYTHING_UNPAVED.joinToString("|")} and surface older today -4 years + or surface older today -8 years + or ( + surface ~ paved|unpaved + and !surface:note + and !note:surface + ) ) """ /* ~paved ways are less likely to change the surface type */ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddRoadSurface.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddRoadSurface.kt index ad45bb101c..c87bfc32b8 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddRoadSurface.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/surface/AddRoadSurface.kt @@ -3,26 +3,22 @@ package de.westnordost.streetcomplete.quests.surface import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ANYTHING_UNPAVED import de.westnordost.streetcomplete.data.meta.updateWithCheckDate +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore +class AddRoadSurface : OsmFilterQuestType() { -class AddRoadSurface(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) : SimpleOverpassQuestType(o) { - override val tagFilters = """ + override val elementFilter = """ ways with highway ~ ${ROADS_WITH_SURFACES.joinToString("|")} - and - ( - !surface - or surface ~ ${ANYTHING_UNPAVED.joinToString("|")} and surface older today -${r * 4} years - or surface older today -${r * 12} years - or - ( - surface ~ paved|unpaved - and !surface:note - and !note:surface - ) + and ( + !surface + or surface ~ ${ANYTHING_UNPAVED.joinToString("|")} and surface older today -4 years + or surface older today -12 years + or ( + surface ~ paved|unpaved + and !surface:note + and !note:surface + ) ) and (access !~ private|no or (foot and foot !~ private|no)) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingBusStop.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingBusStop.kt index b85db1568a..1d7c986fcc 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingBusStop.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingBusStop.kt @@ -2,17 +2,14 @@ package de.westnordost.streetcomplete.quests.tactile_paving import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.data.quest.NoCountriesExcept import de.westnordost.streetcomplete.ktx.toYesNo -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddTactilePavingBusStop(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddTactilePavingBusStop : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with ( (public_transport = platform and (bus = yes or trolleybus = yes or tram = yes)) @@ -22,8 +19,8 @@ class AddTactilePavingBusStop(o: OverpassMapDataAndGeometryApi, r: ResurveyInter and physically_present != no and naptan:BusStopType != HAR and ( !tactile_paving - or tactile_paving = no and tactile_paving older today -${r * 4} years - or tactile_paving older today -${r * 8} years + or tactile_paving = no and tactile_paving older today -4 years + or tactile_paving older today -8 years ) """ override val commitMessage = "Add tactile pavings on bus stops" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalk.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalk.kt index ae9dc9f7dc..1ec00e55e9 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalk.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalk.kt @@ -1,25 +1,35 @@ package de.westnordost.streetcomplete.quests.tactile_paving -import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.elementfilter.filters.RelativeDate -import de.westnordost.streetcomplete.data.elementfilter.filters.TagOlderThan import de.westnordost.streetcomplete.data.meta.updateWithCheckDate +import de.westnordost.streetcomplete.data.elementfilter.toElementFilterExpression import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.quest.NoCountriesExcept -import de.westnordost.streetcomplete.data.elementfilter.getQuestPrintStatement -import de.westnordost.streetcomplete.data.elementfilter.toGlobalOverpassBBox +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.ktx.toYesNo -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddTactilePavingCrosswalk( - private val overpassMapDataApi: OverpassMapDataAndGeometryApi, - private val r: ResurveyIntervalsStore -) : OsmElementQuestType { +class AddTactilePavingCrosswalk : OsmElementQuestType { + + private val crossingFilter by lazy { """ + nodes with + ( + highway = traffic_signals and crossing = traffic_signals and foot != no + or highway = crossing and foot != no + ) + and ( + !tactile_paving + or tactile_paving = no and tactile_paving older today -4 years + or older today -8 years + ) + """.toElementFilterExpression() } + + private val excludedWaysFilter by lazy { """ + ways with + highway = cycleway and foot !~ yes|designated + or highway and access ~ private|no + """.toElementFilterExpression() } override val commitMessage = "Add tactile pavings on crosswalks" override val wikiLink = "Key:tactile_paving" @@ -43,8 +53,14 @@ class AddTactilePavingCrosswalk( override fun getTitle(tags: Map) = R.string.quest_tactilePaving_title_crosswalk - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - return overpassMapDataApi.query(getOverpassQuery(bbox), handler) + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable { + val excludedWayNodeIds = mutableSetOf() + mapData.ways + .filter { excludedWaysFilter.matches(it) } + .flatMapTo(excludedWayNodeIds) { it.nodeIds } + + return mapData.nodes + .filter { crossingFilter.matches(it) && it.id !in excludedWayNodeIds } } override fun isApplicableTo(element: Element): Boolean? = null @@ -54,35 +70,4 @@ class AddTactilePavingCrosswalk( override fun applyAnswerTo(answer: Boolean, changes: StringMapChangesBuilder) { changes.updateWithCheckDate("tactile_paving", answer.toYesNo()) } - - private fun getOverpassQuery(bbox: BoundingBox) = """ - ${bbox.toGlobalOverpassBBox()} - - way[highway = cycleway][foot !~ '^(yes|designated)$']; node(w) -> .exclusive_cycleway_nodes; - way[highway][access ~ '^(private|no)$']; node(w) -> .private_road_nodes; - (.exclusive_cycleway_nodes; .private_road_nodes;) -> .excluded; - - ( - node[highway = traffic_signals][crossing = traffic_signals][foot != no]; - node[highway = crossing][foot != no]; - ) -> .crossings; - - node.crossings[!tactile_paving] -> .crossings_with_unknown_tactile_paving; - node.crossings[tactile_paving = no]${olderThan(4).toOverpassQLString()} -> .old_crossings_without_tactile_paving; - node.crossings${olderThan(8).toOverpassQLString()} -> .very_old_crossings; - - ( - ( - .crossings_with_unknown_tactile_paving; - .old_crossings_without_tactile_paving; - .very_old_crossings; - ); - - .excluded; - ); - - ${getQuestPrintStatement()} - """.trimIndent() - - private fun olderThan(years: Int) = - TagOlderThan("tactile_paving", RelativeDate(-(r * 365 * years).toFloat())) } diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/toilet_availability/AddToiletAvailability.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/toilet_availability/AddToiletAvailability.kt index 5fbeaed908..685fb9a679 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/toilet_availability/AddToiletAvailability.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/toilet_availability/AddToiletAvailability.kt @@ -1,17 +1,16 @@ package de.westnordost.streetcomplete.quests.toilet_availability import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddToiletAvailability(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddToiletAvailability : OsmFilterQuestType() { // only for malls, big stores and rest areas because users should not need to go inside a non-public // place to solve the quest. (Considering malls and department stores public enough) - override val tagFilters = """ + override val elementFilter = """ nodes, ways with ( (shop ~ mall|department_store and name) diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/toilets_fee/AddToiletsFee.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/toilets_fee/AddToiletsFee.kt index 203d2a9915..3d2b3fe420 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/toilets_fee/AddToiletsFee.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/toilets_fee/AddToiletsFee.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.toilets_fee import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddToiletsFee(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddToiletsFee : OsmFilterQuestType() { - override val tagFilters = "nodes, ways with amenity = toilets and access !~ private|customers and !fee" + override val elementFilter = "nodes, ways with amenity = toilets and access !~ private|customers and !fee" override val commitMessage = "Add toilets fee" override val wikiLink = "Key:fee" override val icon = R.drawable.ic_quest_toilet_fee diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/tourism_information/AddInformationToTourism.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/tourism_information/AddInformationToTourism.kt index d6b6ed63a5..9f3bc0a18c 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/tourism_information/AddInformationToTourism.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/tourism_information/AddInformationToTourism.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.tourism_information import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -class AddInformationToTourism(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddInformationToTourism : OsmFilterQuestType() { - override val tagFilters = "nodes, ways, relations with tourism = information and !information" + override val elementFilter = "nodes, ways, relations with tourism = information and !information" override val commitMessage = "Add information type to tourist information" override val wikiLink = "Tag:tourism=information" override val icon = R.drawable.ic_quest_information diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/tracktype/AddTracktype.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/tracktype/AddTracktype.kt index f25fb782fe..9ec332c069 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/tracktype/AddTracktype.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/tracktype/AddTracktype.kt @@ -3,21 +3,18 @@ package de.westnordost.streetcomplete.quests.tracktype import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.ANYTHING_UNPAVED import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddTracktype(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddTracktype : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ ways with highway = track and ( !tracktype - or tracktype != grade1 and tracktype older today -${r * 4} years - or surface ~ ${ANYTHING_UNPAVED.joinToString("|")} and tracktype older today -${r * 4} years - or tracktype older today -${r * 8} years + or tracktype != grade1 and tracktype older today -4 years + or surface ~ ${ANYTHING_UNPAVED.joinToString("|")} and tracktype older today -4 years + or tracktype older today -8 years ) and (access !~ private|no or (foot and foot !~ private|no)) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_button/AddTrafficSignalsButton.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_button/AddTrafficSignalsButton.kt index 18cd7cccb8..db962baa01 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_button/AddTrafficSignalsButton.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_button/AddTrafficSignalsButton.kt @@ -1,15 +1,14 @@ package de.westnordost.streetcomplete.quests.traffic_signals_button import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -class AddTrafficSignalsButton(o: OverpassMapDataAndGeometryApi) : SimpleOverpassQuestType(o) { +class AddTrafficSignalsButton : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with crossing = traffic_signals and highway ~ crossing|traffic_signals and !button_operated """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_sound/AddTrafficSignalsSound.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_sound/AddTrafficSignalsSound.kt index 9816d5ff7d..8338e04233 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_sound/AddTrafficSignalsSound.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_sound/AddTrafficSignalsSound.kt @@ -2,22 +2,19 @@ package de.westnordost.streetcomplete.quests.traffic_signals_sound import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddTrafficSignalsSound(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddTrafficSignalsSound : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with crossing = traffic_signals and highway ~ crossing|traffic_signals and ( !$SOUND_SIGNALS - or $SOUND_SIGNALS = no and $SOUND_SIGNALS older today -${r * 4} years - or $SOUND_SIGNALS older today -${r * 8} years + or $SOUND_SIGNALS = no and $SOUND_SIGNALS older today -4 years + or $SOUND_SIGNALS older today -8 years ) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_vibrate/AddTrafficSignalsVibration.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_vibrate/AddTrafficSignalsVibration.kt index def3fed264..ac73b2db44 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_vibrate/AddTrafficSignalsVibration.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/traffic_signals_vibrate/AddTrafficSignalsVibration.kt @@ -2,22 +2,18 @@ package de.westnordost.streetcomplete.quests.traffic_signals_vibrate import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import de.westnordost.streetcomplete.ktx.toYesNo -import de.westnordost.streetcomplete.quests.YesNoQuestAnswerFragment -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddTrafficSignalsVibration(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddTrafficSignalsVibration : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes with crossing = traffic_signals and highway ~ crossing|traffic_signals and ( !$VIBRATING_BUTTON - or $VIBRATING_BUTTON = no and $VIBRATING_BUTTON older today -${r * 4} years - or $VIBRATING_BUTTON older today -${r * 8} years + or $VIBRATING_BUTTON = no and $VIBRATING_BUTTON older today -4 years + or $VIBRATING_BUTTON older today -8 years ) """ diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/way_lit/AddWayLit.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/way_lit/AddWayLit.kt index 7a54f5e160..f36bd32f76 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/way_lit/AddWayLit.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/way_lit/AddWayLit.kt @@ -2,13 +2,10 @@ package de.westnordost.streetcomplete.quests.way_lit import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddWayLit(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddWayLit : OsmFilterQuestType() { /* Using sidewalk as a tell-tale tag for (urban) streets which reached a certain level of development. I.e. non-urban streets will usually not even be lit in industrialized @@ -17,7 +14,7 @@ class AddWayLit(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) most hike paths and trails. See #427 for discussion. */ - override val tagFilters = """ + override val elementFilter = """ ways with ( ( @@ -33,8 +30,8 @@ class AddWayLit(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) ) and !lit ) - or highway and lit = no and lit older today -${r * 8} years - or highway and lit older today -${r * 16} years + or highway and lit = no and lit older today -8 years + or highway and lit older today -16 years ) and (access !~ private|no or (foot and foot !~ private|no)) and indoor != yes diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessBusiness.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessBusiness.kt index 603465f368..02f4c5dda3 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessBusiness.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessBusiness.kt @@ -2,17 +2,15 @@ package de.westnordost.streetcomplete.quests.wheelchair_access import de.westnordost.osmfeatures.FeatureDictionary import de.westnordost.streetcomplete.R -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi import java.util.concurrent.FutureTask class AddWheelchairAccessBusiness( - o: OverpassMapDataAndGeometryApi, private val featureDictionaryFuture: FutureTask -) : SimpleOverpassQuestType(o) +) : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with ( shop and shop !~ no|vacant diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessOutside.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessOutside.kt index 63147c96ed..ce3d28673e 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessOutside.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessOutside.kt @@ -2,17 +2,14 @@ package de.westnordost.streetcomplete.quests.wheelchair_access import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddWheelchairAccessOutside(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddWheelchairAccessOutside : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with leisure = dog_park - and (!wheelchair or wheelchair older today -${r * 8} years) + and (!wheelchair or wheelchair older today -8 years) """ override val commitMessage = "Add wheelchair access to outside places" override val wikiLink = "Key:wheelchair" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessPublicTransport.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessPublicTransport.kt index 2cd093654c..292626ed06 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessPublicTransport.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessPublicTransport.kt @@ -2,20 +2,17 @@ package de.westnordost.streetcomplete.quests.wheelchair_access import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddWheelchairAccessPublicTransport(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddWheelchairAccessPublicTransport : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with (amenity = bus_station or railway ~ station|subway_entrance) and ( !wheelchair - or wheelchair != yes and wheelchair older today -${r * 4} years - or wheelchair older today -${r * 8} years + or wheelchair != yes and wheelchair older today -4 years + or wheelchair older today -8 years ) """ override val commitMessage = "Add wheelchair access to public transport platforms" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToilets.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToilets.kt index fd48c1a8d5..72c792a1e5 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToilets.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToilets.kt @@ -2,21 +2,18 @@ package de.westnordost.streetcomplete.quests.wheelchair_access import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddWheelchairAccessToilets(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddWheelchairAccessToilets : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways with amenity = toilets and access !~ private|customers and ( !wheelchair - or wheelchair != yes and wheelchair older today -${r * 4} years - or wheelchair older today -${r * 8} years + or wheelchair != yes and wheelchair older today -4 years + or wheelchair older today -8 years ) """ override val commitMessage = "Add wheelchair access to toilets" diff --git a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToiletsPart.kt b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToiletsPart.kt index 77a1f3a803..effc0105dd 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToiletsPart.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/quests/wheelchair_access/AddWheelchairAccessToiletsPart.kt @@ -2,20 +2,17 @@ package de.westnordost.streetcomplete.quests.wheelchair_access import de.westnordost.streetcomplete.R import de.westnordost.streetcomplete.data.meta.updateWithCheckDate -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -class AddWheelchairAccessToiletsPart(o: OverpassMapDataAndGeometryApi, r: ResurveyIntervalsStore) - : SimpleOverpassQuestType(o) { +class AddWheelchairAccessToiletsPart : OsmFilterQuestType() { - override val tagFilters = """ + override val elementFilter = """ nodes, ways, relations with name and toilets = yes and ( !toilets:wheelchair - or toilets:wheelchair != yes and toilets:wheelchair older today -${r * 4} years - or toilets:wheelchair older today -${r * 8} years + or toilets:wheelchair != yes and toilets:wheelchair older today -4 years + or toilets:wheelchair older today -8 years ) """ override val commitMessage = "Add wheelchair access to toilets" diff --git a/app/src/main/java/de/westnordost/streetcomplete/settings/ResurveyIntervalsStore.kt b/app/src/main/java/de/westnordost/streetcomplete/settings/ResurveyIntervalsUpdater.kt similarity index 56% rename from app/src/main/java/de/westnordost/streetcomplete/settings/ResurveyIntervalsStore.kt rename to app/src/main/java/de/westnordost/streetcomplete/settings/ResurveyIntervalsUpdater.kt index 5aaaaa442a..85b48e0b80 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/settings/ResurveyIntervalsStore.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/settings/ResurveyIntervalsUpdater.kt @@ -3,19 +3,21 @@ package de.westnordost.streetcomplete.settings import android.content.SharedPreferences import de.westnordost.streetcomplete.Prefs import de.westnordost.streetcomplete.Prefs.ResurveyIntervals.* +import de.westnordost.streetcomplete.data.elementfilter.filters.RelativeDate import javax.inject.Inject import javax.inject.Singleton /** This class is just to access the user's preference about which multiplier for the resurvey * intervals to use */ -@Singleton class ResurveyIntervalsStore @Inject constructor(private val prefs: SharedPreferences) { - operator fun times(num: Double) = num * multiplier - operator fun times(num: Int) = num * multiplier +@Singleton class ResurveyIntervalsUpdater @Inject constructor(private val prefs: SharedPreferences) { + fun update() { + RelativeDate.MULTIPLIER = multiplier + } - private val multiplier: Double get() = when(intervalsPreference) { - LESS_OFTEN -> 2.0 - DEFAULT -> 1.0 - MORE_OFTEN -> 0.5 + private val multiplier: Float get() = when(intervalsPreference) { + LESS_OFTEN -> 2.0f + DEFAULT -> 1.0f + MORE_OFTEN -> 0.5f } private val intervalsPreference: Prefs.ResurveyIntervals get() = diff --git a/app/src/main/java/de/westnordost/streetcomplete/settings/SettingsFragment.kt b/app/src/main/java/de/westnordost/streetcomplete/settings/SettingsFragment.kt index e8120575e6..15ab688ec8 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/settings/SettingsFragment.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/settings/SettingsFragment.kt @@ -4,7 +4,6 @@ import android.content.Intent import android.content.SharedPreferences import android.os.Bundle import android.view.LayoutInflater -import android.widget.TextView import android.widget.Toast import androidx.appcompat.app.AlertDialog import androidx.appcompat.app.AppCompatDelegate @@ -20,7 +19,6 @@ import de.westnordost.streetcomplete.data.download.tiles.DownloadedTilesDao import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestController import de.westnordost.streetcomplete.data.osmnotes.notequests.OsmNoteQuest import de.westnordost.streetcomplete.data.osmnotes.notequests.OsmNoteQuestController -import de.westnordost.streetcomplete.data.user.UserController import de.westnordost.streetcomplete.ktx.toast import kotlinx.coroutines.* import javax.inject.Inject @@ -31,10 +29,10 @@ class SettingsFragment : PreferenceFragmentCompat(), CoroutineScope by CoroutineScope(Dispatchers.Main) { @Inject internal lateinit var prefs: SharedPreferences - @Inject internal lateinit var userController: UserController @Inject internal lateinit var downloadedTilesDao: DownloadedTilesDao @Inject internal lateinit var osmQuestController: OsmQuestController @Inject internal lateinit var osmNoteQuestController: OsmNoteQuestController + @Inject internal lateinit var resurveyIntervalsUpdater: ResurveyIntervalsUpdater interface Listener { fun onClickedQuestSelection() @@ -114,16 +112,10 @@ class SettingsFragment : PreferenceFragmentCompat(), Prefs.AUTOSYNC -> { if (Prefs.Autosync.valueOf(prefs.getString(Prefs.AUTOSYNC, "ON")!!) != Prefs.Autosync.ON) { val view = LayoutInflater.from(activity).inflate(R.layout.dialog_tutorial_upload, null) - val filled = requireContext().getString(R.string.action_download) - val uploadExplanation = view.findViewById(R.id.tutorialDownloadPanel) - uploadExplanation.text = context!!.getString(R.string.dialog_tutorial_download, filled) - context?.let { - AlertDialog.Builder(it) - .setView(view) - .setPositiveButton(android.R.string.ok, null) - .show() - } - + AlertDialog.Builder(requireContext()) + .setView(view) + .setPositiveButton(android.R.string.ok, null) + .show() } } Prefs.THEME_SELECT -> { @@ -131,6 +123,9 @@ class SettingsFragment : PreferenceFragmentCompat(), AppCompatDelegate.setDefaultNightMode(theme.appCompatNightMode) activity?.recreate() } + Prefs.RESURVEY_INTERVALS -> { + resurveyIntervalsUpdater.update() + } } } diff --git a/app/src/main/java/de/westnordost/streetcomplete/util/ElementGeometryUtils.kt b/app/src/main/java/de/westnordost/streetcomplete/util/ElementGeometryUtils.kt index 38a8119420..ce114edf27 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/util/ElementGeometryUtils.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/util/ElementGeometryUtils.kt @@ -5,4 +5,22 @@ import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGe fun ElementPolylinesGeometry.getOrientationAtCenterLineInDegrees(): Float { val centerLine = polylines.first().centerLineOfPolyline() return centerLine.first.initialBearingTo(centerLine.second).toFloat() -} \ No newline at end of file +} + +/** Returns whether this ElementPolylinesGeometry is near a set of other ElementPolylinesGeometries + * + * Warning: This is computationally very expensive ( for normal ways, O(n³) ), avoid if possible */ +fun ElementPolylinesGeometry.isNear( + maxDistance: Double, + others: Iterable +): Boolean { + val bounds = getBounds().enlargedBy(maxDistance) + return others.any { other -> + bounds.intersect(other.getBounds()) && + polylines.any { polyline -> + other.polylines.any { otherPolyline -> + polyline.distanceTo(otherPolyline) <= maxDistance + } + } + } +} diff --git a/app/src/main/java/de/westnordost/streetcomplete/util/LatLonRaster.kt b/app/src/main/java/de/westnordost/streetcomplete/util/LatLonRaster.kt index bccb048637..2621b2a190 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/util/LatLonRaster.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/util/LatLonRaster.kt @@ -31,7 +31,7 @@ class LatLonRaster(bounds: BoundingBox, private val cellSize: Double) { fun insert(p: LatLon) { val x = longitudeToCellX(p.longitude) val y = latitudeToCellY(p.latitude) - checkBounds(x, y) + if(!isInsideBounds(x, y)) return var list = raster[y * rasterWidth + x] if (list == null) { list = ArrayList() @@ -66,10 +66,8 @@ class LatLonRaster(bounds: BoundingBox, private val cellSize: Double) { return result } - private fun checkBounds(x: Int, y: Int) { - require(x in 0 until rasterWidth) { "Longitude is out of bounds" } - require(y in 0 until rasterHeight) { "Latitude is out of bounds" } - } + private fun isInsideBounds(x: Int, y: Int): Boolean = + x in 0 until rasterWidth && y in 0 until rasterHeight private fun longitudeToCellX(longitude: Double) = floor(normalizeLongitude(longitude - bbox.minLongitude) / cellSize).toInt() diff --git a/app/src/main/java/de/westnordost/streetcomplete/util/SlippyMapMath.kt b/app/src/main/java/de/westnordost/streetcomplete/util/SlippyMapMath.kt index 9129654887..bb4b8d6e8f 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/util/SlippyMapMath.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/util/SlippyMapMath.kt @@ -17,6 +17,8 @@ data class Tile(val x: Int, val y:Int) { tile2lon(x + 1, zoom) ) } + + fun toTilesRect() = TilesRect(x,y,x,y) } /** Returns the minimum rectangle of tiles that encloses all the tiles */ diff --git a/app/src/main/java/de/westnordost/streetcomplete/util/SphericalEarthMath.kt b/app/src/main/java/de/westnordost/streetcomplete/util/SphericalEarthMath.kt index 1a809f74e1..2697095aa4 100644 --- a/app/src/main/java/de/westnordost/streetcomplete/util/SphericalEarthMath.kt +++ b/app/src/main/java/de/westnordost/streetcomplete/util/SphericalEarthMath.kt @@ -5,7 +5,7 @@ package de.westnordost.streetcomplete.util import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.osmapi.map.data.LatLon import de.westnordost.osmapi.map.data.OsmLatLon -import de.westnordost.streetcomplete.ktx.forEachPair +import de.westnordost.streetcomplete.ktx.forEachLine import kotlin.math.* /** Calculate stuff assuming a spherical Earth. The Earth is not spherical, but it is a good @@ -118,14 +118,16 @@ fun LatLon.alongTrackDistanceTo(start: LatLon, end: LatLon, globeRadius: Double /** Returns the shortest distance between this point and the arc between the given points */ fun LatLon.distanceToArc(start: LatLon, end: LatLon, globeRadius: Double = EARTH_RADIUS): Double = - abs(angularDistanceToArc( - start.latitude.toRadians(), - start.longitude.toRadians(), - end.latitude.toRadians(), - end.longitude.toRadians(), - latitude.toRadians(), - longitude.toRadians() - )) * globeRadius + abs( + angularDistanceToArc( + start.latitude.toRadians(), + start.longitude.toRadians(), + end.latitude.toRadians(), + end.longitude.toRadians(), + latitude.toRadians(), + longitude.toRadians() + ) + ) * globeRadius /** Returns the shortest distance between this point and the arcs between the given points */ fun LatLon.distanceToArcs(polyLine: List, globeRadius: Double = EARTH_RADIUS): Double { @@ -133,7 +135,7 @@ fun LatLon.distanceToArcs(polyLine: List, globeRadius: Double = EARTH_RA if (polyLine.size == 1) return distanceTo(polyLine[0]) var shortestDistance = Double.MAX_VALUE - polyLine.forEachPair { first, second -> + polyLine.forEachLine { first, second -> val distance = distanceToArc(first, second, globeRadius) if (distance < shortestDistance) shortestDistance = distance } @@ -142,6 +144,12 @@ fun LatLon.distanceToArcs(polyLine: List, globeRadius: Double = EARTH_RA /* -------------------------------- Polyline extension functions -------------------------------- */ +/** Returns the shortest distance between this polyline and given polyline */ +fun List.distanceTo(polyline: List, globeRadius: Double = EARTH_RADIUS): Double { + require(isNotEmpty()) { "Polyline must not be empty" } + return minOf { it.distanceToArcs(polyline, globeRadius) } +} + /** Returns a bounding box that contains all points */ fun Iterable.enclosingBoundingBox(): BoundingBox { val it = iterator() @@ -173,7 +181,7 @@ fun Iterable.enclosingBoundingBox(): BoundingBox { fun List.measuredLength(globeRadius: Double = EARTH_RADIUS): Double { if (isEmpty()) return 0.0 var length = 0.0 - forEachPair { first, second -> + forEachLine { first, second -> length += first.distanceTo(second, globeRadius) } return length @@ -185,7 +193,7 @@ fun List.centerLineOfPolyline(globeRadius: Double = EARTH_RADIUS): Pair< require(size >= 2) { "positions list must contain at least 2 elements" } var halfDistance = measuredLength() / 2 - forEachPair { first, second -> + forEachLine { first, second -> halfDistance -= first.distanceTo(second, globeRadius) if (halfDistance <= 0) { return Pair(first, second) @@ -222,7 +230,7 @@ fun List.pointOnPolylineFromEnd(distance: Double): LatLon? { private fun List.pointOnPolyline(distance: Double, fromEnd: Boolean): LatLon? { val list = if (fromEnd) this.asReversed() else this var d = 0.0 - list.forEachPair { first, second -> + list.forEachLine { first, second -> val segmentDistance = first.distanceTo(second) if (segmentDistance > 0) { d += segmentDistance @@ -251,7 +259,7 @@ fun List.centerPointOfPolygon(): LatLon { var lat = 0.0 var area = 0.0 val origin = first() - forEachPair { first, second -> + forEachLine { first, second -> // calculating with offsets to avoid rounding imprecision and 180th meridian problem val dx1 = normalizeLongitude(first.longitude - origin.longitude) val dy1 = first.latitude - origin.latitude @@ -280,7 +288,7 @@ fun LatLon.isInPolygon(polygon: List): Boolean { var lastWasIntersectionAtVertex = false val lon = longitude val lat = latitude - polygon.forEachPair { first, second -> + polygon.forEachLine { first, second -> val lat0 = first.latitude val lat1 = second.latitude // scanline check, disregard line segments parallel to the cast ray @@ -307,6 +315,44 @@ fun LatLon.isInPolygon(polygon: List): Boolean { private fun inside(v: Double, bound0: Double, bound1: Double): Boolean = if (bound0 < bound1) v in bound0..bound1 else v in bound1..bound0 + +/** + * Returns the area of a this multipolygon, assuming the outer shell is defined counterclockwise and + * any holes are defined clockwise + */ +fun List>.measuredMultiPolygonArea(globeRadius: Double = EARTH_RADIUS): Double { + return sumOf { it.measuredAreaSigned(globeRadius) } +} + +/** + * Returns the area of a this polygon + */ +fun List.measuredArea(globeRadius: Double = EARTH_RADIUS): Double { + return abs(measuredAreaSigned(globeRadius)) +} + +/** + * Returns the signed area of a this polygon. If it is defined counterclockwise, it'll return + * something positive, clockwise something negative + */ +fun List.measuredAreaSigned(globeRadius: Double = EARTH_RADIUS): Double { + // not closed: area 0 + if (size < 4) return 0.0 + if (first().latitude != last().latitude || first().longitude != last().longitude) return 0.0 + var area = 0.0 + /* The algorithm is basically the same as for the planar case, only the calculation of the area + * for each polygon edge is the polar triangle area */ + forEachLine { first, second -> + area += polarTriangleArea( + first.latitude.toRadians(), + first.longitude.toRadians(), + second.latitude.toRadians(), + second.longitude.toRadians(), + ) + } + return area * (globeRadius * globeRadius) +} + /** * Returns whether the given position is within the given multipolygon. Polygons defined * counterclockwise count as outer shells, polygons defined clockwise count as holes. @@ -333,7 +379,7 @@ fun List.isRingDefinedClockwise(): Boolean { var sum = 0.0 val origin = first() - forEachPair { first, second -> + forEachLine { first, second -> // calculating with offsets to handle 180th meridian val lon0 = normalizeLongitude(first.longitude - origin.longitude) val lat0 = first.latitude - origin.latitude @@ -355,7 +401,62 @@ fun BoundingBox.area(globeRadius: Double = EARTH_RADIUS): Double { return min.distanceTo(minLatMaxLon, globeRadius) * min.distanceTo(maxLatMinLon, globeRadius) } +/** Returns a new bounding box that is [radius] larger than this bounding box */ +fun BoundingBox.enlargedBy(radius: Double, globeRadius: Double = EARTH_RADIUS): BoundingBox { + return BoundingBox( + min.translate(radius, 225.0, globeRadius), + max.translate(radius, 45.0, globeRadius) + ) +} +/** returns whether this bounding box intersects with the other. Works if any of the bounding boxes + * cross the 180th meridian */ +fun BoundingBox.intersect(other: BoundingBox): Boolean = + checkAlignment(other) { bbox1, bbox2 -> bbox1.intersectCanonical(bbox2) } + +/** returns whether this bounding box is completely inside the other, assuming both bounding boxes + * do not cross the 180th meridian */ +fun BoundingBox.isCompletelyInside(other: BoundingBox): Boolean = + checkAlignment(other) { bbox1, bbox2 -> bbox1.isCompletelyInsideCanonical(bbox2) } + +/** returns whether this bounding box intersects with the other, assuming both bounding boxes do + * not cross the 180th meridian */ +private fun BoundingBox.intersectCanonical(other: BoundingBox): Boolean = + maxLongitude >= other.minLongitude && + minLongitude <= other.maxLongitude && + maxLatitude >= other.minLatitude && + minLatitude <= other.maxLatitude + +/** returns whether this bounding box is completely inside the other, assuming both bounding boxes + * do not cross the 180th meridian */ +private fun BoundingBox.isCompletelyInsideCanonical(other: BoundingBox): Boolean = + minLongitude >= other.minLongitude && + minLatitude >= other.minLatitude && + maxLongitude <= other.maxLongitude && + maxLatitude <= other.maxLatitude + + +private inline fun BoundingBox.checkAlignment( + other: BoundingBox, + canonicalCheck: (bbox1: BoundingBox, bbox2: BoundingBox) -> Boolean +): Boolean { + return if(crosses180thMeridian()) { + val these = splitAt180thMeridian() + if (other.crosses180thMeridian()) { + val others = other.splitAt180thMeridian() + these.any { a -> others.any { b -> canonicalCheck(a, b) } } + } else { + these.any { canonicalCheck(it, other) } + } + } else { + if (other.crosses180thMeridian()) { + val others = other.splitAt180thMeridian() + others.any { canonicalCheck(this, it) } + } else { + canonicalCheck(this, other) + } + } +} fun createTranslated(latitude: Double, longitude: Double): LatLon { var lat = latitude @@ -468,3 +569,13 @@ private fun angularDistanceToArc(φ1: Double, λ1: Double, φ2: Double, λ2: Dou if (δat > δ12) return angularDistance(φ2, λ2, φ3, λ3) return δxt } + +/** Returns the signed area of a triangle spanning between the north pole and the two given points. + * */ +private fun polarTriangleArea(φ1: Double, λ1: Double, φ2: Double, λ2: Double): Double { + val tanφ1 = tan((PI / 2 - φ1) / 2) + val tanφ2 = tan((PI / 2 - φ2) / 2) + val Δλ = λ1 - λ2 + val tan = tanφ1 * tanφ2 + return 2 * atan2(tan * sin(Δλ), 1 + tan * cos(Δλ)) +} diff --git a/app/src/main/res/drawable/ic_search_black_128dp.xml b/app/src/main/res/drawable/ic_search_black_128dp.xml new file mode 100644 index 0000000000..524b3e43cf --- /dev/null +++ b/app/src/main/res/drawable/ic_search_black_128dp.xml @@ -0,0 +1,9 @@ + + + diff --git a/app/src/main/res/layout/dialog_tutorial_upload.xml b/app/src/main/res/layout/dialog_tutorial_upload.xml index 73377538ac..b95447aee8 100644 --- a/app/src/main/res/layout/dialog_tutorial_upload.xml +++ b/app/src/main/res/layout/dialog_tutorial_upload.xml @@ -17,15 +17,4 @@ app:drawableTint="@color/text" android:text="@string/dialog_tutorial_upload" android:textAppearance="@android:style/TextAppearance.Theme.Dialog"/> - - diff --git a/app/src/main/res/layout/fragment_quest_download_progress.xml b/app/src/main/res/layout/fragment_download_progress.xml similarity index 100% rename from app/src/main/res/layout/fragment_quest_download_progress.xml rename to app/src/main/res/layout/fragment_download_progress.xml diff --git a/app/src/main/res/layout/fragment_main.xml b/app/src/main/res/layout/fragment_main.xml index a420d9da16..ca03b1d52d 100644 --- a/app/src/main/res/layout/fragment_main.xml +++ b/app/src/main/res/layout/fragment_main.xml @@ -160,10 +160,10 @@ DARK - - "https://lz4.overpass-api.de/api/" - "https://overpass.kumi.systems/api/" - diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 53037653f7..656a64814e 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -162,7 +162,7 @@ "Ice hockey" "Netball" "Rugby" - "Auto-sync" + "Upload answers automatically" "Zoom out" "Zoom in" "Menu" @@ -183,7 +183,7 @@ The app does not share your GPS location with anyone. It is used to automaticall <p><b>Data Usage</b></p> <p> -As mentioned, the app directly communicates with OSM infrastructure. That is, with the <a href=\"https://wiki.openstreetmap.org/wiki/Overpass_API\">Overpass API</a> for downloading quests (which is anonymous) and with the <a href=\"https://wiki.openstreetmap.org/wiki/API_v0.6\">OSM API</a> for uploading changes.<br/> +As mentioned, the app directly communicates with OSM infrastructure.<br/> However, before uploading your changes, the app checks with a <a href=\"https://www.westnordost.de/streetcomplete/banned_versions.txt\">simple text file</a> on my server whether it has been banned from uploading any changes. This is a precautionary measure to be able to keep versions of the app that turn out to have critical bugs from possibly corrupting OSM data.</p>" "<p>To display the map, vector tiles are retrieved from %1$s. See their <a href=\"%2$s\">privacy statement</a> for more information.</p>" <p>Photos you attach to a note are uploaded to my server and deleted some days after that note has been resolved. Their meta-data is stripped before upload.</p> @@ -708,8 +708,6 @@ Otherwise, you can download another keyboard in the app store. Popular keyboards Light Dark System default - Change Overpass server - Requires application restart to apply. May bypass censorship, for example in Russia. This street was tagged as having no sidewalk on either side. In the case that there is a sidewalk after all but it is displayed as a separate way, please answer \"sidewalk\". Open location in another app No other map application installed @@ -925,7 +923,6 @@ Otherwise, you can download another keyboard in the app store. Popular keyboards house number %s: Is there a summit register at %s? Is this defibrillator (AED) inside a building? - With auto-sync off, quests are not downloaded automatically. To do this manually, use the \"%s\" button in the menu at the location you want to download. "Does this pedestrian crossing have an island?" Describe surface diff --git a/app/src/main/res/xml/preferences.xml b/app/src/main/res/xml/preferences.xml index c4f2cd3dca..16143d2f02 100644 --- a/app/src/main/res/xml/preferences.xml +++ b/app/src/main/res/xml/preferences.xml @@ -8,7 +8,7 @@ - - - overpassMapDataDao.get(invocation.getArgument(0) as String) - true - } - - val registry = QuestModule.questTypeRegistry(mock(), overpassMock, mock(), mock(), mock(), mock(), mock()) - - val hamburg = BoundingBox(53.5, 9.9, 53.6, 10.0) - - for (questType in registry.all) { - if (questType is OsmElementQuestType) { - print(questType.javaClass.simpleName + ": ") - questType.download(hamburg, mock()) - println() - } - } - println("Total response time: ${overpassMapDataDao.totalTime/1000}s") - println("Total waiting time: ${overpassMapDataDao.totalWaitTime}s") -} - -private class TestOverpassMapDataDao { - - var totalTime = 0L - var totalWaitTime = 0L - - val osm = OsmConnection( - "https://lz4.overpass-api.de/api/", - "StreetComplete Overpass Query Performance Test", - null, - (180 + 4) * 1000) - - fun get(query: String) { - var time: Long - while(true) { - try { - time = measureTimeMillis { - osm.makeRequest("interpreter", "POST", false, getWriter(query), null) - } - break - } catch (e: OsmApiException) { - if (e.errorCode == 429) { - val status = getStatus() - if (status.availableSlots == 0) { - val waitInSeconds = status.nextAvailableSlotIn ?: 60 - totalWaitTime += waitInSeconds - sleep(waitInSeconds * 1000L) - } - continue - } else throw e - } - } - totalTime += time - val s = "%.1f".format(time/1000.0) - print("${s}s ") - } - - private fun getStatus(): OverpassStatus = osm.makeRequest("status", OverpassStatusParser()) - - private fun getWriter(query: String) = object : ApiRequestWriter { - override fun write(out: OutputStream) { - val utf8 = Charsets.UTF_8 - val request = "data=" + URLEncoder.encode(query, utf8.name()) - out.write(request.toByteArray(utf8)) - } - override fun getContentType() = "application/x-www-form-urlencoded" - } -} - diff --git a/app/src/test/java/de/westnordost/streetcomplete/QuestsOverpassPrinter.kt b/app/src/test/java/de/westnordost/streetcomplete/QuestsOverpassPrinter.kt index 8cbb55394c..668d74f1ce 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/QuestsOverpassPrinter.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/QuestsOverpassPrinter.kt @@ -1,45 +1,22 @@ package de.westnordost.streetcomplete -import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType -import de.westnordost.streetcomplete.data.osm.osmquest.SimpleOverpassQuestType -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi +import de.westnordost.streetcomplete.data.osm.osmquest.OsmFilterQuestType import de.westnordost.streetcomplete.quests.QuestModule -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -import org.mockito.ArgumentMatchers -import org.mockito.ArgumentMatchers.anyInt fun main() { - val overpassMock: OverpassMapDataAndGeometryApi = mock() - on(overpassMock.query(any(), any())).then { invocation -> - var query = invocation.getArgument(0) as String - // make query overpass-turbo friendly - query = query - .replace("0,0,1,1", "{{bbox}}") - .replace("out meta geom 2000;", "out meta geom;") - print("```\n$query\n```\n") - true - } - - val resurveyIntervalsStoreMock: ResurveyIntervalsStore = mock() - on(resurveyIntervalsStoreMock.times(anyInt())).thenAnswer { (it.arguments[0] as Int).toDouble() } - on(resurveyIntervalsStoreMock.times(ArgumentMatchers.anyDouble())).thenAnswer { (it.arguments[0] as Double) } - - val registry = QuestModule.questTypeRegistry( - mock(), overpassMock, resurveyIntervalsStoreMock, mock(), mock(), mock(), mock() - ) - - val bbox = BoundingBox(0.0,0.0,1.0,1.0) + val registry = QuestModule.questTypeRegistry(mock(), mock(), mock(), mock(), mock()) for (questType in registry.all) { if (questType is OsmElementQuestType) { println("### " + questType.javaClass.simpleName) - if (questType is SimpleOverpassQuestType) { - val filters = questType.tagFilters.trimIndent() - println("
\nTag Filters\n\n```\n$filters\n```\n
\n") + if (questType is OsmFilterQuestType) { + val query = "[bbox:{{bbox}}];\n" + questType.filter.toOverpassQLString() + "\n out meta geom;" + println("```\n$query\n```") + } else { + println("Not available, see source code") } - questType.download(bbox, mock()) println() } } diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpressionTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpressionTest.kt index 202190e27c..6abc03c503 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpressionTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFilterExpressionTest.kt @@ -9,6 +9,7 @@ import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.on import org.junit.Assert.* +import java.util.* class ElementFilterExpressionTest { // Tests for toOverpassQLString are in FiltersParserTest @@ -65,7 +66,7 @@ class ElementFilterExpressionTest { @Test fun `matches filter`() { val tagFilter: ElementFilter = mock() - val expr = ElementFilterExpression(listOf(ElementsTypeFilter.NODES), Leaf(tagFilter)) + val expr = ElementFilterExpression(EnumSet.of(ElementsTypeFilter.NODES), Leaf(tagFilter)) on(tagFilter.matches(any())).thenReturn(true) assertTrue(expr.matches(node)) @@ -82,6 +83,15 @@ class ElementFilterExpressionTest { private fun createMatchExpression(vararg elementsTypeFilter: ElementsTypeFilter): ElementFilterExpression { val tagFilter: ElementFilter = mock() on(tagFilter.matches(any())).thenReturn(true) - return ElementFilterExpression(elementsTypeFilter.asList(), Leaf(tagFilter)) + return ElementFilterExpression(createEnumSet(*elementsTypeFilter), Leaf(tagFilter)) + } + + private fun createEnumSet(vararg filters: ElementsTypeFilter): EnumSet { + return when (filters.size) { + 1 -> EnumSet.of(filters[0]) + 2 -> EnumSet.of(filters[0], filters[1]) + 3 -> EnumSet.of(filters[0], filters[1], filters[2]) + else -> throw IllegalStateException() + } } } diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParserAndOverpassQueryCreatorTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParserAndOverpassQueryCreatorTest.kt index 59bc6851fb..22e867d844 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParserAndOverpassQueryCreatorTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/ElementFiltersParserAndOverpassQueryCreatorTest.kt @@ -196,7 +196,7 @@ class ElementFiltersParserAndOverpassQueryCreatorTest { check("nodes with check_date < today -2 days", "node[check_date](if:date(t['check_date']) < date('$twoDaysAgo'));") val twoDaysInFuture = dateDaysAgo(-2f).toCheckDateString() - check("nodes with check_date < today + 0.3 weeks", "node[check_date](if:date(t['check_date']) < date('$twoDaysInFuture'));") + check("nodes with check_date < today + 0.285 weeks", "node[check_date](if:date(t['check_date']) < date('$twoDaysInFuture'));") } @Test fun `element older x days`() { @@ -383,14 +383,14 @@ class ElementFiltersParserAndOverpassQueryCreatorTest { private fun shouldFail(input: String) { try { - ElementFiltersParser().parse(input) + input.toElementFilterExpression() fail() } catch (ignore: ParseException) { } } private fun check(input: String, output: String) { - val expr = ElementFiltersParser().parse(input) + val expr = input.toElementFilterExpression() assertEquals( output.replace("\n","").replace(" ",""), expr.toOverpassQLString().replace("\n","").replace(" ","")) diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtilsKtTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtilsKtTest.kt deleted file mode 100644 index c35b900d3f..0000000000 --- a/app/src/test/java/de/westnordost/streetcomplete/data/elementfilter/OverpassQLUtilsKtTest.kt +++ /dev/null @@ -1,20 +0,0 @@ -package de.westnordost.streetcomplete.data.elementfilter - -import de.westnordost.osmapi.map.data.BoundingBox -import org.junit.Assert.* -import org.junit.Test - -class OverpassQLUtilsKtTest { - - @Test fun `truncates overpass bbox`() { - assertEquals( - "[bbox:0.0001235,1.0001235,2.0001235,3.0001235];", - BoundingBox( - 0.0001234567890123456789, - 1.0001234567890123456789, - 2.0001234567890123456789, - 3.0001234567890123456789 - ).toGlobalOverpassBBox()) - } - -} diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreatorTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreatorTest.kt index d334895498..800c548b0c 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreatorTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/ElementGeometryCreatorTest.kt @@ -1,5 +1,7 @@ package de.westnordost.streetcomplete.data.osm.elementgeometry +import de.westnordost.osmapi.map.MapData +import de.westnordost.osmapi.map.MutableMapData import de.westnordost.osmapi.map.data.* import org.junit.Test @@ -116,6 +118,93 @@ class ElementGeometryCreatorTest { val geom = create(relation(member(SIMPLE_WAY1, SIMPLE_WAY2, SIMPLE_WAY3, WAY_DUPLICATE_NODES))) as ElementPolylinesGeometry assertTrue(geom.polylines.containsAll(listOf(CCW_RING, listOf(P0, P1, P2)))) } + + @Test fun `positions for way`() { + val nodes = listOf( + OsmNode(0, 1, P0, null, null, null), + OsmNode(1, 1, P1, null, null, null) + ) + val mapData = MutableMapData() + mapData.addAll(nodes) + val geom = create(SIMPLE_WAY1, mapData) as ElementPolylinesGeometry + assertEquals(listOf(nodes.map { it.position }), geom.polylines) + } + + @Test fun `returns null for non-existent way`() { + val way = OsmWay(1L, 1, listOf(1,2,3), null) + assertNull(create(way, MutableMapData())) + } + + @Test fun `positions for relation`() { + val relation = OsmRelation(1L, 1, listOf( + OsmRelationMember(0L, "", Element.Type.WAY), + OsmRelationMember(1L, "", Element.Type.WAY), + OsmRelationMember(1L, "", Element.Type.NODE) + ), null) + + val ways = listOf(SIMPLE_WAY1, SIMPLE_WAY2) + val nodesByWayId = mapOf>( + 0L to listOf( + OsmNode(0, 1, P0, null, null, null), + OsmNode(1, 1, P1, null, null, null) + ), + 1L to listOf( + OsmNode(1, 1, P1, null, null, null), + OsmNode(2, 1, P2, null, null, null), + OsmNode(3, 1, P3, null, null, null) + ) + ) + val mapData = MutableMapData() + mapData.addAll(nodesByWayId.values.flatten() + ways) + val positions = listOf(P0, P1, P2, P3) + val geom = create(relation, mapData) as ElementPolylinesGeometry + assertEquals(listOf(positions), geom.polylines) + } + + @Test fun `returns null for non-existent relation`() { + val relation = OsmRelation(1L, 1, listOf( + OsmRelationMember(1L, "", Element.Type.WAY), + OsmRelationMember(2L, "", Element.Type.WAY), + OsmRelationMember(1L, "", Element.Type.NODE) + ), null) + assertNull(create(relation, MutableMapData())) + } + + @Test fun `returns null for relation with a way that's missing from map data`() { + val relation = OsmRelation(1L, 1, listOf( + OsmRelationMember(0L, "", Element.Type.WAY), + OsmRelationMember(1L, "", Element.Type.WAY) + ), null) + val mapData = MutableMapData() + mapData.addAll(listOf( + relation, + OsmWay(0, 0, listOf(0,1), null), + OsmNode(0, 0, P0, null), + OsmNode(1, 0, P1, null) + )) + + assertNull(create(relation, mapData)) + } + + @Test fun `does not return null for relation with a way that's missing from map data if returning incomplete geometries is ok`() { + val relation = OsmRelation(1L, 1, listOf( + OsmRelationMember(0L, "", Element.Type.WAY), + OsmRelationMember(1L, "", Element.Type.WAY) + ), null) + val way = OsmWay(0, 0, listOf(0,1), null) + val mapData = MutableMapData() + mapData.addAll(listOf( + relation, + way, + OsmNode(0, 0, P0, null), + OsmNode(1, 0, P1, null) + )) + + assertEquals( + create(way), + create(relation, mapData, true) + ) + } } private fun create(node: Node) = @@ -127,6 +216,9 @@ private fun create(way: Way) = private fun create(relation: Relation) = ElementGeometryCreator().create(relation, WAY_GEOMETRIES) +private fun create(element: Element, mapData: MapData, allowIncomplete: Boolean = false) = + ElementGeometryCreator().create(element, mapData, allowIncomplete) + private val WAY_AREA = mapOf("area" to "yes") private val O: LatLon = OsmLatLon(1.0, 1.0) diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/OsmApiElementGeometryCreatorTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/OsmApiElementGeometryCreatorTest.kt deleted file mode 100644 index 455077b2e3..0000000000 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/elementgeometry/OsmApiElementGeometryCreatorTest.kt +++ /dev/null @@ -1,108 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.elementgeometry - -import de.westnordost.streetcomplete.data.MapDataApi -import de.westnordost.osmapi.common.errors.OsmNotFoundException -import de.westnordost.osmapi.map.data.* -import de.westnordost.osmapi.map.handler.MapDataHandler -import de.westnordost.streetcomplete.any -import de.westnordost.streetcomplete.eq -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.on -import org.junit.Assert.* -import org.junit.Before -import org.junit.Test -import org.mockito.Mockito.never -import org.mockito.Mockito.verify - -class OsmApiElementGeometryCreatorTest { - - private lateinit var api: MapDataApi - private lateinit var elementCreator: ElementGeometryCreator - private lateinit var creator: OsmApiElementGeometryCreator - - @Before fun setUp() { - api = mock() - elementCreator = mock() - creator = OsmApiElementGeometryCreator(api, elementCreator) - } - - @Test fun `creates for node`() { - val node = OsmNode(1L, 1, 2.0, 3.0, null) - creator.create(node) - verify(elementCreator).create(node) - } - - @Test fun `creates for way`() { - val way = OsmWay(1L, 1, listOf(1,2,3), null) - val positions = listOf( - OsmLatLon(1.0, 2.0), - OsmLatLon(2.0, 4.0), - OsmLatLon(5.0, 6.0) - ) - on(api.getWayComplete(eq(1L), any())).thenAnswer { invocation -> - val handler = (invocation.arguments[1]) as MapDataHandler - handler.handle(way) - way.nodeIds.forEachIndexed { i, nodeId -> - handler.handle(OsmNode(nodeId, 1, positions[i], null)) - } - Any() - } - creator.create(way) - verify(elementCreator).create(way, positions) - } - - @Test fun `returns null for non-existent way`() { - val way = OsmWay(1L, 1, listOf(1,2,3), null) - on(api.getWayComplete(eq(1L), any())).thenThrow(OsmNotFoundException(404, "", "")) - assertNull(creator.create(way)) - verify(elementCreator, never()).create(eq(way), any()) - } - - @Test fun `creates for relation`() { - val relation = OsmRelation(1L, 1, listOf( - OsmRelationMember(1L, "", Element.Type.WAY), - OsmRelationMember(2L, "", Element.Type.WAY), - OsmRelationMember(1L, "", Element.Type.NODE) - ), null) - - val ways = listOf( - OsmWay(1L, 1, listOf(1,2,3), null), - OsmWay(2L, 1, listOf(4,5), null) - ) - val positions = mapOf>( - 1L to listOf( - OsmLatLon(1.0, 2.0), - OsmLatLon(2.0, 4.0), - OsmLatLon(5.0, 6.0) - ), - 2L to listOf( - OsmLatLon(2.0, 1.0), - OsmLatLon(0.0, -1.0) - ) - ) - on(api.getRelationComplete(eq(1L), any())).thenAnswer { invocation -> - val handler = (invocation.arguments[1]) as MapDataHandler - handler.handle(relation) - for (way in ways) { - handler.handle(way) - way.nodeIds.forEachIndexed { i, nodeId -> - handler.handle(OsmNode(nodeId, 1, positions[way.id]!![i], null)) - } - } - Any() - } - creator.create(relation) - verify(elementCreator).create(relation, positions) - } - - @Test fun `returns null for non-existent relation`() { - val relation = OsmRelation(1L, 1, listOf( - OsmRelationMember(1L, "", Element.Type.WAY), - OsmRelationMember(2L, "", Element.Type.WAY), - OsmRelationMember(1L, "", Element.Type.NODE) - ), null) - on(api.getRelationComplete(eq(1L), any())).thenThrow(OsmNotFoundException(404, "", "")) - assertNull(creator.create(relation)) - verify(elementCreator, never()).create(eq(relation), any()) - } -} diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/mapdata/OverpassMapDataAndGeometryApiTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/mapdata/OverpassMapDataAndGeometryApiTest.kt deleted file mode 100644 index 9b027b5cdf..0000000000 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/mapdata/OverpassMapDataAndGeometryApiTest.kt +++ /dev/null @@ -1,53 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.mapdata - -import de.westnordost.streetcomplete.data.OverpassMapDataApi -import de.westnordost.osmapi.overpass.MapDataWithGeometryParser -import de.westnordost.osmapi.overpass.OsmTooManyRequestsException -import de.westnordost.osmapi.overpass.OverpassStatus -import de.westnordost.streetcomplete.any -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.on -import org.junit.Assert.assertFalse -import org.junit.Test -import org.mockito.Mockito.* -import javax.inject.Provider -import kotlin.concurrent.thread - -class OverpassMapDataAndGeometryApiTest { - - @Test fun handleOverpassQuota() { - val provider = mock>() - on(provider.get()).thenReturn(mock()) - - val status = OverpassStatus() - status.availableSlots = 0 - status.nextAvailableSlotIn = 1 - status.maxAvailableSlots = 2 - - val overpass = mock() - on(overpass.getStatus()).thenReturn(status) - doThrow(OsmTooManyRequestsException::class.java).on(overpass).queryElementsWithGeometry(any(), any()) - val dao = OverpassMapDataAndGeometryApi(overpass, mock()) - // the dao will call get(), get an exception in return, ask its status - // then and at least wait for the specified amount of time before calling again - var result = false - val dlThread = thread { - result = dao.query("") { _, _ -> Unit } - } - // sleep the wait time: Downloader should not try to call - // overpass again in this time - Thread.sleep(status.nextAvailableSlotIn!! * 1000L) - verify(overpass, times(1)).getStatus() - verify(overpass, times(1)).queryElementsWithGeometry(any(), any()) - // now we test if dao will call overpass again after that time. It is not really - // defined when the downloader must call overpass again, lets assume 1.5 secs here and - // change it when it fails - Thread.sleep(1500) - verify(overpass, times(2)).getStatus() - verify(overpass, times(2)).queryElementsWithGeometry(any(), any()) - // we are done here, interrupt thread (still part of the test though...) - dlThread.interrupt() - dlThread.join() - assertFalse(result) - } -} diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmApiQuestDownloaderTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmApiQuestDownloaderTest.kt new file mode 100644 index 0000000000..1b8a12d914 --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmApiQuestDownloaderTest.kt @@ -0,0 +1,94 @@ +package de.westnordost.streetcomplete.data.osm.osmquest + +import de.westnordost.countryboundaries.CountryBoundaries +import de.westnordost.osmapi.map.MapDataWithGeometry +import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.data.Element +import de.westnordost.osmapi.map.data.OsmLatLon +import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.streetcomplete.any +import de.westnordost.streetcomplete.data.MapDataApi +import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryCreator +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry +import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.data.osmnotes.NotePositionsSource +import de.westnordost.streetcomplete.data.quest.AllCountries +import de.westnordost.streetcomplete.data.quest.Countries +import de.westnordost.streetcomplete.mock +import de.westnordost.streetcomplete.on +import de.westnordost.streetcomplete.quests.AbstractQuestAnswerFragment +import org.junit.Assert.* +import org.junit.Before +import org.junit.Test +import org.mockito.Mockito.verify +import java.util.concurrent.FutureTask +import javax.inject.Provider + +class OsmApiQuestDownloaderTest { + private lateinit var elementDb: MergedElementDao + private lateinit var osmQuestController: OsmQuestController + private lateinit var countryBoundaries: CountryBoundaries + private lateinit var notePositionsSource: NotePositionsSource + private lateinit var mapDataApi: MapDataApi + private lateinit var mapDataWithGeometry: CachingMapDataWithGeometry + private lateinit var elementGeometryCreator: ElementGeometryCreator + private lateinit var downloader: OsmApiQuestDownloader + + private val bbox = BoundingBox(0.0, 0.0, 1.0, 1.0) + + @Before fun setUp() { + elementDb = mock() + osmQuestController = mock() + on(osmQuestController.replaceInBBox(any(), any(), any())).thenReturn(OsmQuestController.UpdateResult(0,0)) + countryBoundaries = mock() + mapDataApi = mock() + mapDataWithGeometry = mock() + elementGeometryCreator = mock() + notePositionsSource = mock() + val countryBoundariesFuture = FutureTask { countryBoundaries } + countryBoundariesFuture.run() + val mapDataProvider = Provider { mapDataWithGeometry } + downloader = OsmApiQuestDownloader( + elementDb, osmQuestController, countryBoundariesFuture, notePositionsSource, mapDataApi, + mapDataProvider, elementGeometryCreator) + } + + @Test fun `creates quest for element`() { + val pos = OsmLatLon(1.0, 1.0) + val node = OsmNode(5, 0, pos, null) + val geom = ElementPointGeometry(pos) + val questType = TestMapDataQuestType(listOf(node)) + + on(mapDataWithGeometry.getNodeGeometry(5)).thenReturn(geom) + on(osmQuestController.replaceInBBox(any(), any(), any())).thenAnswer { + val createdQuests = it.arguments[0] as List + assertEquals(1, createdQuests.size) + val quest = createdQuests[0] + assertEquals(5, quest.elementId) + assertEquals(Element.Type.NODE, quest.elementType) + assertEquals(geom, quest.geometry) + assertEquals(questType, quest.osmElementQuestType) + OsmQuestController.UpdateResult(1,0) + } + + downloader.download(listOf(questType), bbox) + + verify(elementDb).putAll(any()) + verify(elementDb).deleteUnreferenced() + verify(osmQuestController).replaceInBBox(any(), any(), any()) + } +} + +private class TestMapDataQuestType(private val list: List) : OsmElementQuestType { + + override var enabledInCountries: Countries = AllCountries + + override val icon = 0 + override val commitMessage = "" + override fun getTitle(tags: Map) = 0 + override fun createForm() = object : AbstractQuestAnswerFragment() {} + override fun isApplicableTo(element: Element) = false + override fun applyAnswerTo(answer: String, changes: StringMapChangesBuilder) {} + override fun getApplicableElements(mapData: MapDataWithGeometry): Iterable = list +} diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementUpdateControllerTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementUpdateControllerTest.kt new file mode 100644 index 0000000000..4c82dbb78a --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmElementUpdateControllerTest.kt @@ -0,0 +1,76 @@ +package de.westnordost.streetcomplete.data.osm.osmquest + +import de.westnordost.osmapi.map.data.Element +import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.streetcomplete.data.MapDataApi +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryCreator +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry +import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.mock +import de.westnordost.streetcomplete.on +import org.junit.Before +import org.junit.Test +import org.mockito.Mockito.verify + +class OsmElementUpdateControllerTest { + + private lateinit var mapDataApi: MapDataApi + private lateinit var elementGeometryCreator: ElementGeometryCreator + private lateinit var elementDB: MergedElementDao + private lateinit var questGiver: OsmQuestGiver + private lateinit var c: OsmElementUpdateController + + @Before fun setUp() { + mapDataApi = mock() + elementGeometryCreator = mock() + elementDB = mock() + questGiver = mock() + c = OsmElementUpdateController(mapDataApi, elementGeometryCreator, elementDB, questGiver) + } + + @Test fun delete() { + c.delete(Element.Type.NODE, 123L) + + verify(elementDB).delete(Element.Type.NODE, 123L) + verify(questGiver).deleteQuests(Element.Type.NODE, 123L) + } + + @Test fun update() { + val element = OsmNode(123L, 1, 0.0, 0.0, null) + val point = ElementPointGeometry(element.position) + + on(elementGeometryCreator.create(element)).thenReturn(point) + + c.update(element, null) + + verify(elementDB).put(element) + verify(elementGeometryCreator).create(element) + verify(questGiver).updateQuests(element, point) + } + + @Test fun `update deleted`() { + val element = OsmNode(123L, 1, 0.0, 0.0, null) + + on(elementGeometryCreator.create(element)).thenReturn(null) + + c.update(element, null) + + verify(elementDB).delete(element.type, element.id) + verify(questGiver).deleteQuests(element.type, element.id) + } + + @Test fun recreate() { + val element = OsmNode(123L, 1, 0.0, 0.0, null) + val point = ElementPointGeometry(element.position) + val questType: OsmElementQuestType = mock() + val questTypes = listOf(questType) + on(elementGeometryCreator.create(element)).thenReturn(point) + + c.update(element, questTypes) + + + verify(elementDB).put(element) + verify(elementGeometryCreator).create(element) + verify(questGiver).recreateQuests(element, point, questTypes) + } +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDownloaderTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDownloaderTest.kt deleted file mode 100644 index f5fafe05c0..0000000000 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestDownloaderTest.kt +++ /dev/null @@ -1,117 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.osmquest - -import de.westnordost.countryboundaries.CountryBoundaries -import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.osmapi.map.data.Element -import de.westnordost.osmapi.map.data.OsmLatLon -import de.westnordost.osmapi.map.data.OsmNode -import de.westnordost.streetcomplete.any -import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao -import de.westnordost.streetcomplete.data.quest.AllCountries -import de.westnordost.streetcomplete.data.quest.AllCountriesExcept -import de.westnordost.streetcomplete.data.quest.Countries -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.on -import de.westnordost.streetcomplete.quests.AbstractQuestAnswerFragment -import org.junit.Before -import org.junit.Test -import org.mockito.Mockito.* -import java.util.concurrent.FutureTask - - -class OsmQuestDownloaderTest { - private lateinit var elementDb: MergedElementDao - private lateinit var osmQuestController: OsmQuestController - private lateinit var countryBoundaries: CountryBoundaries - private lateinit var downloader: OsmQuestDownloader - - @Before fun setUp() { - elementDb = mock() - osmQuestController = mock() - on(osmQuestController.replaceInBBox(any(), any(), any())).thenReturn(OsmQuestController.UpdateResult(0,0)) - countryBoundaries = mock() - val countryBoundariesFuture = FutureTask { countryBoundaries } - countryBoundariesFuture.run() - downloader = OsmQuestDownloader(elementDb, osmQuestController, countryBoundariesFuture) - } - - @Test fun `ignore element with invalid geometry`() { - val invalidGeometryElement = ElementWithGeometry( - OsmNode(0, 0, OsmLatLon(1.0, 1.0), null), - null - ) - - val questType = ListBackedQuestType(listOf(invalidGeometryElement)) - - downloader.download(questType, BoundingBox(0.0, 0.0, 1.0, 1.0), setOf()) - } - - @Test fun `ignore at blacklisted position`() { - val blacklistPos = OsmLatLon(0.3, 0.4) - val blacklistElement = ElementWithGeometry( - OsmNode(0, 0, blacklistPos, null), - ElementPointGeometry(blacklistPos) - ) - - val questType = ListBackedQuestType(listOf(blacklistElement)) - - downloader.download(questType, BoundingBox(0.0, 0.0, 1.0, 1.0), setOf(blacklistPos)) - } - - @Test fun `ignore element in country for which this quest is disabled`() { - val pos = OsmLatLon(1.0, 1.0) - val inDisabledCountryElement = ElementWithGeometry( - OsmNode(0, 0, pos, null), - ElementPointGeometry(pos) - ) - - val questType = ListBackedQuestType(listOf(inDisabledCountryElement)) - questType.enabledInCountries = AllCountriesExcept("AA") - // country boundaries say that position is in AA - on(countryBoundaries.isInAny(anyDouble(),anyDouble(),any())).thenReturn(true) - on(countryBoundaries.getContainingIds(anyDouble(),anyDouble(),anyDouble(),anyDouble())).thenReturn(setOf()) - - downloader.download(questType, BoundingBox(0.0, 0.0, 1.0, 1.0), setOf()) - } - - @Test fun `creates quest for element`() { - val pos = OsmLatLon(1.0, 1.0) - val normalElement = ElementWithGeometry( - OsmNode(0, 0, pos, null), - ElementPointGeometry(pos) - ) - - val questType = ListBackedQuestType(listOf(normalElement)) - - on(osmQuestController.replaceInBBox(any(), any(), any())).thenReturn(OsmQuestController.UpdateResult(0,0)) - - downloader.download(questType, BoundingBox(0.0, 0.0, 1.0, 1.0), setOf()) - - verify(elementDb).putAll(any()) - verify(osmQuestController).replaceInBBox(any(), any(), any()) - } -} - -private data class ElementWithGeometry(val element: Element, val geometry: ElementGeometry?) - -private class ListBackedQuestType(private val list: List) : OsmElementQuestType { - - override var enabledInCountries: Countries = AllCountries - - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit): Boolean { - for (e in list) { - handler(e.element, e.geometry) - } - return true - } - - override val icon = 0 - override val commitMessage = "" - override fun getTitle(tags: Map) = 0 - override fun createForm() = object : AbstractQuestAnswerFragment() {} - override fun isApplicableTo(element: Element) = false - override fun applyAnswerTo(answer: String, changes: StringMapChangesBuilder) {} -} diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiverTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiverTest.kt index 5e5a863f4a..5664ddbb17 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiverTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestGiverTest.kt @@ -39,7 +39,7 @@ class OsmQuestGiverTest { osmQuestController = mock() on(osmQuestController.getAllForElement(Element.Type.NODE, 1)).thenReturn(emptyList()) - on(osmQuestController.updateForElement(any(), any(), any(), anyLong())).thenReturn(OsmQuestController.UpdateResult(0,0)) + on(osmQuestController.updateForElement(any(), any(), any(), any(), anyLong())).thenReturn(OsmQuestController.UpdateResult(0,0)) questType = mock() on(questType.enabledInCountries).thenReturn(AllCountries) @@ -61,7 +61,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) - verify(osmQuestController).updateForElement(emptyList(), emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(emptyList(), emptyList(), GEOM, NODE.type, NODE.id) } @Test fun `previous quest blocks new quest`() { @@ -71,7 +71,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) - verify(osmQuestController).updateForElement(emptyList(), emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(emptyList(), emptyList(), GEOM, NODE.type, NODE.id) } @Test fun `not applicable blocks new quest`() { @@ -80,7 +80,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) - verify(osmQuestController).updateForElement(emptyList(), emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(emptyList(), emptyList(), GEOM, NODE.type, NODE.id) } @Test fun `not applicable removes previous quest`() { @@ -92,7 +92,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) - verify(osmQuestController).updateForElement(emptyList(), listOf(123L), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(emptyList(), listOf(123L), GEOM, NODE.type, NODE.id) } @Test fun `applicable adds new quest`() { @@ -100,7 +100,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) val expectedQuest = OsmQuest(questType, NODE.type, NODE.id, GEOM) - verify(osmQuestController).updateForElement(arrayListOf(expectedQuest), emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(arrayListOf(expectedQuest), emptyList(), GEOM, NODE.type, NODE.id) } @Test fun `quest is only enabled in the country the element is in`() { @@ -110,7 +110,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) val expectedQuest = OsmQuest(questType, NODE.type, NODE.id, GEOM) - verify(osmQuestController).updateForElement(arrayListOf(expectedQuest), emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(arrayListOf(expectedQuest), emptyList(), GEOM, NODE.type, NODE.id) } @Test fun `quest is disabled in a country the element is not in`() { @@ -119,7 +119,7 @@ class OsmQuestGiverTest { osmQuestGiver.updateQuests(NODE, GEOM) - verify(osmQuestController).updateForElement(emptyList(), emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(emptyList(), emptyList(), GEOM, NODE.type, NODE.id) } @Test fun `recreate quests`() { @@ -130,7 +130,7 @@ class OsmQuestGiverTest { OsmQuest(questType, NODE.type, NODE.id, GEOM), OsmQuest(questType2, NODE.type, NODE.id, GEOM) ) - verify(osmQuestController).updateForElement(expectedQuests, emptyList(), NODE.type, NODE.id) + verify(osmQuestController).updateForElement(expectedQuests, emptyList(), GEOM, NODE.type, NODE.id) } } diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploaderTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploaderTest.kt index c4cc9a4591..8325870ac2 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploaderTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/OsmQuestsUploaderTest.kt @@ -7,13 +7,11 @@ import de.westnordost.streetcomplete.any import de.westnordost.streetcomplete.data.quest.QuestStatus import de.westnordost.streetcomplete.data.osm.changes.StringMapChanges import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.osm.upload.ChangesetConflictException import de.westnordost.streetcomplete.data.osm.upload.ElementConflictException +import de.westnordost.streetcomplete.data.osm.upload.ElementDeletedException import de.westnordost.streetcomplete.data.user.StatisticsUpdater import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.on @@ -26,39 +24,32 @@ import java.util.concurrent.atomic.AtomicBoolean class OsmQuestsUploaderTest { private lateinit var osmQuestController: OsmQuestController - private lateinit var elementDB: MergedElementDao private lateinit var changesetManager: OpenQuestChangesetsManager - private lateinit var elementGeometryDB: ElementGeometryDao - private lateinit var questGiver: OsmQuestGiver - private lateinit var elementGeometryCreator: OsmApiElementGeometryCreator private lateinit var singleChangeUploader: SingleOsmElementTagChangesUploader private lateinit var statisticsUpdater: StatisticsUpdater + private lateinit var elementUpdateController: OsmElementUpdateController private lateinit var uploader: OsmQuestsUploader @Before fun setUp() { osmQuestController = mock() - elementDB = mock() - on(elementDB.get(any(), anyLong())).thenReturn(createElement()) changesetManager = mock() singleChangeUploader = mock() - elementGeometryDB = mock() - questGiver = mock() - elementGeometryCreator = mock() statisticsUpdater = mock() - on(elementGeometryCreator.create(any())).thenReturn(mock()) - uploader = OsmQuestsUploader(elementDB, elementGeometryDB, changesetManager, questGiver, - elementGeometryCreator, osmQuestController, singleChangeUploader, statisticsUpdater) + elementUpdateController = mock() + uploader = OsmQuestsUploader(changesetManager, elementUpdateController, + osmQuestController, singleChangeUploader, statisticsUpdater) } @Test fun `cancel upload works`() { uploader.upload(AtomicBoolean(true)) - verifyZeroInteractions(changesetManager, singleChangeUploader, elementDB, osmQuestController) + verifyZeroInteractions(elementUpdateController, changesetManager, singleChangeUploader, statisticsUpdater, osmQuestController) } @Test fun `catches ElementConflict exception`() { on(osmQuestController.getAllAnswered()).thenReturn(listOf(createQuest())) on(singleChangeUploader.upload(anyLong(), any(), any())) .thenThrow(ElementConflictException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.upload(AtomicBoolean(false)) @@ -68,12 +59,15 @@ class OsmQuestsUploaderTest { @Test fun `discard if element was deleted`() { val q = createQuest() on(osmQuestController.getAllAnswered()).thenReturn(listOf(q)) - on(elementDB.get(any(), anyLong())).thenReturn(null) + on(singleChangeUploader.upload(anyLong(), any(), any())) + .thenThrow(ElementDeletedException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) verify(uploader.uploadedChangeListener)?.onDiscarded(q.osmElementQuestType.javaClass.simpleName, q.position) + verify(elementUpdateController).delete(any(), anyLong()) } @Test fun `catches ChangesetConflictException exception and tries again once`() { @@ -81,6 +75,7 @@ class OsmQuestsUploaderTest { on(singleChangeUploader.upload(anyLong(), any(), any())) .thenThrow(ChangesetConflictException()) .thenReturn(createElement()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.upload(AtomicBoolean(false)) @@ -95,15 +90,14 @@ class OsmQuestsUploaderTest { on(osmQuestController.getAllAnswered()).thenReturn(quests) on(singleChangeUploader.upload(anyLong(), any(), any())).thenReturn(createElement()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) verify(osmQuestController, times(2)).success(any()) verify(uploader.uploadedChangeListener, times(2))?.onUploaded(any(), any()) - verify(elementDB, times(2)).put(any()) - verify(elementGeometryDB, times(2)).put(any()) - verify(questGiver, times(2)).updateQuests(any(), any()) + verify(elementUpdateController, times(2)).update(any(), isNull()) verify(statisticsUpdater, times(2)).addOne(any(), any()) } @@ -113,13 +107,17 @@ class OsmQuestsUploaderTest { on(osmQuestController.getAllAnswered()).thenReturn(quests) on(singleChangeUploader.upload(anyLong(), any(), any())) .thenThrow(ElementConflictException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) verify(osmQuestController, times(2)).fail(any()) verify(uploader.uploadedChangeListener,times(2))?.onDiscarded(any(), any()) - verifyZeroInteractions(questGiver, elementGeometryCreator, statisticsUpdater) + verify(elementUpdateController, times(2)).get(any(), anyLong()) + verify(elementUpdateController).cleanUp() + verifyNoMoreInteractions(elementUpdateController) + verifyZeroInteractions(statisticsUpdater) } @Test fun `delete unreferenced elements and clean metadata at the end`() { @@ -127,10 +125,11 @@ class OsmQuestsUploaderTest { on(osmQuestController.getAllAnswered()).thenReturn(listOf(quest)) on(singleChangeUploader.upload(anyLong(), any(), any())).thenReturn(createElement()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.upload(AtomicBoolean(false)) - verify(elementDB).deleteUnreferenced() + verify(elementUpdateController).cleanUp() verify(quest.osmElementQuestType).cleanMetadata() } } diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/SimpleOverpassQuestsValidityTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/SimpleOverpassQuestsValidityTest.kt deleted file mode 100644 index a40ac97493..0000000000 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/SimpleOverpassQuestsValidityTest.kt +++ /dev/null @@ -1,28 +0,0 @@ -package de.westnordost.streetcomplete.data.osm.osmquest - -import org.junit.Test - -import de.westnordost.osmapi.map.data.BoundingBox -import de.westnordost.streetcomplete.data.osmnotes.notequests.OsmNoteQuestType -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.quests.QuestModule - -import org.junit.Assert.* - -class SimpleOverpassQuestsValidityTest { - - @Test fun `query valid`() { - val bbox = BoundingBox(0.0, 0.0, 1.0, 1.0) - val questTypes = QuestModule.questTypeRegistry(OsmNoteQuestType(), mock(), mock(), mock(), mock(), mock(), mock()).all - - for (questType in questTypes) { - if (questType is SimpleOverpassQuestType<*>) { - // if this fails and the returned exception is not informative, catch here and record - // the name of the SimpleOverpassQuestType - questType.getOverpassQuery(bbox) - } - } - // parsing the query threw no errors -> valid - assertTrue(true) - } -} diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploaderTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploaderTest.kt index 36e8ab53ec..74bc248f9c 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploaderTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/osmquest/undo/UndoOsmQuestsUploaderTest.kt @@ -6,14 +6,12 @@ import de.westnordost.osmapi.map.data.OsmNode import de.westnordost.streetcomplete.any import de.westnordost.streetcomplete.data.osm.changes.StringMapChanges import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao -import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestGiver +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementUpdateController import de.westnordost.streetcomplete.data.osm.osmquest.SingleOsmElementTagChangesUploader import de.westnordost.streetcomplete.data.osm.upload.ChangesetConflictException import de.westnordost.streetcomplete.data.osm.upload.ElementConflictException +import de.westnordost.streetcomplete.data.osm.upload.ElementDeletedException import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.user.StatisticsUpdater import de.westnordost.streetcomplete.mock @@ -26,40 +24,33 @@ import java.util.concurrent.atomic.AtomicBoolean class UndoOsmQuestsUploaderTest { private lateinit var undoQuestDB: UndoOsmQuestDao - private lateinit var elementDB: MergedElementDao private lateinit var changesetManager: OpenQuestChangesetsManager - private lateinit var elementGeometryDB: ElementGeometryDao - private lateinit var questGiver: OsmQuestGiver - private lateinit var elementGeometryCreator: OsmApiElementGeometryCreator private lateinit var singleChangeUploader: SingleOsmElementTagChangesUploader private lateinit var statisticsUpdater: StatisticsUpdater + private lateinit var elementUpdateController: OsmElementUpdateController private lateinit var uploader: UndoOsmQuestsUploader @Before fun setUp() { undoQuestDB = mock() - elementDB = mock() - on(elementDB.get(any(), anyLong())).thenReturn(createElement()) changesetManager = mock() singleChangeUploader = mock() - elementGeometryDB = mock() - questGiver = mock() - elementGeometryCreator = mock() statisticsUpdater = mock() - on(elementGeometryCreator.create(any())).thenReturn(mock()) - uploader = UndoOsmQuestsUploader(elementDB, elementGeometryDB, changesetManager, questGiver, - elementGeometryCreator, undoQuestDB, singleChangeUploader, statisticsUpdater) + elementUpdateController = mock() + uploader = UndoOsmQuestsUploader(changesetManager, elementUpdateController, + undoQuestDB, singleChangeUploader, statisticsUpdater) } @Test fun `cancel upload works`() { val cancelled = AtomicBoolean(true) uploader.upload(cancelled) - verifyZeroInteractions(changesetManager, singleChangeUploader, elementDB, undoQuestDB) + verifyZeroInteractions(changesetManager, singleChangeUploader, statisticsUpdater, elementUpdateController, undoQuestDB) } @Test fun `catches ElementConflict exception`() { on(undoQuestDB.getAll()).thenReturn(listOf(createUndoQuest())) on(singleChangeUploader.upload(anyLong(), any(), any())) .thenThrow(ElementConflictException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.upload(AtomicBoolean(false)) @@ -69,12 +60,15 @@ class UndoOsmQuestsUploaderTest { @Test fun `discard if element was deleted`() { val q = createUndoQuest() on(undoQuestDB.getAll()).thenReturn(listOf(q)) - on(elementDB.get(any(), anyLong())).thenReturn(null) + on(singleChangeUploader.upload(anyLong(), any(), any())) + .thenThrow(ElementDeletedException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) verify(uploader.uploadedChangeListener)?.onDiscarded(q.osmElementQuestType.javaClass.simpleName, q.position) + verify(elementUpdateController).delete(any(), anyLong()) } @Test fun `catches ChangesetConflictException exception and tries again once`() { @@ -82,6 +76,7 @@ class UndoOsmQuestsUploaderTest { on(singleChangeUploader.upload(anyLong(), any(), any())) .thenThrow(ChangesetConflictException()) .thenReturn(createElement()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.upload(AtomicBoolean(false)) @@ -93,10 +88,12 @@ class UndoOsmQuestsUploaderTest { @Test fun `delete each uploaded quest from local DB and calls listener`() { val quests = listOf(createUndoQuest(), createUndoQuest()) + on(undoQuestDB.getAll()).thenReturn(quests) on(singleChangeUploader.upload(anyLong(), any(), any())) .thenThrow(ElementConflictException()) .thenReturn(createElement()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) @@ -105,11 +102,11 @@ class UndoOsmQuestsUploaderTest { verify(uploader.uploadedChangeListener)?.onUploaded(quests[0].osmElementQuestType.javaClass.simpleName, quests[0].position) verify(uploader.uploadedChangeListener)?.onDiscarded(quests[1].osmElementQuestType.javaClass.simpleName, quests[1].position) - verify(elementDB, times(1)).put(any()) - verify(elementGeometryDB, times(1)).put(any()) - verify(questGiver, times(1)).updateQuests(any(), any()) + verify(elementUpdateController, times(1)).update(any(), isNull()) + verify(elementUpdateController, times(1)).cleanUp() + verify(elementUpdateController, times(2)).get(any(), anyLong()) verify(statisticsUpdater).subtractOne(any(), any()) - verifyNoMoreInteractions(questGiver) + verifyNoMoreInteractions(elementUpdateController) } @Test fun `delete unreferenced elements and clean metadata at the end`() { @@ -117,10 +114,11 @@ class UndoOsmQuestsUploaderTest { on(undoQuestDB.getAll()).thenReturn(listOf(quest)) on(singleChangeUploader.upload(anyLong(), any(), any())).thenReturn(createElement()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(mock()) uploader.upload(AtomicBoolean(false)) - verify(elementDB).deleteUnreferenced() + verify(elementUpdateController).cleanUp() verify(quest.osmElementQuestType).cleanMetadata() } } diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploaderTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploaderTest.kt index 9fe3004b4f..6f0288b177 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploaderTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/splitway/SplitWaysUploaderTest.kt @@ -4,57 +4,47 @@ import de.westnordost.osmapi.map.data.OsmLatLon import de.westnordost.osmapi.map.data.OsmWay import de.westnordost.streetcomplete.on import de.westnordost.streetcomplete.any -import de.westnordost.streetcomplete.data.osm.osmquest.OsmQuestGiver -import de.westnordost.streetcomplete.data.osm.elementgeometry.OsmApiElementGeometryCreator -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometryDao -import de.westnordost.streetcomplete.data.osm.mapdata.MergedElementDao +import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementUpdateController import de.westnordost.streetcomplete.data.osm.upload.ChangesetConflictException import de.westnordost.streetcomplete.data.osm.upload.ElementConflictException +import de.westnordost.streetcomplete.data.osm.upload.ElementDeletedException import de.westnordost.streetcomplete.data.osm.upload.changesets.OpenQuestChangesetsManager import de.westnordost.streetcomplete.data.user.StatisticsUpdater import de.westnordost.streetcomplete.mock import org.junit.Before import org.junit.Test -import org.mockito.ArgumentMatchers import org.mockito.Mockito.* import java.util.concurrent.atomic.AtomicBoolean class SplitWaysUploaderTest { private lateinit var splitWayDB: OsmQuestSplitWayDao - private lateinit var elementDB: MergedElementDao private lateinit var changesetManager: OpenQuestChangesetsManager - private lateinit var elementGeometryDB: ElementGeometryDao - private lateinit var questGiver: OsmQuestGiver - private lateinit var elementGeometryCreator: OsmApiElementGeometryCreator private lateinit var splitSingleOsmWayUploader: SplitSingleWayUploader private lateinit var statisticsUpdater: StatisticsUpdater + private lateinit var elementUpdateController: OsmElementUpdateController private lateinit var uploader: SplitWaysUploader @Before fun setUp() { splitWayDB = mock() - elementDB = mock() - on(elementDB.get(any(), ArgumentMatchers.anyLong())).thenReturn(createElement()) changesetManager = mock() splitSingleOsmWayUploader = mock() - elementGeometryDB = mock() - questGiver = mock() - elementGeometryCreator = mock() + elementUpdateController = mock() statisticsUpdater = mock() - on(elementGeometryCreator.create(any())).thenReturn(mock()) - uploader = SplitWaysUploader(elementDB, elementGeometryDB, changesetManager, questGiver, - elementGeometryCreator, splitWayDB, splitSingleOsmWayUploader, statisticsUpdater) + uploader = SplitWaysUploader(changesetManager, elementUpdateController, splitWayDB, + splitSingleOsmWayUploader, statisticsUpdater) } @Test fun `cancel upload works`() { val cancelled = AtomicBoolean(true) uploader.upload(cancelled) - verifyZeroInteractions(changesetManager, splitSingleOsmWayUploader, elementDB, splitWayDB) + verifyZeroInteractions(changesetManager, splitSingleOsmWayUploader, elementUpdateController, statisticsUpdater, splitWayDB) } @Test fun `catches ElementConflict exception`() { on(splitWayDB.getAll()).thenReturn(listOf(createOsmSplitWay())) on(splitSingleOsmWayUploader.upload(anyLong(), any(), anyList())) .thenThrow(ElementConflictException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(createElement()) uploader.upload(AtomicBoolean(false)) @@ -64,7 +54,9 @@ class SplitWaysUploaderTest { @Test fun `discard if element was deleted`() { val q = createOsmSplitWay() on(splitWayDB.getAll()).thenReturn(listOf(q)) - on(elementDB.get(any(),anyLong())).thenReturn(null) + on(splitSingleOsmWayUploader.upload(anyLong(), any(), any())) + .thenThrow(ElementDeletedException()) + on(elementUpdateController.get(any(), anyLong())).thenReturn(createElement()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) @@ -77,6 +69,7 @@ class SplitWaysUploaderTest { on(splitSingleOsmWayUploader.upload(anyLong(), any(), anyList())) .thenThrow(ChangesetConflictException()) .thenReturn(listOf(createElement())) + on(elementUpdateController.get(any(), anyLong())).thenReturn(createElement()) uploader.upload(AtomicBoolean(false)) @@ -88,10 +81,12 @@ class SplitWaysUploaderTest { @Test fun `delete each uploaded split from local DB and calls listener`() { val quests = listOf(createOsmSplitWay(), createOsmSplitWay()) + on(splitWayDB.getAll()).thenReturn(quests) on(splitSingleOsmWayUploader.upload(anyLong(), any(), anyList())) .thenThrow(ElementConflictException()) .thenReturn(listOf(createElement())) + on(elementUpdateController.get(any(), anyLong())).thenReturn(createElement()) uploader.uploadedChangeListener = mock() uploader.upload(AtomicBoolean(false)) @@ -100,11 +95,11 @@ class SplitWaysUploaderTest { verify(uploader.uploadedChangeListener)?.onUploaded(quests[0].questType.javaClass.simpleName, quests[0].position) verify(uploader.uploadedChangeListener)?.onDiscarded(quests[1].questType.javaClass.simpleName,quests[1].position) - verify(elementDB, times(1)).put(any()) - verify(elementGeometryDB, times(1)).put(any()) - verify(questGiver, times(1)).recreateQuests(any(), any(), any()) + verify(elementUpdateController, times(1)).update(any(), any()) + verify(elementUpdateController).cleanUp() + verify(elementUpdateController, times(2)).get(any(), anyLong()) verify(statisticsUpdater).addOne(any(), any()) - verifyNoMoreInteractions(questGiver) + verifyNoMoreInteractions(elementUpdateController) } @Test fun `delete unreferenced elements and clean metadata at the end`() { @@ -113,10 +108,11 @@ class SplitWaysUploaderTest { on(splitWayDB.getAll()).thenReturn(listOf(quest)) on(splitSingleOsmWayUploader.upload(anyLong(), any(), any())) .thenReturn(listOf(createElement())) + on(elementUpdateController.get(any(), anyLong())).thenReturn(createElement()) uploader.upload(AtomicBoolean(false)) - verify(elementDB).deleteUnreferenced() + verify(elementUpdateController).cleanUp() verify(quest.osmElementQuestType).cleanMetadata() } } diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/osm/upload/changesets/OpenQuestChangesetsManagerTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/osm/upload/changesets/OpenQuestChangesetsManagerTest.kt index 562f1cccea..daffa7ccbb 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/osm/upload/changesets/OpenQuestChangesetsManagerTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/osm/upload/changesets/OpenQuestChangesetsManagerTest.kt @@ -1,12 +1,11 @@ package de.westnordost.streetcomplete.data.osm.upload.changesets import android.content.SharedPreferences +import de.westnordost.osmapi.map.MapDataWithGeometry import de.westnordost.streetcomplete.data.MapDataApi -import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.osmapi.map.data.Element import de.westnordost.streetcomplete.ApplicationConstants import de.westnordost.streetcomplete.any -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder import de.westnordost.streetcomplete.mock @@ -77,8 +76,8 @@ class OpenQuestChangesetsManagerTest { private class TestQuestType : OsmElementQuestType { + override fun getApplicableElements(mapData: MapDataWithGeometry) = emptyList() override fun getTitle(tags: Map) = 0 - override fun download(bbox: BoundingBox, handler: (element: Element, geometry: ElementGeometry?) -> Unit) = false override fun isApplicableTo(element: Element):Boolean? = null override fun applyAnswerTo(answer: String, changes: StringMapChangesBuilder) {} override val icon = 0 diff --git a/app/src/test/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSourceTest.kt b/app/src/test/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSourceTest.kt index 4ef5b97650..eab0deae92 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSourceTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/data/quest/VisibleQuestsSourceTest.kt @@ -54,10 +54,10 @@ class VisibleQuestsSourceTest { } @Test fun getAllVisibleCount() { - on(osmQuestController.getAllVisibleInBBoxCount(bbox, questTypes)).thenReturn(3) + on(osmQuestController.getAllVisibleInBBoxCount(bbox)).thenReturn(3) on(osmNoteQuestController.getAllVisibleInBBoxCount(bbox)).thenReturn(4) - assertEquals(7, source.getAllVisibleCount(bbox, questTypes)) + assertEquals(7, source.getAllVisibleCount(bbox)) } @Test fun getAllVisible() { diff --git a/app/src/test/java/de/westnordost/streetcomplete/ktx/CollectionsTest.kt b/app/src/test/java/de/westnordost/streetcomplete/ktx/CollectionsTest.kt index 2c7944935b..60c9a002c0 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/ktx/CollectionsTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/ktx/CollectionsTest.kt @@ -1,5 +1,7 @@ package de.westnordost.streetcomplete.ktx +import de.westnordost.osmapi.map.data.LatLon +import de.westnordost.osmapi.map.data.OsmLatLon import org.junit.Assert.* import org.junit.Test @@ -49,18 +51,23 @@ class CollectionsTest { assertNull(listOf(1, 2, 3).findPrevious(-1) { true }) } - @Test fun `forEachPair with empty list`() { - listOf().forEachPair { _, _ -> fail() } + @Test fun `forEachLine with empty list`() { + listOf().forEachLine { _, _ -> fail() } } - @Test fun `forEachPair with list with only one element`() { - listOf(1).forEachPair { _, _ -> fail() } + @Test fun `forEachLine with list with only one element`() { + listOf(OsmLatLon(0.0,0.0)).forEachLine { _, _ -> fail() } } - @Test fun `forEachPair with several elements`() { + @Test fun `forEachLine with several elements`() { var counter = 0 - listOf(1,2,3,4).forEachPair { first, second -> - assertEquals(first+1, second) + listOf( + OsmLatLon(0.0,0.0), + OsmLatLon(1.0,0.0), + OsmLatLon(2.0,0.0), + OsmLatLon(3.0,0.0), + ).forEachLine { first, second -> + assertEquals(first.latitude + 1, second.latitude, 0.0) counter++ } assertEquals(3, counter) diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddBuildingLevelsTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddBuildingLevelsTest.kt index e8ca776f32..95f89a0e0f 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddBuildingLevelsTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddBuildingLevelsTest.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.building_levels.AddBuildingLevels import de.westnordost.streetcomplete.quests.building_levels.BuildingLevelsAnswer import org.junit.Test class AddBuildingLevelsTest { - private val questType = AddBuildingLevels(mock()) + private val questType = AddBuildingLevels() @Test fun `apply building levels answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddBusStopShelterTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddBusStopShelterTest.kt index 51f30ed304..b3907fe2ba 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddBusStopShelterTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddBusStopShelterTest.kt @@ -3,7 +3,6 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryDelete -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.bus_stop_shelter.AddBusStopShelter import de.westnordost.streetcomplete.quests.bus_stop_shelter.BusStopShelterAnswer import org.junit.Test @@ -11,7 +10,7 @@ import java.util.* class AddBusStopShelterTest { - private val questType = AddBusStopShelter(mock(), mock()) + private val questType = AddBusStopShelter() @Test fun `apply shelter yes answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddCrossingTypeTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddCrossingTypeTest.kt index 349c1e60be..d686ee6a5f 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddCrossingTypeTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddCrossingTypeTest.kt @@ -3,14 +3,13 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.crossing_type.AddCrossingType import org.junit.Test import java.util.* class AddCrossingTypeTest { - private val questType = AddCrossingType(mock(), mock()) + private val questType = AddCrossingType() @Test fun `apply normal answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddCyclewayTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddCyclewayTest.kt index 0924f6836d..182ca21932 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddCyclewayTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddCyclewayTest.kt @@ -1,32 +1,108 @@ package de.westnordost.streetcomplete.quests import de.westnordost.osmapi.map.data.Element -import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.osmapi.map.data.OsmLatLon +import de.westnordost.osmapi.map.data.OsmWay import de.westnordost.streetcomplete.data.meta.toCheckDate import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryDelete import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.on +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry import de.westnordost.streetcomplete.quests.bikeway.* import de.westnordost.streetcomplete.quests.bikeway.Cycleway.* -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore +import de.westnordost.streetcomplete.util.translate import org.junit.Assert.* -import org.junit.Before import org.junit.Test -import org.mockito.ArgumentMatchers import java.util.* class AddCyclewayTest { - private lateinit var questType: AddCycleway + private val questType = AddCycleway() - @Before fun setUp() { - val r: ResurveyIntervalsStore = mock() - on(r.times(ArgumentMatchers.anyInt())).thenAnswer { (it.arguments[0] as Int).toDouble() } - on(r.times(ArgumentMatchers.anyDouble())).thenAnswer { (it.arguments[0] as Double) } - questType = AddCycleway(mock(), r) + @Test fun `applicable to road with missing cycleway`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "primary" + )) + )) + val p1 = OsmLatLon(0.0,0.0) + val p2 = p1.translate(50.0, 45.0) + mapData.wayGeometriesById[1L] = ElementPolylinesGeometry(listOf(listOf(p1, p2)), p1) + + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to road with nearby cycleway`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2), mapOf( + "highway" to "primary" + )), + OsmWay(2L, 1, listOf(3,4), mapOf( + "highway" to "cycleway" + )) + )) + val p1 = OsmLatLon(0.0,0.0) + val p2 = p1.translate(50.0, 45.0) + val p3 = p1.translate(14.0, 135.0) + val p4 = p3.translate(50.0, 45.0) + + mapData.wayGeometriesById[1L] = ElementPolylinesGeometry(listOf(listOf(p1, p2)), p1) + mapData.wayGeometriesById[2L] = ElementPolylinesGeometry(listOf(listOf(p3, p4)), p3) + + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `applicable to road with cycleway that is far away enough`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2), mapOf( + "highway" to "primary" + )), + OsmWay(2L, 1, listOf(3,4), mapOf( + "highway" to "cycleway" + )) + )) + val p1 = OsmLatLon(0.0,0.0) + val p2 = p1.translate(50.0, 45.0) + val p3 = p1.translate(16.0, 135.0) + val p4 = p3.translate(50.0, 45.0) + + mapData.wayGeometriesById[1L] = ElementPolylinesGeometry(listOf(listOf(p1, p2)), p1) + mapData.wayGeometriesById[2L] = ElementPolylinesGeometry(listOf(listOf(p3, p4)), p3) + + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to road with cycleway that is not old enough`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "primary", + "cycleway" to "track" + ), null, Date()) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `applicable to road with cycleway that is old enough`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "primary", + "cycleway" to "track", + "check_date:cycleway" to "2001-01-01" + ), null, Date()) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to road with cycleway that is old enough but has unknown cycleway tagging`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "primary", + "cycleway" to "whatsthis", + "check_date:cycleway" to "2001-01-01" + ), null, Date()) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } @Test fun `apply cycleway lane answer`() { @@ -489,5 +565,5 @@ class AddCyclewayTest { } private fun createElement(tags: Map, date: Date? = null): Element = - OsmNode(0L, 1, 0.0, 0.0, tags, null, date) + OsmWay(0L, 1, listOf(1,2,3), tags, null, date) } diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxHeightTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxHeightTest.kt index 1d98d65929..aa1c30e294 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxHeightTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxHeightTest.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.max_height.* import org.junit.Test class AddMaxHeightTest { - private val questType = AddMaxHeight(mock()) + private val questType = AddMaxHeight() @Test fun `apply metric height answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxSpeedTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxSpeedTest.kt index d17f447854..4856a723ee 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxSpeedTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxSpeedTest.kt @@ -2,13 +2,12 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.max_speed.* import org.junit.Test class AddMaxSpeedTest { - private val questType = AddMaxSpeed(mock()) + private val questType = AddMaxSpeed() @Test fun `apply no sign answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxWeightTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxWeightTest.kt index 11a123362c..96a26c93ee 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxWeightTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddMaxWeightTest.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.max_weight.* import org.junit.Test class AddMaxWeightTest { - private val questType = AddMaxWeight(mock()) + private val questType = AddMaxWeight() @Test fun `apply metric weight answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddParkingFeeTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddParkingFeeTest.kt index 09e71089e8..d38dfd28d1 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddParkingFeeTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddParkingFeeTest.kt @@ -8,7 +8,6 @@ import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryDelete import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.opening_hours.model.* import de.westnordost.streetcomplete.quests.parking_fee.* import org.junit.Test @@ -16,7 +15,7 @@ import java.util.* class AddParkingFeeTest { - private val questType = AddParkingFee(mock(), mock()) + private val questType = AddParkingFee() private val openingHours = OpeningHoursRuleList(listOf( Rule().apply { diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddPlaceNameTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddPlaceNameTest.kt index aabc53448d..a134e10b53 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddPlaceNameTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddPlaceNameTest.kt @@ -9,7 +9,7 @@ import org.junit.Test class AddPlaceNameTest { - private val questType = AddPlaceName(mock(), mock()) + private val questType = AddPlaceName(mock()) @Test fun `apply no name answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxCollectionTimesTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxCollectionTimesTest.kt index b8859a7b45..085fc26677 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxCollectionTimesTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxCollectionTimesTest.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.opening_hours.model.Weekdays import de.westnordost.streetcomplete.quests.postbox_collection_times.* import org.junit.Test class AddPostboxCollectionTimesTest { - private val questType = AddPostboxCollectionTimes(mock(), mock()) + private val questType = AddPostboxCollectionTimes() @Test fun `apply no signed times answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxRefTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxRefTest.kt index 704cdb727b..efc81625c0 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxRefTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddPostboxRefTest.kt @@ -1,7 +1,6 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.postbox_ref.AddPostboxRef import de.westnordost.streetcomplete.quests.postbox_ref.NoRefVisible import de.westnordost.streetcomplete.quests.postbox_ref.Ref @@ -9,7 +8,7 @@ import org.junit.Test class AddPostboxRefTest { - private val questType = AddPostboxRef(mock()) + private val questType = AddPostboxRef() @Test fun `apply no ref answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddProhibitedForPedestriansTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddProhibitedForPedestriansTest.kt index 79b8c48935..1658b9dedd 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddProhibitedForPedestriansTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddProhibitedForPedestriansTest.kt @@ -9,7 +9,7 @@ import org.junit.Test class AddProhibitedForPedestriansTest { - private val questType = AddProhibitedForPedestrians(mock()) + private val questType = AddProhibitedForPedestrians() @Test fun `apply yes answer`() { questType.verifyAnswer(YES, StringMapEntryAdd("foot", "no")) diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingContainerMaterialsTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingContainerMaterialsTest.kt index 4204376dc0..4c4d4f7090 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingContainerMaterialsTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingContainerMaterialsTest.kt @@ -1,32 +1,104 @@ package de.westnordost.streetcomplete.quests +import de.westnordost.osmapi.map.data.OsmLatLon +import de.westnordost.osmapi.map.data.OsmNode import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryDelete import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.on import de.westnordost.streetcomplete.quests.recycling_material.AddRecyclingContainerMaterials import de.westnordost.streetcomplete.quests.recycling_material.RecyclingMaterials import de.westnordost.streetcomplete.quests.recycling_material.IsWasteContainer import de.westnordost.streetcomplete.quests.recycling_material.RecyclingMaterial.* -import de.westnordost.streetcomplete.settings.ResurveyIntervalsStore -import org.junit.Before +import de.westnordost.streetcomplete.util.translate +import org.junit.Assert.assertEquals import org.junit.Test -import org.mockito.ArgumentMatchers.anyDouble -import org.mockito.ArgumentMatchers.anyInt import java.util.* class AddRecyclingContainerMaterialsTest { - @Before fun setUp() { - val r: ResurveyIntervalsStore = mock() - on(r.times(anyInt())).thenAnswer { (it.arguments[0] as Int).toDouble() } - on(r.times(anyDouble())).thenAnswer { (it.arguments[0] as Double) } - questType = AddRecyclingContainerMaterials(mock(), r) + private val questType = AddRecyclingContainerMaterials() + + @Test fun `applicable to container without recycling materials`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "amenity" to "recycling", + "recycling_type" to "container" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to container with recycling materials`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "amenity" to "recycling", + "recycling_type" to "container", + "recycling:something" to "yes" + )) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `applicable to container with old recycling materials`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "amenity" to "recycling", + "recycling_type" to "container", + "check_date:recycling" to "2001-01-01", + "recycling:plastic_packaging" to "yes", + "recycling:something_else" to "no" + ), null, Date()) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to container with old but unknown recycling materials`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "amenity" to "recycling", + "recycling_type" to "container", + "check_date:recycling" to "2001-01-01", + "recycling:something_else" to "yes" + ), null, Date()) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to container without recycling materials close to another container`() { + val pos1 = OsmLatLon(0.0,0.0) + val pos2 = pos1.translate(19.0, 45.0) + + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, pos1, mapOf( + "amenity" to "recycling", + "recycling_type" to "container" + )), + OsmNode(2L, 1, pos2, mapOf( + "amenity" to "recycling", + "recycling_type" to "container" + )) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } - private lateinit var questType: AddRecyclingContainerMaterials + @Test fun `applicable to container without recycling materials not too close to another container`() { + val pos1 = OsmLatLon(0.0,0.0) + val pos2 = pos1.translate(21.0, 45.0) + + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, pos1, mapOf( + "amenity" to "recycling", + "recycling_type" to "container" + )), + OsmNode(2L, 1, pos2, mapOf( + "amenity" to "recycling", + "recycling_type" to "container", + "recycling:paper" to "yes" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } @Test fun `apply normal answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingTypeTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingTypeTest.kt index 13ac124ff7..d9eb644e80 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingTypeTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddRecyclingTypeTest.kt @@ -2,14 +2,13 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.recycling.AddRecyclingType import de.westnordost.streetcomplete.quests.recycling.RecyclingType import org.junit.Test class AddRecyclingTypeTest { - private val questType = AddRecyclingType(mock()) + private val questType = AddRecyclingType() @Test fun `apply recycling centre answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddSidewalkTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddSidewalkTest.kt index c2ec31dd6b..cc987dcb68 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddSidewalkTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddSidewalkTest.kt @@ -1,16 +1,71 @@ package de.westnordost.streetcomplete.quests +import de.westnordost.osmapi.map.data.OsmLatLon +import de.westnordost.osmapi.map.data.OsmWay import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolylinesGeometry import de.westnordost.streetcomplete.quests.sidewalk.AddSidewalk import de.westnordost.streetcomplete.quests.sidewalk.SeparatelyMapped -import de.westnordost.streetcomplete.quests.sidewalk.SidewalkAnswer import de.westnordost.streetcomplete.quests.sidewalk.SidewalkSides +import de.westnordost.streetcomplete.util.translate +import org.junit.Assert.assertEquals import org.junit.Test class AddSidewalkTest { - private val questType = AddSidewalk(mock()) + private val questType = AddSidewalk() + + @Test fun `applicable to road with missing sidewalk`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "primary", + "lit" to "yes" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to road with nearby footway`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2), mapOf( + "highway" to "primary", + "lit" to "yes" + )), + OsmWay(2L, 1, listOf(3,4), mapOf( + "highway" to "footway" + )) + )) + val p1 = OsmLatLon(0.0,0.0) + val p2 = p1.translate(50.0, 45.0) + val p3 = p1.translate(14.0, 135.0) + val p4 = p3.translate(50.0, 45.0) + + mapData.wayGeometriesById[1L] = ElementPolylinesGeometry(listOf(listOf(p1, p2)), p1) + mapData.wayGeometriesById[2L] = ElementPolylinesGeometry(listOf(listOf(p3, p4)), p3) + + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `applicable to road with footway that is far away enough`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmWay(1L, 1, listOf(1,2), mapOf( + "highway" to "primary", + "lit" to "yes" + )), + OsmWay(2L, 1, listOf(3,4), mapOf( + "highway" to "footway" + )) + )) + val p1 = OsmLatLon(0.0,0.0) + val p2 = p1.translate(50.0, 45.0) + val p3 = p1.translate(16.0, 135.0) + val p4 = p3.translate(50.0, 45.0) + + mapData.wayGeometriesById[1L] = ElementPolylinesGeometry(listOf(listOf(p1, p2)), p1) + mapData.wayGeometriesById[2L] = ElementPolylinesGeometry(listOf(listOf(p3, p4)), p3) + + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } @Test fun `apply no sidewalk answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/AddSportTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/AddSportTest.kt index abf6cca3ba..a8b40d3198 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/AddSportTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/AddSportTest.kt @@ -8,7 +8,7 @@ import org.junit.Test class AddSportTest { - private val questType = AddSport(mock()) + private val questType = AddSport() @Test fun `replace hockey when applying answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/OsmElementQuestType.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/OsmElementQuestType.kt index 58332e1b4d..9899cb3213 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/OsmElementQuestType.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/OsmElementQuestType.kt @@ -1,28 +1,11 @@ package de.westnordost.streetcomplete.quests import de.westnordost.streetcomplete.data.osm.osmquest.OsmElementQuestType - -import de.westnordost.osmapi.map.data.BoundingBox import de.westnordost.streetcomplete.data.osm.changes.StringMapChangesBuilder import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryChange -import org.junit.Assert.fail import org.assertj.core.api.Assertions.* -fun OsmElementQuestType<*>.verifyDownloadYieldsNoQuest(bbox: BoundingBox) { - download(bbox) { element, _ -> - fail("Expected zero elements. Element returned: ${element.type.name}#${element.id}") - } -} - -fun OsmElementQuestType<*>.verifyDownloadYieldsQuest(bbox: BoundingBox) { - var hasQuest = false - download(bbox) { _, _ -> hasQuest = true } - if (!hasQuest) { - fail("Expected nonzero elements. Elements not returned") - } -} - fun OsmElementQuestType.verifyAnswer(tags:Map, answer:T, vararg expectedChanges: StringMapEntryChange) { val cb = StringMapChangesBuilder(tags) this.applyAnswerTo(answer, cb) diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/TestMapDataWithGeometry.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/TestMapDataWithGeometry.kt new file mode 100644 index 0000000000..0aa04f1be8 --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/TestMapDataWithGeometry.kt @@ -0,0 +1,24 @@ +package de.westnordost.streetcomplete.quests + +import de.westnordost.osmapi.map.MapDataWithGeometry +import de.westnordost.osmapi.map.MutableMapData +import de.westnordost.osmapi.map.data.BoundingBox +import de.westnordost.osmapi.map.data.Element +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry + +class TestMapDataWithGeometry(elements: Iterable) : MutableMapData(), MapDataWithGeometry { + + init { + addAll(elements) + handle(BoundingBox(0.0,0.0,1.0,1.0)) + } + + val nodeGeometriesById: MutableMap = mutableMapOf() + val wayGeometriesById: MutableMap = mutableMapOf() + val relationGeometriesById: MutableMap = mutableMapOf() + + override fun getNodeGeometry(id: Long): ElementPointGeometry? = nodeGeometriesById[id] + override fun getWayGeometry(id: Long): ElementGeometry? = wayGeometriesById[id] + override fun getRelationGeometry(id: Long): ElementGeometry? = relationGeometriesById[id] +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/address/AddAddressStreetTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/address/AddAddressStreetTest.kt new file mode 100644 index 0000000000..09427a314a --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/address/AddAddressStreetTest.kt @@ -0,0 +1,48 @@ +package de.westnordost.streetcomplete.quests.address + +import de.westnordost.osmapi.map.data.Element +import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.osmapi.map.data.OsmRelation +import de.westnordost.osmapi.map.data.OsmRelationMember +import de.westnordost.streetcomplete.mock +import de.westnordost.streetcomplete.quests.TestMapDataWithGeometry +import org.junit.Assert.* +import org.junit.Test + +class AddAddressStreetTest { + + private val questType = AddAddressStreet(mock()) + + @Test fun `applicable to place without street name`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "addr:housenumber" to "123" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to place with street name`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "addr:housenumber" to "123", + "addr:street" to "onetwothree", + )) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to place without street name but in a associatedStreet relation`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "addr:housenumber" to "123" + )), + OsmRelation(1L, 1, listOf( + OsmRelationMember(1L, "doesntmatter", Element.Type.NODE) + ), mapOf( + "type" to "associatedStreet" + )), + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopNameTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopNameTest.kt index 904cf98cc6..7cc0ce7874 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopNameTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/bus_stop_name/AddBusStopNameTest.kt @@ -1,20 +1,19 @@ package de.westnordost.streetcomplete.quests.bus_stop_name import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.LocalizedName import de.westnordost.streetcomplete.quests.verifyAnswer import org.junit.Test class AddBusStopNameTest { - private val questType = AddBusStopName(mock()) + private val questType = AddBusStopName() @Test fun `apply no name answer`() { questType.verifyAnswer( NoBusStopName, - StringMapEntryAdd("noname", "yes") + StringMapEntryAdd("name:signed", "no") ) } diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashTypeTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashTypeTest.kt index ac01641583..662a59c8eb 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashTypeTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/car_wash_type/AddCarWashTypeTest.kt @@ -1,14 +1,13 @@ package de.westnordost.streetcomplete.quests.car_wash_type import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.verifyAnswer import org.junit.Test import de.westnordost.streetcomplete.quests.car_wash_type.CarWashType.* class AddCarWashTypeTest { - private val questType = AddCarWashType(mock()) + private val questType = AddCarWashType() @Test fun `only self service`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperatorTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperatorTest.kt index 6e4fd82cc2..0450322873 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperatorTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/clothing_bin_operator/AddClothingBinOperatorTest.kt @@ -1,13 +1,12 @@ package de.westnordost.streetcomplete.quests.clothing_bin_operator import de.westnordost.osmapi.map.data.OsmNode -import de.westnordost.streetcomplete.mock import org.junit.Assert.* import org.junit.Test class AddClothingBinOperatorTest { - private val questType = AddClothingBinOperator(mock()) + private val questType = AddClothingBinOperator() @Test fun `is not applicable to null tags`() { assertFalse(questType.isApplicableTo(create(null))) diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIslandTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIslandTest.kt new file mode 100644 index 0000000000..65efb1781a --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/crossing_island/AddCrossingIslandTest.kt @@ -0,0 +1,35 @@ +package de.westnordost.streetcomplete.quests.crossing_island + +import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.osmapi.map.data.OsmWay +import de.westnordost.streetcomplete.quests.TestMapDataWithGeometry +import org.junit.Assert.* +import org.junit.Test + +class AddCrossingIslandTest { + private val questType = AddCrossingIsland() + + @Test fun `applicable to crossing`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "highway" to "crossing", + "crossing" to "something" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to crossing with private road`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "highway" to "crossing", + "crossing" to "something" + )), + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "residential", + "access" to "private" + )) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumberTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumberTest.kt index 40e47cb85e..16e8a4f7d4 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumberTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/housenumber/AddHousenumberTest.kt @@ -1,15 +1,104 @@ package de.westnordost.streetcomplete.quests.housenumber +import de.westnordost.osmapi.map.data.* import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd -import de.westnordost.streetcomplete.mock +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPointGeometry +import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementPolygonsGeometry +import de.westnordost.streetcomplete.quests.TestMapDataWithGeometry import de.westnordost.streetcomplete.quests.verifyAnswer -import org.junit.Assert.assertFalse -import org.junit.Assert.assertTrue +import org.junit.Assert.* import org.junit.Test class AddHousenumberTest { - private val questType = AddHousenumber(mock()) + private val questType = AddHousenumber() + + @Test fun `does not create quest for generic building`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf("building" to "yes")) to POSITIONS1 + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does not create quest for building with address`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached", + "addr:housenumber" to "123" + )) to POSITIONS1 + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does create quest for building without address`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached" + )) to POSITIONS1 + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does not create quest for building with address node on outline`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached" + )) to POSITIONS1, + OsmNode(2L, 1, P2, mapOf( + "addr:housenumber" to "123" + )) to ElementPointGeometry(P2) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does not create quest for building that is part of a relation with an address`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached" + )) to POSITIONS1, + OsmRelation(2L, 1, listOf( + OsmRelationMember(1L, "something", Element.Type.WAY) + ), mapOf( + "addr:housenumber" to "123" + )) to ElementPointGeometry(P2) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does not create quest for building that is inside an area with an address`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached" + )) to POSITIONS1, + OsmWay(1L, 1, NODES2, mapOf( + "addr:housenumber" to "123", + "amenity" to "school", + )) to POSITIONS2, + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does not create quest for building that contains an address node`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached" + )) to POSITIONS1, + OsmNode(1L, 1, PC, mapOf( + "addr:housenumber" to "123" + )) to ElementPointGeometry(PC), + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `does not create quest for building that intersects bounding box`() { + val mapData = createMapData(mapOf( + OsmWay(1L, 1, NODES1, mapOf( + "building" to "detached" + )) to ElementPolygonsGeometry(listOf(listOf(P1, P2, PO, P4, P1)), PC) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } @Test fun `housenumber regex`() { val r = VALID_HOUSENUMBER_REGEX @@ -88,3 +177,38 @@ class AddHousenumberTest { ) } } + +private val P1 = OsmLatLon(0.25,0.25) +private val P2 = OsmLatLon(0.25,0.75) +private val P3 = OsmLatLon(0.75,0.75) +private val P4 = OsmLatLon(0.75,0.25) + +private val P5 = OsmLatLon(0.1,0.1) +private val P6 = OsmLatLon(0.1,0.9) +private val P7 = OsmLatLon(0.9,0.9) +private val P8 = OsmLatLon(0.9,0.1) + +private val PO = OsmLatLon(1.5, 1.5) +private val PC = OsmLatLon(0.5,0.5) + +private val NODES1 = listOf(1,2,3,4,1) +private val NODES2 = listOf(5,6,7,8,5) + +private val POSITIONS1 = ElementPolygonsGeometry(listOf(listOf(P1, P2, P3, P4, P1)), PC) +private val POSITIONS2 = ElementPolygonsGeometry(listOf(listOf(P5, P6, P7, P8, P5)), PC) + +private fun createMapData(elements: Map): TestMapDataWithGeometry { + val result = TestMapDataWithGeometry(elements.keys) + for((element, geometry) in elements) { + when(element) { + is Node -> + result.nodeGeometriesById[element.id] = geometry as ElementPointGeometry + is Way -> + result.wayGeometriesById[element.id] = geometry + is Relation -> + result.relationGeometriesById[element.id] = geometry + } + } + result.handle(BoundingBox(0.0, 0.0, 1.0, 1.0)) + return result +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/oneway/AddOnewayTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/oneway/AddOnewayTest.kt index cd42f1c07f..a1f6b60d87 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/oneway/AddOnewayTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/oneway/AddOnewayTest.kt @@ -1,143 +1,120 @@ package de.westnordost.streetcomplete.quests.oneway -import de.westnordost.osmapi.map.data.Element import de.westnordost.osmapi.map.data.OsmWay import de.westnordost.osmapi.map.data.Way -import de.westnordost.streetcomplete.any -import de.westnordost.streetcomplete.data.osm.elementgeometry.ElementGeometry -import de.westnordost.streetcomplete.data.osm.mapdata.OverpassMapDataAndGeometryApi -import de.westnordost.streetcomplete.mock -import de.westnordost.streetcomplete.on -import de.westnordost.streetcomplete.quests.verifyDownloadYieldsNoQuest -import de.westnordost.streetcomplete.quests.verifyDownloadYieldsQuest -import org.junit.Before +import de.westnordost.streetcomplete.quests.TestMapDataWithGeometry +import org.junit.Assert.assertEquals import org.junit.Test class AddOnewayTest { - private lateinit var overpassMock: OverpassMapDataAndGeometryApi - private lateinit var questType: AddOneway - - @Before fun setUp() { - overpassMock = mock() - questType = AddOneway(overpassMock) - } + private val questType = AddOneway() @Test fun `does not apply to element without tags`() { - setUpElements(noDeadEndWays(null)) - questType.verifyDownloadYieldsNoQuest(mock()) + val mapData = TestMapDataWithGeometry(noDeadEndWays(null)) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } @Test fun `applies to slim road`() { - setUpElements(noDeadEndWays(mapOf( + val mapData = TestMapDataWithGeometry(noDeadEndWays(mapOf( "highway" to "residential", "width" to "4", "lanes" to "1" ))) - questType.verifyDownloadYieldsQuest(mock()) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) } @Test fun `does not apply to wide road`() { - setUpElements(noDeadEndWays(mapOf( + val mapData = TestMapDataWithGeometry(noDeadEndWays(mapOf( "highway" to "residential", "width" to "5", "lanes" to "1" ))) - questType.verifyDownloadYieldsNoQuest(mock()) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } @Test fun `applies to wider road that has parking lanes`() { - setUpElements(noDeadEndWays(mapOf( + val mapData = TestMapDataWithGeometry(noDeadEndWays(mapOf( "highway" to "residential", "width" to "12", "lanes" to "1", "parking:lane:both" to "perpendicular", "parking:lane:both:perpendicular" to "on_street" ))) - questType.verifyDownloadYieldsQuest(mock()) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) } @Test fun `does not apply to wider road that has parking lanes but not enough`() { - setUpElements(noDeadEndWays(mapOf( + val mapData = TestMapDataWithGeometry(noDeadEndWays(mapOf( "highway" to "residential", "width" to "13", "lanes" to "1", "parking:lane:both" to "perpendicular", "parking:lane:both:perpendicular" to "on_street" ))) - questType.verifyDownloadYieldsNoQuest(mock()) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } @Test fun `applies to wider road that has cycle lanes`() { - setUpElements(noDeadEndWays(mapOf( + val mapData = TestMapDataWithGeometry(noDeadEndWays(mapOf( "highway" to "residential", "width" to "6", "lanes" to "1", "cycleway" to "lane" ))) - questType.verifyDownloadYieldsQuest(mock()) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) } @Test fun `does not apply to slim road with more than one lane`() { - setUpElements(noDeadEndWays(mapOf( + val mapData = TestMapDataWithGeometry(noDeadEndWays(mapOf( "highway" to "residential", "width" to "4", "lanes" to "2" ))) - questType.verifyDownloadYieldsNoQuest(mock()) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } @Test fun `does not apply to dead end road #1`() { - setUpElements(listOf( - way(listOf(1,2), mapOf("highway" to "residential")), - way(listOf(2,3), mapOf( + val mapData = TestMapDataWithGeometry(listOf( + way(1,listOf(1,2), mapOf("highway" to "residential")), + way(2,listOf(2,3), mapOf( "highway" to "residential", "width" to "4", "lanes" to "1" )) )) - questType.verifyDownloadYieldsNoQuest(mock()) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } @Test fun `does not apply to dead end road #2`() { - setUpElements(listOf( - way(listOf(2,3), mapOf( + val mapData = TestMapDataWithGeometry(listOf( + way(1,listOf(2,3), mapOf( "highway" to "residential", "width" to "4", "lanes" to "1" )), - way(listOf(3,4), mapOf("highway" to "residential")) + way(2,listOf(3,4), mapOf("highway" to "residential")) )) - questType.verifyDownloadYieldsNoQuest(mock()) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) } - @Test fun `does not apply to road that ends as an intersection in another`() { - setUpElements(listOf( - way(listOf(1,2), mapOf("highway" to "residential")), - way(listOf(2,3), mapOf( + @Test fun `applies to road that ends as an intersection in another`() { + val mapData = TestMapDataWithGeometry(listOf( + way(1,listOf(1,2), mapOf("highway" to "residential")), + way(2,listOf(2,3), mapOf( "highway" to "residential", "width" to "4", "lanes" to "1" )), - way(listOf(5,3,4), mapOf("highway" to "residential")) + way(3,listOf(5,3,4), mapOf("highway" to "residential")) )) - questType.verifyDownloadYieldsQuest(mock()) - } - - private fun setUpElements(ways: List) { - on(overpassMock.query(any(), any())).then { invocation -> - val callback = invocation.getArgument(1) as (element: Element, geometry: ElementGeometry?) -> Unit - for (way in ways) { - callback(way, null) - } - true - } + assertEquals(1, questType.getApplicableElements(mapData).toList().size) } - private fun way(nodeIds: List, tags: Map?) = OsmWay(1,1, nodeIds, tags) + private fun way(id: Long, nodeIds: List, tags: Map?) = OsmWay(id,1, nodeIds, tags) private fun noDeadEndWays(tags: Map?): List = listOf( - way(listOf(1,2), mapOf("highway" to "residential")), - way(listOf(2,3), tags), - way(listOf(3,4), mapOf("highway" to "residential")) + way(1,listOf(1,2), mapOf("highway" to "residential")), + way(2,listOf(2,3), tags), + way(3,listOf(3,4), mapOf("highway" to "residential")) ) } diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHoursTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHoursTest.kt index 08b19e27a1..aa496bebab 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHoursTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/opening_hours/AddOpeningHoursTest.kt @@ -22,7 +22,7 @@ import java.util.* class AddOpeningHoursTest { - private val questType = AddOpeningHours(mock(), mock(), mock()) + private val questType = AddOpeningHours(mock()) @Test fun `apply description answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrierTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrierTest.kt new file mode 100644 index 0000000000..85a44a7841 --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/railway_crossing/AddRailwayCrossingBarrierTest.kt @@ -0,0 +1,33 @@ +package de.westnordost.streetcomplete.quests.railway_crossing + +import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.osmapi.map.data.OsmWay +import de.westnordost.streetcomplete.quests.TestMapDataWithGeometry +import org.junit.Assert.* +import org.junit.Test + +class AddRailwayCrossingBarrierTest { + private val questType = AddRailwayCrossingBarrier() + + @Test fun `applicable to crossing`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "railway" to "level_crossing" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to crossing with private road`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "railway" to "level_crossing" + )), + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "residential", + "access" to "private" + )) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameTest.kt index 0862566afb..1663c4cd7b 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/road_name/AddRoadNameTest.kt @@ -10,7 +10,7 @@ import org.junit.Test class AddRoadNameTest { - private val questType = AddRoadName(mock(), mock()) + private val questType = AddRoadName(mock()) private val tags = mapOf("highway" to "residential") diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRampTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRampTest.kt index aa44e48599..76cced39e4 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRampTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/steps_ramp/AddStepsRampTest.kt @@ -4,14 +4,13 @@ import de.westnordost.streetcomplete.data.meta.toCheckDateString import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryAdd import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryDelete import de.westnordost.streetcomplete.data.osm.changes.StringMapEntryModify -import de.westnordost.streetcomplete.mock import de.westnordost.streetcomplete.quests.verifyAnswer import org.junit.Test import java.util.* class AddStepsRampTest { - private val questType = AddStepsRamp(mock(), mock()) + private val questType = AddStepsRamp() @Test fun `apply bicycle ramp answer`() { questType.verifyAnswer( diff --git a/app/src/test/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalkTest.kt b/app/src/test/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalkTest.kt new file mode 100644 index 0000000000..4a2a878086 --- /dev/null +++ b/app/src/test/java/de/westnordost/streetcomplete/quests/tactile_paving/AddTactilePavingCrosswalkTest.kt @@ -0,0 +1,33 @@ +package de.westnordost.streetcomplete.quests.tactile_paving + +import de.westnordost.osmapi.map.data.OsmNode +import de.westnordost.osmapi.map.data.OsmWay +import de.westnordost.streetcomplete.quests.TestMapDataWithGeometry +import org.junit.Assert.* +import org.junit.Test + +class AddTactilePavingCrosswalkTest { + private val questType = AddTactilePavingCrosswalk() + + @Test fun `applicable to crossing`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "highway" to "crossing" + )) + )) + assertEquals(1, questType.getApplicableElements(mapData).toList().size) + } + + @Test fun `not applicable to crossing with private road`() { + val mapData = TestMapDataWithGeometry(listOf( + OsmNode(1L, 1, 0.0,0.0, mapOf( + "highway" to "crossing" + )), + OsmWay(1L, 1, listOf(1,2,3), mapOf( + "highway" to "residential", + "access" to "private" + )) + )) + assertEquals(0, questType.getApplicableElements(mapData).toList().size) + } +} \ No newline at end of file diff --git a/app/src/test/java/de/westnordost/streetcomplete/util/SphericalEarthMathTest.kt b/app/src/test/java/de/westnordost/streetcomplete/util/SphericalEarthMathTest.kt index cce7c2b861..72dee43b53 100644 --- a/app/src/test/java/de/westnordost/streetcomplete/util/SphericalEarthMathTest.kt +++ b/app/src/test/java/de/westnordost/streetcomplete/util/SphericalEarthMathTest.kt @@ -1,5 +1,6 @@ package de.westnordost.streetcomplete.util +import de.westnordost.osmapi.map.data.BoundingBox import org.junit.Test import de.westnordost.osmapi.map.data.LatLon @@ -243,6 +244,113 @@ class SphericalEarthMathTest { assertEquals(160.0, bbox.minLongitude, 0.0) } + @Test fun `isCompletelyInside works`() { + val bbox1 = BoundingBox(-1.0, -1.0, 1.0, 1.0) + val bbox2 = BoundingBox(-0.5, -0.5, 0.5, 1.0) + assertTrue(bbox2.isCompletelyInside(bbox1)) + assertFalse(bbox1.isCompletelyInside(bbox2)) + } + + @Test fun `isCompletelyInside works at 180th meridian`() { + val bbox1 = BoundingBox(0.0, 179.0, 1.0, -179.0) + val bbox2 = BoundingBox(0.0, 179.5, 0.5, -179.5) + assertTrue(bbox2.isCompletelyInside(bbox1)) + assertFalse(bbox1.isCompletelyInside(bbox2)) + } + + @Test fun `enlargedBy really enlarges bounding box`() { + val bbox = BoundingBox(0.0, 0.0, 1.0, 1.0) + assertTrue(bbox.isCompletelyInside(bbox.enlargedBy(1.0))) + } + + @Test fun `enlargedBy really enlarges bounding box, even at 180th meridian`() { + val bbox1 = BoundingBox(0.0, 179.0, 1.0, 180.0) + // enlarged bounding box should go over the 180th meridian + assertTrue(bbox1.isCompletelyInside(bbox1.enlargedBy(100.0))) + // here already bbox2 crosses the 180th meridian, maybe this makes a difference + val bbox2 = BoundingBox(0.0, 179.0, 1.0, -179.0) + assertTrue(bbox2.isCompletelyInside(bbox2.enlargedBy(100.0))) + } + + @Test fun `bounding box not on same latitude do not intersect`() { + val bbox = BoundingBox(0.0, 0.0, 1.0, 1.0) + val above = BoundingBox(1.1, 0.0, 1.2, 1.0) + val below = BoundingBox(-0.2, 0.0, -0.1, 1.0) + assertFalse(bbox.intersect(above)) + assertFalse(bbox.intersect(below)) + assertFalse(above.intersect(bbox)) + assertFalse(below.intersect(bbox)) + } + + @Test fun `bounding box not on same longitude do not intersect`() { + val bbox = BoundingBox(0.0, 0.0, 1.0, 1.0) + val left = BoundingBox(0.0, -0.2, 1.0, -0.1) + val right = BoundingBox(0.0, 1.1, 1.0, 1.2) + assertFalse(bbox.intersect(left)) + assertFalse(bbox.intersect(right)) + assertFalse(left.intersect(bbox)) + assertFalse(right.intersect(bbox)) + } + + @Test fun `intersecting bounding boxes`() { + val bbox = BoundingBox(0.0, 0.0, 1.0, 1.0) + val touchLeft = BoundingBox(0.0, -0.1, 1.0, 0.0) + val touchUpperRightCorner = BoundingBox(1.0, 1.0, 1.1, 1.1) + val completelyInside = BoundingBox(0.4, 0.4, 0.8, 0.8) + val intersectLeft = BoundingBox(0.4, -0.5, 0.5, 0.5) + val intersectRight = BoundingBox(0.4, 0.8, 0.5, 1.2) + val intersectTop = BoundingBox(0.9, 0.4, 1.1, 0.8) + val intersectBottom = BoundingBox(-0.2, 0.4, 0.2, 0.6) + assertTrue(bbox.intersect(touchLeft)) + assertTrue(bbox.intersect(touchUpperRightCorner)) + assertTrue(bbox.intersect(completelyInside)) + assertTrue(bbox.intersect(intersectLeft)) + assertTrue(bbox.intersect(intersectRight)) + assertTrue(bbox.intersect(intersectTop)) + assertTrue(bbox.intersect(intersectBottom)) + // and the other way around + assertTrue(touchLeft.intersect(bbox)) + assertTrue(touchUpperRightCorner.intersect(bbox)) + assertTrue(completelyInside.intersect(bbox)) + assertTrue(intersectLeft.intersect(bbox)) + assertTrue(intersectRight.intersect(bbox)) + assertTrue(intersectTop.intersect(bbox)) + assertTrue(intersectBottom.intersect(bbox)) + } + + @Test fun `bounding box not on same longitude do not intersect, even on 180th meridian`() { + val bbox = BoundingBox(0.0, 179.0, 1.0, -179.0) + val other = BoundingBox(0.0, -178.0, 1.0, 178.0) + assertFalse(bbox.intersect(other)) + assertFalse(other.intersect(bbox)) + } + + @Test fun `intersecting bounding boxes at 180th meridian`() { + val bbox = BoundingBox(0.0, 179.5, 1.0, -170.0) + val touchLeft = BoundingBox(0.0, 179.0, 1.0, 180.0) + val touchUpperRightCorner = BoundingBox(1.0, -170.0, 1.1, -169.0) + val completelyInside = BoundingBox(0.4, 179.9, 0.8, -179.9) + val intersectLeft = BoundingBox(0.4, 179.0, 0.5, 179.9) + val intersectRight = BoundingBox(0.4, -179.0, 0.5, -150.0) + val intersectTop = BoundingBox(0.9, 179.9, 1.1, -179.8) + val intersectBottom = BoundingBox(-0.2, 179.9, 0.2, -179.8) + assertTrue(bbox.intersect(touchLeft)) + assertTrue(bbox.intersect(touchUpperRightCorner)) + assertTrue(bbox.intersect(completelyInside)) + assertTrue(bbox.intersect(intersectLeft)) + assertTrue(bbox.intersect(intersectRight)) + assertTrue(bbox.intersect(intersectTop)) + assertTrue(bbox.intersect(intersectBottom)) + // and the other way around + assertTrue(touchLeft.intersect(bbox)) + assertTrue(touchUpperRightCorner.intersect(bbox)) + assertTrue(completelyInside.intersect(bbox)) + assertTrue(intersectLeft.intersect(bbox)) + assertTrue(intersectRight.intersect(bbox)) + assertTrue(intersectTop.intersect(bbox)) + assertTrue(intersectBottom.intersect(bbox)) + } + /* ++++++++++++++++++++++++++++++ test translating of positions +++++++++++++++++++++++++++++ */ @Test fun `translate latitude north`() { checkTranslate(1000, 0) } @@ -392,7 +500,7 @@ class SphericalEarthMathTest { } @Test fun `point at polygon edge is in polygon`() { - val square = p(0.0, 0.0).createSquare(10.0) + val square = p(0.0, 0.0).createCounterClockwiseSquare(10.0) assertTrue(p(0.0, 10.0).isInPolygon(square)) assertTrue(p(10.0, 0.0).isInPolygon(square)) assertTrue(p(-10.0, 0.0).isInPolygon(square)) @@ -400,7 +508,7 @@ class SphericalEarthMathTest { } @Test fun `point at polygon edge at 180th meridian is in polygon`() { - val square = p(180.0, 0.0).createSquare(10.0) + val square = p(180.0, 0.0).createCounterClockwiseSquare(10.0) assertTrue(p(180.0, 10.0).isInPolygon(square)) assertTrue(p(-170.0, 0.0).isInPolygon(square)) assertTrue(p(170.0, 0.0).isInPolygon(square)) @@ -477,20 +585,20 @@ class SphericalEarthMathTest { } @Test fun `point outside polygon is outside polygon`() { - assertFalse(p(0.0, 11.0).isInPolygon(p(0.0, 0.0).createSquare(10.0))) + assertFalse(p(0.0, 11.0).isInPolygon(p(0.0, 0.0).createCounterClockwiseSquare(10.0))) } @Test fun `point outside polygon is outside polygon at 180th meridian`() { - assertFalse(p(-169.0, 0.0).isInPolygon(p(180.0, 0.0).createSquare(10.0))) + assertFalse(p(-169.0, 0.0).isInPolygon(p(180.0, 0.0).createCounterClockwiseSquare(10.0))) } @Test fun `polygon direction does not matter for point-in-polygon check`() { - val square = p(0.0, 0.0).createSquare(10.0).reversed() + val square = p(0.0, 0.0).createCounterClockwiseSquare(10.0).reversed() assertTrue(p(5.0, 5.0).isInPolygon(square)) } @Test fun `polygon direction does not matter for point-in-polygon check at 180th meridian`() { - val square = p(180.0, 0.0).createSquare(10.0).reversed() + val square = p(180.0, 0.0).createCounterClockwiseSquare(10.0).reversed() assertTrue(p(-175.0, 5.0).isInPolygon(square)) } @@ -558,6 +666,30 @@ class SphericalEarthMathTest { assertFalse(p.isInMultipolygon(listOf(way))) } + @Test fun `polygon area is 0 for a polygon with less than 3 edges`() { + val twoEdges = listOf(p(0.0,0.0), p(1.0,0.0), p(1.0,1.0)) + assertEquals(0.0, twoEdges.measuredArea(), 0.0) + } + + @Test fun `polygon area is 0 for a polygon that is not closed`() { + val notClosed = listOf(p(0.0,0.0), p(1.0,0.0), p(1.0,1.0), p(0.0,1.0)) + assertEquals(0.0, notClosed.measuredArea(), 0.0) + } + + @Test fun `polygon area is positive for a counterclockwise polygon`() { + val square = p(0.0,0.0).createCounterClockwiseSquare(1.0) + assertFalse(square.isRingDefinedClockwise()) + assertTrue(square.measuredAreaSigned() > 0) + assertTrue(square.measuredArea() > 0) + } + + @Test fun `polygon area is negative for a clockwise polygon`() { + val square = p(0.0,0.0).createCounterClockwiseSquare(1.0).reversed() + assertTrue(square.isRingDefinedClockwise()) + assertTrue(square.measuredAreaSigned() < 0) + assertTrue(square.measuredArea() > 0) + } + companion object { private val HH = p(10.0, 53.5) } @@ -572,7 +704,7 @@ private val LatLon.y get() = latitude | + | o---o */ -private fun LatLon.createSquare(l: Double) = listOf( +private fun LatLon.createCounterClockwiseSquare(l: Double) = listOf( p(x + l, y + l), p(x + l, y - l), p(x - l, y - l),