Files
gpx.studio/website/src/lib/components/map/layer-control/extension-api.ts

214 lines
6.7 KiB
TypeScript
Raw Normal View History

import { settings } from '$lib/logic/settings';
2025-11-12 12:47:26 +01:00
import { derived, get, writable, type Writable } from 'svelte/store';
2025-11-12 14:48:17 +01:00
import { isSelected, remove, removeAll } from './utils';
import { overlays, overlayTree } from '$lib/assets/layers';
import { browser } from '$app/environment';
2025-11-12 12:47:26 +01:00
import { map } from '$lib/components/map/map';
const { currentOverlays, previousOverlays, selectedOverlayTree } = settings;
export type CustomOverlay = {
2025-11-25 19:18:54 +01:00
extensionName: string;
id: string;
name: string;
tileUrls: string[];
maxZoom?: number;
};
export class ExtensionAPI {
2025-11-12 12:47:26 +01:00
private _overlays: Writable<Map<string, CustomOverlay>> = writable(new Map());
2025-11-12 12:47:26 +01:00
init() {
if (browser && !window.hasOwnProperty('gpxstudio')) {
Object.defineProperty(window, 'gpxstudio', {
value: this,
});
addEventListener('beforeunload', () => {
this.destroy();
});
}
}
2025-11-12 12:47:26 +01:00
ensureLoaded(): Promise<void> {
2025-11-12 14:48:17 +01:00
let unsubscribe: () => void;
const promise = new Promise<void>((resolve) => {
2025-11-12 12:47:26 +01:00
map.onLoad(() => {
2025-11-12 14:48:17 +01:00
unsubscribe = currentOverlays.subscribe((current) => {
if (current) {
resolve();
}
});
});
});
2025-11-12 14:48:17 +01:00
promise.finally(() => {
unsubscribe?.();
});
return promise;
}
addOrUpdateOverlay(overlay: CustomOverlay) {
2025-11-25 19:18:54 +01:00
if (
!overlay.extensionName ||
!overlay.id ||
!overlay.name ||
!overlay.tileUrls ||
overlay.tileUrls.length === 0
) {
throw new Error(
'Overlay must have an extensionName, id, name, and at least one tile URL.'
);
}
overlay.id = this.getOverlayId(overlay.id);
2025-11-12 12:47:26 +01:00
this._overlays.update(($overlays) => {
$overlays.set(overlay.id, overlay);
return $overlays;
});
overlays[overlay.id] = {
version: 8,
sources: {
[overlay.id]: {
type: 'raster',
tiles: overlay.tileUrls,
tileSize: overlay.tileUrls.some((url) => url.includes('512')) ? 512 : 256,
maxzoom: overlay.maxZoom ?? 22,
},
},
layers: [
{
id: overlay.id,
type: 'raster',
source: overlay.id,
},
],
};
2025-11-25 19:18:54 +01:00
if (!overlayTree.overlays.hasOwnProperty(overlay.extensionName)) {
overlayTree.overlays[overlay.extensionName] = {};
}
overlayTree.overlays[overlay.extensionName][overlay.id] = true;
selectedOverlayTree.update((selected) => {
2025-11-25 19:18:54 +01:00
if (!selected.overlays.hasOwnProperty(overlay.extensionName)) {
selected.overlays[overlay.extensionName] = {};
}
selected.overlays[overlay.extensionName][overlay.id] = true;
return selected;
});
const current = get(currentOverlays);
let show = false;
if (current && isSelected(current, overlay.id)) {
2025-11-12 14:48:17 +01:00
show = true;
try {
2025-11-12 12:47:26 +01:00
get(map)?.removeImport(overlay.id);
} catch (e) {
// No reliable way to check if the map is ready to remove sources and layers
}
}
currentOverlays.update((current) => {
2025-11-25 19:18:54 +01:00
if (!current.overlays.hasOwnProperty(overlay.extensionName)) {
current.overlays[overlay.extensionName] = {};
}
current.overlays[overlay.extensionName][overlay.id] = show;
return current;
});
}
2025-11-12 14:48:17 +01:00
filterOverlays(ids: string[]) {
ids = ids.map((id) => this.getOverlayId(id));
const idsToRemove = Array.from(get(this._overlays).keys()).filter(
(id) => !ids.includes(id)
);
2025-11-12 12:47:26 +01:00
currentOverlays.update((current) => {
2025-11-12 14:48:17 +01:00
removeAll(current, idsToRemove);
2025-11-12 12:47:26 +01:00
return current;
});
2025-11-12 12:47:26 +01:00
previousOverlays.update((previous) => {
2025-11-12 14:48:17 +01:00
removeAll(previous, idsToRemove);
2025-11-12 12:47:26 +01:00
return previous;
});
2025-11-12 14:48:17 +01:00
selectedOverlayTree.update((selected) => {
removeAll(selected, idsToRemove);
return selected;
});
Object.keys(overlays).forEach((id) => {
2025-11-12 14:48:17 +01:00
if (idsToRemove.includes(id)) {
delete overlays[id];
}
});
2025-11-12 14:48:17 +01:00
removeAll(overlayTree, idsToRemove);
2025-11-12 12:47:26 +01:00
this._overlays.update(($overlays) => {
$overlays.forEach((_, id) => {
2025-11-12 14:48:17 +01:00
if (idsToRemove.includes(id)) {
2025-11-12 12:47:26 +01:00
$overlays.delete(id);
}
});
return $overlays;
});
}
2025-11-25 19:18:54 +01:00
updateOverlaysOrder(ids: string[]) {
ids = ids.map((id) => this.getOverlayId(id));
selectedOverlayTree.update((selected) => {
let isSelected: Record<string, boolean> = {};
ids.forEach((id) => {
const overlay = get(this._overlays).get(id);
if (
overlay &&
selected.overlays.hasOwnProperty(overlay.extensionName) &&
selected.overlays[overlay.extensionName].hasOwnProperty(id)
) {
isSelected[id] = selected.overlays[overlay.extensionName][id];
delete selected.overlays[overlay.extensionName][id];
}
});
Object.entries(isSelected).forEach(([id, value]) => {
const overlay = get(this._overlays).get(id)!;
selected.overlays[overlay.extensionName][id] = value;
});
return selected;
});
}
2025-11-12 12:47:26 +01:00
isLayerFromExtension = derived(this._overlays, ($overlays) => {
return (id: string) => $overlays.has(id);
});
2025-11-12 12:47:26 +01:00
getLayerName = derived(this._overlays, ($overlays) => {
return (id: string) => $overlays.get(id)?.name || '';
});
private getOverlayId(id: string): string {
return `extension-${id}`;
}
private destroy() {
2025-11-12 12:47:26 +01:00
const ids = Array.from(get(this._overlays).keys());
2025-11-12 14:48:17 +01:00
currentOverlays.update((current) => {
2025-11-12 12:47:26 +01:00
ids.forEach((id) => {
2025-11-12 14:48:17 +01:00
remove(current, id);
});
2025-11-12 14:48:17 +01:00
return current;
});
2025-11-12 14:48:17 +01:00
previousOverlays.update((previous) => {
2025-11-12 12:47:26 +01:00
ids.forEach((id) => {
2025-11-12 14:48:17 +01:00
remove(previous, id);
});
2025-11-12 14:48:17 +01:00
return previous;
});
2025-11-12 14:48:17 +01:00
selectedOverlayTree.update((selected) => {
2025-11-12 12:47:26 +01:00
ids.forEach((id) => {
2025-11-12 14:48:17 +01:00
remove(selected, id);
});
2025-11-12 14:48:17 +01:00
return selected;
});
}
}
2025-11-12 12:47:26 +01:00
export const extensionAPI = new ExtensionAPI();