2023-09-24 18:24:10 +02:00
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" ;
2023-03-11 02:37:07 +01:00
export class AvailableRasterLayers {
public static EditorLayerIndex : ( Feature < Polygon , EditorLayerIndexProperties > &
2023-09-24 18:24:10 +02:00
RasterLayerPolygon ) [ ] = < any > editorlayerindex . features ;
2023-03-11 02:37:07 +01:00
public static globalLayers : RasterLayerPolygon [ ] = globallayers . layers . map (
( properties ) = >
< RasterLayerPolygon > {
type : "Feature" ,
properties ,
2023-09-24 18:24:10 +02:00
geometry : BBox.global.asGeometry ( )
2023-03-11 02:37:07 +01:00
}
2023-09-24 18:24:10 +02:00
) ;
2023-03-11 02:37:07 +01:00
public static readonly osmCartoProperties : RasterLayerProperties = {
id : "osm" ,
name : "OpenStreetMap" ,
url : "https://tile.openstreetmap.org/{z}/{x}/{y}.png" ,
attribution : {
text : "OpenStreetMap" ,
2023-09-24 18:24:10 +02:00
url : "https://openStreetMap.org/copyright"
2023-03-11 02:37:07 +01:00
} ,
best : true ,
max_zoom : 19 ,
min_zoom : 0 ,
2023-09-24 18:24:10 +02:00
category : "osmbasedmap"
} ;
2023-03-11 02:37:07 +01:00
public static readonly osmCarto : RasterLayerPolygon = {
type : "Feature" ,
properties : AvailableRasterLayers.osmCartoProperties ,
2023-09-24 18:24:10 +02:00
geometry : BBox.global.asGeometry ( )
} ;
2023-03-23 00:58:21 +01:00
2023-09-24 18:24:10 +02:00
public static readonly maptilerDefaultLayer : RasterLayerPolygon = {
2023-03-28 05:13:48 +02:00
type : "Feature" ,
2023-05-18 15:56:24 +02:00
properties : {
name : "MapTiler" ,
2023-06-04 22:52:13 +02:00
url : "https://api.maptiler.com/maps/15cc8f61-0353-4be6-b8da-13daea5f7432/style.json?key=GvoVAJgu46I5rZapJuAy" ,
2023-05-18 15:56:24 +02:00
category : "osmbasedmap" ,
id : "maptiler" ,
2023-06-14 00:47:09 +02:00
type : "vector" ,
2023-05-18 15:56:24 +02:00
attribution : {
text : "Maptiler" ,
2023-09-24 18:24:10 +02:00
url : "https://www.maptiler.com/copyright/"
}
2023-03-28 05:13:48 +02:00
} ,
2023-09-24 18:24:10 +02:00
geometry : BBox.global.asGeometry ( )
} ;
public static readonly maptilerCarto : RasterLayerPolygon = {
type : "Feature" ,
properties : {
name : "MapTiler Carto" ,
url : "https://api.maptiler.com/maps/openstreetmap/style.json?key=GvoVAJgu46I5rZapJuAy" ,
category : "osmbasedmap" ,
id : "maptiler.carto" ,
type : "vector" ,
attribution : {
text : "Maptiler" ,
url : "https://www.maptiler.com/copyright/"
}
} ,
geometry : BBox.global.asGeometry ( )
} ;
2023-04-21 17:37:50 +02:00
2023-09-24 18:24:10 +02:00
public static readonly maptilerBackdrop : RasterLayerPolygon = {
type : "Feature" ,
properties : {
name : "MapTiler Backdrop" ,
url : "https://api.maptiler.com/maps/backdrop/style.json?key=GvoVAJgu46I5rZapJuAy" ,
category : "osmbasedmap" ,
id : "maptiler.backdrop" ,
type : "vector" ,
attribution : {
text : "Maptiler" ,
url : "https://www.maptiler.com/copyright/"
}
} ,
geometry : BBox.global.asGeometry ( )
} ;
2023-06-14 00:47:09 +02:00
public static readonly americana : RasterLayerPolygon = {
type : "Feature" ,
properties : {
name : "Americana" ,
url : "https://zelonewolf.github.io/openstreetmap-americana/style.json" ,
category : "osmbasedmap" ,
id : "americana" ,
type : "vector" ,
attribution : {
text : "Americana" ,
2023-09-24 18:24:10 +02:00
url : "https://github.com/ZeLonewolf/openstreetmap-americana/"
}
2023-06-14 00:47:09 +02:00
} ,
2023-09-24 18:24:10 +02:00
geometry : BBox.global.asGeometry ( )
} ;
2023-06-14 00:47:09 +02:00
2023-03-23 00:58:21 +01:00
public static layersAvailableAt (
location : Store < { lon : number ; lat : number } >
) : Store < RasterLayerPolygon [ ] > {
const availableLayersBboxes = Stores . ListStabilized (
location . mapD ( ( loc ) = > {
2023-09-24 18:24:10 +02:00
const lonlat : [ number , number ] = [ loc . lon , loc . lat ] ;
2023-03-23 00:58:21 +01:00
return AvailableRasterLayers . EditorLayerIndex . filter ( ( eliPolygon ) = >
BBox . get ( eliPolygon ) . contains ( lonlat )
2023-09-24 18:24:10 +02:00
) ;
2023-03-23 00:58:21 +01:00
} )
2023-09-24 18:24:10 +02:00
) ;
2023-03-23 00:58:21 +01:00
const available = Stores . ListStabilized (
availableLayersBboxes . map ( ( eliPolygons ) = > {
2023-09-24 18:24:10 +02:00
const loc = location . data ;
const lonlat : [ number , number ] = [ loc . lon , loc . lat ] ;
2023-03-23 00:58:21 +01:00
const matching : RasterLayerPolygon [ ] = eliPolygons . filter ( ( eliPolygon ) = > {
if ( eliPolygon . geometry === null ) {
2023-09-24 18:24:10 +02:00
return true ; // global ELI-layer
2023-03-23 00:58:21 +01:00
}
2023-09-24 18:24:10 +02:00
return GeoOperations . inside ( lonlat , eliPolygon ) ;
} ) ;
matching . push ( . . . AvailableRasterLayers . globalLayers ) ;
matching . unshift ( AvailableRasterLayers . maptilerDefaultLayer ,
AvailableRasterLayers . osmCarto ,
AvailableRasterLayers . maptilerCarto ,
AvailableRasterLayers . maptilerBackdrop ,
AvailableRasterLayers . americana ) ;
return matching ;
2023-03-23 00:58:21 +01:00
} )
2023-09-24 18:24:10 +02:00
) ;
return available ;
2023-03-23 00:58:21 +01:00
}
2023-03-11 02:37:07 +01:00
}
export class RasterLayerUtils {
2023-04-21 17:37:50 +02:00
/ * *
* 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
* /
2023-03-11 02:37:07 +01:00
public static SelectBestLayerAccordingTo (
available : RasterLayerPolygon [ ] ,
2023-04-21 17:37:50 +02:00
preferredCategory : string ,
ignoreLayer? : RasterLayerPolygon
2023-03-11 02:37:07 +01:00
) : RasterLayerPolygon {
2023-09-24 18:24:10 +02:00
let secondBest : RasterLayerPolygon = undefined ;
2023-04-21 17:37:50 +02:00
for ( const rasterLayer of available ) {
if ( rasterLayer === ignoreLayer ) {
2023-09-24 18:24:10 +02:00
continue ;
2023-04-21 17:37:50 +02:00
}
2023-09-24 18:24:10 +02:00
const p = rasterLayer . properties ;
2023-04-21 17:37:50 +02:00
if ( p . category === preferredCategory ) {
if ( p . best ) {
2023-09-24 18:24:10 +02:00
return rasterLayer ;
2023-03-11 02:37:07 +01:00
}
2023-04-21 17:37:50 +02:00
if ( ! secondBest ) {
2023-09-24 18:24:10 +02:00
secondBest = rasterLayer ;
2023-03-11 02:37:07 +01:00
}
2023-04-21 17:37:50 +02:00
}
2023-03-11 02:37:07 +01:00
}
2023-09-24 18:24:10 +02:00
return secondBest ;
2023-03-11 02:37:07 +01:00
}
}
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
* /
2023-09-24 18:24:10 +02:00
readonly name : string ;
2023-03-11 02:37:07 +01:00
/ * *
* Whether the imagery name should be translated
* /
2023-09-24 18:24:10 +02:00
readonly i18n? : boolean ;
2023-06-14 20:39:36 +02:00
readonly type :
| "tms"
| "wms"
| "bing"
| "scanex"
| "wms_endpoint"
| "wmts"
2023-09-24 18:24:10 +02:00
| "vector" ; /* Vector is not actually part of the ELI-spec, we add it for vector layers */
2023-03-11 02:37:07 +01:00
/ * *
* 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"
2023-09-24 18:24:10 +02:00
| "other" ;
2023-03-11 02:37:07 +01:00
/ * *
* A URL template for imagery tiles
* /
2023-09-24 18:24:10 +02:00
readonly url : string ;
readonly min_zoom? : number ;
readonly max_zoom? : number ;
2023-03-11 02:37:07 +01:00
/ * *
* explicit / implicit permission by the owner for use in OSM
* /
2023-09-24 18:24:10 +02:00
readonly permission_osm ? : "explicit" | "implicit" | "no" ;
2023-03-11 02:37:07 +01:00
/ * *
* A URL for the license or permissions for the imagery
* /
2023-09-24 18:24:10 +02:00
readonly license_url? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* A URL for the privacy policy of the operator or false if there is no existing privacy policy for tis imagery .
* /
2023-09-24 18:24:10 +02:00
readonly privacy_policy_url? : string | boolean ;
2023-03-11 02:37:07 +01:00
/ * *
* A unique identifier for the source ; used in imagery_used changeset tag
* /
2023-09-24 18:24:10 +02:00
readonly id : string ;
2023-03-11 02:37:07 +01:00
/ * *
* A short English - language description of the source
* /
2023-09-24 18:24:10 +02:00
readonly description? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* The ISO 3166 - 1 alpha - 2 two letter country code in upper case . Use ZZ for unknown or multiple .
* /
2023-09-24 18:24:10 +02:00
readonly country_code? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* Whether this imagery should be shown in the default world - wide menu
* /
2023-09-24 18:24:10 +02:00
readonly default ? : boolean ;
2023-03-11 02:37:07 +01:00
/ * *
* Whether this imagery is the best source for the region
* /
2023-09-24 18:24:10 +02:00
readonly best? : boolean ;
2023-03-11 02:37:07 +01:00
/ * *
* The age of the oldest imagery or data in the source , as an RFC3339 date or leading portion of one
* /
2023-09-24 18:24:10 +02:00
readonly start_date? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* The age of the newest imagery or data in the source , as an RFC3339 date or leading portion of one
* /
2023-09-24 18:24:10 +02:00
readonly end_date? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* 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
2023-09-24 18:24:10 +02:00
} ;
2023-03-11 02:37:07 +01:00
/ * *
* 'true' if tiles are transparent and can be overlaid on another source
* /
2023-09-24 18:24:10 +02:00
readonly overlay? : boolean & string ;
readonly available_projections? : string [ ] ;
2023-03-11 02:37:07 +01:00
readonly attribution ? : {
readonly url? : string
readonly text? : string
readonly html? : string
readonly required? : boolean
2023-09-24 18:24:10 +02:00
} ;
2023-03-11 02:37:07 +01:00
/ * *
* A URL for an image , that can be displayed in the list of imagery layers next to the name
* /
2023-09-24 18:24:10 +02:00
readonly icon? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* 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 .
* /
2023-09-24 18:24:10 +02:00
readonly eula? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* A URL for an image , that is displayed in the mapview for attribution
* /
2023-09-24 18:24:10 +02:00
readonly "logo-image" ? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* Customized text for the terms of use link ( default is "Background Terms of Use" )
* /
2023-09-24 18:24:10 +02:00
readonly "terms-of-use-text" ? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* 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.
* /
2023-09-24 18:24:10 +02:00
readonly "no-tile-checksum" ? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* header - name attribute specifies a header returned by tile server , that will be shown as ` metadata-key ` attribute in Show Tile Info dialog
* /
2023-09-24 18:24:10 +02:00
readonly "metadata-header" ? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* Set to ` true ` if imagery source is properly aligned and does not need imagery offset adjustments . This is used for OSM based sources too .
* /
2023-09-24 18:24:10 +02:00
readonly "valid-georeference" ? : boolean ;
2023-03-11 02:37:07 +01:00
/ * *
* Size of individual tiles delivered by a TMS service
* /
2023-09-24 18:24:10 +02:00
readonly "tile-size" ? : number ;
2023-03-11 02:37:07 +01:00
/ * *
* Whether tiles status can be accessed by appending / status to the tile URL and can be submitted for re - rendering by appending / dirty .
* /
2023-09-24 18:24:10 +02:00
readonly "mod-tile-features" ? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* 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
2023-09-24 18:24:10 +02:00
} ;
2023-03-11 02:37:07 +01:00
/ * *
* 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
2023-09-24 18:24:10 +02:00
} [ ] ;
2023-03-11 02:37:07 +01:00
/ * *
* format to use when connecting tile server ( when using WMS_ENDPOINT type )
* /
2023-09-24 18:24:10 +02:00
readonly format? : string ;
2023-03-11 02:37:07 +01:00
/ * *
* If ` true ` transparent tiles will be requested from WMS server
* /
2023-09-24 18:24:10 +02:00
readonly transparent? : boolean & string ;
2023-03-11 02:37:07 +01:00
/ * *
* minimum expiry time for tiles in seconds . The larger the value , the longer entry in cache will be considered valid
* /
2023-09-24 18:24:10 +02:00
readonly "minimum-tile-expire" ? : number ;
2023-03-11 02:37:07 +01:00
}