2024-04-14 15:36:41 +02:00
|
|
|
<script lang="ts">
|
|
|
|
import LayerTree from './LayerTree.svelte';
|
|
|
|
|
|
|
|
import { Separator } from '$lib/components/ui/separator';
|
|
|
|
import { ScrollArea } from '$lib/components/ui/scroll-area/index.js';
|
|
|
|
import * as Sheet from '$lib/components/ui/sheet';
|
2024-04-14 16:42:38 +02:00
|
|
|
import * as Accordion from '$lib/components/ui/accordion';
|
2024-06-06 18:11:03 +02:00
|
|
|
import { Label } from '$lib/components/ui/label';
|
|
|
|
import * as Select from '$lib/components/ui/select';
|
2024-06-26 18:46:03 +02:00
|
|
|
import { Slider } from '$lib/components/ui/slider';
|
2024-04-14 15:36:41 +02:00
|
|
|
|
2024-06-26 18:46:03 +02:00
|
|
|
import { basemapTree, overlays, overlayTree } from '$lib/assets/layers';
|
|
|
|
import { isSelected } from '$lib/components/layer-control/utils';
|
2024-05-04 23:50:27 +02:00
|
|
|
import { settings } from '$lib/db';
|
2024-04-24 17:39:56 +02:00
|
|
|
|
|
|
|
import { _ } from 'svelte-i18n';
|
2024-06-06 18:11:03 +02:00
|
|
|
import { writable, get } from 'svelte/store';
|
|
|
|
import { map, setStravaHeatmapURLs } from '$lib/stores';
|
2024-06-06 18:40:08 +02:00
|
|
|
import { browser } from '$app/environment';
|
2024-06-26 17:19:41 +02:00
|
|
|
import CustomLayers from './CustomLayers.svelte';
|
2024-05-04 23:50:27 +02:00
|
|
|
|
2024-06-26 18:46:03 +02:00
|
|
|
const {
|
|
|
|
selectedBasemapTree,
|
|
|
|
selectedOverlayTree,
|
|
|
|
stravaHeatmapColor,
|
|
|
|
currentOverlays,
|
|
|
|
customLayers,
|
|
|
|
opacities
|
|
|
|
} = settings;
|
2024-05-23 11:21:57 +02:00
|
|
|
|
|
|
|
export let open: boolean;
|
2024-06-06 18:11:03 +02:00
|
|
|
|
2024-06-26 18:46:03 +02:00
|
|
|
let selectedOverlay = writable(undefined);
|
|
|
|
let overlayOpacity = writable([1]);
|
|
|
|
|
|
|
|
function setOpacityFromSelection() {
|
|
|
|
if ($selectedOverlay) {
|
|
|
|
let overlayId = $selectedOverlay.value;
|
|
|
|
if ($opacities.hasOwnProperty(overlayId)) {
|
|
|
|
$overlayOpacity = [$opacities[overlayId]];
|
|
|
|
} else {
|
|
|
|
$overlayOpacity = [1];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$overlayOpacity = [1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$: if ($selectedOverlay) {
|
|
|
|
setOpacityFromSelection();
|
|
|
|
}
|
|
|
|
|
2024-06-06 18:11:03 +02:00
|
|
|
const heatmapColors = [
|
|
|
|
{ value: '', label: '' },
|
|
|
|
{ value: 'blue', label: $_('layers.color.blue') },
|
|
|
|
{ value: 'bluered', label: $_('layers.color.bluered') },
|
|
|
|
{ value: 'gray', label: $_('layers.color.gray') },
|
|
|
|
{ value: 'hot', label: $_('layers.color.hot') },
|
|
|
|
{ value: 'orange', label: $_('layers.color.orange') },
|
|
|
|
{ value: 'purple', label: $_('layers.color.purple') }
|
|
|
|
];
|
|
|
|
|
|
|
|
let selectedHeatmapColor = writable(heatmapColors[0]);
|
|
|
|
|
|
|
|
$: if ($selectedHeatmapColor !== heatmapColors[0]) {
|
|
|
|
stravaHeatmapColor.set($selectedHeatmapColor.value);
|
|
|
|
|
|
|
|
// remove and add the heatmap layers
|
|
|
|
let m = get(map);
|
|
|
|
if (m) {
|
|
|
|
let currentStravaLayers = [];
|
2024-06-18 14:23:59 +02:00
|
|
|
if (overlayTree.overlays.world.strava) {
|
|
|
|
for (let layer of Object.keys(overlayTree.overlays.world.strava)) {
|
|
|
|
if (m.getLayer(layer)) {
|
|
|
|
m.removeLayer(layer);
|
|
|
|
currentStravaLayers.push(layer);
|
|
|
|
}
|
|
|
|
if (m.getSource(layer)) {
|
|
|
|
m.removeSource(layer);
|
|
|
|
}
|
2024-06-06 18:11:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (currentStravaLayers.length > 0) {
|
|
|
|
currentOverlays.update(($currentOverlays) => {
|
|
|
|
for (let layer of currentStravaLayers) {
|
|
|
|
$currentOverlays.overlays.world.strava[layer] = false;
|
|
|
|
}
|
|
|
|
return $currentOverlays;
|
|
|
|
});
|
|
|
|
currentOverlays.update(($currentOverlays) => {
|
|
|
|
for (let layer of currentStravaLayers) {
|
|
|
|
$currentOverlays.overlays.world.strava[layer] = true;
|
|
|
|
}
|
|
|
|
return $currentOverlays;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-06 18:40:08 +02:00
|
|
|
$: if ($stravaHeatmapColor && browser) {
|
2024-06-06 18:11:03 +02:00
|
|
|
setStravaHeatmapURLs();
|
|
|
|
if ($stravaHeatmapColor !== get(selectedHeatmapColor).value) {
|
|
|
|
let toSelect = heatmapColors.find(({ value }) => value === $stravaHeatmapColor);
|
|
|
|
if (toSelect) {
|
|
|
|
selectedHeatmapColor.set(toSelect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-04-14 15:36:41 +02:00
|
|
|
</script>
|
|
|
|
|
2024-05-23 11:21:57 +02:00
|
|
|
<Sheet.Root bind:open>
|
|
|
|
<Sheet.Trigger class="hidden" />
|
2024-04-14 15:36:41 +02:00
|
|
|
<Sheet.Content>
|
2024-04-14 15:49:21 +02:00
|
|
|
<Sheet.Header class="h-full">
|
2024-04-24 17:39:56 +02:00
|
|
|
<Sheet.Title>{$_('layers.settings')}</Sheet.Title>
|
2024-04-14 15:36:41 +02:00
|
|
|
<Sheet.Description>
|
2024-04-24 17:39:56 +02:00
|
|
|
{$_('layers.settings_help')}
|
2024-04-14 15:36:41 +02:00
|
|
|
</Sheet.Description>
|
2024-07-12 12:19:03 +02:00
|
|
|
<Accordion.Root class="flex flex-col overflow-hidden" value="layer-selection">
|
|
|
|
<Accordion.Item value="layer-selection" class="flex flex-col overflow-hidden">
|
2024-04-24 17:39:56 +02:00
|
|
|
<Accordion.Trigger>{$_('layers.selection')}</Accordion.Trigger>
|
2024-04-15 10:33:47 +02:00
|
|
|
<Accordion.Content class="grow flex flex-col border rounded">
|
2024-06-27 15:50:15 +02:00
|
|
|
<ScrollArea class="py-2 pl-1 pr-2 min-h-9">
|
2024-04-14 16:42:38 +02:00
|
|
|
<LayerTree
|
|
|
|
layerTree={basemapTree}
|
|
|
|
name="basemapSettings"
|
|
|
|
multiple={true}
|
2024-05-05 18:59:09 +02:00
|
|
|
bind:checked={$selectedBasemapTree}
|
2024-04-14 16:42:38 +02:00
|
|
|
/>
|
|
|
|
</ScrollArea>
|
2024-04-15 10:33:47 +02:00
|
|
|
<Separator />
|
2024-06-27 15:50:15 +02:00
|
|
|
<ScrollArea class="py-2 pl-1 pr-2 min-h-9">
|
2024-04-14 16:42:38 +02:00
|
|
|
<LayerTree
|
|
|
|
layerTree={overlayTree}
|
|
|
|
name="overlaySettings"
|
|
|
|
multiple={true}
|
2024-05-05 18:59:09 +02:00
|
|
|
bind:checked={$selectedOverlayTree}
|
2024-04-14 16:42:38 +02:00
|
|
|
/>
|
|
|
|
</ScrollArea>
|
|
|
|
</Accordion.Content>
|
|
|
|
</Accordion.Item>
|
2024-07-12 12:19:03 +02:00
|
|
|
<Accordion.Item value="overlay-opacity">
|
2024-06-26 18:46:03 +02:00
|
|
|
<Accordion.Trigger>{$_('layers.opacity')}</Accordion.Trigger>
|
|
|
|
<Accordion.Content class="flex flex-col gap-3 overflow-visible">
|
2024-07-12 12:19:03 +02:00
|
|
|
<div class="flex flex-row gap-6 items-center">
|
|
|
|
<Label>
|
|
|
|
{$_('layers.custom_layers.overlay')}
|
|
|
|
</Label>
|
2024-06-26 18:46:03 +02:00
|
|
|
<Select.Root bind:selected={$selectedOverlay}>
|
|
|
|
<Select.Trigger class="h-8 mr-1">
|
|
|
|
<Select.Value />
|
|
|
|
</Select.Trigger>
|
|
|
|
<Select.Content>
|
|
|
|
{#each Object.keys(overlays) as id}
|
|
|
|
{#if isSelected($selectedOverlayTree, id)}
|
|
|
|
<Select.Item value={id}>{$_(`layers.label.${id}`)}</Select.Item>
|
|
|
|
{/if}
|
|
|
|
{/each}
|
|
|
|
{#each Object.entries($customLayers) as [id, layer]}
|
|
|
|
{#if layer.layerType === 'overlay'}
|
|
|
|
<Select.Item value={id}>{layer.name}</Select.Item>
|
|
|
|
{/if}
|
|
|
|
{/each}
|
|
|
|
</Select.Content>
|
|
|
|
</Select.Root>
|
2024-07-12 12:19:03 +02:00
|
|
|
</div>
|
|
|
|
<Label class="flex flex-row gap-6 items-center">
|
2024-06-26 18:46:03 +02:00
|
|
|
{$_('menu.style.opacity')}
|
|
|
|
<div class="p-2 pr-3 grow">
|
|
|
|
<Slider
|
|
|
|
bind:value={$overlayOpacity}
|
|
|
|
min={0.1}
|
|
|
|
max={1}
|
|
|
|
step={0.1}
|
|
|
|
disabled={$selectedOverlay === undefined}
|
|
|
|
onValueChange={() => {
|
|
|
|
if ($selectedOverlay) {
|
|
|
|
$opacities[$selectedOverlay.value] = $overlayOpacity[0];
|
|
|
|
if ($map) {
|
|
|
|
if ($map.getLayer($selectedOverlay.value)) {
|
|
|
|
$map.removeLayer($selectedOverlay.value);
|
|
|
|
$currentOverlays = $currentOverlays;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</Label>
|
|
|
|
</Accordion.Content>
|
|
|
|
</Accordion.Item>
|
2024-07-12 12:19:03 +02:00
|
|
|
<Accordion.Item value="custom-layers">
|
2024-07-10 16:39:48 +02:00
|
|
|
<Accordion.Trigger>{$_('layers.custom_layers.title')}</Accordion.Trigger>
|
|
|
|
<Accordion.Content>
|
|
|
|
<ScrollArea>
|
|
|
|
<CustomLayers />
|
|
|
|
</ScrollArea>
|
|
|
|
</Accordion.Content>
|
|
|
|
</Accordion.Item>
|
2024-07-12 12:19:03 +02:00
|
|
|
<Accordion.Item value="pois" class="hidden">
|
|
|
|
<Accordion.Trigger>{$_('layers.pois')}</Accordion.Trigger>
|
|
|
|
<Accordion.Content></Accordion.Content>
|
|
|
|
</Accordion.Item>
|
|
|
|
<Accordion.Item value="heatmap-color" class="hidden">
|
2024-04-24 17:39:56 +02:00
|
|
|
<Accordion.Trigger>{$_('layers.heatmap')}</Accordion.Trigger>
|
2024-06-10 17:55:48 +02:00
|
|
|
<Accordion.Content class="overflow-visible">
|
2024-07-12 12:19:03 +02:00
|
|
|
<div class="flex flex-row items-center justify-between gap-6">
|
|
|
|
<Label>
|
|
|
|
{$_('menu.style.color')}
|
|
|
|
</Label>
|
2024-06-10 17:55:48 +02:00
|
|
|
<Select.Root bind:selected={$selectedHeatmapColor}>
|
|
|
|
<Select.Trigger class="h-8 mr-1">
|
2024-06-26 18:46:03 +02:00
|
|
|
<Select.Value />
|
2024-06-06 18:11:03 +02:00
|
|
|
</Select.Trigger>
|
|
|
|
<Select.Content>
|
|
|
|
{#each heatmapColors as { value, label }}
|
|
|
|
<Select.Item {value}>{label}</Select.Item>
|
|
|
|
{/each}
|
|
|
|
</Select.Content>
|
|
|
|
</Select.Root>
|
2024-07-12 12:19:03 +02:00
|
|
|
</div>
|
2024-06-06 18:11:03 +02:00
|
|
|
</Accordion.Content>
|
2024-04-14 16:42:38 +02:00
|
|
|
</Accordion.Item>
|
|
|
|
</Accordion.Root>
|
2024-04-14 15:36:41 +02:00
|
|
|
</Sheet.Header>
|
|
|
|
</Sheet.Content>
|
|
|
|
</Sheet.Root>
|