forked from MapComplete/MapComplete
391 lines
No EOL
14 KiB
TypeScript
391 lines
No EOL
14 KiB
TypeScript
import {UIElement} from "../UIElement";
|
|
import {VerticalCombine} from "../Base/VerticalCombine";
|
|
import {VariableUiElement} from "../Base/VariableUIElement";
|
|
import Combine from "../Base/Combine";
|
|
import {
|
|
LayerConfigJson,
|
|
LayoutConfigJson,
|
|
TagRenderingConfigJson
|
|
} from "../../Customizations/JSON/CustomLayoutFromJSON";
|
|
import {TabbedComponent} from "../Base/TabbedComponent";
|
|
import {UIEventSource} from "../../Logic/UIEventSource";
|
|
import {OsmConnection, UserDetails} from "../../Logic/Osm/OsmConnection";
|
|
import {Button} from "../Base/Button";
|
|
import {FixedUiElement} from "../Base/FixedUiElement";
|
|
import {TextField} from "../Input/TextField";
|
|
|
|
|
|
function TagsToString(tags: string | string [] | { k: string, v: string }[]) {
|
|
if (tags === undefined) {
|
|
return undefined;
|
|
}
|
|
if (typeof (tags) == "string") {
|
|
return tags;
|
|
}
|
|
const newTags = [];
|
|
console.log(tags)
|
|
for (const tag of tags) {
|
|
if (typeof (tag) == "string") {
|
|
newTags.push(tag)
|
|
} else {
|
|
newTags.push(tag.k + "=" + tag.v);
|
|
}
|
|
}
|
|
return newTags.join(",");
|
|
}
|
|
|
|
|
|
let createFieldUI: (label: string, key: string, root: any, options?: { deflt?: string }) => UIElement;
|
|
|
|
|
|
class MappingGenerator extends UIElement {
|
|
|
|
private elements: UIElement[];
|
|
|
|
constructor(fullConfig: UIEventSource<LayoutConfigJson>,
|
|
layerConfig: LayerConfigJson,
|
|
tagRendering: TagRenderingConfigJson,
|
|
mapping: { if: string | string[] | { k: string, v: string }[] }) {
|
|
super(undefined);
|
|
this.CreateElements(fullConfig, layerConfig, tagRendering, mapping)
|
|
}
|
|
|
|
private CreateElements(fullConfig: UIEventSource<LayoutConfigJson>, layerConfig: LayerConfigJson,
|
|
tagRendering: TagRenderingConfigJson,
|
|
mapping) {
|
|
{
|
|
const self = this;
|
|
this.elements = [
|
|
createFieldUI("If these tags apply", "if", mapping),
|
|
createFieldUI("Then: show this text", "then", mapping),
|
|
new Button("Remove this mapping", () => {
|
|
for (let i = 0; i < tagRendering.mappings.length; i++) {
|
|
if (tagRendering.mappings[i] === mapping) {
|
|
tagRendering.mappings.splice(i, 1);
|
|
self.elements = [
|
|
new FixedUiElement("Tag mapping removed")
|
|
]
|
|
self.Update();
|
|
break;
|
|
}
|
|
}
|
|
})
|
|
];
|
|
}
|
|
}
|
|
|
|
InnerRender(): string {
|
|
const combine = new VerticalCombine(this.elements);
|
|
combine.clss = "bordered";
|
|
return combine.Render();
|
|
}
|
|
}
|
|
|
|
class TagRenderingGenerator
|
|
extends UIElement {
|
|
|
|
private elements: UIElement[];
|
|
|
|
constructor(fullConfig: UIEventSource<LayoutConfigJson>,
|
|
layerConfig: LayerConfigJson,
|
|
tagRendering: TagRenderingConfigJson,
|
|
isTitle: boolean = false) {
|
|
super(undefined);
|
|
this.CreateElements(fullConfig, layerConfig, tagRendering, isTitle)
|
|
}
|
|
|
|
private CreateElements(fullConfig: UIEventSource<LayoutConfigJson>, layerConfig: LayerConfigJson, tagRendering: TagRenderingConfigJson, isTitle: boolean) {
|
|
|
|
|
|
const self = this;
|
|
this.elements = [
|
|
new FixedUiElement(isTitle ? "<h3>Popup title</h3>" : "<h3>TagRendering/TagQuestion</h3>"),
|
|
createFieldUI("Key", "key", tagRendering),
|
|
createFieldUI("Rendering", "render", tagRendering),
|
|
createFieldUI("Type", "type", tagRendering),
|
|
createFieldUI("Question", "question", tagRendering),
|
|
createFieldUI("Extra tags", "addExtraTags", tagRendering),
|
|
|
|
...(tagRendering.mappings ?? []).map((mapping) => {
|
|
return new MappingGenerator(fullConfig, layerConfig, tagRendering, mapping)
|
|
}),
|
|
new Button("Add mapping", () => {
|
|
if (tagRendering.mappings === undefined) {
|
|
tagRendering.mappings = []
|
|
}
|
|
tagRendering.mappings.push({if: "", then: ""});
|
|
self.CreateElements(fullConfig, layerConfig, tagRendering, isTitle);
|
|
self.Update();
|
|
})
|
|
|
|
]
|
|
|
|
if (!isTitle) {
|
|
const b = new Button("Remove this preset", () => {
|
|
for (let i = 0; i < layerConfig.tagRenderings.length; i++) {
|
|
if (layerConfig.tagRenderings[i] === tagRendering) {
|
|
layerConfig.tagRenderings.splice(i, 1);
|
|
self.elements = [
|
|
new FixedUiElement("Tag rendering removed")
|
|
]
|
|
self.Update();
|
|
break;
|
|
}
|
|
}
|
|
});
|
|
this.elements.push(b);
|
|
}
|
|
|
|
}
|
|
|
|
InnerRender(): string {
|
|
const combine = new VerticalCombine(this.elements);
|
|
combine.clss = "bordered";
|
|
return combine.Render();
|
|
}
|
|
}
|
|
|
|
class PresetGenerator extends UIElement {
|
|
|
|
private elements: UIElement[];
|
|
|
|
constructor(fullConfig: UIEventSource<LayoutConfigJson>, layerConfig: LayerConfigJson,
|
|
preset0: { title?: string, description?: string, icon?: string, tags?: string | string[] | { k: string, v: string }[] }) {
|
|
super(undefined);
|
|
const self = this;
|
|
this.elements = [
|
|
new FixedUiElement("<h3>Preset</h3>"),
|
|
createFieldUI("Title", "title", preset0),
|
|
createFieldUI("Description", "description", preset0, {deflt: layerConfig.description}),
|
|
createFieldUI("icon", "icon", preset0, {deflt: layerConfig.icon}),
|
|
createFieldUI("tags", "tags", preset0, {deflt: TagsToString(layerConfig.overpassTags)}),
|
|
new Button("Remove this preset", () => {
|
|
for (let i = 0; i < layerConfig.presets.length; i++) {
|
|
if (layerConfig.presets[i] === preset0) {
|
|
layerConfig.presets.splice(i, 1);
|
|
self.elements = [
|
|
new FixedUiElement("Preset removed")
|
|
]
|
|
self.Update();
|
|
break;
|
|
}
|
|
}
|
|
})
|
|
]
|
|
|
|
}
|
|
|
|
InnerRender(): string {
|
|
const combine = new VerticalCombine(this.elements);
|
|
combine.clss = "bordered";
|
|
return combine.Render();
|
|
}
|
|
|
|
}
|
|
|
|
class LayerGenerator extends UIElement {
|
|
private fullConfig: UIEventSource<LayoutConfigJson>;
|
|
private layerConfig: UIEventSource<LayerConfigJson>;
|
|
private generateField: ((label: string, key: string, root: any, deflt?: string) => UIElement);
|
|
private uielements: UIElement[];
|
|
|
|
constructor(fullConfig: UIEventSource<LayoutConfigJson>,
|
|
layerConfig: LayerConfigJson) {
|
|
super(undefined);
|
|
this.layerConfig = new UIEventSource<LayerConfigJson>(layerConfig);
|
|
this.fullConfig = fullConfig;
|
|
this.CreateElements(fullConfig, layerConfig)
|
|
|
|
}
|
|
|
|
private CreateElements(fullConfig: UIEventSource<LayoutConfigJson>, layerConfig: LayerConfigJson) {
|
|
const self = this;
|
|
this.uielements = [
|
|
createFieldUI("The name of this layer", "id", layerConfig),
|
|
createFieldUI("A description of objects for this layer", "description", layerConfig),
|
|
createFieldUI("The icon of this layer, either a URL or a base64-encoded svg", "icon", layerConfig),
|
|
createFieldUI("The default stroke color", "color", layerConfig),
|
|
createFieldUI("The minimal needed zoom to start loading", "minzoom", layerConfig),
|
|
createFieldUI("The tags to load from overpass", "overpassTags", layerConfig),
|
|
...layerConfig.presets.map(preset => new PresetGenerator(fullConfig, layerConfig, preset)),
|
|
new Button("Add a preset", () => {
|
|
layerConfig.presets.push({
|
|
icon: undefined,
|
|
title: "",
|
|
description: "",
|
|
tags: TagsToString(layerConfig.overpassTags)
|
|
});
|
|
self.CreateElements(fullConfig, layerConfig);
|
|
self.Update();
|
|
}),
|
|
new TagRenderingGenerator(fullConfig, layerConfig, layerConfig.title ?? {
|
|
key: "",
|
|
addExtraTags: "",
|
|
mappings: [],
|
|
question: "",
|
|
render: "Title",
|
|
type: "text"
|
|
}, true),
|
|
...layerConfig.tagRenderings.map(tr => new TagRenderingGenerator(fullConfig, layerConfig, tr)),
|
|
new Button("Add a tag rendering", () => {
|
|
layerConfig.tagRenderings.push({
|
|
key: "",
|
|
addExtraTags: "",
|
|
mappings: [],
|
|
question: "",
|
|
render: "",
|
|
type: "text"
|
|
});
|
|
self.CreateElements(fullConfig, layerConfig);
|
|
self.Update();
|
|
}),
|
|
|
|
]
|
|
}
|
|
|
|
InnerRender(): string {
|
|
return new VerticalCombine(this.uielements).Render();
|
|
}
|
|
}
|
|
|
|
|
|
class AllLayerComponent extends UIElement {
|
|
|
|
private tabs: TabbedComponent;
|
|
private config: UIEventSource<LayoutConfigJson>;
|
|
|
|
constructor(config: UIEventSource<LayoutConfigJson>) {
|
|
super(undefined);
|
|
this.config = config;
|
|
const self = this;
|
|
let previousLayerAmount = config.data.layers.length;
|
|
config.addCallback((data) => {
|
|
if (data.layers.length != previousLayerAmount) {
|
|
previousLayerAmount = data.layers.length;
|
|
self.UpdateTabs();
|
|
self.Update();
|
|
}
|
|
});
|
|
this.UpdateTabs();
|
|
|
|
}
|
|
|
|
private UpdateTabs() {
|
|
const layerPanes: { header: UIElement | string, content: UIElement | string }[] = [];
|
|
const config = this.config;
|
|
for (const layer of this.config.data.layers) {
|
|
const header = this.config.map(() => {
|
|
return `<img src="${layer?.icon ?? "./assets/help.svg"}">`
|
|
});
|
|
layerPanes.push({
|
|
header: new VariableUiElement(header),
|
|
content: new LayerGenerator(config, layer)
|
|
})
|
|
}
|
|
|
|
|
|
layerPanes.push({
|
|
header: "<img src='./assets/add.svg'>",
|
|
content: new Button("Add a new layer", () => {
|
|
config.data.layers.push({
|
|
id: "",
|
|
title: {
|
|
render: "Title"
|
|
},
|
|
icon: "./assets/bug.svg",
|
|
color: "",
|
|
description: "",
|
|
minzoom: 12,
|
|
overpassTags: "",
|
|
presets: [{}],
|
|
tagRenderings: []
|
|
});
|
|
|
|
config.ping();
|
|
})
|
|
})
|
|
|
|
this.tabs = new TabbedComponent(layerPanes);
|
|
}
|
|
|
|
InnerRender(): string {
|
|
return this.tabs.Render();
|
|
}
|
|
|
|
}
|
|
|
|
|
|
export class ThemeGenerator extends UIElement {
|
|
|
|
private readonly userDetails: UIEventSource<UserDetails>;
|
|
|
|
public readonly themeObject: UIEventSource<LayoutConfigJson>;
|
|
private readonly allQuestionFields: UIElement[];
|
|
public url: UIEventSource<string>;
|
|
|
|
|
|
constructor(connection: OsmConnection, windowHash) {
|
|
super(connection.userDetails);
|
|
this.userDetails = connection.userDetails;
|
|
|
|
const defaultTheme = {layers: [], icon: "./assets/bug.svg"};
|
|
let loadedTheme = undefined;
|
|
if (windowHash !== undefined && windowHash.length > 4) {
|
|
loadedTheme = JSON.parse(atob(windowHash));
|
|
}
|
|
this.themeObject = new UIEventSource<LayoutConfigJson>(loadedTheme ?? defaultTheme);
|
|
const jsonObjectRoot = this.themeObject.data;
|
|
|
|
const base64 = this.themeObject.map(JSON.stringify).map(btoa);
|
|
this.url = base64.map((data) => `https://pietervdvn.github.io/MapComplete/index.html?test=true&userlayout=true#` + data);
|
|
const self = this;
|
|
|
|
createFieldUI = (label, key, root, options) => {
|
|
|
|
const value = new UIEventSource<string>(TagsToString(root[key]) ?? options?.deflt);
|
|
value.addCallback((v) => {
|
|
root[key] = v;
|
|
self.themeObject.ping(); // We assume the root is a part of the themeObject
|
|
})
|
|
return new Combine([
|
|
label,
|
|
new TextField<string>({
|
|
fromString: (str) => str,
|
|
toString: (str) => str,
|
|
value: value
|
|
})]);
|
|
}
|
|
|
|
this.allQuestionFields = [
|
|
createFieldUI("Name of this theme", "name", jsonObjectRoot),
|
|
createFieldUI("Title (shown in the window and in the welcome message)", "title", jsonObjectRoot),
|
|
createFieldUI("Description (shown in the welcome message and various other places)", "description", jsonObjectRoot),
|
|
createFieldUI("The supported language", "language", jsonObjectRoot),
|
|
createFieldUI("startLat", "startLat", jsonObjectRoot),
|
|
createFieldUI("startLon", "startLon", jsonObjectRoot),
|
|
createFieldUI("startzoom", "startZoom", jsonObjectRoot),
|
|
createFieldUI("icon: either a URL to an image file, a relative url to a MapComplete asset ('./asset/help.svg') or a base64-encoded value (including 'data:image/svg+xml;base64,'", "icon", jsonObjectRoot, {deflt: "./assets/bug.svg"}),
|
|
|
|
new AllLayerComponent(this.themeObject)
|
|
]
|
|
|
|
|
|
}
|
|
|
|
InnerRender(): string {
|
|
|
|
if (!this.userDetails.data.loggedIn) {
|
|
return "Not logged in. You need to be logged in to create a theme."
|
|
}
|
|
if (this.userDetails.data.csCount < 500) {
|
|
return "You need at least 500 changesets to create your own theme.";
|
|
}
|
|
|
|
|
|
return new VerticalCombine([
|
|
// new VariableUiElement(this.themeObject.map(JSON.stringify)),
|
|
// new VariableUiElement(this.url.map((url) => `Current URL: <a href="${url}" target="_blank">Click here to open</a>`)),
|
|
...this.allQuestionFields,
|
|
]).Render();
|
|
}
|
|
} |