2023-03-29 17:21:20 +02:00
|
|
|
<script lang="ts">
|
2023-06-14 20:39:36 +02:00
|
|
|
import { UIEventSource } from "../../../Logic/UIEventSource"
|
|
|
|
import type { MapProperties } from "../../../Models/MapProperties"
|
|
|
|
import { Map as MlMap } from "maplibre-gl"
|
|
|
|
import { MapLibreAdaptor } from "../../Map/MapLibreAdaptor"
|
|
|
|
import MaplibreMap from "../../Map/MaplibreMap.svelte"
|
|
|
|
import ToSvelte from "../../Base/ToSvelte.svelte"
|
|
|
|
import Svg from "../../../Svg.js"
|
2023-03-29 17:21:20 +02:00
|
|
|
|
|
|
|
/**
|
2023-04-16 03:42:26 +02:00
|
|
|
* A visualisation to pick a direction on a map background.
|
2023-03-29 17:21:20 +02:00
|
|
|
*/
|
2023-06-14 20:39:36 +02:00
|
|
|
export let value: UIEventSource<undefined | string>
|
|
|
|
export let mapProperties: Partial<MapProperties> & {
|
|
|
|
readonly location: UIEventSource<{ lon: number; lat: number }>
|
|
|
|
}
|
|
|
|
let map: UIEventSource<MlMap> = new UIEventSource<MlMap>(undefined)
|
|
|
|
let mla = new MapLibreAdaptor(map, mapProperties)
|
2023-03-29 17:21:20 +02:00
|
|
|
mla.allowMoving.setData(false)
|
|
|
|
mla.allowZooming.setData(false)
|
2023-06-14 20:39:36 +02:00
|
|
|
let directionElem: HTMLElement | undefined
|
|
|
|
$: value.addCallbackAndRunD((degrees) => {
|
2023-03-29 17:21:20 +02:00
|
|
|
if (directionElem === undefined) {
|
2023-06-14 20:39:36 +02:00
|
|
|
return
|
2023-03-29 17:21:20 +02:00
|
|
|
}
|
2023-06-14 20:39:36 +02:00
|
|
|
directionElem.style.rotate = degrees + "deg"
|
|
|
|
})
|
2023-03-29 17:21:20 +02:00
|
|
|
|
2023-06-14 20:39:36 +02:00
|
|
|
let mainElem: HTMLElement
|
2023-03-29 17:21:20 +02:00
|
|
|
function onPosChange(x: number, y: number) {
|
2023-06-14 20:39:36 +02:00
|
|
|
const rect = mainElem.getBoundingClientRect()
|
|
|
|
const dx = -(rect.left + rect.right) / 2 + x
|
|
|
|
const dy = (rect.top + rect.bottom) / 2 - y
|
|
|
|
const angle = (180 * Math.atan2(dy, dx)) / Math.PI
|
|
|
|
const angleGeo = Math.floor((450 - angle) % 360)
|
|
|
|
value.setData("" + angleGeo)
|
2023-03-29 17:21:20 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 20:39:36 +02:00
|
|
|
let isDown = false
|
2023-03-29 17:21:20 +02:00
|
|
|
</script>
|
|
|
|
|
2023-06-14 20:39:36 +02:00
|
|
|
<div
|
|
|
|
bind:this={mainElem}
|
2023-06-14 20:44:01 +02:00
|
|
|
class="relative h-48 w-48 cursor-pointer overflow-hidden"
|
2023-06-14 20:39:36 +02:00
|
|
|
on:click={(e) => onPosChange(e.x, e.y)}
|
|
|
|
on:mousedown={(e) => {
|
|
|
|
isDown = true
|
|
|
|
onPosChange(e.clientX, e.clientY)
|
|
|
|
}}
|
|
|
|
on:mousemove={(e) => {
|
|
|
|
if (isDown) {
|
2023-03-29 17:21:20 +02:00
|
|
|
onPosChange(e.clientX, e.clientY)
|
2023-06-18 00:48:28 +02:00
|
|
|
e.preventDefault()
|
2023-06-14 20:39:36 +02:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
on:mouseup={() => {
|
|
|
|
isDown = false
|
|
|
|
}}
|
2023-06-18 00:48:28 +02:00
|
|
|
on:touchmove={(e) =>{ onPosChange(e.touches[0].clientX, e.touches[0].clientY); e.preventDefault() }}
|
2023-06-14 20:39:36 +02:00
|
|
|
on:touchstart={(e) => onPosChange(e.touches[0].clientX, e.touches[0].clientY)}
|
|
|
|
>
|
2023-06-14 20:44:01 +02:00
|
|
|
<div class="absolute top-0 left-0 h-full w-full cursor-pointer">
|
2023-06-14 20:39:36 +02:00
|
|
|
<MaplibreMap {map} attribution={false} />
|
2023-03-29 17:21:20 +02:00
|
|
|
</div>
|
|
|
|
|
2023-06-14 20:44:01 +02:00
|
|
|
<div bind:this={directionElem} class="absolute top-0 left-0 h-full w-full">
|
2023-06-14 20:39:36 +02:00
|
|
|
<ToSvelte construct={Svg.direction_stroke_svg} />
|
2023-03-29 17:21:20 +02:00
|
|
|
</div>
|
|
|
|
</div>
|