forked from MapComplete/MapComplete
255 lines
No EOL
11 KiB
TypeScript
255 lines
No EOL
11 KiB
TypeScript
import {UIElement} from "../UIElement";
|
|
import {UIEventSource} from "../../Logic/UIEventSource";
|
|
import {LayoutConfigJson} from "../../Customizations/JSON/LayoutConfigJson";
|
|
import SettingsTable from "./SettingsTable";
|
|
import SingleSetting from "./SingleSetting";
|
|
import {SubtleButton} from "../Base/SubtleButton";
|
|
import Combine from "../Base/Combine";
|
|
import {TextField} from "../Input/TextField";
|
|
import {InputElement} from "../Input/InputElement";
|
|
import MultiLingualTextFields from "../Input/MultiLingualTextFields";
|
|
import CheckBox from "../Input/CheckBox";
|
|
import AndOrTagInput from "../Input/AndOrTagInput";
|
|
import TagRenderingPanel from "./TagRenderingPanel";
|
|
import {DropDown} from "../Input/DropDown";
|
|
import {TagRenderingConfigJson} from "../../Customizations/JSON/TagRenderingConfigJson";
|
|
import {MultiInput} from "../Input/MultiInput";
|
|
import {LayerConfigJson} from "../../Customizations/JSON/LayerConfigJson";
|
|
import PresetInputPanel from "./PresetInputPanel";
|
|
import UserDetails from "../../Logic/Osm/OsmConnection";
|
|
import {FixedUiElement} from "../Base/FixedUiElement";
|
|
import ValidatedTextField from "../Input/ValidatedTextField";
|
|
import Svg from "../../Svg";
|
|
import Constants from "../../Models/Constants";
|
|
|
|
/**
|
|
* Shows the configuration for a single layer
|
|
*/
|
|
export default class LayerPanel extends UIElement {
|
|
private readonly _config: UIEventSource<LayoutConfigJson>;
|
|
|
|
private readonly settingsTable: UIElement;
|
|
private readonly mapRendering: UIElement;
|
|
|
|
private readonly deleteButton: UIElement;
|
|
|
|
public readonly titleRendering: UIElement;
|
|
|
|
public readonly selectedTagRendering: UIEventSource<TagRenderingPanel>
|
|
= new UIEventSource<TagRenderingPanel>(undefined);
|
|
private tagRenderings: UIElement;
|
|
private presetsPanel: UIElement;
|
|
|
|
constructor(config: UIEventSource<LayoutConfigJson>,
|
|
languages: UIEventSource<string[]>,
|
|
index: number,
|
|
currentlySelected: UIEventSource<SingleSetting<any>>,
|
|
userDetails: UserDetails) {
|
|
super();
|
|
this._config = config;
|
|
this.mapRendering = this.setupRenderOptions(config, languages, index, currentlySelected, userDetails);
|
|
|
|
const actualDeleteButton = new SubtleButton(
|
|
Svg.delete_icon_ui(),
|
|
"Yes, delete this layer"
|
|
).onClick(() => {
|
|
config.data.layers.splice(index, 1);
|
|
config.ping();
|
|
});
|
|
|
|
this.deleteButton = new CheckBox(
|
|
new Combine(
|
|
[
|
|
"<h3>Confirm layer deletion</h3>",
|
|
new SubtleButton(
|
|
Svg.close_ui(),
|
|
"No, don't delete"
|
|
),
|
|
"<span class='alert'>Deleting a layer can not be undone!</span>",
|
|
actualDeleteButton
|
|
]
|
|
),
|
|
new SubtleButton(
|
|
Svg.delete_icon_ui(),
|
|
"Remove this layer"
|
|
)
|
|
)
|
|
|
|
function setting(input: InputElement<any>, path: string | string[], name: string, description: string | UIElement): SingleSetting<any> {
|
|
let pathPre = ["layers", index];
|
|
if (typeof (path) === "string") {
|
|
pathPre.push(path);
|
|
} else {
|
|
pathPre = pathPre.concat(path);
|
|
}
|
|
|
|
return new SingleSetting<any>(config, input, pathPre, name, description);
|
|
}
|
|
|
|
|
|
this.settingsTable = new SettingsTable([
|
|
setting(new TextField({placeholder:"Layer id"}), "id", "Id", "An identifier for this layer<br/>This should be a simple, lowercase, human readable string that is used to identify the layer."),
|
|
setting(new MultiLingualTextFields(languages), "name", "Name", "The human-readable name of this layer<br/>Used in the layer control panel and the 'Personal theme'"),
|
|
setting(new MultiLingualTextFields(languages, true), "description", "Description", "A description for this layer.<br/>Shown in the layer selections and in the personal theme"),
|
|
setting(ValidatedTextField.NumberInput("nat", n => n < 23), "minzoom", "Minimal zoom",
|
|
"The minimum zoomlevel needed to load and show this layer."),
|
|
setting(new DropDown("", [
|
|
{value: 0, shown: "Show ways and areas as ways and lines"},
|
|
{value: 2, shown: "Show both the ways/areas and the centerpoints"},
|
|
{value: 1, shown: "Show everything as centerpoint"}]), "wayHandling", "Way handling",
|
|
"Describes how ways and areas are represented on the map: areas can be represented as the area itself, or it can be converted into the centerpoint"),
|
|
setting(ValidatedTextField.NumberInput("int", n => n <= 100), "hideUnderlayingFeaturesMinPercentage", "Max allowed overlap percentage",
|
|
"Consider that we want to show 'Nature Reserves' and 'Forests'. Now, ofter, there are pieces of forest mapped _in_ the nature reserve.<br/>" +
|
|
"Now, showing those pieces of forest overlapping with the nature reserve truly clutters the map and is very user-unfriendly.<br/>" +
|
|
"The features are placed layer by layer. If a feature below a feature on this layer overlaps for more then 'x'-percent, the underlying feature is hidden."),
|
|
setting(new AndOrTagInput(), ["osmSource","overpassTags"], "Overpass query",
|
|
"The tags of the objects to load from overpass"),
|
|
|
|
],
|
|
currentlySelected);
|
|
const self = this;
|
|
|
|
const popupTitleRendering = new TagRenderingPanel(languages, currentlySelected, userDetails, {
|
|
title: "Popup title",
|
|
description: "This is the rendering shown as title in the popup for this element",
|
|
disableQuestions: true
|
|
});
|
|
|
|
new SingleSetting(config, popupTitleRendering, ["layers", index, "title"], "Popup title", "This is the rendering shown as title in the popup");
|
|
this.titleRendering = popupTitleRendering;
|
|
this.registerTagRendering(popupTitleRendering);
|
|
|
|
|
|
const renderings = config.map(config => {
|
|
const layer = config.layers[index] as LayerConfigJson;
|
|
// @ts-ignore
|
|
const renderings : TagRenderingConfigJson[] = layer.tagRenderings ;
|
|
return renderings;
|
|
});
|
|
const tagRenderings = new MultiInput<TagRenderingConfigJson>("Add a tag rendering/question",
|
|
() => ({}),
|
|
() => {
|
|
const tagPanel = new TagRenderingPanel(languages, currentlySelected, userDetails)
|
|
self.registerTagRendering(tagPanel);
|
|
return tagPanel;
|
|
}, renderings,
|
|
{allowMovement: true});
|
|
|
|
tagRenderings.GetValue().addCallback(
|
|
tagRenderings => {
|
|
(config.data.layers[index] as LayerConfigJson).tagRenderings = tagRenderings;
|
|
config.ping();
|
|
}
|
|
)
|
|
|
|
if (userDetails.csCount >= Constants.userJourney.themeGeneratorFullUnlock) {
|
|
|
|
const presetPanel = new MultiInput("Add a preset",
|
|
() => ({tags: [], title: {}}),
|
|
() => new PresetInputPanel(currentlySelected, languages),
|
|
undefined, {allowMovement: true});
|
|
new SingleSetting(config, presetPanel, ["layers", index, "presets"], "Presets", "")
|
|
this.presetsPanel = presetPanel;
|
|
} else {
|
|
this.presetsPanel = new FixedUiElement(`Creating a custom theme which also edits OSM is only unlocked after ${Constants.userJourney.themeGeneratorFullUnlock} changesets`).SetClass("alert");
|
|
}
|
|
|
|
function loadTagRenderings() {
|
|
const values = (config.data.layers[index] as LayerConfigJson).tagRenderings;
|
|
const renderings: TagRenderingConfigJson[] = [];
|
|
for (const value of values) {
|
|
if (typeof (value) !== "string") {
|
|
renderings.push(value);
|
|
}
|
|
|
|
}
|
|
tagRenderings.GetValue().setData(renderings);
|
|
}
|
|
|
|
loadTagRenderings();
|
|
|
|
this.tagRenderings = tagRenderings;
|
|
|
|
|
|
}
|
|
|
|
private setupRenderOptions(config: UIEventSource<LayoutConfigJson>,
|
|
languages: UIEventSource<string[]>,
|
|
index: number,
|
|
currentlySelected: UIEventSource<SingleSetting<any>>,
|
|
userDetails: UserDetails
|
|
): UIElement {
|
|
const iconSelect = new TagRenderingPanel(
|
|
languages, currentlySelected, userDetails,
|
|
{
|
|
title: "Icon",
|
|
description: "A visual representation for this layer and for the points on the map.",
|
|
disableQuestions: true,
|
|
noLanguage: true
|
|
});
|
|
const size = new TagRenderingPanel(languages, currentlySelected, userDetails,
|
|
{
|
|
title: "Icon Size",
|
|
description: "The size of the icons on the map in pixels. Can vary based on the tagging",
|
|
disableQuestions: true,
|
|
noLanguage: true
|
|
});
|
|
const color = new TagRenderingPanel(languages, currentlySelected, userDetails,
|
|
{
|
|
title: "Way and area color",
|
|
description: "The color or a shown way or area. Can vary based on the tagging",
|
|
disableQuestions: true,
|
|
noLanguage: true
|
|
});
|
|
const stroke = new TagRenderingPanel(languages, currentlySelected, userDetails,
|
|
{
|
|
title: "Stroke width",
|
|
description: "The width of lines representing ways and the outline of areas. Can vary based on the tags",
|
|
disableQuestions: true,
|
|
noLanguage: true
|
|
});
|
|
this.registerTagRendering(iconSelect);
|
|
this.registerTagRendering(size);
|
|
this.registerTagRendering(color);
|
|
this.registerTagRendering(stroke);
|
|
|
|
function setting(input: InputElement<any>, path, isIcon: boolean = false): SingleSetting<TagRenderingConfigJson> {
|
|
return new SingleSetting(config, input, ["layers", index, path], undefined, undefined)
|
|
}
|
|
|
|
return new SettingsTable([
|
|
setting(iconSelect, "icon"),
|
|
setting(size, "iconSize"),
|
|
setting(color, "color"),
|
|
setting(stroke, "width")
|
|
], currentlySelected);
|
|
}
|
|
|
|
private registerTagRendering(
|
|
tagRenderingPanel: TagRenderingPanel) {
|
|
|
|
tagRenderingPanel.IsHovered().addCallback(isHovering => {
|
|
if (!isHovering) {
|
|
return;
|
|
}
|
|
this.selectedTagRendering.setData(tagRenderingPanel);
|
|
})
|
|
}
|
|
|
|
InnerRender(): string {
|
|
return new Combine([
|
|
"<h2>General layer settings</h2>",
|
|
this.settingsTable,
|
|
"<h2>Popup contents</h2>",
|
|
this.titleRendering,
|
|
this.tagRenderings,
|
|
"<h2>Presets</h2>",
|
|
"Does this theme support adding a new point?<br/>If this should be the case, add a preset. Make sure that the preset tags do match the overpass-tags, otherwise it might seem like the newly added points dissapear ",
|
|
this.presetsPanel,
|
|
"<h2>Map rendering options</h2>",
|
|
this.mapRendering,
|
|
"<h2>Layer delete</h2>",
|
|
this.deleteButton
|
|
]).Render();
|
|
}
|
|
} |