mirror of
https://github.com/gpxstudio/gpx.studio.git
synced 2025-09-02 16:52:31 +00:00
progress with layer settings
This commit is contained in:
@@ -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']
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -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) {
|
||||||
|
@@ -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();
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
@@ -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
|
||||||
|
@@ -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();
|
||||||
|
@@ -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" />
|
||||||
|
@@ -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>
|
||||||
|
@@ -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>
|
||||||
|
@@ -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>
|
||||||
|
@@ -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),
|
||||||
};
|
};
|
Reference in New Issue
Block a user