| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | <script lang="ts"> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   import { UIEventSource } from "../../Logic/UIEventSource"; | 
					
						
							|  |  |  |   import type { Feature } from "geojson"; | 
					
						
							|  |  |  |   import LayerConfig from "../../Models/ThemeConfig/LayerConfig"; | 
					
						
							|  |  |  |   import ToSvelte from "../Base/ToSvelte.svelte"; | 
					
						
							|  |  |  |   import Svg from "../../Svg.js"; | 
					
						
							|  |  |  |   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"; | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |   import { GeoIndexedStoreForLayer } from "../../Logic/FeatureSource/Actors/GeoIndexedStore"; | 
					
						
							| 
									
										
										
										
											2023-05-14 03:24:13 +02:00
										 |  |  |   import {createEventDispatcher} from "svelte"; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |   export let perLayer: ReadonlyMap<string, GeoIndexedStoreForLayer> | undefined = undefined; | 
					
						
							|  |  |  |   export let bounds: UIEventSource<BBox>; | 
					
						
							|  |  |  |   export let selectedElement: UIEventSource<Feature> | undefined = undefined; | 
					
						
							|  |  |  |   export let selectedLayer: UIEventSource<LayerConfig> | undefined = undefined; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   let searchContents: string = undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let isRunning: boolean = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let inputElement: HTMLInputElement; | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   let feedback: string = undefined; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Hotkeys.RegisterHotkey( | 
					
						
							|  |  |  |     { ctrl: "F" }, | 
					
						
							|  |  |  |     Translations.t.hotkeyDocumentation.selectSearch, | 
					
						
							|  |  |  |     () => { | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |       inputElement?.focus(); | 
					
						
							|  |  |  |       inputElement?.select(); | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-14 03:24:13 +02:00
										 |  |  |   const dispatch = createEventDispatcher<{searchCompleted}>() | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |   async function performSearch() { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       isRunning = true; | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |       searchContents = searchContents?.trim() ?? ""; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |       if (searchContents === "") { | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |       const result = await Geocoding.Search(searchContents, bounds.data); | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |       if (result.length == 0) { | 
					
						
							| 
									
										
										
										
											2023-05-06 12:35:02 +02:00
										 |  |  |         feedback = Translations.t.general.search.nothing.txt; | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |       const poi = result[0]; | 
					
						
							|  |  |  |       const [lat0, lat1, lon0, lon1] = poi.boundingbox; | 
					
						
							|  |  |  |       bounds.set(new BBox([[lon0, lat0], [lon1, lat1]]).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); | 
					
						
							|  |  |  |           selectedElement?.setData(found); | 
					
						
							|  |  |  |           selectedLayer?.setData(layer.layer.layerDef); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-05-14 03:24:13 +02:00
										 |  |  |       dispatch("searchCompleted") | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |     } catch (e) { | 
					
						
							|  |  |  |       console.error(e); | 
					
						
							| 
									
										
										
										
											2023-05-06 12:35:02 +02:00
										 |  |  |       feedback = Translations.t.general.search.error.txt; | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |     } finally { | 
					
						
							|  |  |  |       isRunning = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | </script> | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-24 02:27:55 +02:00
										 |  |  | <div class="flex normal-background rounded-full pl-2 justify-between"> | 
					
						
							| 
									
										
										
										
											2023-05-06 12:35:02 +02:00
										 |  |  |   <form class="w-full"> | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     {#if isRunning} | 
					
						
							|  |  |  |       <Loading>{Translations.t.general.search.searching}</Loading> | 
					
						
							| 
									
										
										
										
											2023-04-14 02:42:57 +02:00
										 |  |  |     {:else if feedback !== undefined} | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |       <div class="alert" on:click={() => feedback = undefined}> | 
					
						
							|  |  |  |         {feedback} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     {:else } | 
					
						
							|  |  |  |       <input | 
					
						
							| 
									
										
										
										
											2023-03-30 04:51:56 +02:00
										 |  |  |         type="search" | 
					
						
							| 
									
										
										
										
											2023-05-06 12:35:02 +02:00
										 |  |  |         class="w-full" | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |         bind:this={inputElement} | 
					
						
							|  |  |  |         on:keypress={keypr => keypr.key === "Enter" ? performSearch() : undefined} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         bind:value={searchContents} | 
					
						
							|  |  |  |         placeholder={Translations.t.general.search.search}> | 
					
						
							|  |  |  |     {/if} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   </form> | 
					
						
							| 
									
										
										
										
											2023-04-24 02:27:55 +02:00
										 |  |  |   <div class="w-6 h-6 self-end" on:click={performSearch}> | 
					
						
							| 
									
										
										
										
											2023-05-08 01:55:21 +02:00
										 |  |  |     <ToSvelte construct={Svg.search_svg}></ToSvelte> | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |   </div> | 
					
						
							|  |  |  | </div> |