2021-07-23 15:56:22 +02:00
import { Translation } from "../../UI/i18n/Translation" ;
2021-03-20 23:45:52 +01:00
import SourceConfig from "./SourceConfig" ;
2021-08-07 23:11:34 +02:00
import TagRenderingConfig from "./TagRenderingConfig" ;
2021-12-04 21:49:17 +01:00
import PresetConfig , { PreciseInput } from "./PresetConfig" ;
2021-08-07 23:11:34 +02:00
import { LayerConfigJson } from "./Json/LayerConfigJson" ;
import Translations from "../../UI/i18n/Translations" ;
import { TagUtils } from "../../Logic/Tags/TagUtils" ;
2021-07-22 11:29:09 +02:00
import FilterConfig from "./FilterConfig" ;
2021-08-07 23:11:34 +02:00
import { Unit } from "../Unit" ;
import DeleteConfig from "./DeleteConfig" ;
2021-10-14 03:46:09 +02:00
import MoveConfig from "./MoveConfig" ;
2021-10-19 02:31:32 +02:00
import PointRenderingConfig from "./PointRenderingConfig" ;
import WithContextLoader from "./WithContextLoader" ;
2021-10-20 02:01:27 +02:00
import LineRenderingConfig from "./LineRenderingConfig" ;
import PointRenderingConfigJson from "./Json/PointRenderingConfigJson" ;
import LineRenderingConfigJson from "./Json/LineRenderingConfigJson" ;
2021-10-22 18:53:07 +02:00
import { TagRenderingConfigJson } from "./Json/TagRenderingConfigJson" ;
2021-10-31 02:08:39 +01:00
import BaseUIElement from "../../UI/BaseUIElement" ;
2021-11-08 02:36:01 +01:00
import Combine from "../../UI/Base/Combine" ;
import Title from "../../UI/Base/Title" ;
import List from "../../UI/Base/List" ;
import Link from "../../UI/Base/Link" ;
2021-11-08 03:00:58 +01:00
import { Utils } from "../../Utils" ;
2022-01-14 19:34:00 +01:00
import { TagsFilter } from "../../Logic/Tags/TagsFilter" ;
import Table from "../../UI/Base/Table" ;
2022-02-01 04:14:54 +01:00
import FilterConfigJson from "./Json/FilterConfigJson" ;
2022-03-24 19:59:46 +01:00
import { And } from "../../Logic/Tags/And" ;
import { Overpass } from "../../Logic/Osm/Overpass" ;
import Constants from "../Constants" ;
2022-04-06 17:28:51 +02:00
import undefinedError = Mocha . utils . undefinedError ;
2021-04-10 23:53:13 +02:00
2021-10-22 18:53:07 +02:00
export default class LayerConfig extends WithContextLoader {
2021-07-23 15:56:22 +02:00
2021-12-04 21:49:17 +01:00
public readonly id : string ;
public readonly name : Translation ;
public readonly description : Translation ;
public readonly source : SourceConfig ;
2021-12-12 02:59:24 +01:00
public readonly calculatedTags : [ string , string , boolean ] [ ] ;
2021-12-04 21:49:17 +01:00
public readonly doNotDownload : boolean ;
2022-01-14 19:34:00 +01:00
public readonly passAllFeatures : boolean ;
2021-12-04 21:49:17 +01:00
public readonly isShown : TagRenderingConfig ;
2021-12-12 17:21:32 +01:00
public minzoom : number ;
public minzoomVisible : number ;
2022-01-14 19:34:00 +01:00
public readonly maxzoom : number ;
2021-12-04 21:49:17 +01:00
public readonly title? : TagRenderingConfig ;
public readonly titleIcons : TagRenderingConfig [ ] ;
2021-10-22 18:53:07 +02:00
2021-10-19 02:31:32 +02:00
public readonly mapRendering : PointRenderingConfig [ ]
2021-10-20 02:01:27 +02:00
public readonly lineRendering : LineRenderingConfig [ ]
2021-10-19 02:31:32 +02:00
2021-07-23 15:56:22 +02:00
public readonly units : Unit [ ] ;
public readonly deletion : DeleteConfig | null ;
2021-10-14 03:46:09 +02:00
public readonly allowMove : MoveConfig | null
2021-07-15 20:47:28 +02:00
public readonly allowSplit : boolean
2021-12-03 02:29:25 +01:00
public readonly shownByDefault : boolean ;
2021-10-25 20:38:57 +02:00
/ * *
* In seconds
* /
public readonly maxAgeOfCache : number
2021-07-23 15:56:22 +02:00
2021-12-04 21:49:17 +01:00
public readonly presets : PresetConfig [ ] ;
2020-11-17 02:22:48 +01:00
2021-12-04 21:49:17 +01:00
public readonly tagRenderings : TagRenderingConfig [ ] ;
public readonly filters : FilterConfig [ ] ;
2022-02-01 04:14:54 +01:00
public readonly filterIsSameAs : string ;
2022-02-07 01:59:07 +01:00
public readonly forceLoad : boolean ;
2022-03-24 19:59:46 +01:00
public readonly syncSelection : "no" | "local" | "theme-only" | "global"
2021-07-23 15:56:22 +02:00
constructor (
json : LayerConfigJson ,
context? : string ,
official : boolean = true
) {
context = context + "." + json . id ;
2022-04-01 12:51:55 +02:00
const translationContext = "layers:" + json . id
2021-10-19 02:31:32 +02:00
super ( json , context )
2021-07-23 15:56:22 +02:00
this . id = json . id ;
2022-01-26 21:40:38 +01:00
if ( json . id === undefined ) {
throw "Not a valid layer: id is undefined: " + JSON . stringify ( json )
2022-01-17 21:33:03 +01:00
}
2021-10-29 01:41:37 +02:00
if ( json . source === undefined ) {
2021-10-31 02:08:39 +01:00
throw "Layer " + this . id + " does not define a source section (" + context + ")"
2021-03-20 23:45:52 +01:00
}
2021-10-25 20:38:57 +02:00
2021-10-29 01:41:37 +02:00
if ( json . source . osmTags === undefined ) {
2021-10-31 02:08:39 +01:00
throw "Layer " + this . id + " does not define a osmTags in the source section - these should always be present, even for geojson layers (" + context + ")"
2021-12-30 22:36:34 +01:00
}
2021-07-23 15:56:22 +02:00
2022-01-14 19:34:00 +01:00
if ( json . id . toLowerCase ( ) !== json . id ) {
2021-12-30 22:36:34 +01:00
throw ` ${ context } : The id of a layer should be lowercase: ${ json . id } `
}
2022-01-14 19:34:00 +01:00
if ( json . id . match ( /[a-z0-9-_]/ ) == null ) {
2021-12-30 22:36:34 +01:00
throw ` ${ context } : The id of a layer should match [a-z0-9-_]*: ${ json . id } `
2021-10-29 01:41:37 +02:00
}
2021-07-23 15:56:22 +02:00
2021-10-29 01:41:37 +02:00
this . maxAgeOfCache = json . source . maxCacheAge ? ? 24 * 60 * 60 * 30
2022-02-14 18:18:05 +01:00
this . syncSelection = json . syncSelection ;
2021-10-29 01:41:37 +02:00
const osmTags = TagUtils . Tag (
json . source . osmTags ,
context + "source.osmTags"
) ;
2021-09-18 02:32:40 +02:00
2022-03-31 02:44:23 +02:00
if ( Constants . priviliged_layers . indexOf ( this . id ) < 0 && osmTags . isNegative ( ) ) {
throw context + "The source states tags which give a very wide selection: it only uses negative expressions, which will result in too much and unexpected data. Add at least one required tag. The tags are:\n\t" + osmTags . asHumanString ( false , false , { } ) ;
}
2021-10-29 01:41:37 +02:00
if ( json . source [ "geoJsonSource" ] !== undefined ) {
throw context + "Use 'geoJson' instead of 'geoJsonSource'" ;
}
if ( json . source [ "geojson" ] !== undefined ) {
throw context + "Use 'geoJson' instead of 'geojson' (the J is a capital letter)" ;
2021-03-24 01:25:57 +01:00
}
2022-03-31 02:44:23 +02:00
2021-03-24 01:25:57 +01:00
2021-10-29 01:41:37 +02:00
this . source = new SourceConfig (
{
osmTags : osmTags ,
2022-03-24 19:59:46 +01:00
geojsonSource : json.source [ "geoJson" ] ,
geojsonSourceLevel : json.source [ "geoJsonZoomLevel" ] ,
2021-10-29 01:41:37 +02:00
overpassScript : json.source [ "overpassScript" ] ,
isOsmCache : json.source [ "isOsmCache" ] ,
2022-02-11 03:57:39 +01:00
mercatorCrs : json.source [ "mercatorCrs" ] ,
idKey : json.source [ "idKey" ]
2021-10-29 01:41:37 +02:00
} ,
2022-04-14 00:53:38 +02:00
Constants . priviliged_layers . indexOf ( this . id ) > 0 ,
2021-10-29 01:41:37 +02:00
json . id
) ;
2021-10-19 02:31:32 +02:00
this . allowSplit = json . allowSplit ? ? false ;
2022-04-01 12:51:55 +02:00
this . name = Translations . T ( json . name , translationContext + ".name" ) ;
2022-04-22 03:17:40 +02:00
if ( json . units !== undefined && ! Array . isArray ( json . units ) ) {
throw "At " + context + ".units: the 'units'-section should be a list; you probably have an object there"
}
2021-10-19 02:31:32 +02:00
this . units = ( json . units ? ? [ ] ) . map ( ( ( unitJson , i ) = > Unit . fromJson ( unitJson , ` ${ context } .unit[ ${ i } ] ` ) ) )
if ( json . description !== undefined ) {
if ( Object . keys ( json . description ) . length === 0 ) {
json . description = undefined ;
}
}
this . description = Translations . T (
json . description ,
2022-04-01 12:51:55 +02:00
translationContext + ".description"
2021-10-19 02:31:32 +02:00
) ;
2021-07-23 15:56:22 +02:00
this . calculatedTags = undefined ;
if ( json . calculatedTags !== undefined ) {
if ( ! official ) {
console . warn (
` Unofficial theme ${ this . id } with custom javascript! This is a security risk `
) ;
2021-01-08 03:57:18 +01:00
}
2021-07-23 15:56:22 +02:00
this . calculatedTags = [ ] ;
for ( const kv of json . calculatedTags ) {
const index = kv . indexOf ( "=" ) ;
2021-12-12 02:59:24 +01:00
let key = kv . substring ( 0 , index ) ;
const isStrict = key . endsWith ( ':' )
2022-01-14 19:34:00 +01:00
if ( isStrict ) {
2021-12-12 02:59:24 +01:00
key = key . substr ( 0 , key . length - 1 )
}
2021-07-23 15:56:22 +02:00
const code = kv . substring ( index + 1 ) ;
2021-01-08 03:57:18 +01:00
2021-10-14 03:46:09 +02:00
try {
new Function ( "feat" , "return " + code + ";" ) ;
} catch ( e ) {
throw ` Invalid function definition: code ${ code } is invalid: ${ e } (at ${ context } ) `
2021-09-18 02:32:40 +02:00
}
2021-12-12 02:59:24 +01:00
this . calculatedTags . push ( [ key , code , isStrict ] ) ;
2020-10-27 01:01:34 +01:00
}
2021-07-23 15:56:22 +02:00
}
2021-01-08 03:57:18 +01:00
2021-07-23 15:56:22 +02:00
this . doNotDownload = json . doNotDownload ? ? false ;
this . passAllFeatures = json . passAllFeatures ? ? false ;
this . minzoom = json . minzoom ? ? 0 ;
2021-07-27 19:39:57 +02:00
this . minzoomVisible = json . minzoomVisible ? ? this . minzoom ;
2021-12-03 02:29:25 +01:00
this . shownByDefault = json . shownByDefault ? ? true ;
2022-02-07 01:59:07 +01:00
this . forceLoad = json . forceLoad ? ? false ;
2021-10-14 03:46:09 +02:00
if ( json . presets !== undefined && json . presets ? . map === undefined ) {
throw "Presets should be a list of items (at " + context + ")"
2021-09-22 20:44:53 +02:00
}
2021-07-14 00:17:15 +02:00
this . presets = ( json . presets ? ? [ ] ) . map ( ( pr , i ) = > {
2021-12-04 21:49:17 +01:00
let preciseInput : PreciseInput = {
2021-10-16 00:43:53 +02:00
preferredBackground : [ "photo" ] ,
snapToLayers : undefined ,
maxSnapDistance : undefined
2021-10-15 19:58:02 +02:00
} ;
2021-08-07 23:11:34 +02:00
if ( pr . preciseInput !== undefined ) {
2021-08-07 21:19:01 +02:00
if ( pr . preciseInput === true ) {
pr . preciseInput = {
preferredBackground : undefined
}
}
2022-01-14 19:34:00 +01:00
2021-08-07 21:19:01 +02:00
let snapToLayers : string [ ] ;
if ( typeof pr . preciseInput . snapToLayer === "string" ) {
snapToLayers = [ pr . preciseInput . snapToLayer ]
} else {
snapToLayers = pr . preciseInput . snapToLayer
}
2021-08-07 23:11:34 +02:00
2022-03-29 00:20:10 +02:00
let preferredBackground : ( "map" | "photo" | "osmbasedmap" | "historicphoto" | string ) [ ]
2021-08-07 21:19:01 +02:00
if ( typeof pr . preciseInput . preferredBackground === "string" ) {
preferredBackground = [ pr . preciseInput . preferredBackground ]
} else {
preferredBackground = pr . preciseInput . preferredBackground
}
preciseInput = {
2022-03-29 00:20:10 +02:00
preferredBackground ,
2021-12-04 21:49:17 +01:00
snapToLayers ,
2021-08-07 21:19:01 +02:00
maxSnapDistance : pr.preciseInput.maxSnapDistance ? ? 10
2021-07-14 00:17:15 +02:00
}
}
2021-08-07 23:11:34 +02:00
const config : PresetConfig = {
2022-04-01 12:51:55 +02:00
title : Translations.T ( pr . title , ` ${ translationContext } .presets. ${ i } .title ` ) ,
2021-08-07 23:11:34 +02:00
tags : pr.tags.map ( ( t ) = > TagUtils . SimpleTag ( t ) ) ,
2022-04-01 12:51:55 +02:00
description : Translations.T ( pr . description , ` ${ translationContext } .presets. ${ i } .description ` ) ,
2021-08-07 21:19:01 +02:00
preciseInput : preciseInput ,
2022-02-09 03:38:40 +01:00
exampleImages : pr.exampleImages
2021-07-24 02:32:33 +02:00
}
2021-08-07 21:19:01 +02:00
return config ;
2021-07-24 02:32:33 +02:00
} ) ;
2021-07-23 15:56:22 +02:00
2021-10-22 18:53:07 +02:00
if ( json . mapRendering === undefined ) {
throw "MapRendering is undefined in " + context
2021-10-21 01:26:20 +02:00
}
2021-09-09 20:26:12 +02:00
2021-11-09 18:22:05 +01:00
if ( json . mapRendering === null ) {
this . mapRendering = [ ]
this . lineRendering = [ ]
} else {
2021-11-09 19:45:26 +01:00
this . mapRendering = Utils . NoNull ( json . mapRendering )
2021-11-09 18:22:05 +01:00
. filter ( r = > r [ "location" ] !== undefined )
. map ( ( r , i ) = > new PointRenderingConfig ( < PointRenderingConfigJson > r , context + ".mapRendering[" + i + "]" ) )
2021-10-20 02:01:27 +02:00
2021-11-09 19:45:26 +01:00
this . lineRendering = Utils . NoNull ( json . mapRendering )
2021-11-09 18:22:05 +01:00
. filter ( r = > r [ "location" ] === undefined )
. map ( ( r , i ) = > new LineRenderingConfig ( < LineRenderingConfigJson > r , context + ".mapRendering[" + i + "]" ) )
const hasCenterRendering = this . mapRendering . some ( r = > r . location . has ( "centroid" ) || r . location . has ( "start" ) || r . location . has ( "end" ) )
if ( this . lineRendering . length === 0 && this . mapRendering . length === 0 ) {
throw ( "The layer " + this . id + " does not have any maprenderings defined and will thus not show up on the map at all. If this is intentional, set maprenderings to 'null' instead of '[]'" )
2022-01-21 03:57:49 +01:00
} else if ( ! hasCenterRendering && this . lineRendering . length === 0 && ! this . source . geojsonSource ? . startsWith ( "https://api.openstreetmap.org/api/0.6/notes.json" ) ) {
2022-03-24 19:59:46 +01:00
throw "The layer " + this . id + " might not render ways. This might result in dropped information (at " + context + ")"
2021-11-09 18:22:05 +01:00
}
}
2021-10-20 02:01:27 +02:00
2022-01-18 18:12:24 +01:00
const missingIds = Utils . NoNull ( json . tagRenderings ) ? . filter ( tr = > typeof tr !== "string" && tr [ "builtin" ] === undefined && tr [ "id" ] === undefined && tr [ "rewrite" ] === undefined ) ? ? [ ] ;
2021-11-08 03:00:58 +01:00
if ( missingIds ? . length > 0 && official ) {
2021-10-14 03:46:09 +02:00
console . error ( "Some tagRenderings of" , this . id , "are missing an id:" , missingIds )
throw "Missing ids in tagrenderings"
}
2022-01-18 18:12:24 +01:00
this . tagRenderings = ( Utils . NoNull ( json . tagRenderings ) ? ? [ ] ) . map ( ( tr , i ) = > new TagRenderingConfig ( < TagRenderingConfigJson > tr , this . id + ".tagRenderings[" + i + "]" ) )
2021-11-08 03:00:58 +01:00
2022-03-24 19:59:46 +01:00
if ( json . filter !== undefined && json . filter !== null && json . filter [ "sameAs" ] !== undefined ) {
2022-02-01 04:14:54 +01:00
this . filterIsSameAs = json . filter [ "sameAs" ]
this . filters = [ ]
2022-03-24 19:59:46 +01:00
} else {
2022-02-01 04:14:54 +01:00
this . filters = ( < FilterConfigJson [ ] > json . filter ? ? [ ] ) . map ( ( option , i ) = > {
2022-04-01 12:51:55 +02:00
return new FilterConfig ( option , ` layers: ${ this . id } .filter. ${ i } ` )
2022-02-01 04:14:54 +01:00
} ) ;
}
2021-10-14 03:46:09 +02:00
2021-11-08 03:00:58 +01:00
{
2021-12-21 18:35:31 +01:00
const duplicateIds = Utils . Dupiclates ( this . filters . map ( f = > f . id ) )
2021-11-08 03:00:58 +01:00
if ( duplicateIds . length > 0 ) {
throw ` Some filters have a duplicate id: ${ duplicateIds } (at ${ context } .filters) `
}
}
2021-10-14 03:46:09 +02:00
if ( json [ "filters" ] !== undefined ) {
throw "Error in " + context + ": use 'filter' instead of 'filters'"
}
2021-07-23 15:56:22 +02:00
2020-11-16 01:59:30 +01:00
2022-01-26 21:40:38 +01:00
this . titleIcons = this . ParseTagRenderings ( ( < TagRenderingConfigJson [ ] > json . titleIcons ) , {
2021-11-11 17:14:03 +01:00
readOnlyMode : true
} ) ;
2021-03-15 16:23:04 +01:00
2021-10-19 02:31:32 +02:00
this . title = this . tr ( "title" , undefined ) ;
this . isShown = this . tr ( "isShown" , "yes" ) ;
2021-10-22 18:53:07 +02:00
2021-07-23 15:56:22 +02:00
this . deletion = null ;
if ( json . deletion === true ) {
json . deletion = { } ;
}
if ( json . deletion !== undefined && json . deletion !== false ) {
this . deletion = new DeleteConfig ( json . deletion , ` ${ context } .deletion ` ) ;
}
2021-03-15 16:23:04 +01:00
2021-10-14 03:46:09 +02:00
this . allowMove = null
if ( json . allowMove === false ) {
this . allowMove = null ;
} else if ( json . allowMove === true ) {
this . allowMove = new MoveConfig ( { } , context + ".allowMove" )
} else if ( json . allowMove !== undefined && json . allowMove !== false ) {
this . allowMove = new MoveConfig ( json . allowMove , context + ".allowMove" )
}
2021-07-23 15:56:22 +02:00
if ( json [ "showIf" ] !== undefined ) {
throw (
"Invalid key on layerconfig " +
this . id +
": showIf. Did you mean 'isShown' instead?"
) ;
}
2021-01-08 03:57:18 +01:00
}
2021-11-07 16:34:51 +01:00
public defaultIcon ( ) : BaseUIElement | undefined {
2021-11-11 17:14:03 +01:00
if ( this . mapRendering === undefined || this . mapRendering === null ) {
2021-11-09 19:45:26 +01:00
return undefined ;
}
2021-10-31 02:08:39 +01:00
const mapRendering = this . mapRendering . filter ( r = > r . location . has ( "point" ) ) [ 0 ]
if ( mapRendering === undefined ) {
return undefined
}
2022-02-09 22:37:21 +01:00
return mapRendering . GetBaseIcon ( this . GetBaseTags ( ) )
}
2022-03-24 19:59:46 +01:00
public GetBaseTags ( ) : any {
2022-02-09 22:37:21 +01:00
return TagUtils . changeAsProperties ( this . source . osmTags . asChange ( { id : "node/-1" } ) )
2021-10-31 02:08:39 +01:00
}
2021-12-05 02:06:14 +01:00
public GenerateDocumentation ( usedInThemes : string [ ] , layerIsNeededBy : Map < string , string [ ] > , dependencies : {
context? : string ;
reason : string ;
neededLayer : string ;
} [ ] , addedByDefault = false , canBeIncluded = true ) : BaseUIElement {
2021-11-08 02:36:01 +01:00
const extraProps = [ ]
2022-04-06 16:12:01 +02:00
extraProps . push ( "This layer is shown at zoomlevel **" + this . minzoom + "** and higher" )
2021-11-08 02:36:01 +01:00
if ( canBeIncluded ) {
2021-11-09 18:22:05 +01:00
if ( addedByDefault ) {
extraProps . push ( "**This layer is included automatically in every theme. This layer might contain no points**" )
2021-11-08 14:18:45 +01:00
}
2022-01-14 19:34:00 +01:00
if ( this . shownByDefault === false ) {
2021-12-07 18:18:24 +01:00
extraProps . push ( 'This layer is not visible by default and must be enabled in the filter by the user. ' )
}
2021-11-08 02:36:01 +01:00
if ( this . title === undefined ) {
2021-12-07 18:18:24 +01:00
extraProps . push ( "This layer cannot be toggled in the filter view. If you import this layer in your theme, override `title` to make this toggleable." )
}
2022-01-14 19:34:00 +01:00
if ( this . title === undefined && this . shownByDefault === false ) {
2021-12-07 18:18:24 +01:00
extraProps . push ( "This layer is not visible by default and the visibility cannot be toggled, effectively resulting in a fully hidden layer. This can be useful, e.g. to calculate some metatags. If you want to render this layer (e.g. for debugging), enable it by setting the URL-parameter layer-<id>=true" )
2021-11-08 02:36:01 +01:00
}
if ( this . name === undefined ) {
extraProps . push ( "Not visible in the layer selection by default. If you want to make this layer toggable, override `name`" )
}
if ( this . mapRendering . length === 0 ) {
extraProps . push ( "Not rendered on the map by default. If you want to rendering this on the map, override `mapRenderings`" )
}
2022-01-14 19:34:00 +01:00
if ( this . source . geojsonSource !== undefined ) {
extraProps . push ( "<img src='../warning.svg' height='1rem'/> This layer is loaded from an external source, namely `" + this . source . geojsonSource + "`" )
}
2021-11-08 02:36:01 +01:00
} else {
extraProps . push ( "This layer can **not** be included in a theme. It is solely used by [special renderings](SpecialRenderings.md) showing a minimap with custom data." )
}
let usingLayer : BaseUIElement [ ] = [ ]
if ( usedInThemes ? . length > 0 && ! addedByDefault ) {
usingLayer = [ new Title ( "Themes using this layer" , 4 ) ,
new List ( ( usedInThemes ? ? [ ] ) . map ( id = > new Link ( id , "https://mapcomplete.osm.be/" + id ) ) )
]
}
2021-12-05 02:06:14 +01:00
for ( const dep of dependencies ) {
2022-01-14 19:34:00 +01:00
extraProps . push ( new Combine ( [ "This layer will automatically load " , new Link ( dep . neededLayer , "./" + dep . neededLayer + ".md" ) , " into the layout as it depends on it: " , dep . reason , "(" + dep . context + ")" ] ) )
2021-12-05 02:06:14 +01:00
}
2022-01-14 19:34:00 +01:00
2022-03-24 19:59:46 +01:00
for ( const revDep of Utils . Dedup ( layerIsNeededBy ? . get ( this . id ) ? ? [ ] ) ) {
2022-01-14 19:34:00 +01:00
extraProps . push ( new Combine ( [ "This layer is needed as dependency for layer" , new Link ( revDep , "#" + revDep ) ] ) )
}
let neededTags : TagsFilter [ ] = [ this . source . osmTags ]
if ( this . source . osmTags [ "and" ] !== undefined ) {
neededTags = this . source . osmTags [ "and" ]
2021-12-05 02:06:14 +01:00
}
2022-01-14 19:34:00 +01:00
let tableRows = Utils . NoNull ( this . tagRenderings . map ( tr = > tr . FreeformValues ( ) )
. map ( values = > {
if ( values == undefined ) {
return undefined
}
const embedded : ( Link | string ) [ ] = values . values ? . map ( v = > Link . OsmWiki ( values . key , v , true ) ) ? ? [ "_no preset options defined, or no values in them_" ]
return [
new Combine ( [
new Link (
2022-01-26 21:40:38 +01:00
"<img src='https://mapcomplete.osm.be/assets/svg/statistics.svg' height='18px'>" ,
"https://taginfo.openstreetmap.org/keys/" + values . key + "#values"
) , Link . OsmWiki ( values . key )
2022-01-14 19:34:00 +01:00
] ) ,
values . type === undefined ? "Multiple choice" : new Link ( values . type , "../SpecialInputElements.md#" + values . type ) ,
new Combine ( embedded )
] ;
} ) )
let quickOverview : BaseUIElement = undefined ;
if ( tableRows . length > 0 ) {
quickOverview = new Combine ( [
"**Warning** This quick overview is incomplete" ,
new Table ( [ "attribute" , "type" , "values which are supported by this layer" ] , tableRows )
] ) . SetClass ( "flex-col flex" )
2021-12-04 21:49:17 +01:00
}
2022-03-24 19:59:46 +01:00
const icon = this . mapRendering
2022-02-09 22:37:21 +01:00
. filter ( mr = > mr . location . has ( "point" ) )
2022-02-14 04:59:57 +01:00
. map ( mr = > mr . icon ? . render ? . txt )
2022-02-09 22:37:21 +01:00
. find ( i = > i !== undefined )
let iconImg = ""
if ( icon !== undefined ) {
// This is for the documentation, so we have to use raw HTML
iconImg = ` <img src='https://mapcomplete.osm.be/ ${ icon } ' height="100px"> `
}
2022-03-24 19:59:46 +01:00
let overpassLink : BaseUIElement = undefined ;
if ( Constants . priviliged_layers . indexOf ( this . id ) < 0 ) {
try {
overpassLink = new Link ( "Execute on overpass" , Overpass . AsOverpassTurboLink ( < TagsFilter > new And ( neededTags ) . optimize ( ) ) )
} catch ( e ) {
console . error ( "Could not generate overpasslink for " + this . id )
}
}
2021-11-08 02:36:01 +01:00
return new Combine ( [
2022-01-14 19:34:00 +01:00
new Combine ( [
new Title ( this . id , 1 ) ,
iconImg ,
this . description ,
"\n"
] ) . SetClass ( "flex flex-col" ) ,
new List ( extraProps ) ,
. . . usingLayer ,
2021-11-09 18:22:05 +01:00
2022-01-14 19:34:00 +01:00
new Title ( "Basic tags for this layer" , 2 ) ,
"Elements must have the all of following tags to be shown on this layer:" ,
new List ( neededTags . map ( t = > t . asHumanString ( true , false , { } ) ) ) ,
2022-03-24 19:59:46 +01:00
overpassLink ,
2022-01-14 19:34:00 +01:00
new Title ( "Supported attributes" , 2 ) ,
quickOverview ,
. . . this . tagRenderings . map ( tr = > tr . GenerateDocumentation ( ) )
] ) . SetClass ( "flex-col" )
2021-11-08 02:36:01 +01:00
}
2021-07-23 15:56:22 +02:00
public CustomCodeSnippets ( ) : string [ ] {
if ( this . calculatedTags === undefined ) {
return [ ] ;
}
return this . calculatedTags . map ( ( code ) = > code [ 1 ] ) ;
2021-07-22 11:29:09 +02:00
}
2022-01-14 19:34:00 +01:00
AllTagRenderings ( ) : TagRenderingConfig [ ] {
return Utils . NoNull ( [ . . . this . tagRenderings , . . . this . titleIcons , this . title , this . isShown ] )
}
2021-01-08 03:57:18 +01:00
2021-10-22 18:53:07 +02:00
public isLeftRightSensitive ( ) : boolean {
2021-10-22 14:01:40 +02:00
return this . lineRendering . some ( lr = > lr . leftRightSensitive )
}
2021-08-07 23:11:34 +02:00
}