| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  | import {TagRenderingOptions} from "../TagRenderingOptions"; | 
					
						
							|  |  |  | import {LayerDefinition, Preset} from "../LayerDefinition"; | 
					
						
							|  |  |  | import {Layout} from "../Layout"; | 
					
						
							|  |  |  | import Translation from "../../UI/i18n/Translation"; | 
					
						
							|  |  |  | import {type} from "os"; | 
					
						
							|  |  |  | import Combine from "../../UI/Base/Combine"; | 
					
						
							|  |  |  | import {UIElement} from "../../UI/UIElement"; | 
					
						
							|  |  |  | import {And, Tag, TagsFilter} from "../../Logic/TagsFilter"; | 
					
						
							|  |  |  | import FixedText from "../Questions/FixedText"; | 
					
						
							|  |  |  | import {ImageCarouselWithUploadConstructor} from "../../UI/Image/ImageCarouselWithUpload"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  | export interface TagRenderingConfigJson { | 
					
						
							|  |  |  |     // If this key is present, then...
 | 
					
						
							|  |  |  |     key?: string, | 
					
						
							|  |  |  |     // Use this string to render
 | 
					
						
							|  |  |  |     render: string, | 
					
						
							|  |  |  |     // One of string, int, nat, float, pfloat, email, phone. Default: string
 | 
					
						
							|  |  |  |     type?: string, | 
					
						
							|  |  |  |     // If it is not known (and no mapping below matches), this question is asked; a textfield is inserted in the rendering above
 | 
					
						
							|  |  |  |     question?: string, | 
					
						
							|  |  |  |     // If a value is added with the textfield, this extra tag is addded. Optional field
 | 
					
						
							|  |  |  |     addExtraTags?: string | string[] | { k: string, v: string }[]; | 
					
						
							|  |  |  |     // Alternatively, these tags are shown if they match - even if the key above is not there
 | 
					
						
							|  |  |  |     // If unknown, these become a radio button
 | 
					
						
							|  |  |  |     mappings?: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if: string, | 
					
						
							|  |  |  |             then: string | 
					
						
							|  |  |  |         }[] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export interface LayerConfigJson { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     id: string; | 
					
						
							|  |  |  |     icon: string; | 
					
						
							|  |  |  |     title: TagRenderingConfigJson; | 
					
						
							|  |  |  |     description: string; | 
					
						
							|  |  |  |     minzoom: number, | 
					
						
							|  |  |  |     color: string; | 
					
						
							|  |  |  |     overpassTags: string | string[] | { k: string, v: string }[]; | 
					
						
							|  |  |  |     presets: [ | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // icon: optional. Uses the layer icon by default
 | 
					
						
							|  |  |  |             icon?: string; | 
					
						
							|  |  |  |             // title: optional. Uses the layer title by default
 | 
					
						
							|  |  |  |             title?: string; | 
					
						
							|  |  |  |             // description: optional. Uses the layer description by default
 | 
					
						
							|  |  |  |             description?: string; | 
					
						
							|  |  |  |             // tags: optional list {k:string, v:string}[]
 | 
					
						
							|  |  |  |             tags?: string | string[] | { k: string, v: string }[] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ], | 
					
						
							|  |  |  |     tagRenderings: TagRenderingConfigJson [] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export interface LayoutConfigJson { | 
					
						
							|  |  |  |     name: string; | 
					
						
							|  |  |  |     title: string; | 
					
						
							|  |  |  |     description: string; | 
					
						
							|  |  |  |     language: string; | 
					
						
							|  |  |  |     layers: LayerConfigJson[], | 
					
						
							|  |  |  |     startZoom: number; | 
					
						
							|  |  |  |     startLat: number; | 
					
						
							|  |  |  |     startLon: number; | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Either a URL or a base64 encoded value (which should include 'data:image/svg+xml;base64,' | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     icon: string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  | export class CustomLayoutFromJSON { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |     public static exampleLayer: LayerConfigJson = { | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |         id: "bookcase", | 
					
						
							|  |  |  |         icon: "", | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |         title: {render: "Bookcase"}, | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |         description: "A small, public cabinet with books. Anyone can leave or take a book", | 
					
						
							|  |  |  |         minzoom: 12, | 
					
						
							|  |  |  |         color: "#0000ff", | 
					
						
							|  |  |  |         overpassTags: "amenity=public_bookcase", | 
					
						
							|  |  |  |         presets: [ | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |                 title: "bookcase" | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |                 // icon: optional. Uses the layer icon by default
 | 
					
						
							|  |  |  |                 // title: optional. Uses the layer title by default
 | 
					
						
							|  |  |  |                 // description: optional. Uses the layer description by default
 | 
					
						
							|  |  |  |                 // tags: optional list {k:string, v:string}[]
 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ], | 
					
						
							|  |  |  |         tagRenderings: [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 // If this key is present, then...
 | 
					
						
							|  |  |  |                 key: "name", | 
					
						
							|  |  |  |                 // Use this string to render
 | 
					
						
							|  |  |  |                 render: "{name}", | 
					
						
							|  |  |  |                 // One of string, int, nat, float, pfloat, email, phone. Default: string
 | 
					
						
							|  |  |  |                 type: "string", | 
					
						
							|  |  |  |                 // If it is not known (and no mapping below matches), this question is asked; a textfield is inserted in the rendering above
 | 
					
						
							|  |  |  |                 question: "Wat is de naam van dit boekenruilkastje?", | 
					
						
							|  |  |  |                 // If a value is added with the textfield, this extra tag is addded. Optional field
 | 
					
						
							|  |  |  |                 addExtraTags: [{ | 
					
						
							|  |  |  |                     "k": "fixme", | 
					
						
							|  |  |  |                     "v": "Added with mapcomplete, to be checked" | 
					
						
							|  |  |  |                 }], | 
					
						
							|  |  |  |                 // Alternatively, these tags are shown if they match - even if the key above is not there
 | 
					
						
							|  |  |  |                 // If unknown, these become a radio button
 | 
					
						
							|  |  |  |                 mappings: [ | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         if: "noname=yes", | 
					
						
							|  |  |  |                         then: "Dit boekenruilkastje heeft geen naam" | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 ] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |     public static exampleLayout: LayoutConfigJson = { | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |         name: "bookcases", | 
					
						
							|  |  |  |         title: "Custom Open bookcases map", | 
					
						
							|  |  |  |         description: "Welcome to a custom layout", | 
					
						
							|  |  |  |         language: "en", | 
					
						
							|  |  |  |         layers: [CustomLayoutFromJSON.exampleLayer], | 
					
						
							|  |  |  |         startZoom: 12, | 
					
						
							|  |  |  |         startLat: 0, | 
					
						
							|  |  |  |         startLon: 0, | 
					
						
							|  |  |  |         icon: "" | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public static FromQueryParam(layoutFromBase64: string): Layout { | 
					
						
							|  |  |  |         if(layoutFromBase64 === "test"){ | 
					
						
							|  |  |  |             console.log(btoa(JSON.stringify(CustomLayoutFromJSON.exampleLayout))); | 
					
						
							|  |  |  |             return CustomLayoutFromJSON.LayoutFromJSON(CustomLayoutFromJSON.exampleLayout); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const spec = JSON.parse(atob(layoutFromBase64)); | 
					
						
							|  |  |  |         return CustomLayoutFromJSON.LayoutFromJSON(spec); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static TagRenderingFromJson(json: any): TagRenderingOptions { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (typeof (json) === "string") { | 
					
						
							|  |  |  |             return new FixedText(json); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let freeform = undefined; | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |         if (json.key !== undefined && json.key !== "" && json.render !== undefined) { | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |             const type = json.type ?? "text"; | 
					
						
							|  |  |  |             freeform = { | 
					
						
							|  |  |  |                 key: json.key, | 
					
						
							|  |  |  |                 template: json.render.replace("{" + json.key + "}", "$" + type + "$"), | 
					
						
							|  |  |  |                 renderTemplate: json.render, | 
					
						
							|  |  |  |                 extraTags: CustomLayoutFromJSON.TagsFromJson(json.addExtraTags), | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let mappings = undefined; | 
					
						
							|  |  |  |         if (json.mappings !== undefined) { | 
					
						
							|  |  |  |             mappings = []; | 
					
						
							|  |  |  |             for (const mapping of json.mappings) { | 
					
						
							|  |  |  |                 mappings.push({ | 
					
						
							|  |  |  |                     k: new And(CustomLayoutFromJSON.TagsFromJson(mapping.if)), txt: mapping.then | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return new TagRenderingOptions({ | 
					
						
							|  |  |  |             question: json.question, | 
					
						
							|  |  |  |             freeform: freeform, | 
					
						
							|  |  |  |             mappings: mappings | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static PresetFromJson(layout: any, preset: any): Preset { | 
					
						
							|  |  |  |         const t = CustomLayoutFromJSON.MaybeTranslation; | 
					
						
							|  |  |  |         const tags = CustomLayoutFromJSON.TagsFromJson; | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             icon: preset.icon ?? layout.icon, | 
					
						
							|  |  |  |             tags: tags(preset.tags) ?? tags(layout.overpassTags), | 
					
						
							|  |  |  |             title: t(preset.title) ?? t(layout.title), | 
					
						
							|  |  |  |             description: t(preset.description) ?? t(layout.description) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static StyleFromJson(layout: any, styleJson: any): ((tags) => { | 
					
						
							|  |  |  |         color: string, | 
					
						
							|  |  |  |         weight?: number, | 
					
						
							|  |  |  |         icon: { | 
					
						
							|  |  |  |             iconUrl: string, | 
					
						
							|  |  |  |             iconSize: number[], | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |     }) { | 
					
						
							|  |  |  |         return (tags) => { | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 color: layout.color, | 
					
						
							|  |  |  |                 weight: 10, | 
					
						
							|  |  |  |                 icon: { | 
					
						
							|  |  |  |                     iconUrl: layout.icon, | 
					
						
							|  |  |  |                     iconSize: [40, 40], | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |     private static TagFromJson(json: string | { k: string, v: string }): Tag { | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |         if (json === undefined) { | 
					
						
							|  |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |         console.log(json) | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |         if (typeof (json) === "string") { | 
					
						
							|  |  |  |             const kv = json.split("="); | 
					
						
							|  |  |  |             return new Tag(kv[0].trim(), kv[1].trim()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return new Tag(json.k.trim(), json.v.trim()) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-08 21:17:17 +02:00
										 |  |  |     private static TagsFromJson(json: string | { k: string, v: string }[]): Tag[] { | 
					
						
							|  |  |  |         if (json === undefined || json === "") { | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (typeof (json) === "string") { | 
					
						
							|  |  |  |             return json.split(",").map(CustomLayoutFromJSON.TagFromJson); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return json.map(CustomLayoutFromJSON.TagFromJson) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static LayerFromJson(json: any): LayerDefinition { | 
					
						
							|  |  |  |         const t = CustomLayoutFromJSON.MaybeTranslation; | 
					
						
							|  |  |  |         const tr = CustomLayoutFromJSON.TagRenderingFromJson; | 
					
						
							|  |  |  |         return new LayerDefinition( | 
					
						
							|  |  |  |             json.id, | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 description: t(json.description), | 
					
						
							|  |  |  |                 name: t(json.title), | 
					
						
							|  |  |  |                 icon: json.icon, | 
					
						
							|  |  |  |                 minzoom: json.minzoom, | 
					
						
							| 
									
										
										
										
											2020-08-08 21:49:39 +02:00
										 |  |  |                 title: tr(json.title) , | 
					
						
							| 
									
										
										
										
											2020-08-08 17:50:43 +02:00
										 |  |  |                 presets: json.presets.map((preset) => { | 
					
						
							|  |  |  |                     return CustomLayoutFromJSON.PresetFromJson(json, preset) | 
					
						
							|  |  |  |                 }), | 
					
						
							|  |  |  |                 elementsToShow: | 
					
						
							|  |  |  |                     [new ImageCarouselWithUploadConstructor()].concat(json.tagRenderings.map(tr)), | 
					
						
							|  |  |  |                 overpassFilter: new And(CustomLayoutFromJSON.TagsFromJson(json.overpassTags)), | 
					
						
							|  |  |  |                 wayHandling: LayerDefinition.WAYHANDLING_CENTER_AND_WAY, | 
					
						
							|  |  |  |                 maxAllowedOverlapPercentage: 0, | 
					
						
							|  |  |  |                 style: CustomLayoutFromJSON.StyleFromJson(json, json.style) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static MaybeTranslation(json: any): Translation | string { | 
					
						
							|  |  |  |         if (json === undefined) { | 
					
						
							|  |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (typeof (json) === "string") { | 
					
						
							|  |  |  |             return json; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return new Translation(json); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static LayoutFromJSON(json: any) { | 
					
						
							|  |  |  |         const t = CustomLayoutFromJSON.MaybeTranslation; | 
					
						
							|  |  |  |         const layout = new Layout(json.name, | 
					
						
							|  |  |  |             [json.language], | 
					
						
							|  |  |  |             t(json.title), | 
					
						
							|  |  |  |             json.layers.map(CustomLayoutFromJSON.LayerFromJson), | 
					
						
							|  |  |  |             json.startZoom, | 
					
						
							|  |  |  |             json.startLat, | 
					
						
							|  |  |  |             json.startLon, | 
					
						
							|  |  |  |             new Combine(['<h3>', t(json.title), '</h3><br/>', t(json.description)]) | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         layout.icon = json.icon; | 
					
						
							|  |  |  |         return layout; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public static TagRenderingOptionsFromJson(spec: any): TagRenderingOptions { | 
					
						
							|  |  |  |         return new TagRenderingOptions(spec); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |