2025-11-11 12:11:38 +01:00
|
|
|
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-11 12:11:38 +01:00
|
|
|
import { isSelected, remove, removeByPrefix, toggle } 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';
|
2025-11-11 12:11:38 +01:00
|
|
|
|
|
|
|
|
const { currentOverlays, previousOverlays, selectedOverlayTree } = settings;
|
|
|
|
|
|
|
|
|
|
export type CustomOverlay = {
|
|
|
|
|
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-11 12:11:38 +01:00
|
|
|
|
2025-11-12 12:47:26 +01:00
|
|
|
init() {
|
2025-11-11 12:11:38 +01:00
|
|
|
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-11 12:11:38 +01:00
|
|
|
return new Promise((resolve) => {
|
2025-11-12 12:47:26 +01:00
|
|
|
map.onLoad(() => {
|
2025-11-11 12:11:38 +01:00
|
|
|
resolve();
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
addOrUpdateOverlay(overlay: CustomOverlay) {
|
|
|
|
|
if (!overlay.id || !overlay.tileUrls || overlay.tileUrls.length === 0) {
|
|
|
|
|
throw new Error('Overlay must have an id 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;
|
|
|
|
|
});
|
2025-11-11 12:11:38 +01:00
|
|
|
|
|
|
|
|
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,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
overlayTree.overlays.world[overlay.id] = true;
|
|
|
|
|
|
|
|
|
|
selectedOverlayTree.update((selected) => {
|
|
|
|
|
selected.overlays.world[overlay.id] = true;
|
|
|
|
|
return selected;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const current = get(currentOverlays);
|
|
|
|
|
if (current && isSelected(current, overlay.id)) {
|
|
|
|
|
try {
|
2025-11-12 12:47:26 +01:00
|
|
|
get(map)?.removeImport(overlay.id);
|
2025-11-11 12:11:38 +01:00
|
|
|
} catch (e) {
|
|
|
|
|
// No reliable way to check if the map is ready to remove sources and layers
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
currentOverlays.update((current) => {
|
|
|
|
|
current.overlays.world[overlay.id] = true;
|
|
|
|
|
return current;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
removeOverlaysWithPrefix(prefix: string) {
|
|
|
|
|
prefix = this.getOverlayId(prefix);
|
|
|
|
|
|
2025-11-12 12:47:26 +01:00
|
|
|
currentOverlays.update((current) => {
|
|
|
|
|
removeByPrefix(current, prefix);
|
|
|
|
|
return current;
|
2025-11-11 12:11:38 +01:00
|
|
|
});
|
2025-11-12 12:47:26 +01:00
|
|
|
previousOverlays.update((previous) => {
|
|
|
|
|
removeByPrefix(previous, prefix);
|
|
|
|
|
return previous;
|
2025-11-11 12:11:38 +01:00
|
|
|
});
|
2025-11-12 12:47:26 +01:00
|
|
|
selectedOverlayTree.update((overlayTree) => {
|
|
|
|
|
removeByPrefix(overlayTree, prefix);
|
|
|
|
|
return overlayTree;
|
2025-11-11 12:11:38 +01:00
|
|
|
});
|
|
|
|
|
Object.keys(overlays).forEach((id) => {
|
|
|
|
|
if (id.startsWith(prefix)) {
|
|
|
|
|
delete overlays[id];
|
|
|
|
|
}
|
|
|
|
|
});
|
2025-11-12 12:47:26 +01:00
|
|
|
removeByPrefix(overlayTree, prefix);
|
|
|
|
|
this._overlays.update(($overlays) => {
|
|
|
|
|
$overlays.forEach((_, id) => {
|
|
|
|
|
if (id.startsWith(prefix)) {
|
|
|
|
|
$overlays.delete(id);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return $overlays;
|
2025-11-11 12:11:38 +01:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
toggleOverlay(id: string) {
|
|
|
|
|
id = this.getOverlayId(id);
|
|
|
|
|
|
|
|
|
|
currentOverlays.update((overlays) => {
|
|
|
|
|
toggle(overlays, id);
|
|
|
|
|
return overlays;
|
|
|
|
|
});
|
|
|
|
|
if (!isSelected(get(selectedOverlayTree), id)) {
|
|
|
|
|
selectedOverlayTree.update((overlays) => {
|
|
|
|
|
toggle(overlays, id);
|
|
|
|
|
return overlays;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-12 12:47:26 +01:00
|
|
|
isLayerFromExtension = derived(this._overlays, ($overlays) => {
|
|
|
|
|
return (id: string) => $overlays.has(id);
|
|
|
|
|
});
|
2025-11-11 12:11:38 +01:00
|
|
|
|
2025-11-12 12:47:26 +01:00
|
|
|
getLayerName = derived(this._overlays, ($overlays) => {
|
|
|
|
|
return (id: string) => $overlays.get(id)?.name || '';
|
|
|
|
|
});
|
2025-11-11 12:11:38 +01:00
|
|
|
|
|
|
|
|
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-11 12:11:38 +01:00
|
|
|
currentOverlays.update((overlays) => {
|
2025-11-12 12:47:26 +01:00
|
|
|
ids.forEach((id) => {
|
2025-11-11 12:11:38 +01:00
|
|
|
remove(overlays, id);
|
|
|
|
|
});
|
|
|
|
|
return overlays;
|
|
|
|
|
});
|
|
|
|
|
previousOverlays.update((overlays) => {
|
2025-11-12 12:47:26 +01:00
|
|
|
ids.forEach((id) => {
|
2025-11-11 12:11:38 +01:00
|
|
|
remove(overlays, id);
|
|
|
|
|
});
|
|
|
|
|
return overlays;
|
|
|
|
|
});
|
|
|
|
|
selectedOverlayTree.update((overlays) => {
|
2025-11-12 12:47:26 +01:00
|
|
|
ids.forEach((id) => {
|
2025-11-11 12:11:38 +01:00
|
|
|
remove(overlays, id);
|
|
|
|
|
});
|
|
|
|
|
return overlays;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-12 12:47:26 +01:00
|
|
|
export const extensionAPI = new ExtensionAPI();
|