MapComplete/Logic/Actors/AvailableBaseLayersImplementation.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

309 lines
11 KiB
TypeScript
Raw Normal View History

2022-09-08 21:40:48 +02:00
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"
2021-10-15 14:52:11 +02:00
2021-11-07 16:34:51 +01:00
export default class AvailableBaseLayersImplementation implements AvailableBaseLayersObj {
2022-09-08 21:40:48 +02:00
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",
2021-10-15 14:52:11 +02:00
19,
2022-09-08 21:40:48 +02:00
false,
false
),
feature: null,
max_zoom: 19,
min_zoom: 0,
isBest: true, // Of course, OpenStreetMap is the best map!
category: "osmbasedmap",
}
2021-10-15 14:52:11 +02:00
2022-09-08 21:40:48 +02:00
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
)
2021-10-15 14:52:11 +02:00
private static LoadRasterIndex(): BaseLayer[] {
const layers: BaseLayer[] = []
// @ts-ignore
2022-09-08 21:40:48 +02:00
const features = editorlayerindex.features
2021-10-15 14:52:11 +02:00
for (const i in features) {
2022-09-08 21:40:48 +02:00
const layer = features[i]
const props = layer.properties
2021-10-15 14:52:11 +02:00
2022-02-08 00:34:07 +01:00
if (props.type === "bing") {
// A lot of work to implement - see https://github.com/pietervdvn/MapComplete/issues/648
2022-09-08 21:40:48 +02:00
continue
2021-10-15 14:52:11 +02:00
}
if (props.id === "MAPNIK") {
// Already added by default
2022-09-08 21:40:48 +02:00
continue
2021-10-15 14:52:11 +02:00
}
if (props.overlay) {
2022-09-08 21:40:48 +02:00
continue
2021-10-15 14:52:11 +02:00
}
if (props.url.toLowerCase().indexOf("apikey") > 0) {
2022-09-08 21:40:48 +02:00
continue
2021-10-15 14:52:11 +02:00
}
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
2022-09-08 21:40:48 +02:00
continue
2021-10-15 14:52:11 +02:00
}
if (props.name === undefined) {
console.warn("Editor layer index: name not defined on ", props)
continue
}
2022-01-26 21:40:38 +01:00
2022-09-08 21:40:48 +02:00
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"
)
2021-10-15 14:52:11 +02:00
// Note: if layer.geometry is null, there is global coverage for this layer
layers.push({
id: props.id,
max_zoom: props.max_zoom ?? 19,
2021-10-15 14:52:11 +02:00
min_zoom: props.min_zoom ?? 1,
name: props.name,
layer: leafletLayer,
feature: layer.geometry !== null ? layer : null,
2021-10-15 14:52:11 +02:00
isBest: props.best ?? false,
2022-09-08 21:40:48 +02:00
category: props.category,
})
2021-10-15 14:52:11 +02:00
}
2022-09-08 21:40:48 +02:00
return layers
2021-10-15 14:52:11 +02:00
}
private static LoadProviderIndex(): BaseLayer[] {
// @ts-ignore
2022-09-08 21:40:48 +02:00
X // Import X to make sure the namespace is not optimized away
2021-10-15 14:52:11 +02:00
function l(id: string, name: string): BaseLayer {
try {
2022-09-08 21:40:48 +02:00
const layer: any = L.tileLayer.provider(id, undefined)
2021-10-15 14:52:11 +02:00
return {
feature: null,
id: id,
name: name,
2022-09-08 21:40:48 +02:00
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,
2021-10-15 14:52:11 +02:00
category: "osmbasedmap",
2022-09-08 21:40:48 +02:00
isBest: false,
2021-10-15 14:52:11 +02:00
}
} catch (e) {
2022-09-08 21:40:48 +02:00
console.error("Could not find provided layer", name, e)
return null
2021-10-15 14:52:11 +02:00
}
}
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)"),
2022-09-08 21:40:48 +02:00
]
return Utils.NoNull(layers)
2021-10-15 14:52:11 +02:00
}
/**
* Converts a layer from the editor-layer-index into a tilelayer usable by leaflet
*/
2022-09-08 21:40:48 +02:00
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}", "")
2021-10-15 14:52:11 +02:00
const subdomainsMatch = url.match(/{switch:[^}]*}/)
2022-09-08 21:40:48 +02:00
let domains: string[] = []
2021-10-15 14:52:11 +02:00
if (subdomainsMatch !== null) {
2022-09-08 21:40:48 +02:00
let domainsStr = subdomainsMatch[0].substr("{switch:".length)
domainsStr = domainsStr.substr(0, domainsStr.length - 1)
domains = domainsStr.split(",")
2021-10-15 14:52:11 +02:00
url = url.replace(/{switch:[^}]*}/, "{s}")
}
if (isWms) {
2022-09-08 21:40:48 +02:00
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
2021-10-15 14:52:11 +02:00
const options = {
maxZoom: Math.max(maxZoom ?? 19, 21),
maxNativeZoom: maxZoom ?? 19,
2021-10-15 14:52:11 +02:00
attribution: attribution + " | ",
subdomains: domains,
uppercase: isUpper,
transparent: false,
2022-09-08 21:40:48 +02:00
}
2021-10-15 14:52:11 +02:00
for (const paramater of paramaters) {
2022-09-08 21:40:48 +02:00
let p = paramater
2021-10-15 14:52:11 +02:00
if (isUpper) {
2022-09-08 21:40:48 +02:00
p = paramater.toUpperCase()
2021-10-15 14:52:11 +02:00
}
2022-09-08 21:40:48 +02:00
options[paramater] = urlObj.searchParams.get(p)
2021-10-15 14:52:11 +02:00
}
if (options.transparent === null) {
2022-09-08 21:40:48 +02:00
options.transparent = false
2021-10-15 14:52:11 +02:00
}
2022-09-08 21:40:48 +02:00
return L.tileLayer.wms(urlObj.protocol + "//" + urlObj.host + urlObj.pathname, options)
2021-10-15 14:52:11 +02:00
}
if (attributionUrl) {
2022-09-08 21:40:48 +02:00
attribution = `<a href='${attributionUrl}' target='_blank'>${attribution}</a>`
2021-10-15 14:52:11 +02:00
}
2022-09-08 21:40:48 +02:00
return L.tileLayer(url, {
attribution: attribution,
maxZoom: Math.max(21, maxZoom ?? 19),
maxNativeZoom: maxZoom ?? 19,
minZoom: 1,
// @ts-ignore
wmts: isWMTS ?? false,
subdomains: domains,
})
2021-10-15 14:52:11 +02:00
}
2022-01-26 21:40:38 +01:00
public AvailableLayersAt(location: Store<Loc>): Store<BaseLayer[]> {
2022-09-08 21:40:48 +02:00
return Stores.ListStabilized(
location.map((currentLocation) => {
2021-11-07 16:34:51 +01:00
if (currentLocation === undefined) {
2022-09-08 21:40:48 +02:00
return this.layerOverview
2021-11-07 16:34:51 +01:00
}
2022-09-08 21:40:48 +02:00
return this.CalculateAvailableLayersAt(currentLocation?.lon, currentLocation?.lat)
})
)
2021-11-07 16:34:51 +01:00
}
2022-09-08 21:40:48 +02:00
public SelectBestLayerAccordingTo(
location: Store<Loc>,
preferedCategory: Store<string | string[]>
): Store<BaseLayer> {
return this.AvailableLayersAt(location).map(
(available) => {
2022-01-26 21:40:38 +01:00
// First float all 'best layers' to the top
2021-11-07 16:34:51 +01:00
available.sort((a, b) => {
2022-09-08 21:40:48 +02:00
if (a.isBest && b.isBest) {
return 0
2021-11-07 16:34:51 +01:00
}
2022-09-08 21:40:48 +02:00
if (!a.isBest) {
return 1
}
return -1
})
2022-01-26 21:40:38 +01:00
if (preferedCategory.data === undefined) {
return available[0]
}
2022-09-08 21:40:48 +02:00
let prefered: string[]
2022-01-26 21:40:38 +01:00
if (typeof preferedCategory.data === "string") {
prefered = [preferedCategory.data]
} else {
2022-09-08 21:40:48 +02:00
prefered = preferedCategory.data
2022-01-26 21:40:38 +01:00
}
2022-09-08 21:40:48 +02:00
prefered.reverse(/*New list, inplace reverse is fine*/)
2022-01-26 21:40:38 +01:00
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) => {
2022-09-08 21:40:48 +02:00
if (a.category === category && b.category === category) {
return 0
}
if (a.category !== category) {
return 1
2022-01-26 21:40:38 +01:00
}
2022-09-08 21:40:48 +02:00
return -1
})
2022-01-26 21:40:38 +01:00
}
return available[0]
2022-09-08 21:40:48 +02:00
},
[preferedCategory]
)
2021-11-07 16:34:51 +01:00
}
private CalculateAvailableLayersAt(lon: number, lat: number): BaseLayer[] {
const availableLayers = [this.osmCarto]
2022-02-11 15:11:50 +01:00
if (lon === undefined || lat === undefined) {
2022-09-08 21:40:48 +02:00
return availableLayers.concat(this.globalLayers)
2022-02-11 15:11:50 +01:00
}
2022-09-08 21:40:48 +02:00
const lonlat: [number, number] = [lon, lat]
2022-02-11 15:11:50 +01:00
for (const layerOverviewItem of this.localLayers) {
2022-09-08 21:40:48 +02:00
const layer = layerOverviewItem
2022-02-11 15:11:50 +01:00
const bbox = BBox.get(layer.feature)
2022-09-08 21:40:48 +02:00
if (!bbox.contains(lonlat)) {
2022-02-11 15:11:50 +01:00
continue
2021-11-07 16:34:51 +01:00
}
2022-02-11 15:11:50 +01:00
if (GeoOperations.inside(lonlat, layer.feature)) {
2022-09-08 21:40:48 +02:00
availableLayers.push(layer)
2021-11-07 16:34:51 +01:00
}
}
2022-09-08 21:40:48 +02:00
return availableLayers.concat(this.globalLayers)
2021-11-07 16:34:51 +01:00
}
2022-09-08 21:40:48 +02:00
}