import BaseLayer from "../../Models/BaseLayer"; import {Store, Stores} from "../UIEventSource"; import Loc from "../../Models/Loc"; import {GeoOperations} from "../GeoOperations"; import * as editorlayerindex from "../../assets/editor-layer-index.json"; import * as L from "leaflet"; import {TileLayer} from "leaflet"; import * as X from "leaflet-providers"; import {Utils} from "../../Utils"; import {AvailableBaseLayersObj} from "./AvailableBaseLayers"; import {BBox} from "../BBox"; export default class AvailableBaseLayersImplementation implements AvailableBaseLayersObj { public readonly osmCarto: BaseLayer = { id: "osm", name: "OpenStreetMap", layer: () => AvailableBaseLayersImplementation.CreateBackgroundLayer("osm", "OpenStreetMap", "https://tile.openstreetmap.org/{z}/{x}/{y}.png", "OpenStreetMap", "https://openStreetMap.org/copyright", 19, false, false), feature: null, max_zoom: 19, min_zoom: 0, isBest: true, // Of course, OpenStreetMap is the best map! category: "osmbasedmap" } public readonly layerOverview = AvailableBaseLayersImplementation.LoadRasterIndex().concat(AvailableBaseLayersImplementation.LoadProviderIndex()); public readonly globalLayers = this.layerOverview.filter(layer => layer.feature?.geometry === undefined || layer.feature?.geometry === null) public readonly localLayers = this.layerOverview.filter(layer => layer.feature?.geometry !== undefined && layer.feature?.geometry !== null) private static LoadRasterIndex(): BaseLayer[] { const layers: BaseLayer[] = [] // @ts-ignore const features = editorlayerindex.features; for (const i in features) { const layer = features[i]; const props = layer.properties; if (props.type === "bing") { // A lot of work to implement - see https://github.com/pietervdvn/MapComplete/issues/648 continue; } if (props.id === "MAPNIK") { // Already added by default continue; } if (props.overlay) { continue; } if (props.url.toLowerCase().indexOf("apikey") > 0) { continue; } if (props.max_zoom < 19) { // We want users to zoom to level 19 when adding a point // If they are on a layer which hasn't enough precision, they can not zoom far enough. This is confusing, so we don't use this layer continue; } if (props.name === undefined) { console.warn("Editor layer index: name not defined on ", props) continue } const leafletLayer: () => TileLayer = () => AvailableBaseLayersImplementation.CreateBackgroundLayer( props.id, props.name, props.url, props.name, props.license_url, props.max_zoom, props.type.toLowerCase() === "wms", props.type.toLowerCase() === "wmts" ) // Note: if layer.geometry is null, there is global coverage for this layer layers.push({ id: props.id, max_zoom: props.max_zoom ?? 19, min_zoom: props.min_zoom ?? 1, name: props.name, layer: leafletLayer, feature: layer.geometry !== null ? layer : null, isBest: props.best ?? false, category: props.category }); } return layers; } private static LoadProviderIndex(): BaseLayer[] { // @ts-ignore X; // Import X to make sure the namespace is not optimized away function l(id: string, name: string): BaseLayer { try { const layer: any = L.tileLayer.provider(id, undefined); return { feature: null, id: id, name: name, layer: () => L.tileLayer.provider(id, { maxNativeZoom: layer.options?.maxZoom, maxZoom: Math.max(layer.options?.maxZoom ?? 19, 21) }), min_zoom: 1, max_zoom: layer.options.maxZoom, category: "osmbasedmap", isBest: false } } catch (e) { console.error("Could not find provided layer", name, e); return null; } } const layers = [ l("Stamen.TonerLite", "Toner Lite (by Stamen)"), l("Stamen.TonerBackground", "Toner Background - no labels (by Stamen)"), l("Stamen.Watercolor", "Watercolor (by Stamen)"), l("Stadia.OSMBright", "Osm Bright (by Stadia)"), l("CartoDB.Positron", "Positron (by CartoDB)"), l("CartoDB.PositronNoLabels", "Positron - no labels (by CartoDB)"), l("CartoDB.Voyager", "Voyager (by CartoDB)"), l("CartoDB.VoyagerNoLabels", "Voyager - no labels (by CartoDB)"), ]; return Utils.NoNull(layers); } /** * Converts a layer from the editor-layer-index into a tilelayer usable by leaflet */ private static CreateBackgroundLayer(id: string, name: string, url: string, attribution: string, attributionUrl: string, maxZoom: number, isWms: boolean, isWMTS?: boolean): TileLayer { url = url.replace("{zoom}", "{z}") .replace("&BBOX={bbox}", "") .replace("&bbox={bbox}", ""); const subdomainsMatch = url.match(/{switch:[^}]*}/) let domains: string[] = []; if (subdomainsMatch !== null) { let domainsStr = subdomainsMatch[0].substr("{switch:".length); domainsStr = domainsStr.substr(0, domainsStr.length - 1); domains = domainsStr.split(","); url = url.replace(/{switch:[^}]*}/, "{s}") } if (isWms) { url = url.replace("&SRS={proj}", ""); url = url.replace("&srs={proj}", ""); const paramaters = ["format", "layers", "version", "service", "request", "styles", "transparent", "version"]; const urlObj = new URL(url); const isUpper = urlObj.searchParams["LAYERS"] !== null; const options = { maxZoom: Math.max(maxZoom ?? 19, 21), maxNativeZoom: maxZoom ?? 19, attribution: attribution + " | ", subdomains: domains, uppercase: isUpper, transparent: false, }; for (const paramater of paramaters) { let p = paramater; if (isUpper) { p = paramater.toUpperCase(); } options[paramater] = urlObj.searchParams.get(p); } if (options.transparent === null) { options.transparent = false; } return L.tileLayer.wms(urlObj.protocol + "//" + urlObj.host + urlObj.pathname, options); } if (attributionUrl) { attribution = `${attribution}`; } return L.tileLayer(url, { attribution: attribution, maxZoom: Math.max(21, maxZoom ?? 19), maxNativeZoom: maxZoom ?? 19, minZoom: 1, // @ts-ignore wmts: isWMTS ?? false, subdomains: domains }); } public AvailableLayersAt(location: Store): Store { return Stores.ListStabilized(location.map( (currentLocation) => { if (currentLocation === undefined) { return this.layerOverview; } return this.CalculateAvailableLayersAt(currentLocation?.lon, currentLocation?.lat); })); } public SelectBestLayerAccordingTo(location: Store, preferedCategory: Store): Store { return this.AvailableLayersAt(location) .map(available => { // First float all 'best layers' to the top available.sort((a, b) => { if (a.isBest && b.isBest) { return 0; } if (!a.isBest) { return 1 } return -1; } ) if (preferedCategory.data === undefined) { return available[0] } let prefered: string [] if (typeof preferedCategory.data === "string") { prefered = [preferedCategory.data] } else { prefered = preferedCategory.data; } prefered.reverse(/*New list, inplace reverse is fine*/); for (const category of prefered) { //Then sort all 'photo'-layers to the top. Stability of the sorting will force a 'best' photo layer on top available.sort((a, b) => { if (a.category === category && b.category === category) { return 0; } if (a.category !== category) { return 1 } return -1; } ) } return available[0] }, [preferedCategory]) } private CalculateAvailableLayersAt(lon: number, lat: number): BaseLayer[] { const availableLayers = [this.osmCarto] if (lon === undefined || lat === undefined) { return availableLayers.concat(this.globalLayers); } const lonlat : [number, number] = [lon, lat]; for (const layerOverviewItem of this.localLayers) { const layer = layerOverviewItem; const bbox = BBox.get(layer.feature) if(!bbox.contains(lonlat)){ continue } if (GeoOperations.inside(lonlat, layer.feature)) { availableLayers.push(layer); } } return availableLayers.concat(this.globalLayers); } }