| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | <script lang="ts"> | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  |   import { UIEventSource } from "../../Logic/UIEventSource" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   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" | 
					
						
							| 
									
										
										
										
											2023-12-13 02:16:53 +01:00
										 |  |  |   import { placeholder } from "../../Utils/placeholder" | 
					
						
							| 
									
										
										
										
											2023-12-15 01:46:01 +01:00
										 |  |  |   import { SearchIcon } from "@rgossiaux/svelte-heroicons/solid" | 
					
						
							| 
									
										
										
										
											2024-01-01 03:29:57 +01:00
										 |  |  |   import { ariaLabel } from "../../Utils/ariaLabel" | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   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-06-14 20:39:36 +02:00
										 |  |  |   ) | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   let isRunning: boolean = false | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   let inputElement: HTMLInputElement | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   let feedback: string = undefined | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   function focusOnSearch() { | 
					
						
							| 
									
										
										
										
											2023-12-06 17:27:30 +01:00
										 |  |  |     requestAnimationFrame(() => { | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  |       inputElement?.focus() | 
					
						
							|  |  |  |       inputElement?.select() | 
					
						
							| 
									
										
										
										
											2023-12-06 17:27:30 +01:00
										 |  |  |     }) | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Hotkeys.RegisterHotkey({ ctrl: "F" }, Translations.t.hotkeyDocumentation.selectSearch, () => { | 
					
						
							|  |  |  |     feedback = undefined | 
					
						
							|  |  |  |     focusOnSearch() | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   }) | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   const dispatch = createEventDispatcher<{ searchCompleted; searchIsValid: boolean }>() | 
					
						
							|  |  |  |   $: { | 
					
						
							|  |  |  |     if (!searchContents?.trim()) { | 
					
						
							|  |  |  |       dispatch("searchIsValid", false) | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       dispatch("searchIsValid", true) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |   async function performSearch() { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       isRunning = true | 
					
						
							|  |  |  |       searchContents = searchContents?.trim() ?? "" | 
					
						
							| 
									
										
										
										
											2023-05-18 15:44:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +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() | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |         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) | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |       ) | 
					
						
							|  |  |  |       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) | 
					
						
							| 
									
										
										
										
											2023-12-06 17:27:30 +01:00
										 |  |  |           if (found === undefined) { | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  |             continue | 
					
						
							| 
									
										
										
										
											2023-12-06 17:27:30 +01:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  |           selectedElement?.setData(found) | 
					
						
							|  |  |  |           console.log("Found an element that probably matches:", selectedElement?.data) | 
					
						
							|  |  |  |           break | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |       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() | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |     } finally { | 
					
						
							|  |  |  |       isRunning = false | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +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"> | 
					
						
							| 
									
										
										
										
											2023-12-21 01:46:18 +01:00
										 |  |  |   <form class="flex w-full flex-wrap"> | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |     {#if isRunning} | 
					
						
							|  |  |  |       <Loading>{Translations.t.general.search.searching}</Loading> | 
					
						
							|  |  |  |     {:else} | 
					
						
							|  |  |  |       <input | 
					
						
							|  |  |  |         type="search" | 
					
						
							|  |  |  |         class="w-full" | 
					
						
							|  |  |  |         bind:this={inputElement} | 
					
						
							| 
									
										
										
										
											2023-12-21 01:46:18 +01:00
										 |  |  |         on:keypress={(keypr) => { | 
					
						
							|  |  |  |           feedback = undefined | 
					
						
							|  |  |  |           return keypr.key === "Enter" ? performSearch() : undefined | 
					
						
							|  |  |  |         }} | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |         bind:value={searchContents} | 
					
						
							| 
									
										
										
										
											2023-12-13 02:16:53 +01:00
										 |  |  |         use:placeholder={Translations.t.general.search.search} | 
					
						
							| 
									
										
										
										
											2024-01-01 03:29:57 +01:00
										 |  |  |         use:ariaLabel={Translations.t.general.search.search} | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |       /> | 
					
						
							| 
									
										
										
										
											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} | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |     {/if} | 
					
						
							|  |  |  |   </form> | 
					
						
							| 
									
										
										
										
											2023-12-19 22:21:34 +01:00
										 |  |  |   <SearchIcon aria-hidden="true" class="h-6 w-6 self-end" on:click={performSearch} /> | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | </div> |