import Combine from "./Base/Combine" import {FixedUiElement} from "./Base/FixedUiElement" import BaseUIElement from "./BaseUIElement" import Title from "./Base/Title" import Table from "./Base/Table" import {RenderingSpecification, SpecialVisualization, SpecialVisualizationState,} from "./SpecialVisualization" import {HistogramViz} from "./Popup/HistogramViz" import {MinimapViz} from "./Popup/MinimapViz" import {ShareLinkViz} from "./Popup/ShareLinkViz" import {UploadToOsmViz} from "./Popup/UploadToOsmViz" import {MultiApplyViz} from "./Popup/MultiApplyViz" import {AddNoteCommentViz} from "./Popup/AddNoteCommentViz" import {PlantNetDetectionViz} from "./Popup/PlantNetDetectionViz" import {ConflateButton, ImportWayButton} from "./Popup/ImportButton" import TagApplyButton from "./Popup/TagApplyButton" import {CloseNoteButton} from "./Popup/CloseNoteButton" import {MapillaryLinkVis} from "./Popup/MapillaryLinkVis" import {Store, Stores, UIEventSource} from "../Logic/UIEventSource" import AllTagsPanel from "./Popup/AllTagsPanel.svelte" import AllImageProviders from "../Logic/ImageProviders/AllImageProviders" import {ImageCarousel} from "./Image/ImageCarousel" import {ImageUploadFlow} from "./Image/ImageUploadFlow" import {VariableUiElement} from "./Base/VariableUIElement" import {Utils} from "../Utils" import Wikidata, {WikidataResponse} from "../Logic/Web/Wikidata" import {Translation} from "./i18n/Translation" import Translations from "./i18n/Translations" import ReviewForm from "./Reviews/ReviewForm" import ReviewElement from "./Reviews/ReviewElement" import OpeningHoursVisualization from "./OpeningHours/OpeningHoursVisualization" import LiveQueryHandler from "../Logic/Web/LiveQueryHandler" import {SubtleButton} from "./Base/SubtleButton" import Svg from "../Svg" import {OpenIdEditor, OpenJosm} from "./BigComponents/CopyrightPanel" import Hash from "../Logic/Web/Hash" import NoteCommentElement from "./Popup/NoteCommentElement" import ImgurUploader from "../Logic/ImageProviders/ImgurUploader" import FileSelectorButton from "./Input/FileSelectorButton" import {LoginToggle} from "./Popup/LoginButton" import Toggle from "./Input/Toggle" import {SubstitutedTranslation} from "./SubstitutedTranslation" import List from "./Base/List" import StatisticsPanel from "./BigComponents/StatisticsPanel" import AutoApplyButton from "./Popup/AutoApplyButton" import {LanguageElement} from "./Popup/LanguageElement" import FeatureReviews from "../Logic/Web/MangroveReviews" import Maproulette from "../Logic/Maproulette" import SvelteUIElement from "./Base/SvelteUIElement" import {BBoxFeatureSourceForLayer} from "../Logic/FeatureSource/Sources/TouchesBboxFeatureSource" import QuestionViz from "./Popup/QuestionViz" import {Feature, Point} from "geojson" import {GeoOperations} from "../Logic/GeoOperations" import CreateNewNote from "./Popup/CreateNewNote.svelte" import AddNewPoint from "./Popup/AddNewPoint/AddNewPoint.svelte" import UserProfile from "./BigComponents/UserProfile.svelte" import LanguagePicker from "./LanguagePicker" import Link from "./Base/Link" import LayerConfig from "../Models/ThemeConfig/LayerConfig" import TagRenderingConfig from "../Models/ThemeConfig/TagRenderingConfig" import NearbyImages, {NearbyImageOptions, P4CPicture, SelectOneNearbyImage,} from "./Popup/NearbyImages" import {Tag} from "../Logic/Tags/Tag" import ChangeTagAction from "../Logic/Osm/Actions/ChangeTagAction" import {And} from "../Logic/Tags/And" import {SaveButton} from "./Popup/SaveButton" import Lazy from "./Base/Lazy" import {CheckBox} from "./Input/Checkboxes" import Slider from "./Input/Slider" import DeleteWizard from "./Popup/DeleteWizard" import {OsmId, OsmTags, WayId} from "../Models/OsmFeature" import MoveWizard from "./Popup/MoveWizard" import SplitRoadWizard from "./Popup/SplitRoadWizard" import {ExportAsGpxViz} from "./Popup/ExportAsGpxViz" import WikipediaPanel from "./Wikipedia/WikipediaPanel.svelte" import TagRenderingEditable from "./Popup/TagRendering/TagRenderingEditable.svelte"; import {ImportPointButton} from "./Popup/ImportButtons/ImportPointButton"; class NearbyImageVis implements SpecialVisualization { // Class must be in SpecialVisualisations due to weird cyclical import that breaks the tests args: { name: string; defaultValue?: string; doc: string; required?: boolean }[] = [ { name: "mode", defaultValue: "expandable", doc: "Indicates how this component is initialized. Options are: \n\n- `open`: always show and load the pictures\n- `collapsable`: show the pictures, but a user can collapse them\n- `expandable`: shown by default; but a user can collapse them.", }, { name: "mapillary", defaultValue: "true", doc: "If 'true', includes a link to mapillary on this location.", }, ] docs = "A component showing nearby images loaded from various online services such as Mapillary. In edit mode and when used on a feature, the user can select an image to add to the feature" funcName = "nearby_images" constr( state: SpecialVisualizationState, tagSource: UIEventSource>, args: string[], feature: Feature, layer: LayerConfig ): BaseUIElement { const t = Translations.t.image.nearbyPictures const mode: "open" | "expandable" | "collapsable" = args[0] const [lon, lat] = GeoOperations.centerpointCoordinates(feature) const id: string = tagSource.data["id"] const canBeEdited: boolean = !!id?.match("(node|way|relation)/-?[0-9]+") const selectedImage = new UIEventSource(undefined) let saveButton: BaseUIElement = undefined if (canBeEdited) { const confirmText: BaseUIElement = new SubstitutedTranslation( t.confirm, tagSource, state ) const onSave = async () => { console.log("Selected a picture...", selectedImage.data) const osmTags = selectedImage.data.osmTags const tags: Tag[] = [] for (const key in osmTags) { tags.push(new Tag(key, osmTags[key])) } await state?.changes?.applyAction( new ChangeTagAction(id, new And(tags), tagSource.data, { theme: state?.layout.id, changeType: "link-image", }) ) } saveButton = new SaveButton( selectedImage, state, confirmText, t.noImageSelected ) .onClick(onSave) .SetClass("flex justify-end") } const nearby = new Lazy(() => { const towardsCenter = new CheckBox(t.onlyTowards, false) const maxSearchRadius = 100 const stepSize = 10 const defaultValue = Math.floor(maxSearchRadius / (2 * stepSize)) * stepSize const fromOsmPreferences = state?.osmConnection ?.GetPreference("nearby-images-radius", "" + defaultValue) .sync( (s) => Number(s), [], (i) => "" + i ) const radiusValue = new UIEventSource(fromOsmPreferences.data) radiusValue.addCallbackAndRunD((v) => fromOsmPreferences.setData(v)) const radius = new Slider(stepSize, maxSearchRadius, { value: radiusValue, step: 10, }) const alreadyInTheImage = AllImageProviders.LoadImagesFor(tagSource) const options: NearbyImageOptions & { value } = { lon, lat, searchRadius: maxSearchRadius, shownRadius: radius.GetValue(), value: selectedImage, blacklist: alreadyInTheImage, towardscenter: towardsCenter.GetValue(), maxDaysOld: 365 * 3, } const slideshow = canBeEdited ? new SelectOneNearbyImage(options, state.indexedFeatures) : new NearbyImages(options, state.indexedFeatures) const controls = new Combine([ towardsCenter, new Combine([ new VariableUiElement( radius.GetValue().map((radius) => t.withinRadius.Subs({radius})) ), radius, ]).SetClass("flex justify-between"), ]).SetClass("flex flex-col") return new Combine([ slideshow, controls, saveButton, new MapillaryLinkVis().constr(state, tagSource, [], feature).SetClass("mt-6"), ]) }) let withEdit: BaseUIElement = nearby if (canBeEdited) { withEdit = new Combine([t.hasMatchingPicture, nearby]).SetClass("flex flex-col") } if (mode === "open") { return withEdit } const toggleState = new UIEventSource(mode === "collapsable") return new Toggle( new Combine([new Title(t.title), withEdit]), new Title(t.browseNearby).onClick(() => toggleState.setData(true)), toggleState ) } } class StealViz implements SpecialVisualization { // Class must be in SpecialVisualisations due to weird cyclical import that breaks the tests funcName = "steal" docs = "Shows a tagRendering from a different object as if this was the object itself" args = [ { name: "featureId", doc: "The key of the attribute which contains the id of the feature from which to use the tags", required: true, }, { name: "tagRenderingId", doc: "The layer-id and tagRenderingId to render. Can be multiple value if ';'-separated (in which case every value must also contain the layerId, e.g. `layerId.tagRendering0; layerId.tagRendering1`). Note: this can cause layer injection", required: true, }, ] constr(state: SpecialVisualizationState, featureTags, args) { const [featureIdKey, layerAndtagRenderingIds] = args const tagRenderings: [LayerConfig, TagRenderingConfig][] = [] for (const layerAndTagRenderingId of layerAndtagRenderingIds.split(";")) { const [layerId, tagRenderingId] = layerAndTagRenderingId.trim().split(".") const layer = state.layout.layers.find((l) => l.id === layerId) const tagRendering = layer.tagRenderings.find((tr) => tr.id === tagRenderingId) tagRenderings.push([layer, tagRendering]) } if (tagRenderings.length === 0) { throw "Could not create stolen tagrenddering: tagRenderings not found" } return new VariableUiElement( featureTags.map((tags) => { const featureId = tags[featureIdKey] if (featureId === undefined) { return undefined } const otherTags = state.featureProperties.getStore(featureId) const otherFeature = state.indexedFeatures.featuresById.data.get(featureId); const elements: BaseUIElement[] = [] for (const [layer, tagRendering] of tagRenderings) { elements.push(new SvelteUIElement(TagRenderingEditable, { config: tagRendering, tags: otherTags, selectedElement: otherFeature, state, layer })) } if (elements.length === 1) { return elements[0] } return new Combine(elements).SetClass("flex flex-col") }, [state.indexedFeatures.featuresById]) ) } getLayerDependencies(args): string[] { const [_, tagRenderingId] = args if (tagRenderingId.indexOf(".") < 0) { throw "Error: argument 'layerId.tagRenderingId' of special visualisation 'steal' should contain a dot" } const [layerId, __] = tagRenderingId.split(".") return [layerId] } } export default class SpecialVisualizations { public static specialVisualizations: SpecialVisualization[] = SpecialVisualizations.initList() static undoEncoding(str: string) { return str .trim() .replace(/&LPARENS/g, "(") .replace(/&RPARENS/g, ")") .replace(/&LBRACE/g, "{") .replace(/&RBRACE/g, "}") .replace(/&COMMA/g, ",") } /** * * For a given string, returns a specification what parts are fixed and what parts are special renderings. * Note that _normal_ substitutions are ignored. * * // Return empty list on empty input * SpecialVisualizations.constructSpecification("") // => [] * * // Advanced cases with commas, braces and newlines should be handled without problem * const templates = SpecialVisualizations.constructSpecification("{send_email(&LBRACEemail&RBRACE,Broken bicycle pump,Hello&COMMA\n\nWith this email&COMMA I'd like to inform you that the bicycle pump located at https://mapcomplete.osm.be/cyclofix?lat=&LBRACE_lat&RBRACE&lon=&LBRACE_lon&RBRACE&z=18#&LBRACEid&RBRACE is broken.\n\n Kind regards,Report this bicycle pump as broken)}") * const templ = > templates[0] * templ.func.funcName // => "send_email" * templ.args[0] = "{email}" */ public static constructSpecification( template: string, extraMappings: SpecialVisualization[] = [] ): RenderingSpecification[] { if (template === "") { return [] } if (template["type"] !== undefined) { console.trace( "Got a non-expanded template while constructing the specification:", template ) throw "Got a non-expanded template while constructing the specification" } const allKnownSpecials = extraMappings.concat(SpecialVisualizations.specialVisualizations) for (const knownSpecial of allKnownSpecials) { // Note: the '.*?' in the regex reads as 'any character, but in a non-greedy way' const matched = template.match( new RegExp(`(.*){${knownSpecial.funcName}\\((.*?)\\)(:.*)?}(.*)`, "s") ) if (matched != null) { // We found a special component that should be brought to live const partBefore = SpecialVisualizations.constructSpecification( matched[1], extraMappings ) const argument = matched[2] /* .trim() // We don't trim, as spaces might be relevant, e.g. "what is ... of {title()}"*/ const style = matched[3]?.substring(1) ?? "" const partAfter = SpecialVisualizations.constructSpecification( matched[4], extraMappings ) const args = knownSpecial.args.map((arg) => arg.defaultValue ?? "") if (argument.length > 0) { const realArgs = argument.split(",").map((str) => this.undoEncoding(str)) for (let i = 0; i < realArgs.length; i++) { if (args.length <= i) { args.push(realArgs[i]) } else { args[i] = realArgs[i] } } } const element: RenderingSpecification = { args: args, style: style, func: knownSpecial, } return [...partBefore, element, ...partAfter] } } // Let's to a small sanity check to help the theme designers: if (template.search(/{[^}]+\([^}]*\)}/) >= 0) { // Hmm, we might have found an invalid rendering name let suggestion = "" if (SpecialVisualizations.specialVisualizations?.length > 0) { suggestion = "did you mean one of: " + SpecialVisualizations.specialVisualizations .map((sp) => sp.funcName + "()") .join(", ") } console.warn( "Found a suspicious special rendering value in: ", template, suggestion ) } // IF we end up here, no changes have to be made - except to remove any resting {} return [template] } public static DocumentationFor(viz: string | SpecialVisualization): BaseUIElement | undefined { if (typeof viz === "string") { viz = SpecialVisualizations.specialVisualizations.find((sv) => sv.funcName === viz) } if (viz === undefined) { return undefined } return new Combine([ new Title(viz.funcName, 3), viz.docs, viz.args.length > 0 ? new Table( ["name", "default", "description"], viz.args.map((arg) => { let defaultArg = arg.defaultValue ?? "_undefined_" if (defaultArg == "") { defaultArg = "_empty string_" } return [arg.name, defaultArg, arg.doc] }) ) : undefined, new Title("Example usage of " + viz.funcName, 4), new FixedUiElement( viz.example ?? "`{" + viz.funcName + "(" + viz.args.map((arg) => arg.defaultValue).join(",") + ")}`" ).SetClass("literal-code"), ]) } public static HelpMessage() { const helpTexts = SpecialVisualizations.specialVisualizations.map((viz) => SpecialVisualizations.DocumentationFor(viz) ) return new Combine([ new Combine([ new Title("Special tag renderings", 1), "In a tagrendering, some special values are substituted by an advanced UI-element. This allows advanced features and visualizations to be reused by custom themes or even to query third-party API's.", "General usage is `{func_name()}`, `{func_name(arg, someotherarg)}` or `{func_name(args):cssStyle}`. Note that you _do not_ need to use quotes around your arguments, the comma is enough to separate them. This also implies you cannot use a comma in your args", new Title("Using expanded syntax", 4), `Instead of using \`{"render": {"en": "{some_special_visualisation(some_arg, some other really long message, more args)} , "nl": "{some_special_visualisation(some_arg, een boodschap in een andere taal, more args)}}\`, one can also write`, new FixedUiElement( JSON.stringify( { render: { special: { type: "some_special_visualisation", argname: "some_arg", message: { en: "some other really long message", nl: "een boodschap in een andere taal", }, other_arg_name: "more args", }, before: { en: "Some text to prefix before the special element (e.g. a title)", nl: "Een tekst om voor het element te zetten (bv. een titel)", }, after: { en: "Some text to put after the element, e.g. a footer", }, }, }, null, " " ) ).SetClass("code"), 'In other words: use `{ "before": ..., "after": ..., "special": {"type": ..., "argname": ...argvalue...}`. The args are in the `special` block; an argvalue can be a string, a translation or another value. (Refer to class `RewriteSpecial` in case of problems)', ]).SetClass("flex flex-col"), ...helpTexts, ]).SetClass("flex flex-col") } // noinspection JSUnusedGlobalSymbols public static renderExampleOfSpecial( state: SpecialVisualizationState, s: SpecialVisualization ): BaseUIElement { const examples = s.structuredExamples === undefined ? [] : s.structuredExamples().map((e) => { return s.constr( state, new UIEventSource>(e.feature.properties), e.args, e.feature, undefined ) }) return new Combine([new Title(s.funcName), s.docs, ...examples]) } private static initList(): SpecialVisualization[] { const specialVisualizations: SpecialVisualization[] = [ new QuestionViz(), { funcName: "add_new_point", docs: "An element which allows to add a new point on the 'last_click'-location. Only makes sense in the layer `last_click`", args: [], constr(state: SpecialVisualizationState, _, __, feature): BaseUIElement { let [lon, lat] = GeoOperations.centerpointCoordinates(feature) return new SvelteUIElement(AddNewPoint, { state, coordinate: {lon, lat}, }) }, }, { funcName: "user_profile", args: [], docs: "A component showing information about the currently logged in user (username, profile description, profile picture + link to edit them). Mostly meant to be used in the 'user-settings'", constr(state: SpecialVisualizationState): BaseUIElement { return new SvelteUIElement(UserProfile, { osmConnection: state.osmConnection, }) }, }, { funcName: "language_picker", args: [], docs: "A component to set the language of the user interface", constr(state: SpecialVisualizationState): BaseUIElement { return new LanguagePicker( state.layout.language, Translations.t.general.pickLanguage.Clone() ) }, }, { funcName: "logout", args: [], docs: "Shows a button where the user can log out", constr(state: SpecialVisualizationState): BaseUIElement { return new SubtleButton(Svg.logout_svg(), Translations.t.general.logout, { imgSize: "w-6 h-6", }).onClick(() => { state.osmConnection.LogOut() }) }, }, new HistogramViz(), new StealViz(), new MinimapViz(), { funcName: "split_button", docs: "Adds a button which allows to split a way", args: [], constr( state: SpecialVisualizationState, tagSource: UIEventSource> ): BaseUIElement { return new VariableUiElement( tagSource .map((tags) => tags.id) .map((id) => { if (id.startsWith("way/")) { return new SplitRoadWizard(id, state) } return undefined }) ) }, }, { funcName: "move_button", docs: "Adds a button which allows to move the object to another location. The config will be read from the layer config", args: [], constr( state: SpecialVisualizationState, tagSource: UIEventSource>, argument: string[], feature: Feature, layer: LayerConfig ): BaseUIElement { if (feature.geometry.type !== "Point") { return undefined } return new MoveWizard( >feature, >tagSource, state, layer.allowMove ) }, }, { funcName: "delete_button", docs: "Adds a button which allows to delete the object at this location. The config will be read from the layer config", args: [], constr( state: SpecialVisualizationState, tagSource: UIEventSource>, argument: string[], feature: Feature, layer: LayerConfig ): BaseUIElement { return new VariableUiElement( tagSource .map((tags) => tags.id) .map( (id) => new DeleteWizard( id, >tagSource, state, layer.deletion // Reading the configuration from the layerconfig is a bit cheating and should be factored out ) ) ) }, }, new ShareLinkViz(), new ExportAsGpxViz(), new UploadToOsmViz(), new MultiApplyViz(), new AddNoteCommentViz(), { funcName: "open_note", args: [], docs: "Creates a new map note on the given location. This options is placed in the 'last_click'-popup automatically if the 'notes'-layer is enabled", constr( state: SpecialVisualizationState, tagSource: UIEventSource>, argument: string[], feature: Feature ): BaseUIElement { const [lon, lat] = GeoOperations.centerpointCoordinates(feature) return new SvelteUIElement(CreateNewNote, {state, coordinate: {lon, lat}}) }, }, new CloseNoteButton(), new PlantNetDetectionViz(), new TagApplyButton(), new ImportPointButton(), new ImportWayButton(), new ConflateButton(), new NearbyImageVis(), { funcName: "wikipedia", docs: "A box showing the corresponding wikipedia article(s) - based on the **wikidata** tag.", args: [ { name: "keyToShowWikipediaFor", doc: "Use the wikidata entry from this key to show the wikipedia article for. Multiple keys can be given (separated by ';'), in which case the first matching value is used", defaultValue: "wikidata;wikipedia", }, ], example: "`{wikipedia()}` is a basic example, `{wikipedia(name:etymology:wikidata)}` to show the wikipedia page of whom the feature was named after. Also remember that these can be styled, e.g. `{wikipedia():max-height: 10rem}` to limit the height", constr: (_, tagsSource, args) => { const keys = args[0].split(";").map((k) => k.trim()) const wikiIds: Store = tagsSource.map((tags) => { const key = keys.find((k) => tags[k] !== undefined && tags[k] !== "") return tags[key]?.split(";")?.map((id) => id.trim()) }) return new SvelteUIElement(WikipediaPanel, { wikiIds, }) }, }, { funcName: "wikidata_label", docs: "Shows the label of the corresponding wikidata-item", args: [ { name: "keyToShowWikidataFor", doc: "Use the wikidata entry from this key to show the label", defaultValue: "wikidata", }, ], example: "`{wikidata_label()}` is a basic example, `{wikipedia(name:etymology:wikidata)}` to show the label itself", constr: (_, tagsSource, args) => new VariableUiElement( tagsSource .map((tags) => tags[args[0]]) .map((wikidata) => { wikidata = Utils.NoEmpty( wikidata?.split(";")?.map((wd) => wd.trim()) ?? [] )[0] const entry = Wikidata.LoadWikidataEntry(wikidata) return new VariableUiElement( entry.map((e) => { if (e === undefined || e["success"] === undefined) { return wikidata } const response = e["success"] return Translation.fromMap(response.labels) }) ) }) ), }, new MapillaryLinkVis(), new LanguageElement(), { funcName: "all_tags", docs: "Prints all key-value pairs of the object - used for debugging", args: [], constr: (state, tags: UIEventSource) => new SvelteUIElement(AllTagsPanel, {tags, state}), }, { funcName: "image_carousel", docs: "Creates an image carousel for the given sources. An attempt will be made to guess what source is used. Supported: Wikidata identifiers, Wikipedia pages, Wikimedia categories, IMGUR (with attribution, direct links)", args: [ { name: "image_key", defaultValue: AllImageProviders.defaultKeys.join(","), doc: "The keys given to the images, e.g. if image is given, the first picture URL will be added as image, the second as image:0, the third as image:1, etc... Multiple values are allowed if ';'-separated ", }, ], constr: (state, tags, args) => { let imagePrefixes: string[] = undefined if (args.length > 0) { imagePrefixes = [].concat(...args.map((a) => a.split(","))) } return new ImageCarousel( AllImageProviders.LoadImagesFor(tags, imagePrefixes), tags, state ) }, }, { funcName: "image_upload", docs: "Creates a button where a user can upload an image to IMGUR", args: [ { name: "image-key", doc: "Image tag to add the URL to (or image-tag:0, image-tag:1 when multiple images are added)", defaultValue: "image", }, { name: "label", doc: "The text to show on the button", defaultValue: "Add image", }, ], constr: (state, tags, args) => { return new ImageUploadFlow(tags, state, args[0], args[1]) }, }, { funcName: "reviews", docs: "Adds an overview of the mangrove-reviews of this object. Mangrove.Reviews needs - in order to identify the reviewed object - a coordinate and a name. By default, the name of the object is given, but this can be overwritten", example: "`{reviews()}` for a vanilla review, `{reviews(name, play_forest)}` to review a play forest. If a name is known, the name will be used as identifier, otherwise 'play_forest' is used", args: [ { name: "subjectKey", defaultValue: "name", doc: "The key to use to determine the subject. If specified, the subject will be tags[subjectKey]", }, { name: "fallback", doc: "The identifier to use, if tags[subjectKey] as specified above is not available. This is effectively a fallback value", }, ], constr: (state, tags, args, feature) => { const nameKey = args[0] ?? "name" let fallbackName = args[1] const mangrove = FeatureReviews.construct( feature, tags, state.userRelatedState.mangroveIdentity, { nameKey: nameKey, fallbackName, } ) const form = new ReviewForm((r) => mangrove.createReview(r), state) return new ReviewElement(mangrove, form) }, }, { funcName: "opening_hours_table", docs: "Creates an opening-hours table. Usage: {opening_hours_table(opening_hours)} to create a table of the tag 'opening_hours'.", args: [ { name: "key", defaultValue: "opening_hours", doc: "The tagkey from which the table is constructed.", }, { name: "prefix", defaultValue: "", doc: "Remove this string from the start of the value before parsing. __Note: use `&LPARENs` to indicate `(` if needed__", }, { name: "postfix", defaultValue: "", doc: "Remove this string from the end of the value before parsing. __Note: use `&RPARENs` to indicate `)` if needed__", }, ], example: "A normal opening hours table can be invoked with `{opening_hours_table()}`. A table for e.g. conditional access with opening hours can be `{opening_hours_table(access:conditional, no @ &LPARENS, &RPARENS)}`", constr: (state, tagSource: UIEventSource, args) => { return new OpeningHoursVisualization( tagSource, state, args[0], args[1], args[2] ) }, }, { funcName: "live", docs: "Downloads a JSON from the given URL, e.g. '{live(example.org/data.json, shorthand:x.y.z, other:a.b.c, shorthand)}' will download the given file, will create an object {shorthand: json[x][y][z], other: json[a][b][c] out of it and will return 'other' or 'json[a][b][c]. This is made to use in combination with tags, e.g. {live({url}, {url:format}, needed_value)}", example: "{live({url},{url:format},hour)} {live(https://data.mobility.brussels/bike/api/counts/?request=live&featureID=CB2105,hour:data.hour_cnt;day:data.day_cnt;year:data.year_cnt,hour)}", args: [ { name: "Url", doc: "The URL to load", required: true, }, { name: "Shorthands", doc: "A list of shorthands, of the format 'shorthandname:path.path.path'. separated by ;", }, { name: "path", doc: "The path (or shorthand) that should be returned", }, ], constr: (_, tagSource: UIEventSource, args) => { const url = args[0] const shorthands = args[1] const neededValue = args[2] const source = LiveQueryHandler.FetchLiveData(url, shorthands.split(";")) return new VariableUiElement( source.map((data) => data[neededValue] ?? "Loading...") ) }, }, { funcName: "canonical", docs: "Converts a short, canonical value into the long, translated text including the unit. This only works if a `unit` is defined for the corresponding value. The unit specification will be included in the text. ", example: "If the object has `length=42`, then `{canonical(length)}` will be shown as **42 meter** (in english), **42 metre** (in french), ...", args: [ { name: "key", doc: "The key of the tag to give the canonical text for", required: true, }, ], constr: (state, tagSource, args) => { const key = args[0] return new VariableUiElement( tagSource .map((tags) => tags[key]) .map((value) => { if (value === undefined) { return undefined } const allUnits = [].concat( ...(state?.layout?.layers?.map((lyr) => lyr.units) ?? []) ) const unit = allUnits.filter((unit) => unit.isApplicableToKey(key) )[0] if (unit === undefined) { return value } return unit.asHumanLongValue(value) }) ) }, }, { funcName: "export_as_geojson", docs: "Exports the selected feature as GeoJson-file", args: [], constr: (state, tagSource, tagsSource, feature, layer) => { const t = Translations.t.general.download return new SubtleButton( Svg.download_svg(), new Combine([ t.downloadFeatureAsGeojson.SetClass("font-bold text-lg"), t.downloadGeoJsonHelper.SetClass("subtle"), ]).SetClass("flex flex-col") ).onClick(() => { console.log("Exporting as Geojson") const tags = tagSource.data const title = layer?.title?.GetRenderValue(tags)?.Subs(tags)?.txt ?? "geojson" const data = JSON.stringify(feature, null, " ") Utils.offerContentsAsDownloadableFile( data, title + "_mapcomplete_export.geojson", { mimetype: "application/vnd.geo+json", } ) }) }, }, { funcName: "open_in_iD", docs: "Opens the current view in the iD-editor", args: [], constr: (state, feature) => { return new OpenIdEditor(state.mapProperties, undefined, feature.data.id) }, }, { funcName: "open_in_josm", docs: "Opens the current view in the JOSM-editor", args: [], constr: (state) => { return new OpenJosm(state.osmConnection, state.mapProperties.bounds) }, }, { funcName: "clear_location_history", docs: "A button to remove the travelled track information from the device", args: [], constr: (state) => { return new SubtleButton( Svg.delete_icon_svg().SetStyle("height: 1.5rem"), Translations.t.general.removeLocationHistory ).onClick(() => { state.historicalUserLocations.features.setData([]) Hash.hash.setData(undefined) }) }, }, { funcName: "visualize_note_comments", docs: "Visualises the comments for notes", args: [ { name: "commentsKey", doc: "The property name of the comments, which should be stringified json", defaultValue: "comments", }, { name: "start", doc: "Drop the first 'start' comments", defaultValue: "0", }, ], constr: (state, tags, args) => new VariableUiElement( tags .map((tags) => tags[args[0]]) .map((commentsStr) => { const comments: any[] = JSON.parse(commentsStr) const startLoc = Number(args[1] ?? 0) if (!isNaN(startLoc) && startLoc > 0) { comments.splice(0, startLoc) } return new Combine( comments .filter((c) => c.text !== "") .map((c) => new NoteCommentElement(c)) ).SetClass("flex flex-col") }) ), }, { funcName: "add_image_to_note", docs: "Adds an image to a node", args: [ { name: "Id-key", doc: "The property name where the ID of the note to close can be found", defaultValue: "id", }, ], example: " The following example sets the status to '2' (false positive)\n" + "\n" + "```json\n" + "{\n" + ' "id": "mark_duplicate",\n' + ' "render": {\n' + ' "special": {\n' + ' "type": "maproulette_set_status",\n' + ' "message": {\n' + ' "en": "Mark as not found or false positive"\n' + " },\n" + ' "status": "2",\n' + ' "image": "close"\n' + " }\n" + " }\n" + "}\n" + "```", constr: (state, tags, args) => { const isUploading = new UIEventSource(false) const t = Translations.t.notes const id = tags.data[args[0] ?? "id"] const uploader = new ImgurUploader(async (url) => { isUploading.setData(false) await state.osmConnection.addCommentToNote(id, url) NoteCommentElement.addCommentTo(url, tags, state) }) const label = new Combine([ Svg.camera_plus_svg().SetClass("block w-12 h-12 p-1 text-4xl "), Translations.t.image.addPicture, ]).SetClass( "p-2 border-4 border-black rounded-full font-bold h-full align-center w-full flex justify-center" ) const fileSelector = new FileSelectorButton(label) fileSelector.GetValue().addCallback((filelist) => { isUploading.setData(true) uploader.uploadMany("Image for osm.org/note/" + id, "CC0", filelist) }) const ti = Translations.t.image const uploadPanel = new Combine([ fileSelector, ti.respectPrivacy.SetClass("text-sm"), ]).SetClass("flex flex-col") return new LoginToggle( new Toggle( Translations.t.image.uploadingPicture.SetClass("alert"), uploadPanel, isUploading ), t.loginToAddPicture, state ) }, }, { funcName: "title", args: [], docs: "Shows the title of the popup. Useful for some cases, e.g. 'What is phone number of {title()}?'", example: "`What is the phone number of {title()}`, which might automatically become `What is the phone number of XYZ`.", constr: (state, tagsSource) => new VariableUiElement( tagsSource.map((tags) => { const layer = state.layout.getMatchingLayer(tags) const title = layer?.title?.GetRenderValue(tags) if (title === undefined) { return undefined } return new SubstitutedTranslation(title, tagsSource, state).SetClass("px-1") }) ), }, { funcName: "maproulette_task", args: [], constr(state, tagSource) { let parentId = tagSource.data.mr_challengeId if (parentId === undefined) { console.warn("Element ", tagSource.data.id, " has no mr_challengeId") return undefined } let challenge = Stores.FromPromise( Utils.downloadJsonCached( `https://maproulette.org/api/v2/challenge/${parentId}`, 24 * 60 * 60 * 1000 ) ) return new VariableUiElement( challenge.map((challenge) => { let listItems: BaseUIElement[] = [] let title: BaseUIElement if (challenge?.name) { title = new Title(challenge.name) } if (challenge?.description) { listItems.push(new FixedUiElement(challenge.description)) } if (challenge?.instruction) { listItems.push(new FixedUiElement(challenge.instruction)) } if (listItems.length === 0) { return undefined } else { return [title, new List(listItems)] } }) ) }, docs: "Fetches the metadata of MapRoulette campaign that this task is part of and shows those details (namely `title`, `description` and `instruction`).\n\nThis reads the property `mr_challengeId` to detect the parent campaign.", }, { funcName: "maproulette_set_status", docs: "Change the status of the given MapRoulette task", args: [ { name: "message", doc: "A message to show to the user", }, { name: "image", doc: "Image to show", defaultValue: "confirm", }, { name: "message_confirm", doc: "What to show when the task is closed, either by the user or was already closed.", }, { name: "status", doc: "A statuscode to apply when the button is clicked. 1 = `close`, 2 = `false_positive`, 3 = `skip`, 4 = `deleted`, 5 = `already fixed` (on the map, e.g. for duplicates), 6 = `too hard`", defaultValue: "1", }, { name: "maproulette_id", doc: "The property name containing the maproulette id", defaultValue: "mr_taskId", }, ], constr: (state, tagsSource, args) => { let [message, image, message_closed, status, maproulette_id_key] = args if (image === "") { image = "confirm" } if (Svg.All[image] !== undefined || Svg.All[image + ".svg"] !== undefined) { if (image.endsWith(".svg")) { image = image.substring(0, image.length - 4) } image = Svg[image + "_ui"]() } const failed = new UIEventSource(false) const closeButton = new SubtleButton(image, message).OnClickWithLoading( Translations.t.general.loading, async () => { const maproulette_id = tagsSource.data[maproulette_id_key] ?? tagsSource.data.id try { await Maproulette.singleton.closeTask( Number(maproulette_id), Number(status), { tags: `MapComplete MapComplete:${state.layout.id}`, } ) tagsSource.data["mr_taskStatus"] = Maproulette.STATUS_MEANING[Number(status)] tagsSource.data.status = status tagsSource.ping() } catch (e) { console.error(e) failed.setData(true) } } ) let message_closed_element = undefined if (message_closed !== undefined && message_closed !== "") { message_closed_element = new FixedUiElement(message_closed) } return new VariableUiElement( tagsSource .map( (tgs) => tgs["status"] ?? Maproulette.STATUS_MEANING[tgs["mr_taskStatus"]] ) .map(Number) .map( (status) => { if (failed.data) { return new FixedUiElement( "ERROR - could not close the MapRoulette task" ).SetClass("block alert") } if (status === Maproulette.STATUS_OPEN) { return closeButton } return message_closed_element ?? "Closed!" }, [failed] ) ) }, }, { funcName: "statistics", docs: "Show general statistics about the elements currently in view. Intended to use on the `current_view`-layer", args: [], constr: (state) => { return new Combine( state.layout.layers .filter((l) => l.name !== null) .map( (l) => { const fs = state.perLayer.get(l.id) const bbox = state.mapProperties.bounds const fsBboxed = new BBoxFeatureSourceForLayer(fs, bbox) return new StatisticsPanel(fsBboxed) }, [state.mapProperties.bounds] ) ) }, }, { funcName: "send_email", docs: "Creates a `mailto`-link where some fields are already set and correctly escaped. The user will be promted to send the email", args: [ { name: "to", doc: "Who to send the email to?", required: true, }, { name: "subject", doc: "The subject of the email", required: true, }, { name: "body", doc: "The text in the email", required: true, }, { name: "button_text", doc: "The text shown on the button in the UI", required: true, }, ], constr(__, tags, args) { return new VariableUiElement( tags.map((tags) => { const [to, subject, body, button_text] = args.map((str) => Utils.SubstituteKeys(str, tags) ) const url = "mailto:" + to + "?subject=" + encodeURIComponent(subject) + "&body=" + encodeURIComponent(body) return new SubtleButton(Svg.envelope_svg(), button_text, { url, }) }) ) }, }, { funcName: "link", docs: "Construct a link. By using the 'special' visualisation notation, translation should be easier", args: [ { name: "text", doc: "Text to be shown", required: true, }, { name: "href", doc: "The URL to link to", required: true, }, { name: "class", doc: "CSS-classes to add to the element", }, ], constr( state: SpecialVisualizationState, tagSource: UIEventSource>, args: string[] ): BaseUIElement { const [text, href, classnames] = args return new VariableUiElement( tagSource.map((tags) => new Link( Utils.SubstituteKeys(text, tags), Utils.SubstituteKeys(href, tags), true ).SetClass(classnames) ) ) }, }, { funcName: "multi", docs: "Given an embedded tagRendering (read only) and a key, will read the keyname as a JSON-list. Every element of this list will be considered as tags and rendered with the tagRendering", example: "```json\n" + JSON.stringify( { render: { special: { type: "multi", key: "_doors_from_building_properties", tagrendering: { en: "The building containing this feature has a door of width {entrance:width}", }, }, }, }, null, " " ) + "\n```", args: [ { name: "key", doc: "The property to read and to interpret as a list of properties", required: true, }, { name: "tagrendering", doc: "An entire tagRenderingConfig", required: true, }, ], constr(state, featureTags, args) { const [key, tr] = args const translation = new Translation({"*": tr}) return new VariableUiElement( featureTags.map((tags) => { const properties: object[] = JSON.parse(tags[key]) const elements = [] for (const property of properties) { const subsTr = new SubstitutedTranslation( translation, new UIEventSource(property), state ) elements.push(subsTr) } return new List(elements) }) ) }, }, ] specialVisualizations.push(new AutoApplyButton(specialVisualizations)) const invalid = specialVisualizations .map((sp, i) => ({sp, i})) .filter((sp) => sp.sp.funcName === undefined) if (invalid.length > 0) { throw ( "Invalid special visualisation found: funcName is undefined for " + invalid.map((sp) => sp.i).join(", ") + '. Did you perhaps type \n funcName: "funcname" // type declaration uses COLON\ninstead of:\n funcName = "funcName" // value definition uses EQUAL' ) } return specialVisualizations } }