MapComplete/src/UI/Popup/AddNewPoint/AddNewPoint.svelte

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

368 lines
13 KiB
Svelte
Raw Normal View History

<script lang="ts">
2023-06-15 16:12:46 +02:00
/**
* This component ties together all the steps that are needed to create a new point.
* There are many subcomponents which help with that
*/
2023-09-28 23:50:27 +02:00
import type { SpecialVisualizationState } from "../../SpecialVisualization"
import PresetList from "./PresetList.svelte"
import type PresetConfig from "../../../Models/ThemeConfig/PresetConfig"
import LayerConfig from "../../../Models/ThemeConfig/LayerConfig"
import Tr from "../../Base/Tr.svelte"
import SubtleButton from "../../Base/SubtleButton.svelte"
import FromHtml from "../../Base/FromHtml.svelte"
import Translations from "../../i18n/Translations.js"
import TagHint from "../TagHint.svelte"
import { And } from "../../../Logic/Tags/And.js"
import LoginToggle from "../../Base/LoginToggle.svelte"
import Constants from "../../../Models/Constants.js"
import FilteredLayer from "../../../Models/FilteredLayer"
import { Store, UIEventSource } from "../../../Logic/UIEventSource"
import { EyeIcon, EyeOffIcon } from "@rgossiaux/svelte-heroicons/solid"
import LoginButton from "../../Base/LoginButton.svelte"
import NewPointLocationInput from "../../BigComponents/NewPointLocationInput.svelte"
import CreateNewNodeAction from "../../../Logic/Osm/Actions/CreateNewNodeAction"
import { OsmWay } from "../../../Logic/Osm/OsmObject"
import { Tag } from "../../../Logic/Tags/Tag"
import type { WayId } from "../../../Models/OsmFeature"
import Loading from "../../Base/Loading.svelte"
import type { GlobalFilter } from "../../../Models/GlobalFilter"
import { onDestroy } from "svelte"
import NextButton from "../../Base/NextButton.svelte"
import BackButton from "../../Base/BackButton.svelte"
import ToSvelte from "../../Base/ToSvelte.svelte"
import Svg from "../../../Svg"
import OpenBackgroundSelectorButton from "../../BigComponents/OpenBackgroundSelectorButton.svelte"
import { twJoin } from "tailwind-merge"
2023-09-28 23:50:27 +02:00
export let coordinate: { lon: number; lat: number }
export let state: SpecialVisualizationState
2023-06-15 16:12:46 +02:00
let selectedPreset: {
preset: PresetConfig
layer: LayerConfig
icon: string
tags: Record<string, string>
2023-09-28 23:50:27 +02:00
} = undefined
let checkedOfGlobalFilters: number = 0
let confirmedCategory = false
2023-06-15 16:12:46 +02:00
$: if (selectedPreset === undefined) {
2023-09-28 23:50:27 +02:00
confirmedCategory = false
creating = false
checkedOfGlobalFilters = 0
2023-06-15 16:12:46 +02:00
}
2023-09-28 23:50:27 +02:00
let flayer: FilteredLayer = undefined
let layerIsDisplayed: UIEventSource<boolean> | undefined = undefined
let layerHasFilters: Store<boolean> | undefined = undefined
let globalFilter: UIEventSource<GlobalFilter[]> = state.layerState.globalFilters
let _globalFilter: GlobalFilter[] = []
2023-06-15 16:12:46 +02:00
onDestroy(
globalFilter.addCallbackAndRun((globalFilter) => {
2023-09-28 23:50:27 +02:00
console.log("Global filters are", globalFilter)
_globalFilter = globalFilter ?? []
2023-06-15 16:12:46 +02:00
})
2023-09-28 23:50:27 +02:00
)
2023-06-15 16:12:46 +02:00
$: {
2023-09-28 23:50:27 +02:00
flayer = state.layerState.filteredLayers.get(selectedPreset?.layer?.id)
layerIsDisplayed = flayer?.isDisplayed
layerHasFilters = flayer?.hasFilter
2023-06-15 16:12:46 +02:00
}
2023-09-28 23:50:27 +02:00
const t = Translations.t.general.add
2023-09-28 23:50:27 +02:00
const zoom = state.mapProperties.zoom
2023-09-28 23:50:27 +02:00
const isLoading = state.dataIsLoading
let preciseCoordinate: UIEventSource<{ lon: number; lat: number }> = new UIEventSource(undefined)
let snappedToObject: UIEventSource<string> = new UIEventSource<string>(undefined)
2023-06-15 16:12:46 +02:00
// Small helper variable: if the map is tapped, we should let the 'Next'-button grab some attention as users have to click _that_ to continue, not the map
2023-09-28 23:50:27 +02:00
let preciseInputIsTapped = false
2023-09-28 23:50:27 +02:00
let creating = false
2023-06-15 16:12:46 +02:00
/**
* Call when the user should restart the flow by clicking on the map, e.g. because they disabled filters.
* Will delete the lastclick-location
*/
function abort() {
2023-09-28 23:50:27 +02:00
state.selectedElement.setData(undefined)
2023-06-15 16:12:46 +02:00
// When aborted, we force the contributors to place the pin _again_
// This is because there might be a nearby object that was disabled; this forces them to re-evaluate the map
2023-09-28 23:50:27 +02:00
state.lastClickObject.features.setData([])
preciseInputIsTapped = false
2023-06-15 16:12:46 +02:00
}
2023-06-15 16:12:46 +02:00
async function confirm() {
2023-09-28 23:50:27 +02:00
creating = true
const location: { lon: number; lat: number } = preciseCoordinate.data
const snapTo: WayId | undefined = <WayId>snappedToObject.data
2023-06-15 16:12:46 +02:00
const tags: Tag[] = selectedPreset.preset.tags.concat(
..._globalFilter.map((f) => f?.onNewPoint?.tags ?? [])
2023-09-28 23:50:27 +02:00
)
console.log("Creating new point at", location, "snapped to", snapTo, "with tags", tags)
2023-09-28 23:50:27 +02:00
let snapToWay: undefined | OsmWay = undefined
if (snapTo !== undefined && snapTo !== null) {
2023-09-28 23:50:27 +02:00
const downloaded = await state.osmObjectDownloader.DownloadObjectAsync(snapTo, 0)
2023-06-15 16:12:46 +02:00
if (downloaded !== "deleted") {
2023-09-28 23:50:27 +02:00
snapToWay = downloaded
2023-06-15 16:12:46 +02:00
}
}
2023-06-15 16:12:46 +02:00
const newElementAction = new CreateNewNodeAction(tags, location.lat, location.lon, {
theme: state.layout?.id ?? "unkown",
changeType: "create",
snapOnto: snapToWay,
2023-09-28 23:50:27 +02:00
reusePointWithinMeters: 1,
})
await state.changes.applyAction(newElementAction)
state.newFeatures.features.ping()
2023-06-15 16:12:46 +02:00
// The 'changes' should have created a new point, which added this into the 'featureProperties'
2023-09-28 23:50:27 +02:00
const newId = newElementAction.newElementId
console.log("Applied pending changes, fetching store for", newId)
const tagsStore = state.featureProperties.getStore(newId)
if (!tagsStore) {
2023-09-28 23:50:27 +02:00
console.error("Bug: no tagsStore found for", newId)
}
2023-06-15 16:12:46 +02:00
{
// Set some metainfo
2023-09-28 23:50:27 +02:00
const properties = tagsStore.data
2023-06-15 16:12:46 +02:00
if (snapTo) {
// metatags (starting with underscore) are not uploaded, so we can safely mark this
2023-09-28 23:50:27 +02:00
delete properties["_referencing_ways"]
properties["_referencing_ways"] = `["${snapTo}"]`
2023-06-15 16:12:46 +02:00
}
2023-09-28 23:50:27 +02:00
properties["_backend"] = state.osmConnection.Backend()
properties["_last_edit:timestamp"] = new Date().toISOString()
const userdetails = state.osmConnection.userDetails.data
properties["_last_edit:contributor"] = userdetails.name
properties["_last_edit:uid"] = "" + userdetails.uid
tagsStore.ping()
2023-05-14 03:24:13 +02:00
}
2023-09-28 23:50:27 +02:00
const feature = state.indexedFeatures.featuresById.data.get(newId)
console.log("Selecting feature", feature, "and opening their popup")
abort()
state.selectedLayer.setData(selectedPreset.layer)
state.selectedElement.setData(feature)
tagsStore.ping()
}
function confirmSync() {
2023-09-28 23:50:27 +02:00
confirm()
.then((_) => console.debug("New point successfully handled"))
.catch((e) => console.error("Handling the new point went wrong due to", e))
2023-06-15 16:12:46 +02:00
}
</script>
<LoginToggle ignoreLoading={true} {state}>
2023-06-15 16:12:46 +02:00
<!-- This component is basically one big if/then/else flow checking for many conditions and edge cases that (in some cases) have to be handled;
1. the first (and outermost) is of course: are we logged in?
2. What do we want to add?
3. Are all elements of this category visible? (i.e. there are no filters possibly hiding this, is the data still loading, ...) -->
2023-06-15 16:12:46 +02:00
<LoginButton osmConnection={state.osmConnection} slot="not-logged-in">
<Tr slot="message" t={Translations.t.general.add.pleaseLogin} />
</LoginButton>
2023-10-16 14:27:05 +02:00
{#if $zoom < Constants.minZoomLevelToAddNewPoint}
2023-06-15 16:12:46 +02:00
<div class="alert">
<Tr t={Translations.t.general.add.zoomInFurther} />
</div>
2023-10-16 14:27:05 +02:00
{:else if $isLoading}
<div class="alert">
<Loading>
<Tr t={Translations.t.general.add.stillLoading} />
</Loading>
</div>
2023-06-15 16:12:46 +02:00
{:else if selectedPreset === undefined}
<!-- First, select the correct preset -->
<PresetList
{state}
on:select={(event) => {
selectedPreset = event.detail
}}
2023-06-15 16:12:46 +02:00
/>
{:else if !$layerIsDisplayed}
<!-- Check that the layer is enabled, so that we don't add a duplicate -->
<div class="alert flex items-center justify-center">
<EyeOffIcon class="w-8" />
<Tr
t={Translations.t.general.add.layerNotEnabled.Subs({ layer: selectedPreset.layer.name })}
/>
</div>
2023-06-15 16:12:46 +02:00
<div class="flex flex-wrap-reverse md:flex-nowrap">
<button
class="flex w-full gap-x-1"
on:click={() => {
abort()
state.guistate.openFilterView(selectedPreset.layer)
}}
2023-06-15 16:12:46 +02:00
>
<ToSvelte construct={Svg.layers_svg().SetClass("w-12")} />
<Tr t={Translations.t.general.add.openLayerControl} />
</button>
2023-06-15 16:12:46 +02:00
<button
class="primary flex w-full gap-x-1"
on:click={() => {
layerIsDisplayed.setData(true)
abort()
}}
2023-06-15 16:12:46 +02:00
>
<EyeIcon class="w-12" />
<Tr t={Translations.t.general.add.enableLayer.Subs({ name: selectedPreset.layer.name })} />
</button>
</div>
{:else if $layerHasFilters}
<!-- Some filters are enabled. The feature to add might already be mapped, but hidden -->
<div class="alert flex items-center justify-center">
<EyeOffIcon class="w-8" />
<Tr t={Translations.t.general.add.disableFiltersExplanation} />
</div>
<div class="flex flex-wrap-reverse md:flex-nowrap">
<button
class="primary flex w-full gap-x-1"
on:click={() => {
abort()
state.layerState.filteredLayers.get(selectedPreset.layer.id).disableAllFilters()
}}
2023-06-15 16:12:46 +02:00
>
<EyeOffIcon class="w-12" />
<Tr t={Translations.t.general.add.disableFilters} />
</button>
<button
class="flex w-full gap-x-1"
on:click={() => {
abort()
state.guistate.openFilterView(selectedPreset.layer)
}}
2023-06-15 16:12:46 +02:00
>
<ToSvelte construct={Svg.layers_svg().SetClass("w-12")} />
<Tr t={Translations.t.general.add.openLayerControl} />
</button>
</div>
{:else if !confirmedCategory}
<!-- Second, confirm the category -->
<h2 class="mr-12">
<Tr
t={Translations.t.general.add.confirmTitle.Subs({ title: selectedPreset.preset.title })}
/>
</h2>
2023-06-15 16:12:46 +02:00
<Tr t={Translations.t.general.add.confirmIntro} />
2023-06-15 16:12:46 +02:00
{#if selectedPreset.preset.description}
<Tr t={selectedPreset.preset.description} />
{/if}
2023-06-15 16:12:46 +02:00
{#if selectedPreset.preset.exampleImages}
<h3>
{#if selectedPreset.preset.exampleImages.length === 1}
<Tr t={Translations.t.general.example} />
{:else}
<Tr t={Translations.t.general.examples} />
{/if}
</h3>
<span class="flex flex-wrap items-stretch">
{#each selectedPreset.preset.exampleImages as src}
2023-06-15 16:12:46 +02:00
<img {src} class="m-1 h-64 w-auto rounded-lg" />
{/each}
</span>
2023-06-15 16:12:46 +02:00
{/if}
<TagHint
embedIn={(tags) => t.presetInfo.Subs({ tags })}
{state}
tags={new And(selectedPreset.preset.tags)}
/>
2023-06-15 16:12:46 +02:00
<div class="flex w-full flex-wrap-reverse md:flex-nowrap">
<BackButton on:click={() => (selectedPreset = undefined)} clss="w-full">
<Tr t={t.backToSelect} />
</BackButton>
2023-06-15 16:12:46 +02:00
<NextButton on:click={() => (confirmedCategory = true)} clss="primary w-full">
<div slot="image" class="relative">
<FromHtml src={selectedPreset.icon} />
<img class="absolute bottom-0 right-0 h-4 w-4" src="./assets/svg/confirm.svg" />
</div>
<div class="w-full">
<Tr t={selectedPreset.text} />
2023-05-14 03:24:13 +02:00
</div>
2023-06-15 16:12:46 +02:00
</NextButton>
</div>
{:else if _globalFilter?.length > 0 && _globalFilter?.length > checkedOfGlobalFilters}
<Tr t={_globalFilter[checkedOfGlobalFilters].onNewPoint?.safetyCheck} cls="mx-12" />
<SubtleButton
on:click={() => {
checkedOfGlobalFilters = checkedOfGlobalFilters + 1
}}
2023-06-15 16:12:46 +02:00
>
<img
slot="image"
src={_globalFilter[checkedOfGlobalFilters].onNewPoint?.icon ?? "./assets/svg/confirm.svg"}
class="h-12 w-12"
/>
<Tr
slot="message"
t={_globalFilter[checkedOfGlobalFilters].onNewPoint?.confirmAddNew.Subs({
preset: selectedPreset.preset,
})}
2023-06-15 16:12:46 +02:00
/>
</SubtleButton>
<SubtleButton
on:click={() => {
globalFilter.setData([])
abort()
}}
2023-06-15 16:12:46 +02:00
>
<img slot="image" src="./assets/svg/close.svg" class="h-8 w-8" />
<Tr slot="message" t={Translations.t.general.cancel} />
</SubtleButton>
{:else if !creating}
<div class="relative w-full p-1">
<div class="h-96 max-h-screen w-full overflow-hidden rounded-xl">
<NewPointLocationInput
on:click={() => {
preciseInputIsTapped = true
}}
value={preciseCoordinate}
snappedTo={snappedToObject}
{state}
{coordinate}
targetLayer={selectedPreset.layer}
snapToLayers={selectedPreset.preset.preciseInput.snapToLayers}
/>
</div>
2023-06-15 16:12:46 +02:00
<div
class={twJoin(
!preciseInputIsTapped && "hidden",
"absolute top-0 flex w-full justify-center p-12"
)}
>
<NextButton on:click={confirmSync} clss="primary w-fit">
2023-06-15 16:12:46 +02:00
<div class="flex w-full justify-end gap-x-2">
<Tr t={Translations.t.general.add.confirmLocation} />
</div>
</NextButton>
</div>
2023-06-15 16:12:46 +02:00
<div class="absolute bottom-0 left-0 p-4">
<OpenBackgroundSelectorButton {state} />
</div>
</div>
<div class="flex flex-wrap-reverse md:flex-nowrap">
<BackButton on:click={() => (selectedPreset = undefined)} clss="w-full">
<Tr t={t.backToSelect} />
</BackButton>
2023-06-15 16:12:46 +02:00
<NextButton on:click={confirm} clss={"primary w-full"}>
<div class="flex w-full justify-end gap-x-2">
<Tr t={Translations.t.general.add.confirmLocation} />
</div>
2023-06-15 16:12:46 +02:00
</NextButton>
</div>
{:else}
<Loading>Creating point...</Loading>
{/if}
</LoginToggle>