Files
gpx.studio/website/src/lib/components/map/layer-control/LayerControl.svelte

236 lines
8.2 KiB
Svelte
Raw Normal View History

2025-06-21 21:07:36 +02:00
<script lang="ts">
import CustomControl from '$lib/components/map/custom-control/CustomControl.svelte';
import LayerTree from './LayerTree.svelte';
2025-10-23 18:58:33 +02:00
import { OverpassLayer } from './overpass-layer';
2025-06-21 21:07:36 +02:00
import { Separator } from '$lib/components/ui/separator';
import { ScrollArea } from '$lib/components/ui/scroll-area/index.js';
import { Layers } from '@lucide/svelte';
import { basemaps, defaultBasemap, overlays } from '$lib/assets/layers';
2025-10-17 23:54:45 +02:00
import { settings } from '$lib/logic/settings';
import { map } from '$lib/components/map/map';
import { customBasemapUpdate, getLayers } from './utils';
2025-06-21 21:07:36 +02:00
import type { ImportSpecification, StyleSpecification } from 'mapbox-gl';
2025-10-23 18:42:10 +02:00
import { untrack } from 'svelte';
2025-06-21 21:07:36 +02:00
let container: HTMLDivElement;
2025-10-17 23:54:45 +02:00
let overpassLayer: OverpassLayer;
2025-06-21 21:07:36 +02:00
const {
currentBasemap,
previousBasemap,
currentOverlays,
currentOverpassQueries,
selectedBasemapTree,
selectedOverlayTree,
selectedOverpassTree,
customLayers,
opacities,
} = settings;
function setStyle() {
2025-10-17 23:54:45 +02:00
if (!$map) {
2025-06-21 21:07:36 +02:00
return;
}
2025-10-17 23:54:45 +02:00
let basemap = basemaps.hasOwnProperty($currentBasemap)
? basemaps[$currentBasemap]
2025-10-22 21:54:22 +02:00
: ($customLayers[$currentBasemap]?.value ?? basemaps[defaultBasemap]);
2025-10-17 23:54:45 +02:00
$map.removeImport('basemap');
2025-06-21 21:07:36 +02:00
if (typeof basemap === 'string') {
2025-10-17 23:54:45 +02:00
$map.addImport({ id: 'basemap', url: basemap }, 'overlays');
2025-06-21 21:07:36 +02:00
} else {
2025-10-17 23:54:45 +02:00
$map.addImport(
2025-06-21 21:07:36 +02:00
{
id: 'basemap',
url: '',
data: basemap as StyleSpecification,
},
'overlays'
);
}
}
$effect(() => {
2025-10-17 23:54:45 +02:00
if ($map && ($currentBasemap || $customBasemapUpdate)) {
2025-10-23 18:42:10 +02:00
untrack(() => setStyle());
2025-06-21 21:07:36 +02:00
}
});
function addOverlay(id: string) {
2025-10-17 23:54:45 +02:00
if (!$map) {
2025-06-21 21:07:36 +02:00
return;
}
try {
2025-10-17 23:54:45 +02:00
let overlay = $customLayers.hasOwnProperty(id) ? $customLayers[id].value : overlays[id];
2025-06-21 21:07:36 +02:00
if (typeof overlay === 'string') {
2025-10-17 23:54:45 +02:00
$map.addImport({ id, url: overlay });
2025-06-21 21:07:36 +02:00
} else {
2025-10-17 23:54:45 +02:00
if ($opacities.hasOwnProperty(id)) {
2025-06-21 21:07:36 +02:00
overlay = {
...overlay,
layers: (overlay as StyleSpecification).layers.map((layer) => {
if (layer.type === 'raster') {
if (!layer.paint) {
layer.paint = {};
}
2025-10-17 23:54:45 +02:00
layer.paint['raster-opacity'] = $opacities[id];
2025-06-21 21:07:36 +02:00
}
return layer;
}),
};
}
2025-10-17 23:54:45 +02:00
$map.addImport({
2025-06-21 21:07:36 +02:00
id,
url: '',
data: overlay as StyleSpecification,
});
}
} catch (e) {
// No reliable way to check if the map is ready to add sources and layers
}
}
function updateOverlays() {
2025-10-17 23:54:45 +02:00
if ($map && $currentOverlays && $opacities) {
let overlayLayers = getLayers($currentOverlays);
2025-06-21 21:07:36 +02:00
try {
let activeOverlays =
2025-10-17 23:54:45 +02:00
$map
2025-06-21 21:07:36 +02:00
.getStyle()
2025-10-17 23:54:45 +02:00
.imports?.reduce(
2025-06-21 21:07:36 +02:00
(
acc: Record<string, ImportSpecification>,
imprt: ImportSpecification
) => {
if (
!['basemap', 'overlays', 'glyphs-and-sprite'].includes(imprt.id)
) {
acc[imprt.id] = imprt;
}
return acc;
},
{}
) || {};
let toRemove = Object.keys(activeOverlays).filter((id) => !overlayLayers[id]);
toRemove.forEach((id) => {
2025-10-17 23:54:45 +02:00
$map?.removeImport(id);
2025-06-21 21:07:36 +02:00
});
let toAdd = Object.entries(overlayLayers)
.filter(([id, selected]) => selected && !activeOverlays.hasOwnProperty(id))
.map(([id]) => id);
toAdd.forEach((id) => {
addOverlay(id);
});
} catch (e) {
// No reliable way to check if the map is ready to add sources and layers
}
}
}
$effect(() => {
2025-10-17 23:54:45 +02:00
if ($map && $currentOverlays && $opacities) {
2025-10-23 18:42:10 +02:00
untrack(() => updateOverlays());
2025-06-21 21:07:36 +02:00
}
});
2025-10-17 23:54:45 +02:00
map.onLoad((_map: mapboxgl.Map) => {
if (overpassLayer) {
overpassLayer.remove();
}
overpassLayer = new OverpassLayer(_map);
overpassLayer.add();
2025-10-23 18:42:10 +02:00
let first = true;
_map.on('style.import.load', () => {
if (!first) return;
first = false;
updateOverlays();
});
2025-10-17 23:54:45 +02:00
});
2025-06-21 21:07:36 +02:00
let open = $state(false);
function openLayerControl() {
open = true;
}
function closeLayerControl() {
open = false;
}
let cancelEvents = $state(false);
</script>
<CustomControl class="group min-w-[29px] min-h-[29px] overflow-hidden">
<!-- svelte-ignore a11y_no_static_element_interactions -->
<div
bind:this={container}
class="size-full"
onmouseenter={openLayerControl}
onmouseleave={closeLayerControl}
onpointerenter={() => {
if (!open) {
cancelEvents = true;
openLayerControl();
setTimeout(() => {
cancelEvents = false;
}, 500);
}
}}
>
<div
class="flex flex-row justify-center items-center delay-100 transition-[opacity] duration-0 {open
? 'opacity-0 size-0 delay-0'
: 'w-[29px] h-[29px]'}"
>
<Layers size="20" />
</div>
<div
class="transition-[grid-template-rows grid-template-cols] grid grid-rows-[0fr] grid-cols-[0fr] duration-150 h-full {open
? 'grid-rows-[1fr] grid-cols-[1fr]'
: ''} {cancelEvents ? 'pointer-events-none' : ''}"
>
2025-10-18 18:51:11 +02:00
<ScrollArea class="overflow-hidden">
2025-06-21 21:07:36 +02:00
<div class="h-fit">
2025-10-18 18:51:11 +02:00
<div class="p-2 ml-1">
2025-06-21 21:07:36 +02:00
<LayerTree
2025-10-17 23:54:45 +02:00
layerTree={$selectedBasemapTree}
2025-06-21 21:07:36 +02:00
name="basemaps"
2025-10-17 23:54:45 +02:00
selected={$currentBasemap}
2025-06-21 21:07:36 +02:00
onselect={(value) => {
2025-10-17 23:54:45 +02:00
$previousBasemap = $currentBasemap;
$currentBasemap = value;
2025-06-21 21:07:36 +02:00
}}
/>
</div>
<Separator class="w-full" />
2025-10-18 18:51:11 +02:00
<div class="p-2 ml-1">
2025-10-17 23:54:45 +02:00
{#if $currentOverlays}
2025-06-21 21:07:36 +02:00
<LayerTree
2025-10-17 23:54:45 +02:00
layerTree={$selectedOverlayTree}
2025-06-21 21:07:36 +02:00
name="overlays"
multiple={true}
2025-10-17 23:54:45 +02:00
bind:checked={$currentOverlays}
2025-06-21 21:07:36 +02:00
/>
{/if}
</div>
<Separator class="w-full" />
2025-10-18 18:51:11 +02:00
<div class="p-2 ml-1">
2025-10-17 23:54:45 +02:00
{#if $currentOverpassQueries}
2025-06-21 21:07:36 +02:00
<LayerTree
2025-10-17 23:54:45 +02:00
layerTree={$selectedOverpassTree}
2025-06-21 21:07:36 +02:00
name="overpass"
multiple={true}
2025-10-17 23:54:45 +02:00
bind:checked={$currentOverpassQueries}
2025-06-21 21:07:36 +02:00
/>
{/if}
</div>
</div>
</ScrollArea>
</div>
</div>
</CustomControl>
<svelte:window
on:click={(e) => {
if (open && !cancelEvents && !container.contains(e.target)) {
closeLayerControl();
}
}}
/>