update extension api

This commit is contained in:
vcoppe
2025-11-12 12:47:26 +01:00
parent 50a5cb23f5
commit 110f23bdf1
5 changed files with 63 additions and 42 deletions

View File

@@ -20,7 +20,7 @@
import CustomLayers from './CustomLayers.svelte'; import CustomLayers from './CustomLayers.svelte';
import { settings } from '$lib/logic/settings'; import { settings } from '$lib/logic/settings';
import { untrack } from 'svelte'; import { untrack } from 'svelte';
import { extensionAPI } from './extension-api'; import { extensionAPI } from '$lib/components/map/layer-control/extension-api';
const { const {
selectedBasemapTree, selectedBasemapTree,
@@ -33,6 +33,8 @@
opacities, opacities,
} = settings; } = settings;
const { isLayerFromExtension, getLayerName } = extensionAPI;
let { open = $bindable() }: { open: boolean } = $props(); let { open = $bindable() }: { open: boolean } = $props();
let accordionValue: string | undefined = $state(undefined); let accordionValue: string | undefined = $state(undefined);
@@ -161,8 +163,8 @@
<Select.Trigger class="h-8 mr-1 w-full"> <Select.Trigger class="h-8 mr-1 w-full">
{#if selectedOverlay} {#if selectedOverlay}
{#if isSelected($selectedOverlayTree, selectedOverlay)} {#if isSelected($selectedOverlayTree, selectedOverlay)}
{#if extensionAPI.isLayerFromExtension(selectedOverlay)} {#if $isLayerFromExtension(selectedOverlay)}
{extensionAPI.getLayerName(selectedOverlay)} {$getLayerName(selectedOverlay)}
{:else} {:else}
{i18n._(`layers.label.${selectedOverlay}`)} {i18n._(`layers.label.${selectedOverlay}`)}
{/if} {/if}
@@ -175,8 +177,8 @@
{#each Object.keys(overlays) as id} {#each Object.keys(overlays) as id}
{#if isSelected($selectedOverlayTree, id)} {#if isSelected($selectedOverlayTree, id)}
<Select.Item value={id}> <Select.Item value={id}>
{#if extensionAPI.isLayerFromExtension(id)} {#if $isLayerFromExtension(id)}
{extensionAPI.getLayerName(id)} {$getLayerName(id)}
{:else} {:else}
{i18n._(`layers.label.${id}`)} {i18n._(`layers.label.${id}`)}
{/if} {/if}

View File

@@ -26,6 +26,7 @@
} = $props(); } = $props();
const { customLayers } = settings; const { customLayers } = settings;
const { isLayerFromExtension, getLayerName } = extensionAPI;
$effect.pre(() => { $effect.pre(() => {
if (checked !== undefined) { if (checked !== undefined) {
@@ -73,8 +74,8 @@
<Label for="{name}-{id}" class="flex flex-row items-center gap-1"> <Label for="{name}-{id}" class="flex flex-row items-center gap-1">
{#if $customLayers.hasOwnProperty(id)} {#if $customLayers.hasOwnProperty(id)}
{$customLayers[id].name} {$customLayers[id].name}
{:else if extensionAPI.isLayerFromExtension(id)} {:else if $isLayerFromExtension(id)}
{extensionAPI.getLayerName(id)} {$getLayerName(id)}
{:else} {:else}
{i18n._(`layers.label.${id}`)} {i18n._(`layers.label.${id}`)}
{/if} {/if}

View File

@@ -1,9 +1,9 @@
import { map, type MapboxGLMap } from '$lib/components/map/map';
import { settings } from '$lib/logic/settings'; import { settings } from '$lib/logic/settings';
import { get } from 'svelte/store'; import { derived, get, writable, type Writable } from 'svelte/store';
import { isSelected, remove, removeByPrefix, toggle } from './utils'; import { isSelected, remove, removeByPrefix, toggle } from './utils';
import { overlays, overlayTree } from '$lib/assets/layers'; import { overlays, overlayTree } from '$lib/assets/layers';
import { browser } from '$app/environment'; import { browser } from '$app/environment';
import { map } from '$lib/components/map/map';
const { currentOverlays, previousOverlays, selectedOverlayTree } = settings; const { currentOverlays, previousOverlays, selectedOverlayTree } = settings;
@@ -15,11 +15,9 @@ export type CustomOverlay = {
}; };
export class ExtensionAPI { export class ExtensionAPI {
private _map: MapboxGLMap; private _overlays: Writable<Map<string, CustomOverlay>> = writable(new Map());
private _overlays: Map<string, CustomOverlay> = new Map();
constructor(map: MapboxGLMap) { init() {
this._map = map;
if (browser && !window.hasOwnProperty('gpxstudio')) { if (browser && !window.hasOwnProperty('gpxstudio')) {
Object.defineProperty(window, 'gpxstudio', { Object.defineProperty(window, 'gpxstudio', {
value: this, value: this,
@@ -30,9 +28,9 @@ export class ExtensionAPI {
} }
} }
async ensureLoaded(): Promise<void> { ensureLoaded(): Promise<void> {
return new Promise((resolve) => { return new Promise((resolve) => {
this._map.onLoad(() => { map.onLoad(() => {
resolve(); resolve();
}); });
}); });
@@ -44,7 +42,10 @@ export class ExtensionAPI {
} }
overlay.id = this.getOverlayId(overlay.id); overlay.id = this.getOverlayId(overlay.id);
this._overlays.set(overlay.id, overlay); this._overlays.update(($overlays) => {
$overlays.set(overlay.id, overlay);
return $overlays;
});
overlays[overlay.id] = { overlays[overlay.id] = {
version: 8, version: 8,
@@ -75,7 +76,7 @@ export class ExtensionAPI {
const current = get(currentOverlays); const current = get(currentOverlays);
if (current && isSelected(current, overlay.id)) { if (current && isSelected(current, overlay.id)) {
try { try {
get(this._map)?.removeImport(overlay.id); get(map)?.removeImport(overlay.id);
} catch (e) { } catch (e) {
// No reliable way to check if the map is ready to remove sources and layers // No reliable way to check if the map is ready to remove sources and layers
} }
@@ -90,27 +91,31 @@ export class ExtensionAPI {
removeOverlaysWithPrefix(prefix: string) { removeOverlaysWithPrefix(prefix: string) {
prefix = this.getOverlayId(prefix); prefix = this.getOverlayId(prefix);
currentOverlays.update((overlays) => { currentOverlays.update((current) => {
removeByPrefix(overlays, prefix); removeByPrefix(current, prefix);
return overlays; return current;
}); });
previousOverlays.update((overlays) => { previousOverlays.update((previous) => {
removeByPrefix(overlays, prefix); removeByPrefix(previous, prefix);
return overlays; return previous;
}); });
selectedOverlayTree.update((overlays) => { selectedOverlayTree.update((overlayTree) => {
removeByPrefix(overlays, prefix); removeByPrefix(overlayTree, prefix);
return overlays; return overlayTree;
}); });
Object.keys(overlays).forEach((id) => { Object.keys(overlays).forEach((id) => {
if (id.startsWith(prefix)) { if (id.startsWith(prefix)) {
delete overlays[id]; delete overlays[id];
} }
}); });
Object.keys(overlayTree.overlays.world).forEach((id) => { removeByPrefix(overlayTree, prefix);
if (id.startsWith(prefix)) { this._overlays.update(($overlays) => {
delete overlayTree.overlays.world[id]; $overlays.forEach((_, id) => {
} if (id.startsWith(prefix)) {
$overlays.delete(id);
}
});
return $overlays;
}); });
} }
@@ -129,34 +134,34 @@ export class ExtensionAPI {
} }
} }
isLayerFromExtension(id: string): boolean { isLayerFromExtension = derived(this._overlays, ($overlays) => {
return this._overlays.has(id); return (id: string) => $overlays.has(id);
} });
getLayerName(id: string): string { getLayerName = derived(this._overlays, ($overlays) => {
const overlay = this._overlays.get(id); return (id: string) => $overlays.get(id)?.name || '';
return overlay ? overlay.name : ''; });
}
private getOverlayId(id: string): string { private getOverlayId(id: string): string {
return `extension-${id}`; return `extension-${id}`;
} }
private destroy() { private destroy() {
const ids = Array.from(get(this._overlays).keys());
currentOverlays.update((overlays) => { currentOverlays.update((overlays) => {
this._overlays.forEach((_, id) => { ids.forEach((id) => {
remove(overlays, id); remove(overlays, id);
}); });
return overlays; return overlays;
}); });
previousOverlays.update((overlays) => { previousOverlays.update((overlays) => {
this._overlays.forEach((_, id) => { ids.forEach((id) => {
remove(overlays, id); remove(overlays, id);
}); });
return overlays; return overlays;
}); });
selectedOverlayTree.update((overlays) => { selectedOverlayTree.update((overlays) => {
this._overlays.forEach((_, id) => { ids.forEach((id) => {
remove(overlays, id); remove(overlays, id);
}); });
return overlays; return overlays;
@@ -164,4 +169,4 @@ export class ExtensionAPI {
} }
} }
export const extensionAPI = new ExtensionAPI(map); export const extensionAPI = new ExtensionAPI();

View File

@@ -71,7 +71,7 @@ export function removeByPrefix(node: LayerTreeType, prefix: string) {
if (key.startsWith(prefix)) { if (key.startsWith(prefix)) {
delete node[key]; delete node[key];
} else if (typeof node[key] !== 'boolean') { } else if (typeof node[key] !== 'boolean') {
remove(node[key], prefix); removeByPrefix(node[key], prefix);
} }
}); });
return node; return node;

View File

@@ -0,0 +1,13 @@
<script lang="ts">
import { browser } from '$app/environment';
import { extensionAPI } from '$lib/components/map/layer-control/extension-api';
import type { Snippet } from 'svelte';
let { children }: { children: Snippet } = $props();
if (browser) {
extensionAPI.init();
}
</script>
{@render children()}