318 lines
12 KiB
TypeScript
318 lines
12 KiB
TypeScript
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"
|
|
import Constants from "./Constants"
|
|
|
|
export class AvailableRasterLayers {
|
|
public static EditorLayerIndex: (Feature<Polygon, EditorLayerIndexProperties> &
|
|
RasterLayerPolygon)[] = (<any>editorlayerindex.features).filter(
|
|
(l) => l.properties.id !== "Bing"
|
|
)
|
|
public static globalLayers: RasterLayerPolygon[] = globallayers.layers
|
|
.filter(
|
|
(properties) =>
|
|
properties.id !== "osm.carto" && properties.id !== "Bing" /*Added separately*/
|
|
)
|
|
.map(
|
|
(properties) =>
|
|
<RasterLayerPolygon>{
|
|
type: "Feature",
|
|
properties,
|
|
geometry: BBox.global.asGeometry(),
|
|
}
|
|
)
|
|
public static bing: RasterLayerPolygon = (<any>editorlayerindex.features).find(
|
|
(l) => l.properties.id === "Bing"
|
|
)
|
|
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 maptilerDefaultLayer: RasterLayerPolygon = {
|
|
type: "Feature",
|
|
properties: {
|
|
name: "MapTiler",
|
|
url:
|
|
"https://api.maptiler.com/maps/15cc8f61-0353-4be6-b8da-13daea5f7432/style.json?key=" +
|
|
Constants.maptilerApiKey,
|
|
category: "osmbasedmap",
|
|
id: "maptiler",
|
|
type: "vector",
|
|
attribution: {
|
|
text: "Maptiler",
|
|
url: "https://www.maptiler.com/copyright/",
|
|
},
|
|
},
|
|
geometry: BBox.global.asGeometry(),
|
|
}
|
|
|
|
public static layersAvailableAt(
|
|
location: Store<{ lon: number; lat: number }>,
|
|
enableBing?: Store<boolean>
|
|
): Store<RasterLayerPolygon[]> {
|
|
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)
|
|
)
|
|
})
|
|
)
|
|
return 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.push(AvailableRasterLayers.maptilerDefaultLayer)
|
|
if (enableBing?.data) {
|
|
matching.push(AvailableRasterLayers.bing)
|
|
}
|
|
matching.push(...AvailableRasterLayers.globalLayers)
|
|
return matching
|
|
},
|
|
[enableBing]
|
|
)
|
|
)
|
|
}
|
|
|
|
public static allIds(): Set<string> {
|
|
const all: string[] = []
|
|
all.push(...AvailableRasterLayers.globalLayers.map((l) => l.properties.id))
|
|
all.push(...AvailableRasterLayers.EditorLayerIndex.map((l) => l.properties.id))
|
|
all.push(this.osmCarto.properties.id)
|
|
all.push(this.maptilerDefaultLayer.properties.id)
|
|
return new Set<string>(all)
|
|
}
|
|
}
|
|
|
|
export class RasterLayerUtils {
|
|
/**
|
|
* Selects, from the given list of available rasterLayerPolygons, a rasterLayer.
|
|
* This rasterlayer will be of type 'preferredCategory' and will be of the 'best'-layer (if available).
|
|
* Returns 'undefined' if no such layer is available
|
|
* @param available
|
|
* @param preferredCategory
|
|
* @param ignoreLayer
|
|
*/
|
|
public static SelectBestLayerAccordingTo(
|
|
available: RasterLayerPolygon[],
|
|
preferredCategory: string,
|
|
ignoreLayer?: RasterLayerPolygon
|
|
): RasterLayerPolygon {
|
|
let secondBest: RasterLayerPolygon = undefined
|
|
for (const rasterLayer of available) {
|
|
if (rasterLayer === ignoreLayer) {
|
|
continue
|
|
}
|
|
const p = rasterLayer.properties
|
|
if (p.category === preferredCategory) {
|
|
if (p.best) {
|
|
return rasterLayer
|
|
}
|
|
if (!secondBest) {
|
|
secondBest = rasterLayer
|
|
}
|
|
}
|
|
}
|
|
return secondBest
|
|
}
|
|
}
|
|
|
|
export type RasterLayerPolygon = Feature<Polygon, RasterLayerProperties>
|
|
|
|
/**
|
|
* 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"
|
|
| "vector" /* Vector is not actually part of the ELI-spec, we add it for vector layers */
|
|
/**
|
|
* 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
|
|
}
|