progress with layer settings

This commit is contained in:
vcoppe
2024-05-04 23:50:27 +02:00
parent 35b0adc7e5
commit b3d016e2af
10 changed files with 272 additions and 145 deletions

View File

@@ -372,10 +372,11 @@ export const opacities: { [key: string]: number; } = {
}; };
export type LayerTreeType = string[] | { [key: string]: LayerTreeType; }; export type LayerTreeType = string[] | { [key: string]: LayerTreeType; };
export type CollapsedInfoTreeType = { export type CollapsedInfoTreeType<T> = {
self: boolean; self: T;
children: { [key: string]: CollapsedInfoTreeType; }; children: { [key: string]: CollapsedInfoTreeType<T>; };
}; };
export type CheckedInfoTreeType = { [key: string]: boolean | CheckedInfoTreeType };
export const basemapTree: LayerTreeType = { export const basemapTree: LayerTreeType = {
basemaps: { basemaps: {
@@ -408,7 +409,51 @@ export const overlayTree: LayerTreeType = {
}, },
} }
export const layerKeys: { [key: string]: string[]; } = {
mapboxOutdoors: ['basemaps', 'world'],
mapboxSatellite: ['basemaps', 'world'],
openStreetMap: ['basemaps', 'world'],
openTopoMap: ['basemaps', 'world'],
openHikingMap: ['basemaps', 'world'],
cyclOSM: ['basemaps', 'world'],
cyclOSMlite: ['overlays', 'world', 'cyclOSM'],
waymarkedTrailsHiking: ['overlays', 'world', 'waymarked_trails'],
waymarkedTrailsCycling: ['overlays', 'world', 'waymarked_trails'],
waymarkedTrailsMTB: ['overlays', 'world', 'waymarked_trails'],
waymarkedTrailsSkating: ['overlays', 'world', 'waymarked_trails'],
waymarkedTrailsHorseRiding: ['overlays', 'world', 'waymarked_trails'],
waymarkedTrailsWinter: ['overlays', 'world', 'waymarked_trails'],
bgMountains: ['basemaps', 'countries', 'bulgaria'],
finlandTopo: ['basemaps', 'countries', 'finland'],
ignPlanV2: ['basemaps', 'countries', 'france'],
ignFrScan25: ['basemaps', 'countries', 'france'],
ignSatellite: ['basemaps', 'countries', 'france'],
ignFrCadastre: ['overlays', 'countries', 'france'],
ignSlope: ['overlays', 'countries', 'france'],
linz: ['basemaps', 'countries', 'new_zealand'],
linzTopo: ['basemaps', 'countries', 'new_zealand'],
norwayTopo: ['basemaps', 'countries', 'norway'],
swisstopo: ['basemaps', 'countries', 'switzerland'],
swisstopoSlope: ['overlays', 'countries', 'switzerland'],
swisstopoCycling: ['overlays', 'countries', 'switzerland'],
swisstopoMountainBike: ['overlays', 'countries', 'switzerland'],
swedenTopo: ['basemaps', 'countries', 'sweden'],
ordnanceSurvey: ['basemaps', 'countries', 'united_kingdom'],
usgs: ['basemaps', 'countries', 'united_states'],
};
export const defaultBasemap = 'mapboxOutdoors'; export const defaultBasemap = 'mapboxOutdoors';
export const defaultAvailableBasemaps = ['mapboxOutdoors', 'mapboxSatellite', 'openStreetMap', 'openTopoMap', 'openHikingMap', 'cyclOSM']; export const defaultBasemapTree: LayerTreeType = {
export const defaultAvailableOverlays = ['cyclOSMlite', 'waymarkedTrailsHiking', 'waymarkedTrailsCycling', 'waymarkedTrailsMTB', 'waymarkedTrailsSkating', 'waymarkedTrailsHorseRiding', 'waymarkedTrailsWinter']; basemaps: {
world: ['mapboxOutdoors', 'mapboxSatellite', 'openStreetMap', 'openTopoMap', 'openHikingMap', 'cyclOSM']
}
};
export const defaultOverlayTree: LayerTreeType = {
overlays: {
world: {
cyclOSM: ['cyclOSMlite'],
waymarked_trails: ['waymarkedTrailsHiking', 'waymarkedTrailsCycling', 'waymarkedTrailsMTB']
}
}
}

View File

@@ -53,7 +53,7 @@
} }
onMount(() => { onMount(() => {
$map = new mapboxgl.Map({ let newMap = new mapboxgl.Map({
container: 'map', container: 'map',
style: { version: 8, sources: {}, layers: [] }, style: { version: 8, sources: {}, layers: [] },
projection: { name: 'mercator' }, projection: { name: 'mercator' },
@@ -63,16 +63,19 @@
logoPosition: 'bottom-right', logoPosition: 'bottom-right',
boxZoom: false boxZoom: false
}); });
newMap.on('load', () => {
$map = newMap; // only set the store after the map has loaded
});
$map.addControl( newMap.addControl(
new mapboxgl.AttributionControl({ new mapboxgl.AttributionControl({
compact: true compact: true
}) })
); );
$map.addControl(new mapboxgl.NavigationControl()); newMap.addControl(new mapboxgl.NavigationControl());
$map.addControl( newMap.addControl(
new MapboxGeocoder({ new MapboxGeocoder({
accessToken: mapboxgl.accessToken, accessToken: mapboxgl.accessToken,
mapboxgl: mapboxgl, mapboxgl: mapboxgl,
@@ -82,7 +85,7 @@
}) })
); );
$map.addControl( newMap.addControl(
new mapboxgl.GeolocateControl({ new mapboxgl.GeolocateControl({
positionOptions: { positionOptions: {
enableHighAccuracy: true enableHighAccuracy: true
@@ -93,10 +96,10 @@
}) })
); );
$map.addControl(scaleControl); newMap.addControl(scaleControl);
$map.on('style.load', toggleTerrain); newMap.on('style.load', toggleTerrain);
$map.on('pitchstart', toggleTerrain); newMap.on('pitchstart', toggleTerrain);
}); });
$: if ($map) { $: if ($map) {

View File

@@ -23,7 +23,16 @@
let showDistanceMarkers = false; let showDistanceMarkers = false;
let showDirectionMarkers = false; let showDirectionMarkers = false;
const { distanceUnits, velocityUnits, temperatureUnits, mode } = settings; const {
distanceUnits,
velocityUnits,
temperatureUnits,
mode,
currentBasemap,
previousBasemap,
currentOverlays,
previousOverlays
} = settings;
$: if ($mode === 'system') { $: if ($mode === 'system') {
resetMode(); resetMode();
} else { } else {
@@ -184,7 +193,6 @@
<svelte:window <svelte:window
on:keydown={(e) => { on:keydown={(e) => {
e.stopImmediatePropagation();
if (e.key === 'n' && (e.metaKey || e.ctrlKey)) { if (e.key === 'n' && (e.metaKey || e.ctrlKey)) {
createFile(); createFile();
e.preventDefault(); e.preventDefault();
@@ -217,6 +225,16 @@
} else if (e.key === 'a' && (e.metaKey || e.ctrlKey)) { } else if (e.key === 'a' && (e.metaKey || e.ctrlKey)) {
$selectFiles.selectAllFiles(); $selectFiles.selectAllFiles();
e.preventDefault(); e.preventDefault();
} else if (e.key === 'F1') {
[$currentBasemap, $previousBasemap] = [$previousBasemap, $currentBasemap];
e.preventDefault();
} else if (e.key === 'F2') {
if ($currentOverlays.length > 0) {
[$currentOverlays, $previousOverlays] = [[], $currentOverlays];
} else {
[$currentOverlays, $previousOverlays] = [$previousOverlays, []];
}
e.preventDefault();
} }
}} }}
/> />

View File

@@ -8,16 +8,12 @@
let container: HTMLDivElement | null = null; let container: HTMLDivElement | null = null;
$: if ($map && container) { $: if ($map && container) {
$map.on('load', () => {
if ($map && container) {
if (position.includes('right')) container.classList.add('float-right'); if (position.includes('right')) container.classList.add('float-right');
else container.classList.add('float-left'); else container.classList.add('float-left');
container.classList.remove('hidden'); container.classList.remove('hidden');
let control = new CustomControl(container); let control = new CustomControl(container);
$map.addControl(control, position); $map.addControl(control, position);
} }
});
}
</script> </script>
<div <div

View File

@@ -68,6 +68,7 @@ export class GPXLayer {
} }
let addedSource = false; let addedSource = false;
try {
if (!this.map.getSource(this.fileId)) { if (!this.map.getSource(this.fileId)) {
let data = this.getGeoJSON(); let data = this.getGeoJSON();
@@ -98,6 +99,9 @@ export class GPXLayer {
this.map.on('mouseenter', this.fileId, toPointerCursor); this.map.on('mouseenter', this.fileId, toPointerCursor);
this.map.on('mouseleave', this.fileId, toDefaultCursor); this.map.on('mouseleave', this.fileId, toDefaultCursor);
} }
} catch (e) { // No reliable way to check if the map is ready to add sources and layers
return;
}
if (!addedSource) { if (!addedSource) {
let source = this.map.getSource(this.fileId); let source = this.map.getSource(this.fileId);
@@ -138,8 +142,12 @@ export class GPXLayer {
this.map.off('mouseleave', this.fileId, toDefaultCursor); this.map.off('mouseleave', this.fileId, toDefaultCursor);
this.map.off('style.load', this.updateBinded); this.map.off('style.load', this.updateBinded);
if (this.map.getLayer(this.fileId)) {
this.map.removeLayer(this.fileId); this.map.removeLayer(this.fileId);
}
if (this.map.getSource(this.fileId)) {
this.map.removeSource(this.fileId); this.map.removeSource(this.fileId);
}
this.markers.forEach((marker) => { this.markers.forEach((marker) => {
marker.remove(); marker.remove();

View File

@@ -8,28 +8,75 @@
import { Layers } from 'lucide-svelte'; import { Layers } from 'lucide-svelte';
import { import { basemaps, overlays, opacities } from '$lib/assets/layers';
basemaps, import { settings } from '$lib/db';
basemapTree,
overlays,
overlayTree,
opacities,
defaultBasemap
} from '$lib/assets/layers';
import { map } from '$lib/stores'; import { map } from '$lib/stores';
import { get, writable } from 'svelte/store';
const {
currentBasemap,
previousBasemap,
currentOverlays,
selectedBasemapTree,
selectedOverlayTree
} = settings;
$: if ($map) { $: if ($map) {
$map.on('load', () => { // Set style depending on the current basemap
$map.setStyle(basemaps[defaultBasemap]); $map.setStyle(basemaps[$currentBasemap], {
diff: false
}); });
} }
let addOverlayLayer: { [key: string]: () => void } = {}; $: if ($map) {
// Add or remove overlay layers depending on the current overlays
// Object.keys(overlays).forEach((id) => {
// if ($currentOverlays.includes(id)) {
// if (!addOverlayLayer.hasOwnProperty(id)) {
// addOverlayLayer[id] = addOverlayLayerForId(id);
// }
// if (!$map.getLayer(id)) {
// addOverlayLayer[id]();
// $map.on('style.load', addOverlayLayer[id]);
// }
// } else if ($map.getLayer(id)) {
// $map.removeLayer(id);
// $map.off('style.load', addOverlayLayer[id]);
// }
// });
console.log($currentOverlays);
}
let selectedBasemap = writable(get(currentBasemap));
selectedBasemap.subscribe((value) => {
// Updates coming from radio buttons
if (value !== get(currentBasemap)) {
previousBasemap.set(get(currentBasemap));
currentBasemap.set(value);
}
});
currentBasemap.subscribe((value) => {
// Updates coming from the database, or from the user swapping basemaps
selectedBasemap.set(value);
});
let selectedOverlays = writable(get(currentOverlays));
selectedOverlays.subscribe((value) => {
// Updates coming from checkboxes
if (value != get(currentOverlays)) {
currentOverlays.set(value);
}
});
currentOverlays.subscribe((value) => {
// Updates coming from the database, or from the user toggling overlays
selectedOverlays.set(value);
});
let addOverlayLayer: { [key: string]: () => void } = {};
function addOverlayLayerForId(id: string) { function addOverlayLayerForId(id: string) {
return () => { return () => {
if ($map) { if ($map) {
try {
if (!$map.getSource(id)) { if (!$map.getSource(id)) {
$map.addSource(id, overlays[id]); $map.addSource(id, overlays[id]);
} }
@@ -45,6 +92,9 @@
: {}) : {})
} }
}); });
} catch (e) {
// No reliable way to check if the map is ready to add sources and layers
}
} }
}; };
} }
@@ -63,37 +113,18 @@
<div class="h-fit"> <div class="h-fit">
<div class="p-2"> <div class="p-2">
<LayerTree <LayerTree
layerTree={basemapTree} layerTree={$selectedBasemapTree}
name="basemaps" name="basemaps"
onValueChange={(id) => { bind:selected={$selectedBasemap}
if ($map) {
$map.setStyle(basemaps[id], {
diff: false
});
}
}}
/> />
</div> </div>
<Separator class="w-full" /> <Separator class="w-full" />
<div class="p-2"> <div class="p-2">
<LayerTree <LayerTree
layerTree={overlayTree} layerTree={$selectedOverlayTree}
name="overlays" name="overlays"
multiple={true} multiple={true}
onValueChange={(id, checked) => { bind:checked={$selectedOverlays}
if (!addOverlayLayer.hasOwnProperty(id)) {
addOverlayLayer[id] = addOverlayLayerForId(id);
}
if ($map) {
if (checked) {
addOverlayLayer[id]();
$map.on('style.load', addOverlayLayer[id]);
} else {
$map.removeLayer(id);
$map.off('style.load', addOverlayLayer[id]);
}
}
}}
/> />
</div> </div>
<Separator class="w-full" /> <Separator class="w-full" />

View File

@@ -9,16 +9,21 @@
import { Settings } from 'lucide-svelte'; import { Settings } from 'lucide-svelte';
import { import { basemapTree, overlayTree, type CollapsedInfoTreeType } from '$lib/assets/layers';
basemaps, import { settings } from '$lib/db';
basemapTree,
overlays,
overlayTree,
opacities,
defaultBasemap
} from '$lib/assets/layers';
import { _ } from 'svelte-i18n'; import { _ } from 'svelte-i18n';
const { selectedBasemapTree, selectedOverlayTree } = settings;
let checkedBasemaps: CollapsedInfoTreeType<{ [key: string]: boolean }> = {
self: {},
children: {}
};
let checkedOverlays: CollapsedInfoTreeType<{ [key: string]: boolean }> = {
self: {},
children: {}
};
</script> </script>
<Sheet.Root> <Sheet.Root>
@@ -43,9 +48,10 @@
layerTree={basemapTree} layerTree={basemapTree}
name="basemapSettings" name="basemapSettings"
multiple={true} multiple={true}
onValueChange={(id) => { onValueChange={(id, checked) => {
// TODO console.log('basemap', id, checked);
}} }}
bind:checked={checkedBasemaps}
/> />
</ScrollArea> </ScrollArea>
<Separator /> <Separator />
@@ -55,8 +61,9 @@
name="overlaySettings" name="overlaySettings"
multiple={true} multiple={true}
onValueChange={(id, checked) => { onValueChange={(id, checked) => {
// TODO console.log('overlay', id, checked);
}} }}
bind:checked={checkedOverlays}
/> />
</ScrollArea> </ScrollArea>
</Accordion.Content> </Accordion.Content>

View File

@@ -1,19 +1,25 @@
<script lang="ts"> <script lang="ts">
import LayerTreeNode from './LayerTreeNode.svelte'; import LayerTreeNode from './LayerTreeNode.svelte';
import { type CollapsedInfoTreeType, type LayerTreeType } from '$lib/assets/layers'; import {
type CheckedInfoTreeType,
type CollapsedInfoTreeType,
type LayerTreeType
} from '$lib/assets/layers';
export let layerTree: LayerTreeType; export let layerTree: LayerTreeType;
export let name: string; export let name: string;
export let selected: string | undefined = undefined;
export let multiple: boolean = false; export let multiple: boolean = false;
export let onValueChange: (id: string, checked: boolean) => void; let open: CollapsedInfoTreeType<boolean> = {
let open: CollapsedInfoTreeType = {
self: true, self: true,
children: {} children: {}
}; };
export let checked: CheckedInfoTreeType = {};
</script> </script>
<form>
<fieldset class="min-w-64"> <fieldset class="min-w-64">
<LayerTreeNode {name} node={layerTree} {multiple} {onValueChange} bind:open /> <LayerTreeNode {name} node={layerTree} bind:selected {multiple} bind:open bind:checked />
</fieldset> </fieldset>
</form>

View File

@@ -6,24 +6,20 @@
import { ChevronDown, ChevronUp } from 'lucide-svelte'; import { ChevronDown, ChevronUp } from 'lucide-svelte';
import { type CollapsedInfoTreeType, type LayerTreeType } from '$lib/assets/layers'; import {
type CheckedInfoTreeType,
type CollapsedInfoTreeType,
type LayerTreeType
} from '$lib/assets/layers';
import { _ } from 'svelte-i18n'; import { _ } from 'svelte-i18n';
export let name: string; export let name: string;
export let node: LayerTreeType; export let node: LayerTreeType;
export let selected: string | undefined = undefined;
export let multiple: boolean = false; export let multiple: boolean = false;
export let onValueChange: (id: string, checked: boolean) => void; export let open: CollapsedInfoTreeType<boolean>;
let checked: { [key: string]: boolean } = {};
if (multiple && Array.isArray(node)) {
node.forEach((id) => {
checked[id] = false;
});
}
export let open: CollapsedInfoTreeType;
if (!Array.isArray(node)) { if (!Array.isArray(node)) {
Object.keys(node).forEach((id) => { Object.keys(node).forEach((id) => {
if (!open.children.hasOwnProperty(id)) { if (!open.children.hasOwnProperty(id)) {
@@ -34,6 +30,23 @@
} }
}); });
} }
export let checked: CheckedInfoTreeType;
if (Array.isArray(node)) {
if (multiple) {
node.forEach((id) => {
if (!checked.hasOwnProperty(id)) {
checked[id] = false;
}
});
}
} else {
Object.keys(node).forEach((id) => {
if (!checked.hasOwnProperty(id)) {
checked[id] = {};
}
});
}
</script> </script>
{#if Array.isArray(node)} {#if Array.isArray(node)}
@@ -42,27 +55,18 @@
<div class="flex flex-row items-center gap-2"> <div class="flex flex-row items-center gap-2">
{#if multiple} {#if multiple}
<Checkbox <Checkbox
{id} id="{name}-{id}"
{name} {name}
value={id} value={id}
bind:checked={checked[id]} bind:checked={checked[id]}
on:click={() => {
onValueChange(id, !checked[id]);
}}
class="scale-90" class="scale-90"
/> />
{:else} {:else}
<input <input id="{name}-{id}" type="radio" {name} value={id} bind:group={selected} />
type="radio"
{id}
{name}
value={id}
on:change={() => {
onValueChange(id, true);
}}
/>
{/if} {/if}
<Label for={id}>{$_(`layers.label.${id}`)}</Label> <Label for="{name}-{id}" class="flex flex-row items-center gap-1"
>{$_(`layers.label.${id}`)}</Label
>
</div> </div>
{/each} {/each}
</div> </div>
@@ -87,9 +91,10 @@
<svelte:self <svelte:self
node={node[id]} node={node[id]}
{name} {name}
bind:selected
{multiple} {multiple}
{onValueChange}
bind:open={open.children[id]} bind:open={open.children[id]}
bind:checked={checked[id]}
/> />
</Collapsible.Content> </Collapsible.Content>
</Collapsible.Root> </Collapsible.Root>

View File

@@ -4,6 +4,7 @@ import { type FreezedObject, type Patch, produceWithPatches, applyPatches } from
import { writable, get, derived, type Readable, type Writable } from 'svelte/store'; import { writable, get, derived, type Readable, type Writable } from 'svelte/store';
import { fileOrder, selectedFiles } from './stores'; import { fileOrder, selectedFiles } from './stores';
import { mode } from 'mode-watcher'; import { mode } from 'mode-watcher';
import { defaultBasemap, defaultBasemapTree, defaultOverlayTree } from './assets/layers';
class Database extends Dexie { class Database extends Dexie {
@@ -266,6 +267,7 @@ function dexieSettingStore(setting: string, initial: any): Writable<any> {
let store = writable(initial); let store = writable(initial);
liveQuery(() => db.settings.get(setting)).subscribe(value => { liveQuery(() => db.settings.get(setting)).subscribe(value => {
if (value !== undefined) { if (value !== undefined) {
console.log('setting', setting, 'changed to', value);
store.set(value); store.set(value);
} }
}); });
@@ -293,4 +295,10 @@ export const settings = {
routing: dexieSettingStore('routing', true), routing: dexieSettingStore('routing', true),
routingProfile: dexieSettingStore('routingProfile', 'bike'), routingProfile: dexieSettingStore('routingProfile', 'bike'),
privateRoads: dexieSettingStore('privateRoads', false), privateRoads: dexieSettingStore('privateRoads', false),
currentBasemap: dexieSettingStore('currentBasemap', defaultBasemap),
previousBasemap: dexieSettingStore('previousBasemap', defaultBasemap),
selectedBasemapTree: dexieSettingStore('selectedBasemapTree', defaultBasemapTree),
currentOverlays: dexieSettingStore('currentOverlays', {}),
previousOverlays: dexieSettingStore('previousOverlays', {}),
selectedOverlayTree: dexieSettingStore('selectedOverlayTree', defaultOverlayTree),
}; };