import { Feature, Polygon } from "geojson" import * as editorlayerindex from "../assets/editor-layer-index.json" import * as globallayers from "../assets/global-raster-layers.json" import { BBox } from "../Logic/BBox" import { Store, Stores } from "../Logic/UIEventSource" import { GeoOperations } from "../Logic/GeoOperations" import { RasterLayerProperties } from "./RasterLayerProperties" export class AvailableRasterLayers { public static EditorLayerIndex: (Feature & RasterLayerPolygon)[] = editorlayerindex.features public static globalLayers: RasterLayerPolygon[] = globallayers.layers.map( (properties) => { type: "Feature", properties, geometry: BBox.global.asGeometry(), } ) public static readonly osmCartoProperties: RasterLayerProperties = { id: "osm", name: "OpenStreetMap", url: "https://tile.openstreetmap.org/{z}/{x}/{y}.png", attribution: { text: "OpenStreetMap", url: "https://openStreetMap.org/copyright", }, best: true, max_zoom: 19, min_zoom: 0, category: "osmbasedmap", } public static readonly osmCarto: RasterLayerPolygon = { type: "Feature", properties: AvailableRasterLayers.osmCartoProperties, geometry: BBox.global.asGeometry(), } public static readonly maplibre: RasterLayerPolygon = { type: "Feature", properties: { name: "MapLibre", url: null, }, geometry: BBox.global.asGeometry(), } public static layersAvailableAt( location: Store<{ lon: number; lat: number }> ): Store { const availableLayersBboxes = Stores.ListStabilized( location.mapD((loc) => { const lonlat: [number, number] = [loc.lon, loc.lat] return AvailableRasterLayers.EditorLayerIndex.filter((eliPolygon) => BBox.get(eliPolygon).contains(lonlat) ) }) ) const available = Stores.ListStabilized( availableLayersBboxes.map((eliPolygons) => { const loc = location.data const lonlat: [number, number] = [loc.lon, loc.lat] const matching: RasterLayerPolygon[] = eliPolygons.filter((eliPolygon) => { if (eliPolygon.geometry === null) { return true // global ELI-layer } return GeoOperations.inside(lonlat, eliPolygon) }) matching.unshift(AvailableRasterLayers.osmCarto) matching.unshift(AvailableRasterLayers.maplibre) matching.push(...AvailableRasterLayers.globalLayers) return matching }) ) return available } } export class RasterLayerUtils { public static SelectBestLayerAccordingTo( available: RasterLayerPolygon[], preferredCategory: string | string[] ): RasterLayerPolygon { available = [...available] if (preferredCategory === undefined) { return available[0] } let prefered: string[] if (typeof preferredCategory === "string") { prefered = [preferredCategory] } else { prefered = preferredCategory } for (let i = prefered.length - 1; i >= 0; i--) { const category = prefered[i] //Then sort all layers of the preferred type to the top. Stability of the sorting will force a 'best' photo layer on top available.sort((ap, bp) => { const a = ap.properties const b = bp.properties if (a.category === category && b.category === category) { return 0 } if (a.category !== category) { return 1 } return -1 }) } const best = available.find((l) => l.properties.best) if (best) { return best } return available[0] } } export type RasterLayerPolygon = Feature /** * Information about a raster tile layer * * Based on the spec here https://github.com/osmlab/editor-layer-index/blob/gh-pages/schema.json * which was then converted with http://borischerny.com/json-schema-to-typescript-browser/ */ export interface EditorLayerIndexProperties extends RasterLayerProperties { /** * The name of the imagery source */ readonly name: string /** * Whether the imagery name should be translated */ readonly i18n?: boolean readonly type: "tms" | "wms" | "bing" | "scanex" | "wms_endpoint" | "wmts" /** * A rough categorisation of different types of layers. See https://github.com/osmlab/editor-layer-index/blob/gh-pages/CONTRIBUTING.md#categories for a description of the individual categories. */ readonly category?: | "photo" | "map" | "historicmap" | "osmbasedmap" | "historicphoto" | "qa" | "elevation" | "other" /** * A URL template for imagery tiles */ readonly url: string readonly min_zoom?: number readonly max_zoom?: number /** * explicit/implicit permission by the owner for use in OSM */ readonly permission_osm?: "explicit" | "implicit" | "no" /** * A URL for the license or permissions for the imagery */ readonly license_url?: string /** * A URL for the privacy policy of the operator or false if there is no existing privacy policy for tis imagery. */ readonly privacy_policy_url?: string | boolean /** * A unique identifier for the source; used in imagery_used changeset tag */ readonly id: string /** * A short English-language description of the source */ readonly description?: string /** * The ISO 3166-1 alpha-2 two letter country code in upper case. Use ZZ for unknown or multiple. */ readonly country_code?: string /** * Whether this imagery should be shown in the default world-wide menu */ readonly default?: boolean /** * Whether this imagery is the best source for the region */ readonly best?: boolean /** * The age of the oldest imagery or data in the source, as an RFC3339 date or leading portion of one */ readonly start_date?: string /** * The age of the newest imagery or data in the source, as an RFC3339 date or leading portion of one */ readonly end_date?: string /** * HTTP header to check for information if the tile is invalid */ readonly no_tile_header?: { /** * This interface was referenced by `undefined`'s JSON-Schema definition * via the `patternProperty` "^.*$". */ [k: string]: string[] | null } /** * 'true' if tiles are transparent and can be overlaid on another source */ readonly overlay?: boolean & string readonly available_projections?: string[] readonly attribution?: { readonly url?: string readonly text?: string readonly html?: string readonly required?: boolean } /** * A URL for an image, that can be displayed in the list of imagery layers next to the name */ readonly icon?: string /** * A link to an EULA text that has to be accepted by the user, before the imagery source is added. Can contain {lang} to be replaced by a current user language wiki code (like FR:) or an empty string for the default English text. */ readonly eula?: string /** * A URL for an image, that is displayed in the mapview for attribution */ readonly "logo-image"?: string /** * Customized text for the terms of use link (default is "Background Terms of Use") */ readonly "terms-of-use-text"?: string /** * Specify a checksum for tiles, which aren't real tiles. `type` is the digest type and can be MD5, SHA-1, SHA-256, SHA-384 and SHA-512, value is the hex encoded checksum in lower case. To create a checksum save the tile as file and upload it to e.g. https://defuse.ca/checksums.htm. */ readonly "no-tile-checksum"?: string /** * header-name attribute specifies a header returned by tile server, that will be shown as `metadata-key` attribute in Show Tile Info dialog */ readonly "metadata-header"?: string /** * Set to `true` if imagery source is properly aligned and does not need imagery offset adjustments. This is used for OSM based sources too. */ readonly "valid-georeference"?: boolean /** * Size of individual tiles delivered by a TMS service */ readonly "tile-size"?: number /** * Whether tiles status can be accessed by appending /status to the tile URL and can be submitted for re-rendering by appending /dirty. */ readonly "mod-tile-features"?: string /** * HTTP headers to be sent to server. It has two attributes header-name and header-value. May be specified multiple times. */ readonly "custom-http-headers"?: { readonly "header-name"?: string readonly "header-value"?: string } /** * Default layer to open (when using WMS_ENDPOINT type). Contains list of layer tag with two attributes - name and style, e.g. `"default-layers": ["layer": { name="Basisdata_NP_Basiskart_JanMayen_WMTS_25829" "style":"default" } ]` (not allowed in `mirror` attribute) */ readonly "default-layers"?: { layer?: { "layer-name"?: string "layer-style"?: string [k: string]: unknown } [k: string]: unknown }[] /** * format to use when connecting tile server (when using WMS_ENDPOINT type) */ readonly format?: string /** * If `true` transparent tiles will be requested from WMS server */ readonly transparent?: boolean & string /** * minimum expiry time for tiles in seconds. The larger the value, the longer entry in cache will be considered valid */ readonly "minimum-tile-expire"?: number }