MapComplete/src/UI/Popup/Notes/CreateNewNote.svelte

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

164 lines
5.5 KiB
Svelte
Raw Normal View History

<script lang="ts">
/**
* UIcomponent to create a new note at the given location
*/
2023-12-26 22:30:27 +01:00
import type { SpecialVisualizationState } from "../../SpecialVisualization"
import { UIEventSource } from "../../../Logic/UIEventSource"
import { LocalStorageSource } from "../../../Logic/Web/LocalStorageSource"
import ValidatedInput from "../../InputElement/ValidatedInput.svelte"
import SubtleButton from "../../Base/SubtleButton.svelte"
import Tr from "../../Base/Tr.svelte"
import Translations from "../../i18n/Translations.js"
2023-10-16 14:27:05 +02:00
import type { Feature, Point } from "geojson"
2023-12-26 22:30:27 +01:00
import LoginToggle from "../../Base/LoginToggle.svelte"
import FilteredLayer from "../../../Models/FilteredLayer"
import NewPointLocationInput from "../../BigComponents/NewPointLocationInput.svelte"
import Layers from "../../../assets/svg/Layers.svelte"
import AddSmall from "../../../assets/svg/AddSmall.svelte"
import type { OsmTags } from "../../../Models/OsmFeature"
import Loading from "../../Base/Loading.svelte"
2024-06-21 12:30:07 +02:00
import NextButton from "../../Base/NextButton.svelte"
import Note from "../../../assets/svg/Note.svelte"
import TitledPanel from "../../Base/TitledPanel.svelte"
2023-10-16 14:27:05 +02:00
export let coordinate: UIEventSource<{ lon: number; lat: number }>
export let state: SpecialVisualizationState
let comment: UIEventSource<string> = LocalStorageSource.get("note-text")
2023-10-16 14:27:05 +02:00
let created = false
2023-10-16 14:27:05 +02:00
let notelayer: FilteredLayer = state.layerState.filteredLayers.get("note")
2023-10-16 14:27:05 +02:00
let hasFilter = notelayer?.hasFilter
let isDisplayed = notelayer?.isDisplayed
2023-12-26 22:30:27 +01:00
let submitted = false
let textEntered = false
function enableNoteLayer() {
2023-10-16 14:27:05 +02:00
state.guistate.closeAll()
isDisplayed.setData(true)
}
async function uploadNote() {
2023-12-26 22:30:27 +01:00
submitted = true
2023-10-16 14:27:05 +02:00
let txt = comment.data
if (txt === undefined || txt === "") {
2023-10-16 14:27:05 +02:00
return
}
2023-10-16 14:27:05 +02:00
const loc = coordinate.data
txt += "\n\n #MapComplete #" + state?.theme?.id
2023-10-16 14:27:05 +02:00
const id = await state?.osmConnection?.openNote(loc.lat, loc.lon, txt)
console.log("Created a note, got id", id)
2023-12-26 22:30:27 +01:00
const feature = <Feature<Point, OsmTags>>{
type: "Feature",
geometry: {
type: "Point",
2023-10-16 14:27:05 +02:00
coordinates: [loc.lon, loc.lat],
},
properties: {
id: "" + id.id,
date_created: new Date().toISOString(),
_first_comment: txt,
comments: JSON.stringify([
{
text: txt,
html: txt,
user: state.osmConnection?.userDetails?.data?.name,
2023-10-16 14:27:05 +02:00
uid: state.osmConnection?.userDetails?.data?.uid,
},
]),
},
}
// Normally, the 'Changes' will generate the new element. The 'notes' are an exception to this
2023-10-16 14:27:05 +02:00
state.newFeatures.features.data.push(feature)
state.newFeatures.features.ping()
state.selectedElement?.setData(feature)
2023-09-28 23:50:27 +02:00
if (state.featureProperties.trackFeature) {
2023-10-16 14:27:05 +02:00
state.featureProperties.trackFeature(feature)
}
2023-10-16 14:27:05 +02:00
comment.setData("")
created = true
state.selectedElement.setData(feature)
}
</script>
{#if notelayer === undefined}
<div class="alert">
This theme does not include the layer 'note'. As a result, no nodes can be created
</div>
2023-12-26 22:30:27 +01:00
{:else if submitted}
2023-12-30 15:24:30 +01:00
<Loading />
{:else if created}
<div class="thanks">
<Tr t={Translations.t.notes.isCreated} />
</div>
{:else}
<TitledPanel>
<Tr slot="title" t={Translations.t.notes.createNoteTitle} />
{#if !$isDisplayed}
<div class="alert">
<Tr t={Translations.t.notes.noteLayerNotEnabled} />
</div>
<SubtleButton on:click={enableNoteLayer}>
<Layers slot="image" class="mr-4 h-8 w-8" />
<Tr slot="message" t={Translations.t.notes.noteLayerDoEnable} />
</SubtleButton>
{:else if $hasFilter}
<!-- ...but a filter is set ...-->
<div class="alert">
<Tr t={Translations.t.notes.noteLayerHasFilters} />
</div>
<SubtleButton on:click={() => notelayer.disableAllFilters()}>
2023-12-01 15:23:28 +01:00
<Layers class="mr-4 h-8 w-8" />
<Tr slot="message" t={Translations.t.notes.disableAllNoteFilters} />
</SubtleButton>
{:else}
<!-- The layer with notes is displayed without filters, so we can add a note without worrying for duplicates -->
<div class="h-full flex flex-col justify-between">
2024-06-24 13:11:35 +02:00
<form
class="flex flex-col rounded-sm p-2"
2024-06-24 13:11:35 +02:00
on:submit|preventDefault={uploadNote}
>
2023-12-26 22:30:27 +01:00
<label class="neutral-label">
<Tr t={Translations.t.notes.createNoteIntro} />
<div class="w-full p-1">
<ValidatedInput autofocus={true} type="text" value={comment} />
</div>
2023-12-26 22:30:27 +01:00
</label>
2023-12-26 22:30:27 +01:00
<LoginToggle {state}>
<span slot="loading"><!--empty: don't show a loading message--></span>
<div slot="not-logged-in" class="alert">
<Tr t={Translations.t.notes.warnAnonymous} />
</div>
</LoginToggle>
{#if $comment?.length >= 3}
2024-06-21 12:30:07 +02:00
<NextButton on:click={uploadNote} clss="self-end primary">
2023-12-26 22:30:27 +01:00
<AddSmall slot="image" class="mr-4 h-8 w-8" />
2024-06-21 12:30:07 +02:00
<Tr t={Translations.t.notes.createNote} />
</NextButton>
2023-12-26 22:30:27 +01:00
{:else}
<div class="alert">
<Tr t={Translations.t.notes.textNeeded} />
</div>
{/if}
</form>
<div class="h-56 w-full">
<NewPointLocationInput value={coordinate} {state}>
<div class="h-20 w-full pb-10" slot="image">
<Note class="h-10 w-full" />
</div>
</NewPointLocationInput>
</div>
</div>
{/if}
</TitledPanel>
{/if}