forked from MapComplete/MapComplete
340 lines
No EOL
12 KiB
TypeScript
340 lines
No EOL
12 KiB
TypeScript
import {Layout} from "../Layout";
|
|
import {LayoutConfigJson} from "./LayoutConfigJson";
|
|
import {AndOrTagConfigJson} from "./TagConfigJson";
|
|
import {And, Or, RegexTag, Tag, TagsFilter} from "../../Logic/Tags";
|
|
import {TagRenderingConfigJson} from "./TagRenderingConfigJson";
|
|
import {TagRenderingOptions} from "../TagRenderingOptions";
|
|
import Translation from "../../UI/i18n/Translation";
|
|
import {LayerConfigJson} from "./LayerConfigJson";
|
|
import {LayerDefinition, Preset} from "../LayerDefinition";
|
|
import {TagDependantUIElementConstructor} from "../UIElementConstructor";
|
|
import Combine from "../../UI/Base/Combine";
|
|
import {ImageCarouselWithUploadConstructor} from "../../UI/Image/ImageCarouselWithUpload";
|
|
import {ImageCarouselConstructor} from "../../UI/Image/ImageCarousel";
|
|
import * as drinkingWater from "../../assets/layers/drinking_water/drinking_water.json";
|
|
import * as ghostbikes from "../../assets/layers/ghost_bike/ghost_bike.json"
|
|
import * as viewpoint from "../../assets/layers/viewpoint/viewpoint.json"
|
|
import * as bike_parking from "../../assets/layers/bike_parking/bike_parking.json"
|
|
import {Utils} from "../../Utils";
|
|
|
|
export class FromJSON {
|
|
|
|
public static sharedLayers: Map<string, LayerDefinition> = FromJSON.getSharedLayers();
|
|
|
|
private static getSharedLayers() {
|
|
const sharedLayers = new Map<string, LayerDefinition>();
|
|
|
|
const sharedLayersList = [
|
|
FromJSON.Layer(drinkingWater),
|
|
FromJSON.Layer(ghostbikes),
|
|
FromJSON.Layer(viewpoint),
|
|
FromJSON.Layer(bike_parking),
|
|
];
|
|
|
|
for (const layer of sharedLayersList) {
|
|
sharedLayers.set(layer.id, layer);
|
|
}
|
|
|
|
return sharedLayers;
|
|
}
|
|
|
|
public static FromBase64(layoutFromBase64: string): Layout {
|
|
return FromJSON.LayoutFromJSON(JSON.parse(atob(layoutFromBase64)));
|
|
}
|
|
|
|
public static LayoutFromJSON(json: LayoutConfigJson): Layout {
|
|
const tr = FromJSON.Translation;
|
|
|
|
const layers = json.layers.map(FromJSON.Layer);
|
|
const roaming: TagDependantUIElementConstructor[] = json.roamingRenderings?.map(FromJSON.TagRendering) ?? [];
|
|
for (const layer of layers) {
|
|
layer.elementsToShow.push(...roaming);
|
|
}
|
|
|
|
const layout = new Layout(
|
|
json.id,
|
|
typeof (json.language) === "string" ? [json.language] : json.language,
|
|
tr(json.title),
|
|
layers,
|
|
json.startZoom,
|
|
json.startLat,
|
|
json.startLon,
|
|
new Combine(["<h3>", tr(json.title), "</h3>", tr(json.description)]),
|
|
);
|
|
|
|
layout.widenFactor = json.widenFactor ?? 0.07;
|
|
layout.icon = json.icon;
|
|
layout.maintainer = json.maintainer;
|
|
layout.version = json.version;
|
|
layout.socialImage = json.socialImage;
|
|
layout.changesetMessage = json.changesetmessage;
|
|
return layout;
|
|
}
|
|
|
|
public static Translation(json: string | any): string | Translation {
|
|
if (json === undefined) {
|
|
return undefined;
|
|
}
|
|
if (typeof (json) === "string") {
|
|
return json;
|
|
}
|
|
const tr = {};
|
|
let keyCount = 0;
|
|
for (let key in json) {
|
|
keyCount ++;
|
|
tr[key] = json[key]; // I'm doing this wrong, I know
|
|
}
|
|
if(keyCount == 0){
|
|
return undefined;
|
|
}
|
|
return new Translation(tr);
|
|
}
|
|
|
|
public static TagRendering(json: TagRenderingConfigJson | string): TagDependantUIElementConstructor {
|
|
return FromJSON.TagRenderingWithDefault(json, "", undefined);
|
|
}
|
|
|
|
public static TagRenderingWithDefault(json: TagRenderingConfigJson | string, propertyName, defaultValue: string): TagDependantUIElementConstructor {
|
|
if (json === undefined) {
|
|
if(defaultValue !== undefined){
|
|
console.log(`Using default value ${defaultValue} for ${propertyName}`)
|
|
return FromJSON.TagRendering(defaultValue);
|
|
}
|
|
throw `Tagrendering ${propertyName} is undefined...`
|
|
}
|
|
|
|
if (typeof json === "string") {
|
|
|
|
switch (json) {
|
|
case "picture": {
|
|
return new ImageCarouselWithUploadConstructor()
|
|
}
|
|
case "pictures": {
|
|
return new ImageCarouselWithUploadConstructor()
|
|
}
|
|
case "image": {
|
|
return new ImageCarouselWithUploadConstructor()
|
|
}
|
|
case "images": {
|
|
return new ImageCarouselWithUploadConstructor()
|
|
}
|
|
case "picturesNoUpload": {
|
|
return new ImageCarouselConstructor()
|
|
}
|
|
}
|
|
|
|
|
|
return new TagRenderingOptions({
|
|
freeform: {
|
|
key: "id",
|
|
renderTemplate: json,
|
|
template: "$$$"
|
|
}
|
|
});
|
|
}
|
|
|
|
let template = FromJSON.Translation(json.render);
|
|
|
|
let freeform = undefined;
|
|
if (json.freeform?.key && json.freeform.key !== "") {
|
|
// Setup the freeform
|
|
if (template === undefined) {
|
|
console.error("Freeform.key is defined, but render is not. This is not allowed.", json)
|
|
throw "Freeform is defined, but render is not. This is not allowed."
|
|
}
|
|
|
|
freeform = {
|
|
template: `$${json.freeform.type ?? "string"}$`,
|
|
renderTemplate: template,
|
|
key: json.freeform.key
|
|
};
|
|
if (json.freeform.addExtraTags) {
|
|
freeform.extraTags = new And(json.freeform.addExtraTags.map(FromJSON.SimpleTag))
|
|
}
|
|
} else if (json.render) {
|
|
// Template (aka rendering) is defined, but freeform.key is not. We allow an input as string
|
|
freeform = {
|
|
template: undefined, // Template to ask is undefined -> we block asking for this key
|
|
renderTemplate: template,
|
|
key: "id" // every object always has an id
|
|
}
|
|
}
|
|
|
|
const mappings = json.mappings?.map(mapping => (
|
|
{
|
|
k: FromJSON.Tag(mapping.if),
|
|
txt: FromJSON.Translation(mapping.then),
|
|
hideInAnswer: mapping.hideInAnswer
|
|
})
|
|
);
|
|
|
|
if(template === undefined && (mappings === undefined || mappings.length === 0)){
|
|
console.error("Empty tagrendering detected: no mappings nor template given", json)
|
|
throw "Empty tagrendering detected: no mappings nor template given"
|
|
}
|
|
|
|
|
|
let rendering = new TagRenderingOptions({
|
|
question: FromJSON.Translation(json.question),
|
|
freeform: freeform,
|
|
mappings: mappings
|
|
});
|
|
|
|
if (json.condition) {
|
|
return rendering.OnlyShowIf(FromJSON.Tag(json.condition));
|
|
}
|
|
|
|
return rendering;
|
|
}
|
|
|
|
public static SimpleTag(json: string): Tag {
|
|
const tag = Utils.SplitFirst(json, "=");
|
|
return new Tag(tag[0], tag[1]);
|
|
}
|
|
|
|
public static Tag(json: AndOrTagConfigJson | string): TagsFilter {
|
|
if(json === undefined){
|
|
throw "Error while parsing a tag: nothing defined. Make sure all the tags are defined and at least one tag is present in a complex expression"
|
|
}
|
|
if (typeof (json) == "string") {
|
|
const tag = json as string;
|
|
if (tag.indexOf("!~") >= 0) {
|
|
const split = Utils.SplitFirst(tag, "!~");
|
|
if (split[1] === "*") {
|
|
split[1] = "..*"
|
|
}
|
|
return new RegexTag(
|
|
split[0],
|
|
new RegExp("^" + split[1] + "$"),
|
|
true
|
|
);
|
|
}
|
|
if (tag.indexOf("!=") >= 0) {
|
|
const split = Utils.SplitFirst(tag, "!=");
|
|
if (split[1] === "*") {
|
|
split[1] = "..*"
|
|
}
|
|
return new RegexTag(
|
|
split[0],
|
|
new RegExp("^" + split[1] + "$"),
|
|
true
|
|
);
|
|
}
|
|
if (tag.indexOf("~") >= 0) {
|
|
const split = Utils.SplitFirst(tag, "~");
|
|
if (split[1] === "*") {
|
|
split[1] = "..*"
|
|
}
|
|
return new RegexTag(
|
|
split[0],
|
|
new RegExp("^" + split[1] + "$")
|
|
);
|
|
}
|
|
const split = Utils.SplitFirst(tag, "=");
|
|
return new Tag(split[0], split[1])
|
|
}
|
|
if (json.and !== undefined) {
|
|
return new And(json.and.map(FromJSON.Tag));
|
|
}
|
|
if (json.or !== undefined) {
|
|
return new Or(json.or.map(FromJSON.Tag));
|
|
}
|
|
}
|
|
|
|
public static Layer(json: LayerConfigJson | string): LayerDefinition {
|
|
|
|
if (typeof (json) === "string") {
|
|
const cached = FromJSON.sharedLayers.get(json);
|
|
if (cached) {
|
|
return cached;
|
|
}
|
|
|
|
throw "Layer not yet loaded..."
|
|
}
|
|
|
|
console.log("Parsing layer", json)
|
|
const tr = FromJSON.Translation;
|
|
const overpassTags = FromJSON.Tag(json.overpassTags);
|
|
const icon = FromJSON.TagRenderingWithDefault(json.icon, "layericon", "./assets/bug.svg");
|
|
const iconSize = FromJSON.TagRenderingWithDefault(json.iconSize, "iconSize", "40,40,center");
|
|
const color = FromJSON.TagRenderingWithDefault(json.color, "layercolor", "#0000ff");
|
|
const width = FromJSON.TagRenderingWithDefault(json.width, "layerwidth", "10");
|
|
|
|
let tagRenderingDefs = json.tagRenderings ?? [];
|
|
if (tagRenderingDefs.indexOf("images") < 0) {
|
|
tagRenderingDefs = ["images", ...tagRenderingDefs];
|
|
}
|
|
let tagRenderings = tagRenderingDefs.map(FromJSON.TagRendering);
|
|
|
|
|
|
const renderTags = {"id": "node/-1"}
|
|
const presets: Preset[] = json?.presets?.map(preset => {
|
|
return ({
|
|
title: tr(preset.title),
|
|
description: tr(preset.description),
|
|
tags: preset.tags.map(FromJSON.SimpleTag)
|
|
});
|
|
}) ?? [];
|
|
|
|
function style(tags) {
|
|
const iconSizeStr =
|
|
iconSize.GetContent(tags).txt.split(",");
|
|
const iconwidth = Number(iconSizeStr[0]);
|
|
const iconheight = Number(iconSizeStr[1]);
|
|
const iconmode = iconSizeStr[2];
|
|
const iconAnchor = [iconwidth / 2, iconheight / 2] // x, y
|
|
// If iconAnchor is set to [0,0], then the top-left of the icon will be placed at the geographical location
|
|
if (iconmode.indexOf("left") >= 0) {
|
|
iconAnchor[0] = 0;
|
|
}
|
|
if (iconmode.indexOf("right") >= 0) {
|
|
iconAnchor[0] = iconwidth;
|
|
}
|
|
|
|
if (iconmode.indexOf("top") >= 0) {
|
|
iconAnchor[1] = 0;
|
|
}
|
|
if (iconmode.indexOf("bottom") >= 0) {
|
|
iconAnchor[1] = iconheight;
|
|
}
|
|
|
|
// the anchor is always set from the center of the point
|
|
// x, y with x going right and y going down if the values are bigger
|
|
const popupAnchor = [0, 3 - iconAnchor[1]];
|
|
|
|
return {
|
|
color: color.GetContent(tags).txt,
|
|
weight: width.GetContent(tags).txt,
|
|
icon: {
|
|
iconUrl: icon.GetContent(tags).txt,
|
|
iconSize: [iconwidth, iconheight],
|
|
popupAnchor: popupAnchor,
|
|
iconAnchor: iconAnchor
|
|
},
|
|
}
|
|
}
|
|
|
|
const layer = new LayerDefinition(
|
|
json.id,
|
|
{
|
|
name: tr(json.name),
|
|
description: tr(json.description),
|
|
icon: icon.GetContent(renderTags).txt,
|
|
overpassFilter: overpassTags,
|
|
|
|
title: FromJSON.TagRendering(json.title),
|
|
minzoom: json.minzoom,
|
|
presets: presets,
|
|
elementsToShow: tagRenderings,
|
|
style: style,
|
|
wayHandling: json.wayHandling
|
|
|
|
}
|
|
);
|
|
console.log("Parsed layer is ", layer);
|
|
return layer;
|
|
|
|
}
|
|
|
|
|
|
} |