diff --git a/Customizations/JSON/CustomLayoutFromJSON.ts b/Customizations/JSON/CustomLayoutFromJSON.ts new file mode 100644 index 000000000..34f52db90 --- /dev/null +++ b/Customizations/JSON/CustomLayoutFromJSON.ts @@ -0,0 +1,221 @@ +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"; + + +export class CustomLayoutFromJSON { + + public static exampleLayer = { + id: "bookcase", + icon: "", + title: "Bookcase", + description: "A small, public cabinet with books. Anyone can leave or take a book", + minzoom: 12, + color: "#0000ff", + overpassTags: "amenity=public_bookcase", + presets: [ + { + // 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" + } + ] + } + ] + } + + public static exampleLayout = { + 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; + if (json.key !== undefined && json.render !== undefined) { + 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], + }, + } + }; + } + + private static TagFromJson(json: any): Tag { + if (json === undefined) { + return undefined; + } + 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()) + } + + private static TagsFromJson(json: any): Tag[] { + if (json === undefined) { + 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, + title: tr(json.title), + 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(['