MapComplete/src/UI/BigComponents/Geosearch.svelte

137 lines
4.5 KiB
Svelte
Raw Normal View History

2023-03-28 05:13:48 +02:00
<script lang="ts">
2023-12-19 22:21:34 +01:00
import { UIEventSource } from "../../Logic/UIEventSource"
import type { Feature } from "geojson"
import Translations from "../i18n/Translations"
import Loading from "../Base/Loading.svelte"
import Hotkeys from "../Base/Hotkeys"
import { Geocoding } from "../../Logic/Osm/Geocoding"
import { BBox } from "../../Logic/BBox"
import { GeoIndexedStoreForLayer } from "../../Logic/FeatureSource/Actors/GeoIndexedStore"
import { createEventDispatcher, onDestroy } from "svelte"
import { placeholder } from "../../Utils/placeholder"
import { SearchIcon } from "@rgossiaux/svelte-heroicons/solid"
import { ariaLabel } from "../../Utils/ariaLabel"
import { GeoLocationState } from "../../Logic/State/GeoLocationState"
2023-03-28 05:13:48 +02:00
export let perLayer: ReadonlyMap<string, GeoIndexedStoreForLayer> | undefined = undefined
export let bounds: UIEventSource<BBox>
export let selectedElement: UIEventSource<Feature> | undefined = undefined
2023-03-28 05:13:48 +02:00
export let geolocationState: GeoLocationState | undefined = undefined
export let clearAfterView: boolean = true
let searchContents: string = ""
export let triggerSearch: UIEventSource<any> = new UIEventSource<any>(undefined)
onDestroy(
triggerSearch.addCallback((_) => {
performSearch()
2023-12-21 01:46:18 +01:00
})
)
2023-03-28 05:13:48 +02:00
let isRunning: boolean = false
2023-03-28 05:13:48 +02:00
let inputElement: HTMLInputElement
let feedback: string = undefined
2023-12-19 22:21:34 +01:00
function focusOnSearch() {
requestAnimationFrame(() => {
2023-12-19 22:21:34 +01:00
inputElement?.focus()
inputElement?.select()
})
2023-12-19 22:21:34 +01:00
}
Hotkeys.RegisterHotkey({ ctrl: "F" }, Translations.t.hotkeyDocumentation.selectSearch, () => {
feedback = undefined
focusOnSearch()
})
const dispatch = createEventDispatcher<{ searchCompleted; searchIsValid: boolean }>()
$: {
if (!searchContents?.trim()) {
dispatch("searchIsValid", false)
} else {
dispatch("searchIsValid", true)
}
}
async function performSearch() {
try {
isRunning = true
geolocationState?.allowMoving.setData(true)
geolocationState?.requestMoment.setData(undefined) // If the GPS is still searching for a fix, we say that we don't want tozoom to it anymore
searchContents = searchContents?.trim() ?? ""
2023-05-18 15:44:54 +02:00
if (searchContents === "") {
return
}
const result = await Geocoding.Search(searchContents, bounds.data)
if (result.length == 0) {
feedback = Translations.t.general.search.nothing.txt
2023-12-19 22:21:34 +01:00
focusOnSearch()
return
}
const poi = result[0]
const [lat0, lat1, lon0, lon1] = poi.boundingbox
bounds.set(
new BBox([
[lon0, lat0],
[lon1, lat1],
2023-12-21 01:46:18 +01:00
]).pad(0.01)
)
if (perLayer !== undefined) {
const id = poi.osm_type + "/" + poi.osm_id
const layers = Array.from(perLayer?.values() ?? [])
for (const layer of layers) {
const found = layer.features.data.find((f) => f.properties.id === id)
if (found === undefined) {
2023-12-19 22:21:34 +01:00
continue
}
2023-12-19 22:21:34 +01:00
selectedElement?.setData(found)
console.log("Found an element that probably matches:", selectedElement?.data)
break
}
}
if (clearAfterView) {
searchContents = ""
}
dispatch("searchIsValid", false)
dispatch("searchCompleted")
} catch (e) {
console.error(e)
feedback = Translations.t.general.search.error.txt
2023-12-19 22:21:34 +01:00
focusOnSearch()
} finally {
isRunning = false
2023-03-28 05:13:48 +02:00
}
}
2023-03-28 05:13:48 +02:00
</script>
2023-06-14 20:44:01 +02:00
<div class="normal-background flex justify-between rounded-full pl-2">
2024-09-02 03:47:54 +02:00
<form class="flex w-full flex-wrap items-center ">
{#if isRunning}
<Loading>{Translations.t.general.search.searching}</Loading>
{:else}
<input
type="search"
2024-09-02 03:47:54 +02:00
style="border: none !important;"
class="w-full outline-none border-none mx-2"
bind:this={inputElement}
2023-12-21 01:46:18 +01:00
on:keypress={(keypr) => {
feedback = undefined
return keypr.key === "Enter" ? performSearch() : undefined
}}
bind:value={searchContents}
use:placeholder={Translations.t.general.search.search}
use:ariaLabel={Translations.t.general.search.search}
/>
2023-12-19 22:21:34 +01:00
{#if feedback !== undefined}
<!-- The feedback is _always_ shown for screenreaders and to make sure that the searchfield can still be selected by tabbing-->
2023-12-21 01:46:18 +01:00
<div class="alert" role="alert" aria-live="assertive">
2023-12-19 22:21:34 +01:00
{feedback}
</div>
{/if}
{/if}
</form>
2024-09-02 03:47:54 +02:00
<SearchIcon aria-hidden="true" class="h-6 w-6 mx-2 self-center" on:click={performSearch} />
2023-03-28 05:13:48 +02:00
</div>