Themes: improve 'auto-type' special visualisation

This commit is contained in:
Pieter Vander Vennet 2024-08-23 13:00:26 +02:00
parent c291b16406
commit 658db35617
7 changed files with 140 additions and 165 deletions

View file

@ -2,10 +2,13 @@
"id": "tourism_accomodation",
"name": {
"en": "Tourism accomodation",
"de": "Touristische Unterkunft"
"de": "Touristische Unterkunft",
"nl": "Accomodatie voor toeristen"
},
"description": {
"en": "Various types of lodging facilities"
"en": "Various types of lodging facilities",
"de": "Verschiedene Arten von Unterkünften",
"nl": "Verschillende soorten overnachtingsmogelijkheden voor toeristen"
},
"minzoom": 12,
"title": {
@ -26,37 +29,44 @@
"if": "tourism=hostel",
"then": {
"en": "Hostel {name}",
"de": "Hostel {name}"
"de": "Hostel {name}",
"nl": "Herberg {name}"
}
},
{
"if": "guest_house=bed_and_breakfast",
"then": {
"en": "B&B {name}"
"en": "B&B {name}",
"de": "B&B {name}",
"nl": "B&B {name}"
}
},
{
"if": "tourism=guest_house",
"then": {
"en": "Guest house {name}"
"en": "Guest house {name}",
"de": "Ferienhaus {name}"
}
},
{
"if": "tourism=motel",
"then": {
"en": "Motel {name}"
"en": "Motel {name}",
"de": "Motel {name}"
}
},
{
"if": "tourism=apartment",
"then": {
"en": "Apartment {name}"
"en": "Apartment {name}",
"de": "Ferienwohnung {name}"
}
},
{
"if": "tourism=chalet",
"then": {
"en": "Vacation home {name}"
"en": "Vacation home {name}",
"de": "Ferienhaus {name}"
}
}
],
@ -131,7 +141,8 @@
},
"description": {
"en": "A hotel is an establishment that provides paid lodging, usually on a short-term basis. Rooms are not shared with strangers.",
"de": "Ein Hotel ist eine Einrichtung, die eine bezahlte Unterkunft anbietet, in der Regel für einen kurzen Zeitraum. Die Zimmer werden nicht mit Fremden geteilt."
"de": "Ein Hotel ist eine Einrichtung, die eine bezahlte Unterkunft anbietet, in der Regel für einen kurzen Zeitraum. Die Zimmer werden nicht mit Fremden geteilt.",
"nl": "En hotel is een gebouw waar je tegen betaling kan overnachten voor een korte periode. Je krijgt je eigen kamer."
},
"tags": [
"tourism=hotel"
@ -143,11 +154,13 @@
],
"title": {
"en": "a hostel",
"de": "Ein Hostel"
"de": "Ein Hostel",
"nl": "een herberg"
},
"description": {
"en": "A hostel is a type of tourist accommodation where people can sleep in a room shared with strangers",
"de": "Ein Hostel ist eine Art von Touristenherberge, in der man in einem Zimmer schlafen kann, das man mit Fremden teilt"
"de": "Ein Hostel ist eine Art von Touristenherberge, in der man in einem Zimmer schlafen kann, das man mit Fremden teilt",
"nl": "Een herberg is een gebouw waar je enkele dagen kan blijven. Je deelt een kamer met onbekenden."
}
},
{
@ -155,10 +168,14 @@
"tourism=apartment"
],
"title": {
"en": "a vacation apartment"
"en": "a vacation apartment",
"de": "eine Ferienwohnung",
"nl": "een vakantie-appartement"
},
"description": {
"en": "A furnished apartment or flat with cooking and bathroom facilities in a shared building that can be rented for vacations, typically without breakfast or reception desk"
"en": "A furnished apartment or flat with cooking and bathroom facilities in a shared building that can be rented for vacations, typically without breakfast or reception desk",
"de": "Ein möbliertes Appartement oder eine Wohnung mit Kochgelegenheit und Bad in einem Gemeinschaftsgebäude, das für Ferienzwecke gemietet werden kann, normalerweise ohne Frühstück oder Rezeption",
"nl": "Een bemeubeld apparement met kookgelegenheid en een badkamer in een groter gebouw. Het appartement kan gehuurd worden voor vakanties. Er is geen receptie of ontbijt voorzien."
}
},
{
@ -166,10 +183,12 @@
"tourism=chalet"
],
"title": {
"en": "a vacation chalet"
"en": "a vacation chalet",
"de": "ein Ferienchalet"
},
"description": {
"en": "A holiday cottage or vacation home with cooking and bathroom facilities that can be rented for holiday vacations, typically without breakfast or reception desk"
"en": "A holiday cottage or vacation home with cooking and bathroom facilities that can be rented for holiday vacations, typically without breakfast or reception desk",
"de": "Ein Ferienhaus oder eine Ferienwohnung mit Kochgelegenheit und Bad, das bzw. die für Ferienaufenthalte gemietet werden kann, in der Regel ohne Frühstück oder Rezeption"
}
},
{
@ -177,10 +196,13 @@
"tourism=motel"
],
"title": {
"en": "a motel"
"en": "a motel",
"de": "ein Motel",
"nl": "een motel"
},
"description": {
"en": "A motel is an establishment that provides paid lodging, usually on a short-term basis, with convenient parking for motor cars at or close to the room. They are typically cheaper then a hotel"
"en": "A motel is an establishment that provides paid lodging, usually on a short-term basis, with convenient parking for motor cars at or close to the room. They are typically cheaper then a hotel",
"de": "Ein Motel ist eine Einrichtung, die eine bezahlte Unterkunft anbietet, in der Regel für kurze Zeit, mit bequemen Parkplätzen für Kraftfahrzeuge am oder in der Nähe des Zimmers. Sie sind in der Regel billiger als ein Hotel"
}
},
{
@ -189,10 +211,12 @@
"guest_house=bed_and_breakfast"
],
"title": {
"en": "a bed-and-breakfast"
"en": "a bed-and-breakfast",
"de": "eine Frühstückspension"
},
"description": {
"en": "A bed-and-breakfast in a guesthouse is a small lodging establishment. These are often a few rooms within a private family home where the owners also accommodate the guests. There is no reception desk, nor is there staff at all times. In some cases, check-in happens remotely by sharing a code to get in."
"en": "A bed-and-breakfast in a guesthouse is a small lodging establishment. These are often a few rooms within a private family home where the owners also accommodate the guests. There is no reception desk, nor is there staff at all times. In some cases, check-in happens remotely by sharing a code to get in.",
"de": "Ein Bed-and-Breakfast in einem Gästehaus ist ein kleiner Beherbergungsbetrieb. Oft handelt es sich um einige Zimmer in einem Privathaus, in dem die Eigentümer auch die Gäste beherbergen. Es gibt keine Rezeption und es ist auch nicht ständig Personal anwesend. In einigen Fällen erfolgt das Einchecken aus der Ferne, indem ein Zugangscode mitgeteilt wird."
}
},
{
@ -200,7 +224,8 @@
"tourism=guest_house"
],
"title": {
"en": "a small-scale lodging facility, typically operated by the owner"
"en": "a small-scale lodging facility, typically operated by the owner",
"de": "eine kleine Beherbergungseinrichtung, die in der Regel vom Eigentümer betrieben wird"
}
}
],
@ -211,11 +236,13 @@
"id": "name",
"question": {
"en": "What is the name of this {title()}?",
"de": "Wie lautet der Name von {title()}?"
"de": "Wie lautet der Name von {title()}?",
"nl": "Wat is de naam van deze {title()}"
},
"render": {
"en": "{name}",
"de": "{name}"
"de": "{name}",
"nl": "{name}"
},
"freeform": {
"key": "name"
@ -234,43 +261,57 @@
"options": [
{
"question": {
"en": "All types"
"en": "All types",
"de": "Alle Arten",
"nl": "Alle types"
}
},
{
"osmTags": "tourism=hotel",
"question": {
"en": "Hotels"
"en": "Hotels",
"de": "Hotels",
"nl": "Hotels"
}
},
{
"osmTags": "tourism=hostel",
"question": {
"en": "Hostels"
"en": "Hostels",
"de": "Hostels / Herbergen",
"nl": "Hostels"
}
},
{
"osmTags": "tourism=guest_house",
"question": {
"en": "Guest houses and Bed&Breakfasts"
"en": "Guest houses and Bed&Breakfasts",
"de": "Gasthäuser und Bed & Breakfasts",
"nl": "Gastenkamers en bed-en-breakfasts"
}
},
{
"osmTags": "tourism=motel",
"question": {
"en": "Motels"
"en": "Motels",
"de": "Motels",
"nl": "Motels"
}
},
{
"osmTags": "tourism=chalet",
"question": {
"en": "Vacation home"
"en": "Vacation home",
"de": "Ferienhaus",
"nl": "Vakantiehuisje"
}
},
{
"osmTags": "tourism=apartment",
"question": {
"en": "Vacation apartment"
"en": "Vacation apartment",
"de": "Ferienwohnung",
"nl": "Vakantie-appartement"
}
}
]

View file

@ -701,8 +701,8 @@
},
"preset_type": {
"question": "Of what type is this object?",
"typeDescription": "This is a {title}. {description}",
"typeTitle": "This is a {title}"
"typeDescription": "This is <b>{title}</b>. <div class='subtle'>{description}</div>",
"typeTitle": "This is <b>{title}</b>"
},
"privacy": {
"editingIntro": "When you make a change to the map, this change is recorded on OpenStreetMap and is publicly available to anyone. A changeset made with MapComplete includes the following data:",

View file

@ -234,6 +234,13 @@ export class TagUtils {
return properties
}
static asProperties(tags: TagsFilter | TagsFilter[], baseproperties: Record<string, string>= {}) {
if(Array.isArray(tags)){
tags = new And(tags)
}
return TagUtils.changeAsProperties(tags.asChange(baseproperties))
}
static changeAsProperties(kvs: { k: string; v: string }[]): Record<string, string> {
const tags: Record<string, string> = {}
for (const kv of kvs) {

View file

@ -346,7 +346,7 @@ export default class LayerConfig extends WithContextLoader {
this.popupInFloatover = json.popupInFloatover ?? false
}
public defaultIcon(): BaseUIElement | undefined {
public defaultIcon(properties?: Record<string, string>): BaseUIElement | undefined {
if (this.mapRendering === undefined || this.mapRendering === null) {
return undefined
}
@ -354,7 +354,7 @@ export default class LayerConfig extends WithContextLoader {
if (mapRendering === undefined) {
return undefined
}
return mapRendering.GetBaseIcon(this.GetBaseTags())
return mapRendering.GetBaseIcon(properties ?? this.GetBaseTags())
}
public GetBaseTags(): Record<string, string> {

View file

@ -53,8 +53,6 @@ export class ShareLinkViz implements SpecialVisualization {
}
}
return new SvelteUIElement(ShareButton, { generateShareData, text }).SetClass(
"w-full h-full"
)
return new SvelteUIElement(ShareButton, { generateShareData, text })
}
}

View file

@ -7,6 +7,14 @@
import LayerConfig from "../../../Models/ThemeConfig/LayerConfig"
import { twJoin } from "tailwind-merge"
import Marker from "../../Map/Marker.svelte"
import ToSvelte from "../../Base/ToSvelte.svelte"
import { And } from "../../../Logic/Tags/And"
import { TagUtils } from "../../../Logic/Tags/TagUtils"
import BaseUIElement from "../../BaseUIElement"
import type { Mapping } from "../../../Models/ThemeConfig/TagRenderingConfig"
import SvelteUIElement from "../../Base/SvelteUIElement"
import Icon from "../../Map/Icon.svelte"
import { TagsFilter } from "../../../Logic/Tags/TagsFilter"
export let selectedElement: Feature
export let tags: UIEventSource<Record<string, string>>
@ -32,27 +40,41 @@
}
const emojiHeights = {
"small":"2rem",
"medium":"3rem",
"large":"5rem"
"small": "2rem",
"medium": "3rem",
"large": "5rem",
}
</script>
function getAutoIcon(mapping: {if?: TagsFilter }): BaseUIElement {
for (const preset of layer.presets) {
if (!new And(preset.tags).shadows(mapping.if)) {
continue
}
return layer.defaultIcon(TagUtils.asProperties(preset.tags))
}
return undefined
}
</script>
{#if mapping.icon !== undefined}
<div class="inline-flex items-center">
{#if mapping.icon === "auto"}
<div class="w-8 h-8 shrink-0 mr-2">
<ToSvelte construct={() => getAutoIcon(mapping)} />
</div>
{:else}
<Marker
icons={mapping.icon}
size={twJoin(
size={twJoin("shrink-0",
`mapping-icon-${mapping.iconClass ?? "small"}-height mapping-icon-${
mapping.iconClass ?? "small"
}-width`,
"shrink-0"
)}
}-width`)}
emojiHeight={ emojiHeights[mapping.iconClass] ?? "2rem"}
clss={`mapping-icon-${mapping.iconClass ?? "small"}`}
/>
{/if}
<SpecialTranslation t={mapping.then} {tags} {state} {layer} feature={selectedElement} {clss} />
</div>
{:else if mapping.then !== undefined}

View file

@ -2031,41 +2031,6 @@ export default class SpecialVisualizations {
return new VariableUiElement(translation)
},
},
{
funcName: "preset_type_select",
docs: "An editable tag rendering which allows to change the type",
args: [],
constr(
state: SpecialVisualizationState,
tags: UIEventSource<Record<string, string>>,
argument: string[],
selectedElement: Feature,
layer: LayerConfig
): SvelteUIElement {
const t = Translations.t.preset_type
const question: QuestionableTagRenderingConfigJson = {
id: layer.id + "-type",
question: t.question.translations,
mappings: layer.presets.map((pr) => {
return {
if: new And(pr.tags).asJson(),
then: (pr.description ? t.typeDescription : t.typeTitle).Subs({
title: pr.title,
description: pr.description,
}).translations,
}
}),
}
const config = new TagRenderingConfig(question)
return new SvelteUIElement(TagRenderingEditable, {
config,
tags,
selectedElement,
state,
layer,
})
},
},
{
funcName: "pending_changes",
docs: "A module showing the pending changes, with the option to clear the pending changes",
@ -2149,15 +2114,14 @@ export default class SpecialVisualizations {
const question: QuestionableTagRenderingConfigJson = {
id: layer.id + "-type",
question: t.question.translations,
mappings: layer.presets.map((pr) => {
return {
mappings: layer.presets.map((pr) => ({
if: new And(pr.tags).asJson(),
icon: "auto",
then: (pr.description ? t.typeDescription : t.typeTitle).Subs({
title: pr.title,
description: pr.description,
}).translations,
}
}),
})),
}
const config = new TagRenderingConfig(question)
return new SvelteUIElement(TagRenderingEditable, {
@ -2168,75 +2132,7 @@ export default class SpecialVisualizations {
layer,
})
},
},
{
funcName: "pending_changes",
docs: "A module showing the pending changes, with the option to clear the pending changes",
args: [],
constr(
state: SpecialVisualizationState,
tagSource: UIEventSource<Record<string, string>>,
argument: string[],
feature: Feature,
layer: LayerConfig
): BaseUIElement {
return new SvelteUIElement(PendingChangesIndicator, { state, compact: false })
},
},
{
funcName: "clear_caches",
docs: "A button which clears the locally downloaded data and the service worker. Login status etc will be kept",
args: [
{
name: "text",
required: true,
doc: "The text to show on the button",
},
],
constr(
state: SpecialVisualizationState,
tagSource: UIEventSource<Record<string, string>>,
argument: string[],
feature: Feature,
layer: LayerConfig
): SvelteUIElement {
return new SvelteUIElement<any, any, any>(ClearCaches, {
msg: argument[0] ?? "Clear local caches",
})
},
},
{
funcName: "group",
docs: "A collapsable group (accordion)",
args: [
{
name: "header",
doc: "The _identifier_ of a single tagRendering. This will be used as header",
},
{
name: "labels",
doc: "A `;`-separated list of either identifiers or label names. All tagRenderings matching this value will be shown in the accordion",
},
],
constr(
state: SpecialVisualizationState,
tags: UIEventSource<Record<string, string>>,
argument: string[],
selectedElement: Feature,
layer: LayerConfig
): SvelteUIElement {
const [header, labelsStr] = argument
const labels = labelsStr.split(";").map((x) => x.trim())
return new SvelteUIElement<any, any, any>(GroupedView, {
state,
tags,
selectedElement,
layer,
header,
labels,
})
},
},
}
]
specialVisualizations.push(new AutoApplyButton(specialVisualizations))
@ -2245,6 +2141,7 @@ export default class SpecialVisualizations {
const invalid = specialVisualizations
.map((sp, i) => ({ sp, i }))
.filter((sp) => sp.sp.funcName === undefined || !sp.sp.funcName.match(regex))
if (invalid.length > 0) {
throw (
"Invalid special visualisation found: funcName is undefined or doesn't match " +
@ -2254,6 +2151,16 @@ export default class SpecialVisualizations {
)
}
const allNames = specialVisualizations.map(f => f.funcName)
const seen = new Set<string>()
for (let name of allNames) {
name = name.toLowerCase()
if(seen.has(name)){
throw "Invalid special visualisations: detected a duplicate name: "+name
}
seen.add(name)
}
return specialVisualizations
}
}