mirror of
https://github.com/gpxstudio/gpx.studio.git
synced 2025-12-02 10:02:12 +00:00
update extension api
This commit is contained in:
@@ -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}
|
||||||
|
|||||||
@@ -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}
|
||||||
|
|||||||
@@ -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();
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
13
website/src/routes/[[language]]/app/+layout.svelte
Normal file
13
website/src/routes/[[language]]/app/+layout.svelte
Normal 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()}
|
||||||
Reference in New Issue
Block a user