Accessibility: improve keyboard only flow (see #1181); remove some legacy use of Svelte

This commit is contained in:
Pieter Vander Vennet 2023-12-06 17:27:30 +01:00
parent d1a6c11513
commit 4ee83cfe5c
35 changed files with 613 additions and 683 deletions

View file

@ -351,6 +351,7 @@
"if": { "if": {
"and": [ "and": [
"seasonal!=no", "seasonal!=no",
"seasonal~*",
{ {
"or": [ "or": [
{ {

View file

@ -1,33 +1,13 @@
{ {
"id": "mapcomplete-changes", "id": "mapcomplete-changes",
"title": { "title": {
"en": "Changes made with MapComplete", "en": "Changes made with MapComplete"
"ca": "Canvis fets amb MapComplete",
"cs": "Změny provedené pomocí MapComplete",
"de": "Mit MapComplete erstellte Änderungen",
"es": "Cambios realizados con MapComplete",
"fr": "Changements faits avec MapComplete",
"nl": "Wijzigingen gemaakt met MapComplete",
"pl": "Zmiany wprowadzone za pomocą MapComplete"
}, },
"shortDescription": { "shortDescription": {
"en": "Show changes made with MapComplete", "en": "Shows changes made by MapComplete"
"ca": "Mostra els canvis fets amb MapComplete",
"cs": "Zobrazení změn provedených pomocí nástroje MapComplete",
"de": "Mit MapComplete erstellte Änderungen anzeigen",
"es": "Mostrar cambios realizados con MapComplete",
"nl": "Toon wijzigingen gemaakt met MapComplete",
"pl": "Pokaż zmiany wprowadzone za pomocą MapComplete"
}, },
"description": { "description": {
"en": "This maps shows all the changes made with MapComplete", "en": "This maps shows all the changes made with MapComplete"
"ca": "Aquest mapa mostra tots els canvis fets amb MapComplete",
"cs": "Tato mapa zobrazuje všechny změny provedené pomocí MapComplete",
"de": "Diese Karte zeigt alle mit MapComplete vorgenommenen Änderungen",
"es": "Este mapa muestra todos los cambios realizados con MapComplete",
"fr": "Cette carte montre tous les changements faits avec MapComplete",
"nl": "Deze kaart toont alle wijzigingen die met MapComplete gemaakt werden",
"pl": "Ta mapa pokazuje wszystkie zmiany wprowadzone za pomocą MapComplete"
}, },
"icon": "./assets/svg/logo.svg", "icon": "./assets/svg/logo.svg",
"hideFromOverview": true, "hideFromOverview": true,
@ -40,13 +20,7 @@
{ {
"id": "mapcomplete-changes", "id": "mapcomplete-changes",
"name": { "name": {
"en": "Changeset centers", "en": "Changeset centers"
"ca": "Centre del conjunt de canvis",
"cs": "Centrum změn",
"de": "Zentrum der Änderungssätze",
"es": "Centro del conjunto de cambios",
"nl": "Centerpunt van changeset",
"pl": "Centra zmian"
}, },
"minzoom": 0, "minzoom": 0,
"source": { "source": {
@ -57,85 +31,41 @@
}, },
"title": { "title": {
"render": { "render": {
"en": "Changeset for {theme}", "en": "Changeset for {theme}"
"ca": "Conjunt de canvis per a {theme}",
"cs": "Změna pro {theme}",
"de": "Änderungssatz für {theme}",
"es": "Conjunto de cambios para {theme}",
"fr": "Groupe de modifications pour {theme}",
"pl": "Zestaw zmian dla {theme}"
} }
}, },
"description": { "description": {
"en": "Show all MapComplete changes", "en": "Shows all MapComplete changes"
"ca": "Mostra tots els canvis de MapComplete",
"cs": "Zobrazit všechny změny MapComplete",
"de": "Alle MapComplete-Änderungen anzeigen",
"es": "Mostrar todos los cambios de MapComplete",
"nl": "Toon alle MapComplete wijzigingen",
"pl": "Wyświetl wszystkie zmiany MapComplete"
}, },
"tagRenderings": [ "tagRenderings": [
{ {
"id": "show_changeset_id", "id": "show_changeset_id",
"render": { "render": {
"en": "Changeset <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>", "en": "Changeset <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>"
"ca": "Conjunt de canvi <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>",
"cs": "Změny <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>",
"de": "Änderungssatz <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>",
"es": "Conjunto de cambios <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>",
"fr": "Groupe de modifications <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>",
"pl": "Zestaw zmian <a href='https://openstreetmap.org/changeset/{id}' target='_blank'>{id}</a>"
} }
}, },
{ {
"id": "contributor", "id": "contributor",
"question": { "question": {
"en": "Which contributor made this change?", "en": "What contributor did make this change?"
"ca": "Quin col·laborador va fer aquest canvi?",
"cs": "Který přispěvatel tuto změnu provedl?",
"de": "Wer hat diese Änderung vorgenommen?",
"es": "¿Qué contribuidor hizo este cambio?",
"fr": "Quel contributeur a fait cette modification ?",
"nl": "Welke bijdrager maakte deze wijziging?",
"pl": "Który współautor dokonał tej zmiany?"
}, },
"freeform": { "freeform": {
"key": "user" "key": "user"
}, },
"render": { "render": {
"en": "Change made by <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>", "en": "Change made by <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>"
"ca": "Canvi fet per <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>",
"cs": "Změna provedená <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>",
"de": "Änderung von <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>",
"es": "Cambio realizado por <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>",
"fr": "Modification faite par <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>",
"nl": "Wijziging gemaakt door <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>",
"pl": "Zmiana dokonana przez <a href='https://openstreetmap.org/user/{user}' target='_blank'>{user}</a>"
} }
}, },
{ {
"id": "theme-id", "id": "theme-id",
"question": { "question": {
"en": "What theme was used to make this change?", "en": "What theme was used to make this change?"
"ca": "Quin tema es va utilitzar per fer aquest canvi?",
"cs": "Jaké téma bylo použito k provedení této změny?",
"de": "Welches Thema wurde für diese Änderung verwendet?",
"es": "¿Qué tema se utilizó para realizar este cambio?",
"fr": "Quel thème a été utilisé pour faire cette modification ?",
"pl": "Jakiego tematu użyto do wprowadzenia tej zmiany?"
}, },
"freeform": { "freeform": {
"key": "theme" "key": "theme"
}, },
"render": { "render": {
"en": "Change with theme <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>", "en": "Change with theme <a href='https://mapcomplete.org/{theme}'>{theme}</a>"
"ca": "Canvi amb el tema <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>",
"cs": "Změna s motivem <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>",
"de": "Geändert mit Thema <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>",
"es": "Cambio con tema <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>",
"fr": "Modifié avec le thème <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>",
"pl": "Zmiana za pomocą motywu <a href='https://mapcomplete.osm.be/{theme}'>{theme}</a>"
} }
}, },
{ {
@ -144,45 +74,19 @@
"key": "locale" "key": "locale"
}, },
"question": { "question": {
"en": "What locale (language) was this change made in?", "en": "What locale (language) was this change made in?"
"ca": "Amb quina configuració regional (idioma) s'ha fet aquest canvi?",
"cs": "V jakém národním prostředí (jazyce) byla tato změna provedena?",
"de": "In welcher Benutzersprache wurde diese Änderung vorgenommen?",
"es": "¿En qué configuración regional (idioma) se realizó este cambio?",
"fr": "En quelle langue est-ce que ce changement a été fait ?",
"nl": "In welke locale (taal) werd deze wijziging gemaakt?",
"pl": "W jakim języku wprowadzono tę zmianę?"
}, },
"render": { "render": {
"en": "User locale is {locale}", "en": "User locale is {locale}"
"ca": "La configuració regional de l'usuari és {locale}",
"cs": "Uživatelské prostředí je {locale}",
"de": "Benutzersprache {locale}",
"es": "La configuración regional del usuario es {locale}",
"nl": "De gebruikerstaal is {locale}",
"pl": "Ustawienia regionalne użytkownika to {locale}"
} }
}, },
{ {
"id": "host", "id": "host",
"render": { "render": {
"en": "Change made with <a href='{host}'>{host}</a>", "en": "Change with with <a href='{host}'>{host}</a>"
"ca": "Canviat fet amb <a href='{host}'>{host}</a>",
"cs": "Změna provedená pomocí <a href='{host}'>{host}</a>",
"de": "Geändert über <a href='{host}'>{host}</a>",
"es": "Cambio realizado con <a href='{host}'>{host}</a>",
"fr": "Modification faite avec <a href='{host}'>{host}</a>",
"nl": "Wijziging gemaakt met <a href='{host}'>{host}</a>",
"pl": "Zmiana dokonana w <a href='{host}'>{host}</a>"
}, },
"question": { "question": {
"en": "What host (website) was this change made with?", "en": "What host (website) was this change made with?"
"ca": "Amb quin amfitrió (lloc web) es va fer aquest canvi?",
"cs": "U jakého hostitele (webové stránky) byla tato změna provedena?",
"de": "Über welchen Host (Webseite) wurde diese Änderung vorgenommen?",
"es": "¿Con qué host (página web) se realizó este cambio?",
"nl": "Met welke host (website) werd deze wijziging gemaakt?",
"pl": "Na jakim hoście (stronie internetowej) dokonano tej zmiany?"
}, },
"freeform": { "freeform": {
"key": "host" "key": "host"
@ -203,22 +107,10 @@
{ {
"id": "version", "id": "version",
"question": { "question": {
"en": "What version of MapComplete was used to make this change?", "en": "What version of MapComplete was used to make this change?"
"ca": "Quina versió de MapComplete es va utilitzar per fer aquest canvi?",
"cs": "Jaká verze aplikace MapComplete byla použita k provedení této změny?",
"de": "Mit welcher Version von MapComplete wurde diese Änderung gemacht?",
"es": "¿Qué versión de MapComplete se usó para realizar este cambio?",
"fr": "Quelle version de MapComplete a été utilisée pour faire cette modification ?",
"pl": "Która wersja MapComplete została wykorzystana, aby zrobić tę zmianę?"
}, },
"render": { "render": {
"en": "Made with {editor}", "en": "Made with {editor}"
"ca": "Fet amb {editor}",
"cs": "Vyrobeno pomocí {editor}",
"de": "Erstellt mit {editor}",
"es": "Realizado con {editor}",
"fr": "Fait avec {editor}",
"pl": "Zrobione za pomocą {editor}"
}, },
"freeform": { "freeform": {
"key": "editor" "key": "editor"
@ -568,13 +460,7 @@
} }
], ],
"question": { "question": {
"en": "Theme name contains {search}", "en": "Themename contains {search}"
"ca": "El nom del tema conté {search}",
"cs": "Název motivu obsahuje {search}",
"de": "Themenname enthält {search}",
"es": "El nombre del tema contiene {search}",
"nl": "Themenaam bevat {search}",
"pl": "Nazwa tematu zawiera {search}"
} }
} }
] ]
@ -590,7 +476,7 @@
} }
], ],
"question": { "question": {
"en": "Theme name does <b>not</b> contain {search}" "en": "Themename does <b>not</b> contain {search}"
} }
} }
] ]
@ -606,13 +492,7 @@
} }
], ],
"question": { "question": {
"en": "Made by contributor {search}", "en": "Made by contributor {search}"
"ca": "Fet pel col·laborador {search}",
"cs": "Vytvořil přispěvatel {search}",
"de": "Erstellt von {search}",
"es": "Hecho por el colaborador {search}",
"nl": "Gemaakt door bijdrager {search}",
"pl": "Wykonane przez współautora {search}"
} }
} }
] ]
@ -628,13 +508,7 @@
} }
], ],
"question": { "question": {
"en": "<b>Not</b> made by contributor {search}", "en": "<b>Not</b> made by contributor {search}"
"ca": "<b>No</b> fet pel col·laborador {search}",
"cs": "<b>Není</b> vytvořeno přispěvatelem {search}",
"de": "<b>Nicht</b> erstellt von {search}",
"es": "<b>No</b> hecho por el colaborador {search}",
"nl": "<b>Niet</b> gemaakt door bijdrager {search}",
"pl": "<b>Nie</b> wykonane przez współautora {search}"
} }
} }
] ]
@ -651,13 +525,7 @@
} }
], ],
"question": { "question": {
"en": "Made before {search}", "en": "Made before {search}"
"ca": "Fet abans de {search}",
"cs": "Vytvořeno před {search}",
"de": "Erstellt vor {search}",
"es": "Hecho antes de {search}",
"nl": "Gemaakt voor {search}",
"pl": "Stworzone przed {search}"
} }
} }
] ]
@ -674,13 +542,7 @@
} }
], ],
"question": { "question": {
"en": "Made after {search}", "en": "Made after {search}"
"ca": "Fet després de {search}",
"cs": "Vytvořeno po {search}",
"de": "Erstellt nach {search}",
"es": "Hecho después de {search}",
"nl": "Gemaakt na {search}",
"pl": "Stworzone po {search}"
} }
} }
] ]
@ -696,14 +558,7 @@
} }
], ],
"question": { "question": {
"en": "User language (iso-code) {search}", "en": "User language (iso-code) {search}"
"ca": "Idioma de l'usuari (codi iso) {search}",
"cs": "Jazyk uživatele (iso-kód) {search}",
"de": "Benutzersprache (ISO-Code) {search}",
"es": "Use idioma (ISO-code) {search}",
"fr": "Langage utilisateur (code-ISO) {search}",
"nl": "De taal van de bijdrager is {search}",
"pl": "Język użytkownika (kod iso) {search}"
} }
} }
] ]
@ -719,13 +574,7 @@
} }
], ],
"question": { "question": {
"en": "Made with host {search}", "en": "Made with host {search}"
"ca": "Fet amb l'amfitrió {search}",
"cs": "Vytvořeno pomocí hostitele {search}",
"de": "Erstellt mit Host {search}",
"es": "Hecho con el host {search}",
"nl": "Gemaakt met host {search}",
"pl": "Wykonane z hostem {search}"
} }
} }
] ]
@ -736,14 +585,7 @@
{ {
"osmTags": "add-image>0", "osmTags": "add-image>0",
"question": { "question": {
"en": "Changeset added at least one image", "en": "Changeset added at least one image"
"ca": "El conjunt de canvis ha afegit almenys una imatge",
"cs": "Sada změn přidala alespoň jeden obrázek",
"de": "Im Änderungssatz wurde mindestens ein Bild hinzugefügt",
"es": "El conjunto de cambios ha añadido al menos una imagen",
"fr": "Le groupe de modifications a ajouté au moins une image",
"nl": "Changeset bevat minstens één afbeelding",
"pl": "Zestaw zmian dodał co najmniej jedno zdjęcie"
} }
} }
] ]
@ -754,7 +596,7 @@
{ {
"osmTags": "theme!=grb", "osmTags": "theme!=grb",
"question": { "question": {
"en": "Made with host {search}" "en": "Exclude GRB theme"
} }
} }
] ]
@ -765,7 +607,7 @@
{ {
"osmTags": "theme!=etymology", "osmTags": "theme!=etymology",
"question": { "question": {
"en": "Changeset added at least one image" "en": "Exclude etymology theme"
} }
} }
] ]
@ -780,13 +622,7 @@
{ {
"id": "link_to_more", "id": "link_to_more",
"render": { "render": {
"en": "More statistics can be found <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>here</a>", "en": "More statistics can be found <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>here</a>"
"ca": "Es pot trobar més estadística <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>aquí</a>",
"cs": "Další statistiky najdete <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>here</a>",
"de": "Mehr Statistiken gibt es <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>hier</a>",
"es": "Puede encontrar más estadísticas <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>aquí</a>",
"fr": "D'autres statistiques sont disponibles <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>ici</a>",
"pl": "Więcej statystyk można znaleźć <a href='https://github.com/pietervdvn/MapComplete/tree/develop/Docs/Tools/graphs' target='_blank'>tutaj</a>"
} }
}, },
{ {

2
package-lock.json generated
View file

@ -1,6 +1,6 @@
{ {
"name": "mapcomplete", "name": "mapcomplete",
"version": "0.36.1", "version": "0.36.2",
"lockfileVersion": 2, "lockfileVersion": 2,
"requires": true, "requires": true,
"packages": { "packages": {

View file

@ -1,6 +1,6 @@
{ {
"name": "mapcomplete", "name": "mapcomplete",
"version": "0.36.1", "version": "0.36.2",
"repository": "https://github.com/pietervdvn/MapComplete", "repository": "https://github.com/pietervdvn/MapComplete",
"description": "A small website to edit OSM easily", "description": "A small website to edit OSM easily",
"bugs": "https://github.com/pietervdvn/MapComplete/issues", "bugs": "https://github.com/pietervdvn/MapComplete/issues",

View file

@ -777,10 +777,6 @@ video {
float: left; float: left;
} }
.m-8 {
margin: 2rem;
}
.m-4 { .m-4 {
margin: 1rem; margin: 1rem;
} }
@ -793,6 +789,10 @@ video {
margin: 0px; margin: 0px;
} }
.m-8 {
margin: 2rem;
}
.m-2 { .m-2 {
margin: 0.5rem; margin: 0.5rem;
} }
@ -1188,6 +1188,10 @@ video {
max-height: 6rem; max-height: 6rem;
} }
.max-h-64 {
max-height: 16rem;
}
.max-h-7 { .max-h-7 {
max-height: 1.75rem; max-height: 1.75rem;
} }
@ -1266,6 +1270,10 @@ video {
width: 3.5rem; width: 3.5rem;
} }
.w-auto {
width: auto;
}
.w-5 { .w-5 {
width: 1.25rem; width: 1.25rem;
} }
@ -1283,10 +1291,6 @@ video {
width: 12rem; width: 12rem;
} }
.w-auto {
width: auto;
}
.max-w-full { .max-w-full {
max-width: 100%; max-width: 100%;
} }
@ -1685,6 +1689,10 @@ video {
border-style: dotted; border-style: dotted;
} }
.border-none {
border-style: none;
}
.border-black { .border-black {
--tw-border-opacity: 1; --tw-border-opacity: 1;
border-color: rgb(0 0 0 / var(--tw-border-opacity)); border-color: rgb(0 0 0 / var(--tw-border-opacity));
@ -2234,6 +2242,11 @@ body {
font-family: "Helvetica Neue", Arial, sans-serif; font-family: "Helvetica Neue", Arial, sans-serif;
} }
.focusable {
/* Not a 'real' class, but rather an indication to FloatOver and ModalRight to, when they open, grab the focus */
border: 1px solid red
}
svg, svg,
img { img {
box-sizing: content-box; box-sizing: content-box;

View file

@ -4,7 +4,7 @@ import { RegexTag } from "../src/Logic/Tags/RegexTag"
import { ImmutableStore } from "../src/Logic/UIEventSource" import { ImmutableStore } from "../src/Logic/UIEventSource"
import { BBox } from "../src/Logic/BBox" import { BBox } from "../src/Logic/BBox"
import * as fs from "fs" import * as fs from "fs"
import { writeFileSync } from "fs" import { existsSync, mkdirSync, readFileSync, writeFileSync } from "fs"
import { Feature } from "geojson" import { Feature } from "geojson"
import ScriptUtils from "./ScriptUtils" import ScriptUtils from "./ScriptUtils"
import { Imgur } from "../src/Logic/ImageProviders/Imgur" import { Imgur } from "../src/Logic/ImageProviders/Imgur"
@ -181,6 +181,32 @@ export default class GenerateImageAnalysis extends Script {
} }
} }
async downloadViews(datapath: string): Promise<void> {
const { allImages, imageSource } = this.loadImageUrls(datapath)
console.log("Detected", allImages.size, "images")
const results: [string, number][] = []
const today = new Date().toISOString().substring(0, "YYYY-MM-DD".length)
const viewDir = datapath + "/views_" + today
if (!existsSync(viewDir)) {
mkdirSync(viewDir)
}
for (const image of Array.from(allImages)) {
const cachedView = viewDir + "/" + image.replace(/\\/g, "_")
let attribution: LicenseInfo
if (existsSync(cachedView)) {
attribution = JSON.parse(readFileSync(cachedView, "utf8"))
} else {
attribution = await Imgur.singleton.DownloadAttribution(image)
writeFileSync(cachedView, JSON.stringify(attribution))
}
results.push([image, attribution.views])
}
const targetpath = datapath + "/views.csv"
console.log("Writing views to", targetpath)
fs.writeFileSync(targetpath, results.map((r) => r.join(",")).join("\n"))
}
async downloadImage(url: string, imagePath: string): Promise<boolean> { async downloadImage(url: string, imagePath: string): Promise<boolean> {
const filenameLong = url.replace(/[\/:.\-%]/g, "_") + ".jpg" const filenameLong = url.replace(/[\/:.\-%]/g, "_") + ".jpg"
const targetPathLong = imagePath + "/" + filenameLong const targetPathLong = imagePath + "/" + filenameLong
@ -391,6 +417,7 @@ export default class GenerateImageAnalysis extends Script {
await this.downloadData(datapath, cached) await this.downloadData(datapath, cached)
await this.downloadMetadata(datapath) await this.downloadMetadata(datapath)
await this.downloadViews(datapath)
await this.downloadAllImages(datapath, imageBackupPath) await this.downloadAllImages(datapath, imageBackupPath)
this.analyze(datapath) this.analyze(datapath)
} }

View file

@ -21,6 +21,7 @@ interface TagsUpdaterState {
osmObjectDownloader: OsmObjectDownloader osmObjectDownloader: OsmObjectDownloader
indexedFeatures: IndexedFeatureSource indexedFeatures: IndexedFeatureSource
} }
export default class SelectedElementTagsUpdater { export default class SelectedElementTagsUpdater {
private static readonly metatags = new Set([ private static readonly metatags = new Set([
"timestamp", "timestamp",
@ -42,6 +43,84 @@ export default class SelectedElementTagsUpdater {
}) })
} }
public static applyUpdate(latestTags: OsmTags, id: string, state: TagsUpdaterState) {
try {
const leftRightSensitive = state.layout.isLeftRightSensitive()
if (leftRightSensitive) {
SimpleMetaTagger.removeBothTagging(latestTags)
}
const pendingChanges = state.changes.pendingChanges.data
.filter((change) => change.type + "/" + change.id === id)
.filter((change) => change.tags !== undefined)
for (const pendingChange of pendingChanges) {
const tagChanges = pendingChange.tags
for (const tagChange of tagChanges) {
const key = tagChange.k
const v = tagChange.v
if (v === undefined || v === "") {
delete latestTags[key]
} else {
latestTags[key] = v
}
}
}
// With the changes applied, we merge them onto the upstream object
let somethingChanged = false
const currentTagsSource = state.featureProperties.getStore(id)
if (currentTagsSource === undefined) {
console.warn("No tags store found for", id, "cannot update tags")
return
}
const currentTags = currentTagsSource.data
for (const key in latestTags) {
let osmValue = latestTags[key]
if (typeof osmValue === "number") {
osmValue = "" + osmValue
}
const localValue = currentTags[key]
if (localValue !== osmValue) {
somethingChanged = true
currentTags[key] = osmValue
}
}
for (const currentKey in currentTags) {
if (currentKey.startsWith("_")) {
continue
}
if (SelectedElementTagsUpdater.metatags.has(currentKey)) {
continue
}
if (currentKey in latestTags) {
continue
}
console.log("Removing key as deleted upstream", currentKey)
delete currentTags[currentKey]
somethingChanged = true
}
if (somethingChanged) {
console.log(
"Detected upstream changes to the object " +
id +
" when opening it, updating..."
)
currentTagsSource.ping()
} else {
console.debug("Fetched latest tags for ", id, "but detected no changes")
}
return currentTags
} catch (e) {
console.error("Updating the tags of selected element ", id, "failed due to", e)
}
}
private installCallback(state: TagsUpdaterState) { private installCallback(state: TagsUpdaterState) {
state.selectedElement.addCallbackAndRunD(async (s) => { state.selectedElement.addCallbackAndRunD(async (s) => {
let id = s.properties?.id let id = s.properties?.id
@ -90,77 +169,4 @@ export default class SelectedElementTagsUpdater {
} }
}) })
} }
public static applyUpdate(latestTags: OsmTags, id: string, state: TagsUpdaterState) {
try {
const leftRightSensitive = state.layout.isLeftRightSensitive()
if (leftRightSensitive) {
SimpleMetaTagger.removeBothTagging(latestTags)
}
const pendingChanges = state.changes.pendingChanges.data
.filter((change) => change.type + "/" + change.id === id)
.filter((change) => change.tags !== undefined)
for (const pendingChange of pendingChanges) {
const tagChanges = pendingChange.tags
for (const tagChange of tagChanges) {
const key = tagChange.k
const v = tagChange.v
if (v === undefined || v === "") {
delete latestTags[key]
} else {
latestTags[key] = v
}
}
}
// With the changes applied, we merge them onto the upstream object
let somethingChanged = false
const currentTagsSource = state.featureProperties.getStore(id)
const currentTags = currentTagsSource.data
for (const key in latestTags) {
let osmValue = latestTags[key]
if (typeof osmValue === "number") {
osmValue = "" + osmValue
}
const localValue = currentTags[key]
if (localValue !== osmValue) {
somethingChanged = true
currentTags[key] = osmValue
}
}
for (const currentKey in currentTags) {
if (currentKey.startsWith("_")) {
continue
}
if (SelectedElementTagsUpdater.metatags.has(currentKey)) {
continue
}
if (currentKey in latestTags) {
continue
}
console.log("Removing key as deleted upstream", currentKey)
delete currentTags[currentKey]
somethingChanged = true
}
if (somethingChanged) {
console.log(
"Detected upstream changes to the object " +
id +
" when opening it, updating..."
)
currentTagsSource.ping()
} else {
console.debug("Fetched latest tags for ", id, "but detected no changes")
}
return currentTags
} catch (e) {
console.error("Updating the tags of selected element ", id, "failed due to", e)
}
}
} }

View file

@ -108,7 +108,7 @@ export class ImageUploadManager {
description, description,
file, file,
targetKey, targetKey,
tags.data["_orig_theme"] tags?.data?.["_orig_theme"]
) )
if (!isNaN(Number(featureId))) { if (!isNaN(Number(featureId))) {
// This is a map note // This is a map note

View file

@ -97,7 +97,10 @@ export abstract class Store<T> implements Readable<T> {
abstract map<J>(f: (t: T) => J): Store<J> abstract map<J>(f: (t: T) => J): Store<J>
abstract map<J>(f: (t: T) => J, extraStoresToWatch: Store<any>[]): Store<J> abstract map<J>(f: (t: T) => J, extraStoresToWatch: Store<any>[]): Store<J>
public mapD<J>(f: (t: Exclude<T, undefined | null>) => J, extraStoresToWatch?: Store<any>[]): Store<J> { public mapD<J>(
f: (t: Exclude<T, undefined | null>) => J,
extraStoresToWatch?: Store<any>[]
): Store<J> {
return this.map((t) => { return this.map((t) => {
if (t === undefined) { if (t === undefined) {
return undefined return undefined
@ -105,7 +108,7 @@ export abstract class Store<T> implements Readable<T> {
if (t === null) { if (t === null) {
return null return null
} }
return f(<Exclude<T, undefined | null>> t) return f(<Exclude<T, undefined | null>>t)
}, extraStoresToWatch) }, extraStoresToWatch)
} }
@ -201,24 +204,36 @@ export abstract class Store<T> implements Readable<T> {
mapped.addCallbackAndRun((newEventSource) => { mapped.addCallbackAndRun((newEventSource) => {
if (newEventSource === null) { if (newEventSource === null) {
sink.setData(null) sink.setData(null)
} else if (newEventSource === undefined) { return
}
if (newEventSource === undefined) {
sink.setData(undefined) sink.setData(undefined)
} else if (!seenEventSources.has(newEventSource)) { return
seenEventSources.add(newEventSource) }
newEventSource.addCallbackAndRun((resultData) => { if (seenEventSources.has(newEventSource)) {
if (mapped.data === newEventSource) {
sink.setData(resultData)
}
})
} else {
// Already seen, so we don't have to add a callback, just update the value // Already seen, so we don't have to add a callback, just update the value
sink.setData(newEventSource.data) sink.setData(newEventSource.data)
return
} }
seenEventSources.add(newEventSource)
newEventSource.addCallbackAndRun((resultData) => {
if (mapped.data === newEventSource) {
sink.setData(resultData)
}
})
}) })
return sink return sink
} }
public bindD<X>(f: (t: Exclude<T, undefined | null>) => Store<X>): Store<X> {
return this.bind((t) => {
if (t === undefined || t === null) {
return <undefined | null>t
}
return f(<Exclude<T, undefined | null>>t)
})
}
public stabilized(millisToStabilize): Store<T> { public stabilized(millisToStabilize): Store<T> {
if (Utils.runningFromConsole) { if (Utils.runningFromConsole) {
return this return this
@ -771,7 +786,10 @@ export class UIEventSource<T> extends Store<T> implements Writable<T> {
* Monoidal map which results in a read-only store. 'undefined' is passed 'as is' * Monoidal map which results in a read-only store. 'undefined' is passed 'as is'
* Given a function 'f', will construct a new UIEventSource where the contents will always be "f(this.data)' * Given a function 'f', will construct a new UIEventSource where the contents will always be "f(this.data)'
*/ */
public mapD<J>(f: (t: Exclude<T, undefined | null>) => J, extraSources: Store<any>[] = []): Store<J | undefined> { public mapD<J>(
f: (t: Exclude<T, undefined | null>) => J,
extraSources: Store<any>[] = []
): Store<J | undefined> {
return new MappedStore( return new MappedStore(
this, this,
(t) => { (t) => {
@ -781,11 +799,13 @@ export class UIEventSource<T> extends Store<T> implements Writable<T> {
if (t === null) { if (t === null) {
return null return null
} }
return f(<Exclude<T, undefined | null>> t) return f(<Exclude<T, undefined | null>>t)
}, },
extraSources, extraSources,
this._callbacks, this._callbacks,
(this.data === undefined || this.data === null) ?(<undefined | null> this.data) : f(<any> this.data) this.data === undefined || this.data === null
? <undefined | null>this.data
: f(<any>this.data)
) )
} }

View file

@ -79,6 +79,11 @@ export class MenuState {
this.highlightedUserSetting.setData(undefined) this.highlightedUserSetting.setData(undefined)
} }
}) })
this.menuViewTab.addCallbackD((tab) => {
if (tab !== "settings") {
this.highlightedUserSetting.setData(undefined)
}
})
this.themeViewTab.addCallbackAndRun((tab) => { this.themeViewTab.addCallbackAndRun((tab) => {
if (tab !== "filters") { if (tab !== "filters") {
this.highlightedLayerInFilters.setData(undefined) this.highlightedLayerInFilters.setData(undefined)

View file

@ -475,9 +475,12 @@ export default class ThemeViewState implements SpecialVisualizationState {
{ nomod: "Escape", onUp: true }, { nomod: "Escape", onUp: true },
Translations.t.hotkeyDocumentation.closeSidebar, Translations.t.hotkeyDocumentation.closeSidebar,
() => { () => {
if (this.previewedImage.data !== undefined) {
this.previewedImage.setData(undefined)
return
}
this.selectedElement.setData(undefined) this.selectedElement.setData(undefined)
this.guistate.closeAll() this.guistate.closeAll()
this.previewedImage.setData(undefined)
this.focusOnMap() this.focusOnMap()
} }
) )

View file

@ -36,7 +36,9 @@
dispatcher("submit", e.dataTransfer.files) dispatcher("submit", e.dataTransfer.files)
}} }}
> >
<label class={twMerge(cls, drawAttention ? "glowing-shadow" : "")} for={"fileinput" + id}> <label class={twMerge(cls, drawAttention ? "glowing-shadow" : "")}
tabindex="0" for={"fileinput" + id}
on:click={() => {console.log("Clicked", inputElement); inputElement.click()}}>
<slot /> <slot />
</label> </label>
<input <input

View file

@ -1,5 +1,5 @@
<script lang="ts"> <script lang="ts">
import { createEventDispatcher } from "svelte" import { createEventDispatcher, onMount } from "svelte";
import { XCircleIcon } from "@rgossiaux/svelte-heroicons/solid" import { XCircleIcon } from "@rgossiaux/svelte-heroicons/solid"
import { twMerge } from "tailwind-merge" import { twMerge } from "tailwind-merge"
@ -9,6 +9,12 @@
const dispatch = createEventDispatcher<{ close }>() const dispatch = createEventDispatcher<{ close }>()
export let extraClasses = "p-4 md:p-6" export let extraClasses = "p-4 md:p-6"
let mainContent: HTMLElement
onMount(() => {
console.log("Mounting floatover")
mainContent?.focus()
})
</script> </script>
<div <div
@ -18,7 +24,7 @@
dispatch("close") dispatch("close")
}} }}
> >
<div class="content normal-background" on:click|stopPropagation={() => {}}> <div bind:this={mainContent} class="content normal-background" on:click|stopPropagation={() => {}}>
<div class="h-full rounded-xl"> <div class="h-full rounded-xl">
<slot /> <slot />
</div> </div>

View file

@ -1,25 +1,36 @@
<script lang="ts"> <script lang="ts">
import { createEventDispatcher } from "svelte" import { createEventDispatcher, onMount } from "svelte";
import { XCircleIcon } from "@rgossiaux/svelte-heroicons/solid" import { XCircleIcon } from "@rgossiaux/svelte-heroicons/solid";
import { Utils } from "../../Utils";
/** /**
* The slotted element will be shown on the right side * The slotted element will be shown on the right side
*/ */
const dispatch = createEventDispatcher<{ close }>() const dispatch = createEventDispatcher<{ close }>();
let mainContent: HTMLElement;
onMount(() => {
window.setTimeout(
() => Utils.focusOnFocusableChild(mainContent), 250
)
})
</script> </script>
<div <div
bind:this={mainContent}
class="absolute top-0 right-0 h-screen w-full overflow-y-auto drop-shadow-2xl md:w-6/12 lg:w-5/12 xl:w-4/12" class="absolute top-0 right-0 h-screen w-full overflow-y-auto drop-shadow-2xl md:w-6/12 lg:w-5/12 xl:w-4/12"
style="max-width: 100vw; max-height: 100vh" style="max-width: 100vw; max-height: 100vh"
> >
<div class="normal-background m-0 flex flex-col"> <div class="normal-background m-0 flex flex-col">
<slot name="close-button"> <slot name="close-button">
<div <button
class="absolute right-10 top-10 h-8 w-8 cursor-pointer" class="absolute right-10 top-10 h-8 w-8 cursor-pointer"
on:click={() => dispatch("close")} on:click={() => dispatch("close")}
> >
<XCircleIcon /> <XCircleIcon />
</div> </button>
</slot> </slot>
<slot /> <slot />
</div> </div>

View file

@ -30,7 +30,7 @@
} }
</script> </script>
<div class="tabbedgroup flex h-full w-full"> <div class="tabbedgroup flex h-full w-full focusable">
<TabGroup <TabGroup
class="flex h-full w-full flex-col" class="flex h-full w-full flex-col"
defaultIndex={1} defaultIndex={1}

View file

@ -19,6 +19,7 @@
href={LinkToWeblate.hrefToWeblate($language, context)} href={LinkToWeblate.hrefToWeblate($language, context)}
target="_blank" target="_blank"
class="weblate-link mx-1" class="weblate-link mx-1"
tabindex="-1"
> >
<Translate class="font-gray" /> <Translate class="font-gray" />
</a> </a>
@ -27,6 +28,7 @@
href={LinkToWeblate.hrefToWeblate($language, context)} href={LinkToWeblate.hrefToWeblate($language, context)}
class="weblate-link hidden-on-mobile mx-1" class="weblate-link hidden-on-mobile mx-1"
target="_blank" target="_blank"
tabindex="-1"
> >
<Translate class="font-gray inline-block" /> <Translate class="font-gray inline-block" />
</a> </a>

View file

@ -1,7 +1,6 @@
<script lang="ts"> <script lang="ts">
import { UIEventSource } from "../../Logic/UIEventSource" import { UIEventSource } from "../../Logic/UIEventSource"
import type { Feature } from "geojson" import type { Feature } from "geojson"
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
import ToSvelte from "../Base/ToSvelte.svelte" import ToSvelte from "../Base/ToSvelte.svelte"
import Svg from "../../Svg.js" import Svg from "../../Svg.js"
import Translations from "../i18n/Translations" import Translations from "../i18n/Translations"
@ -15,7 +14,6 @@
export let perLayer: ReadonlyMap<string, GeoIndexedStoreForLayer> | undefined = undefined export let perLayer: ReadonlyMap<string, GeoIndexedStoreForLayer> | undefined = undefined
export let bounds: UIEventSource<BBox> export let bounds: UIEventSource<BBox>
export let selectedElement: UIEventSource<Feature> | undefined = undefined export let selectedElement: UIEventSource<Feature> | undefined = undefined
export let selectedLayer: UIEventSource<LayerConfig> | undefined = undefined
export let clearAfterView: boolean = true export let clearAfterView: boolean = true
@ -34,8 +32,11 @@
let feedback: string = undefined let feedback: string = undefined
Hotkeys.RegisterHotkey({ ctrl: "F" }, Translations.t.hotkeyDocumentation.selectSearch, () => { Hotkeys.RegisterHotkey({ ctrl: "F" }, Translations.t.hotkeyDocumentation.selectSearch, () => {
feedback = undefined
requestAnimationFrame(() => {
inputElement?.focus() inputElement?.focus()
inputElement?.select() inputElement?.select()
})
}) })
const dispatch = createEventDispatcher<{ searchCompleted; searchIsValid: boolean }>() const dispatch = createEventDispatcher<{ searchCompleted; searchIsValid: boolean }>()
@ -73,8 +74,12 @@
const layers = Array.from(perLayer?.values() ?? []) const layers = Array.from(perLayer?.values() ?? [])
for (const layer of layers) { for (const layer of layers) {
const found = layer.features.data.find((f) => f.properties.id === id) const found = layer.features.data.find((f) => f.properties.id === id)
selectedElement?.setData(found) if (found === undefined) {
selectedLayer?.setData(layer.layer.layerDef) continue;
}
selectedElement?.setData(found);
console.log("Found an element that probably matches:", selectedElement?.data);
break;
} }
} }
if (clearAfterView) { if (clearAfterView) {

View file

@ -1,35 +1,25 @@
<script lang="ts"> <script lang="ts">
import type { Feature } from "geojson" import type { Feature } from "geojson";
import { UIEventSource } from "../../Logic/UIEventSource" import { Store, UIEventSource } from "../../Logic/UIEventSource";
import LayerConfig from "../../Models/ThemeConfig/LayerConfig" import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
import type { SpecialVisualizationState } from "../SpecialVisualization" import type { SpecialVisualizationState } from "../SpecialVisualization";
import TagRenderingAnswer from "../Popup/TagRendering/TagRenderingAnswer.svelte" import TagRenderingAnswer from "../Popup/TagRendering/TagRenderingAnswer.svelte";
import { onDestroy } from "svelte" import Translations from "../i18n/Translations";
import Translations from "../i18n/Translations" import Tr from "../Base/Tr.svelte";
import Tr from "../Base/Tr.svelte" import { XCircleIcon } from "@rgossiaux/svelte-heroicons/solid";
import { XCircleIcon } from "@rgossiaux/svelte-heroicons/solid"
export let state: SpecialVisualizationState export let state: SpecialVisualizationState;
export let layer: LayerConfig export let layer: LayerConfig;
export let selectedElement: Feature export let selectedElement: Feature;
export let tags: UIEventSource<Record<string, string>> let tags: UIEventSource<Record<string, string>> = state.featureProperties.getStore(selectedElement.properties.id);
$: {
tags = state.featureProperties.getStore(selectedElement.properties.id);
}
let _tags: Record<string, string> let metatags: Store<Record<string, string>> = state.userRelatedState.preferencesAsTags;
onDestroy(
tags.addCallbackAndRun((tags) => {
_tags = tags
})
)
let _metatags: Record<string, string>
onDestroy(
state.userRelatedState.preferencesAsTags.addCallbackAndRun((tags) => {
_metatags = tags
})
)
</script> </script>
{#if _tags._deleted === "yes"} {#if $tags._deleted === "yes"}
<Tr t={Translations.t.delete.isDeleted} /> <Tr t={Translations.t.delete.isDeleted} />
{:else} {:else}
<div <div
@ -44,7 +34,7 @@
class="no-weblate title-icons links-as-button mr-2 flex flex-row flex-wrap items-center gap-x-0.5 p-1 pt-0.5 sm:pt-1" class="no-weblate title-icons links-as-button mr-2 flex flex-row flex-wrap items-center gap-x-0.5 p-1 pt-0.5 sm:pt-1"
> >
{#each layer.titleIcons as titleIconConfig} {#each layer.titleIcons as titleIconConfig}
{#if (titleIconConfig.condition?.matchesProperties(_tags) ?? true) && (titleIconConfig.metacondition?.matchesProperties( { ..._metatags, ..._tags } ) ?? true) && titleIconConfig.IsKnown(_tags)} {#if (titleIconConfig.condition?.matchesProperties($tags) ?? true) && (titleIconConfig.metacondition?.matchesProperties({ ...$metatags, ...$tags }) ?? true) && titleIconConfig.IsKnown($tags)}
<div class={titleIconConfig.renderIconClass ?? "flex h-8 w-8 items-center"}> <div class={titleIconConfig.renderIconClass ?? "flex h-8 w-8 items-center"}>
<TagRenderingAnswer <TagRenderingAnswer
config={titleIconConfig} config={titleIconConfig}
@ -59,15 +49,15 @@
{/each} {/each}
</div> </div>
</div> </div>
<XCircleIcon
class="h-8 w-8 cursor-pointer" <button on:click={() => state.selectedElement.setData(undefined)} class="border-none p-0">
on:click={() => state.selectedElement.setData(undefined)} <XCircleIcon class="h-8 w-8" />
/> </button>
</div> </div>
{/if} {/if}
<style> <style>
:global(.title-icons a) { :global(.title-icons a) {
display: block !important; display: block !important;
} }
</style> </style>

View file

@ -1,19 +1,23 @@
<script lang="ts"> <script lang="ts">
import type { Feature } from "geojson" import type { Feature } from "geojson"
import { UIEventSource } from "../../Logic/UIEventSource" import { Store, UIEventSource } from "../../Logic/UIEventSource";
import LayerConfig from "../../Models/ThemeConfig/LayerConfig" import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
import type { SpecialVisualizationState } from "../SpecialVisualization" import type { SpecialVisualizationState } from "../SpecialVisualization"
import TagRenderingEditable from "../Popup/TagRendering/TagRenderingEditable.svelte" import TagRenderingEditable from "../Popup/TagRendering/TagRenderingEditable.svelte"
import { onDestroy } from "svelte" import { onDestroy } from "svelte"
import Translations from "../i18n/Translations" import Translations from "../i18n/Translations"
import Tr from "../Base/Tr.svelte" import Tr from "../Base/Tr.svelte"
import TagRenderingConfig from "../../Models/ThemeConfig/TagRenderingConfig";
export let state: SpecialVisualizationState export let state: SpecialVisualizationState
export let layer: LayerConfig export let layer: LayerConfig
export let selectedElement: Feature export let selectedElement: Feature
export let tags: UIEventSource<Record<string, string>>
export let highlightedRendering: UIEventSource<string> = undefined export let highlightedRendering: UIEventSource<string> = undefined
let tags: UIEventSource<Record<string, string>> = state.featureProperties.getStore(selectedElement.properties.id)
$: {
tags = state.featureProperties.getStore(selectedElement.properties.id)
}
let _metatags: Record<string, string> let _metatags: Record<string, string>
onDestroy( onDestroy(
state.userRelatedState.preferencesAsTags.addCallbackAndRun((tags) => { state.userRelatedState.preferencesAsTags.addCallbackAndRun((tags) => {
@ -21,20 +25,12 @@
}) })
) )
let knownTagRenderings = layer.tagRenderings.filter( let knownTagRenderings: Store<TagRenderingConfig[]> = tags.mapD(tgs => layer.tagRenderings.filter(
(config) => (config) =>
(config.condition?.matchesProperties($tags) ?? true) && (config.condition?.matchesProperties(tgs) ?? true) &&
config.metacondition?.matchesProperties({ ...$tags, ..._metatags } ?? true) && config.metacondition?.matchesProperties({ ...tgs, ..._metatags } ?? true) &&
config.IsKnown($tags) config.IsKnown(tgs)
) ))
$: {
knownTagRenderings = layer.tagRenderings.filter(
(config) =>
(config.condition?.matchesProperties($tags) ?? true) &&
config.metacondition?.matchesProperties({ ...$tags, ..._metatags } ?? true) &&
config.IsKnown($tags)
)
}
</script> </script>
{#if $tags._deleted === "yes"} {#if $tags._deleted === "yes"}
@ -43,8 +39,8 @@
<Tr t={Translations.t.general.returnToTheMap} /> <Tr t={Translations.t.general.returnToTheMap} />
</button> </button>
{:else} {:else}
<div class="flex h-full flex-col gap-y-2 overflow-y-auto p-1 px-2"> <div class="flex h-full flex-col gap-y-2 overflow-y-auto p-1 px-2 focusable" tabindex="-1">
{#each knownTagRenderings as config (config.id)} {#each $knownTagRenderings as config (config.id)}
<TagRenderingEditable <TagRenderingEditable
{tags} {tags}
{config} {config}
@ -52,7 +48,7 @@
{selectedElement} {selectedElement}
{layer} {layer}
{highlightedRendering} {highlightedRendering}
clss={knownTagRenderings.length === 1 ? "h-full" : "tr-length-" + knownTagRenderings.length} clss={$knownTagRenderings.length === 1 ? "h-full" : "tr-length-" + $knownTagRenderings.length}
/> />
{/each} {/each}
</div> </div>

View file

@ -23,7 +23,6 @@
export let state: ThemeViewState export let state: ThemeViewState
let layout = state.layout let layout = state.layout
let selectedElement = state.selectedElement let selectedElement = state.selectedElement
let selectedLayer = state.selectedLayer
let triggerSearch: UIEventSource<any> = new UIEventSource<any>(undefined) let triggerSearch: UIEventSource<any> = new UIEventSource<any>(undefined)
let searchEnabled = false let searchEnabled = false
@ -116,7 +115,6 @@
}} }}
perLayer={state.perLayer} perLayer={state.perLayer}
{selectedElement} {selectedElement}
{selectedLayer}
{triggerSearch} {triggerSearch}
/> />
</div> </div>

View file

@ -27,7 +27,6 @@
} }
function select() { function select() {
state.selectedLayer.setData(favConfig);
state.selectedElement.setData(feature); state.selectedElement.setData(feature);
center(); center();
} }

View file

@ -13,11 +13,12 @@
} }
let imgEl: HTMLImageElement let imgEl: HTMLImageElement
export let imgClass: string = undefined
</script> </script>
<div class="relative"> <div class="relative">
<img bind:this={imgEl} src={image.url} on:error={(event) => { <img bind:this={imgEl} src={image.url} class={imgClass ?? ""} on:error={(event) => {
if(fallbackImage){ if(fallbackImage){
imgEl.src = fallbackImage imgEl.src = fallbackImage
} }

View file

@ -7,9 +7,10 @@ import ImageAttribution from "./ImageAttribution.svelte"
import ImagePreview from "./ImagePreview.svelte" import ImagePreview from "./ImagePreview.svelte"
import { DownloadIcon } from "@rgossiaux/svelte-heroicons/solid" import { DownloadIcon } from "@rgossiaux/svelte-heroicons/solid"
import { Utils } from "../../Utils" import { Utils } from "../../Utils"
import { twMerge } from "tailwind-merge";
export let image: ProvidedImage export let image: ProvidedImage
export let clss: string = undefined
async function download() { async function download() {
const response = await fetch(image.url) const response = await fetch(image.url)
const blob = await response.blob() const blob = await response.blob()
@ -20,7 +21,7 @@ async function download() {
</script> </script>
<div class="w-full h-full relative"> <div class={twMerge("w-full h-full relative", clss)}>
<div class="absolute top-0 left-0 w-full h-full overflow-hidden"> <div class="absolute top-0 left-0 w-full h-full overflow-hidden">
<ImagePreview image={image} /> <ImagePreview image={image} />
</div> </div>

View file

@ -13,7 +13,7 @@
$: { $: {
if (panzoomEl) { if (panzoomEl) {
panzoomInstance = panzoom(panzoomEl, { bounds: true, panzoomInstance = panzoom(panzoomEl, { bounds: true,
boundsPadding: 1, boundsPadding: 0.49,
minZoom: 1, minZoom: 1,
maxZoom: 25, maxZoom: 25,
initialZoom: 1.2 initialZoom: 1.2

View file

@ -64,7 +64,9 @@
</script> </script>
<div class="flex w-fit shrink-0 flex-col"> <div class="flex w-fit shrink-0 flex-col">
<AttributedImage image={providedImage} /> <div on:click={() => state.previewedImage.setData(providedImage)}>
<AttributedImage image={providedImage} imgClass="max-h-64 w-auto"/>
</div>
{#if linkable} {#if linkable}
<label> <label>
<input bind:checked={isLinked} type="checkbox" /> <input bind:checked={isLinked} type="checkbox" />

View file

@ -1,52 +1,49 @@
<script lang="ts"> <script lang="ts">
import { Store } from "../../Logic/UIEventSource" import { Store } from "../../Logic/UIEventSource";
import type { OsmTags } from "../../Models/OsmFeature" import type { OsmTags } from "../../Models/OsmFeature";
import type { SpecialVisualizationState } from "../SpecialVisualization" import type { SpecialVisualizationState } from "../SpecialVisualization";
import type { Feature } from "geojson" import type { Feature } from "geojson";
import LayerConfig from "../../Models/ThemeConfig/LayerConfig" import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
import Translations from "../i18n/Translations" import Translations from "../i18n/Translations";
import Tr from "../Base/Tr.svelte" import Tr from "../Base/Tr.svelte";
import NearbyImages from "./NearbyImages.svelte" import NearbyImages from "./NearbyImages.svelte";
import Svg from "../../Svg" import { XCircleIcon } from "@babeard/svelte-heroicons/solid";
import ToSvelte from "../Base/ToSvelte.svelte" import Camera_plus from "../../assets/svg/Camera_plus.svelte";
import { XCircleIcon } from "@babeard/svelte-heroicons/solid" import LoginToggle from "../Base/LoginToggle.svelte";
import exp from "constants"
import Camera_plus from "../../assets/svg/Camera_plus.svelte"
import LoginToggle from "../Base/LoginToggle.svelte"
export let tags: Store<OsmTags> export let tags: Store<OsmTags>;
export let state: SpecialVisualizationState export let state: SpecialVisualizationState;
export let lon: number export let lon: number;
export let lat: number export let lat: number;
export let feature: Feature export let feature: Feature;
export let linkable: boolean = true export let linkable: boolean = true;
export let layer: LayerConfig export let layer: LayerConfig;
const t = Translations.t.image.nearby const t = Translations.t.image.nearby;
let expanded = false let expanded = false;
</script> </script>
<LoginToggle {state}> <LoginToggle {state}>
{#if expanded} {#if expanded}
<NearbyImages {tags} {state} {lon} {lat} {feature} {linkable}> <NearbyImages {tags} {state} {lon} {lat} {feature} {linkable}>
<XCircleIcon <button slot="corner"
slot="corner" class="h-6 w-6 cursor-pointer no-image-background p-0 border-none"
class="h-6 w-6 cursor-pointer" on:click={() => {
on:click={() => {
expanded = false expanded = false
}} }}>
/> <XCircleIcon />
</NearbyImages> </button>
{:else} </NearbyImages>
<button {:else}
class="flex w-full items-center" <button
on:click={() => { class="flex w-full items-center"
on:click={() => {
expanded = true expanded = true
}} }}
> >
<Camera_plus class="mr-2 block h-8 w-8 p-1" /> <Camera_plus class="mr-2 block h-8 w-8 p-1" />
<Tr t={t.seeNearby} /> <Tr t={t.seeNearby} />
</button> </button>
{/if} {/if}
</LoginToggle> </LoginToggle>

View file

@ -56,9 +56,9 @@
multiple={true} multiple={true}
on:submit={(e) => handleFiles(e.detail)} on:submit={(e) => handleFiles(e.detail)}
> >
<div class="flex items-center"> <div class="flex items-center" >
{#if image !== undefined} {#if image !== undefined}
<img src={image} /> <img src={image} aria-hidden="true" />
{:else} {:else}
<Camera_plus class="block h-12 w-12 p-1 text-4xl" /> <Camera_plus class="block h-12 w-12 p-1 text-4xl" />
{/if} {/if}
@ -70,15 +70,15 @@
</div> </div>
</FileSelector> </FileSelector>
<div class="text-sm"> <div class="text-sm">
<Tr t={t.respectPrivacy} /> <button
<a class="link small "
class="cursor-pointer"
on:click={() => { on:click={() => {
state.guistate.openUsersettings("picture-license") state.guistate.openUsersettings("picture-license")
}} }}
> >
<Tr t={t.currentLicense.Subs({ license: $licenseStore })} /> <Tr t={t.currentLicense.Subs({ license: $licenseStore })} />
</a> </button>
<Tr t={t.respectPrivacy} />
</div> </div>
</div> </div>
</LoginToggle> </LoginToggle>

View file

@ -302,19 +302,23 @@ export class MapLibreAdaptor implements MapProperties, ExportableMap {
rescaleIcons: number, rescaleIcons: number,
pixelRatio: number pixelRatio: number
) { ) {
const marker = element const style = element.style.transform
const style = marker.style.transform let x = element.getBoundingClientRect().x
let x = marker.getBoundingClientRect().x let y = element.getBoundingClientRect().y
let y = marker.getBoundingClientRect().y element.style.transform = ""
marker.style.transform = ""
const offset = style.match(/translate\(([-0-9]+)%, ?([-0-9]+)%\)/) const offset = style.match(/translate\(([-0-9]+)%, ?([-0-9]+)%\)/)
const w = marker.style.width const w = element.style.width
const h = element.style.height
// Force a wider view for icon badges // Force a wider view for icon badges
marker.style.width = marker.getBoundingClientRect().width * 4 + "px" element.style.width = element.getBoundingClientRect().width * 4 + "px"
const svgSource = await htmltoimage.toSvg(marker) element.style.height = element.getBoundingClientRect().height + "px"
const svgSource = await htmltoimage.toSvg(element)
const img = await MapLibreAdaptor.createImage(svgSource) const img = await MapLibreAdaptor.createImage(svgSource)
marker.style.width = w element.style.width = w
element.style.height = h
if (offset && rescaleIcons !== 1) { if (offset && rescaleIcons !== 1) {
const [_, __, relYStr] = offset const [_, __, relYStr] = offset
const relY = Number(relYStr) const relY = Number(relYStr)

View file

@ -410,8 +410,13 @@ class LineRenderingLayer {
this._listenerInstalledOn.add(id) this._listenerInstalledOn.add(id)
tags.addCallbackAndRunD((properties) => { tags.addCallbackAndRunD((properties) => {
// Make sure to use 'getSource' here, the layer names are different! // Make sure to use 'getSource' here, the layer names are different!
if (map.getSource(this._layername) === undefined) { try {
return true if (map.getSource(this._layername) === undefined) {
return true
}
} catch (e) {
console.debug("Could not fetch source for", this._layername)
return
} }
map.setFeatureState( map.setFeatureState(
{ source: this._layername, id }, { source: this._layername, id },

View file

@ -5,14 +5,13 @@
import TagRenderingMapping from "./TagRenderingMapping.svelte" import TagRenderingMapping from "./TagRenderingMapping.svelte"
import type { SpecialVisualizationState } from "../../SpecialVisualization" import type { SpecialVisualizationState } from "../../SpecialVisualization"
import type { Feature } from "geojson" import type { Feature } from "geojson"
import { UIEventSource } from "../../../Logic/UIEventSource" import { Store, UIEventSource } from "../../../Logic/UIEventSource";
import { onDestroy } from "svelte" import { onDestroy } from "svelte"
import LayerConfig from "../../../Models/ThemeConfig/LayerConfig" import LayerConfig from "../../../Models/ThemeConfig/LayerConfig"
import { twMerge } from "tailwind-merge" import { twMerge } from "tailwind-merge"
export let tags: UIEventSource<Record<string, string> | undefined> export let tags: UIEventSource<Record<string, string> | undefined>
let _tags: Record<string, string> let _tags: Record<string, string>
let trs: { then: Translation; icon?: string; iconClass?: string }[]
export let state: SpecialVisualizationState export let state: SpecialVisualizationState
export let selectedElement: Feature export let selectedElement: Feature
@ -23,22 +22,18 @@
if (config === undefined) { if (config === undefined) {
throw "Config is undefined in tagRenderingAnswer" throw "Config is undefined in tagRenderingAnswer"
} }
onDestroy( let trs : Store<{then: Translation, icon?: string, iconClass?: string}[]> = tags.mapD(tags => Utils.NoNull(config?.GetRenderValues(tags)))
tags.addCallbackAndRun((tags) => {
_tags = tags
trs = Utils.NoNull(config?.GetRenderValues(_tags))
})
)
</script> </script>
{#if config !== undefined && (config?.condition === undefined || config.condition.matchesProperties(_tags))} {#if config !== undefined && (config?.condition === undefined || config.condition.matchesProperties($tags))}
<div class={twMerge("link-underline inline-block w-full", config?.classes, extraClasses)}> <div class={twMerge("link-underline inline-block w-full", config?.classes, extraClasses)}>
{#if trs.length === 1} {#if $trs.length === 1}
<TagRenderingMapping mapping={trs[0]} {tags} {state} {selectedElement} {layer} /> <TagRenderingMapping mapping={$trs[0]} {tags} {state} {selectedElement} {layer} />
{/if} {/if}
{#if trs.length > 1} {#if $trs.length > 1}
<ul> <ul>
{#each trs as mapping} {#each $trs as mapping}
<li> <li>
<TagRenderingMapping {mapping} {tags} {state} {selectedElement} {layer} /> <TagRenderingMapping {mapping} {tags} {state} {selectedElement} {layer} />
</li> </li>

View file

@ -1,40 +1,41 @@
<script lang="ts"> <script lang="ts">
import TagRenderingConfig from "../../../Models/ThemeConfig/TagRenderingConfig" import TagRenderingConfig from "../../../Models/ThemeConfig/TagRenderingConfig";
import { Store, UIEventSource } from "../../../Logic/UIEventSource" import { Store, UIEventSource } from "../../../Logic/UIEventSource";
import type { Feature } from "geojson" import type { Feature } from "geojson";
import type { SpecialVisualizationState } from "../../SpecialVisualization" import type { SpecialVisualizationState } from "../../SpecialVisualization";
import TagRenderingAnswer from "./TagRenderingAnswer.svelte" import TagRenderingAnswer from "./TagRenderingAnswer.svelte";
import { PencilAltIcon, XCircleIcon } from "@rgossiaux/svelte-heroicons/solid" import { PencilAltIcon, XCircleIcon } from "@rgossiaux/svelte-heroicons/solid";
import TagRenderingQuestion from "./TagRenderingQuestion.svelte" import TagRenderingQuestion from "./TagRenderingQuestion.svelte";
import { onDestroy } from "svelte" import { onDestroy } from "svelte";
import Tr from "../../Base/Tr.svelte" import Tr from "../../Base/Tr.svelte";
import Translations from "../../i18n/Translations.js" import Translations from "../../i18n/Translations.js";
import LayerConfig from "../../../Models/ThemeConfig/LayerConfig" import LayerConfig from "../../../Models/ThemeConfig/LayerConfig";
import { Utils } from "../../../Utils" import { Utils } from "../../../Utils";
import { twMerge } from "tailwind-merge" import { twMerge } from "tailwind-merge";
export let config: TagRenderingConfig
export let tags: UIEventSource<Record<string, string>>
export let selectedElement: Feature | undefined
export let state: SpecialVisualizationState
export let layer: LayerConfig = undefined
export let editingEnabled: Store<boolean> | undefined = state?.featureSwitchUserbadge export let config: TagRenderingConfig;
export let tags: UIEventSource<Record<string, string>>;
export let selectedElement: Feature | undefined;
export let state: SpecialVisualizationState;
export let layer: LayerConfig = undefined;
export let highlightedRendering: UIEventSource<string> = undefined export let editingEnabled: Store<boolean> | undefined = state?.featureSwitchUserbadge;
export let clss
export let highlightedRendering: UIEventSource<string> = undefined;
export let clss;
/** /**
* Indicates if this tagRendering currently shows the attribute or asks the question to _change_ the property * Indicates if this tagRendering currently shows the attribute or asks the question to _change_ the property
*/ */
export let editMode = !config.IsKnown(tags.data) // || showQuestionIfUnknown; export let editMode = !config.IsKnown(tags.data); // || showQuestionIfUnknown;
if (tags) { if (tags) {
onDestroy( onDestroy(
tags.addCallbackD((tags) => { tags.addCallbackD((tags) => {
editMode = !config.IsKnown(tags) editMode = !config.IsKnown(tags);
}) })
) );
} }
let htmlElem: HTMLDivElement let htmlElem: HTMLDivElement;
$: { $: {
if (editMode && htmlElem !== undefined && config.IsKnown(tags)) { if (editMode && htmlElem !== undefined && config.IsKnown(tags)) {
// EditMode switched to true yet the answer is already known, so the person wants to make a change // EditMode switched to true yet the answer is already known, so the person wants to make a change
@ -42,32 +43,36 @@
// Some delay is applied to give Svelte the time to render the _question_ // Some delay is applied to give Svelte the time to render the _question_
window.setTimeout(() => { window.setTimeout(() => {
Utils.scrollIntoView(<any>htmlElem) Utils.scrollIntoView(<any>htmlElem);
}, 50) }, 50);
} }
} }
const _htmlElement = new UIEventSource<HTMLElement>(undefined) const _htmlElement = new UIEventSource<HTMLElement>(undefined);
$: _htmlElement.setData(htmlElem) $: _htmlElement.setData(htmlElem);
function setHighlighting() { function setHighlighting() {
if (highlightedRendering === undefined) { if (highlightedRendering === undefined) {
return return;
} }
if (htmlElem === undefined) { if (htmlElem === undefined) {
return return;
} }
const highlighted = highlightedRendering.data const highlighted = highlightedRendering.data;
if (config.id === highlighted) { if (config.id === highlighted) {
htmlElem.classList.add("glowing-shadow") htmlElem.classList.add("glowing-shadow");
htmlElem.tabIndex = "-1";
console.log("Scrolling to", htmlElem);
htmlElem.scrollIntoView({ behavior: "smooth" });
Utils.focusOnFocusableChild(htmlElem);
} else { } else {
htmlElem.classList.remove("glowing-shadow") htmlElem.classList.remove("glowing-shadow");
} }
} }
if (highlightedRendering) { if (highlightedRendering) {
onDestroy(highlightedRendering?.addCallbackAndRun(() => setHighlighting())) onDestroy(highlightedRendering?.addCallbackAndRun(() => setHighlighting()));
onDestroy(_htmlElement.addCallbackAndRun(() => setHighlighting())) onDestroy(_htmlElement.addCallbackAndRun(() => setHighlighting()));
} }
</script> </script>
@ -84,13 +89,13 @@
> >
<Tr t={Translations.t.general.cancel} /> <Tr t={Translations.t.general.cancel} />
</button> </button>
<XCircleIcon <button slot="upper-right"
slot="upper-right" class="h-8 w-8 cursor-pointer border-none p-0"
class="h-8 w-8 cursor-pointer" on:click={() => {
on:click={() => {
editMode = false editMode = false
}} }}>
/> <XCircleIcon />
</button>
</TagRenderingQuestion> </TagRenderingQuestion>
{:else} {:else}
<div class="low-interaction flex items-center justify-between overflow-hidden rounded px-2"> <div class="low-interaction flex items-center justify-between overflow-hidden rounded px-2">

View file

@ -136,7 +136,6 @@
function onSave() { function onSave() {
if (selectedTags === undefined) { if (selectedTags === undefined) {
console.log("SelectedTags is undefined, ignoring 'onSave'-event");
return; return;
} }
if (layer === undefined || (layer?.source === null && layer.id !== "favourite")) { if (layer === undefined || (layer?.source === null && layer.id !== "favourite")) {

View file

@ -1,138 +1,116 @@
<script lang="ts"> <script lang="ts">
import { Store, UIEventSource } from "../Logic/UIEventSource" import { Store, UIEventSource } from "../Logic/UIEventSource";
import { Map as MlMap } from "maplibre-gl" import { Map as MlMap } from "maplibre-gl";
import MaplibreMap from "./Map/MaplibreMap.svelte" import MaplibreMap from "./Map/MaplibreMap.svelte";
import FeatureSwitchState from "../Logic/State/FeatureSwitchState" import FeatureSwitchState from "../Logic/State/FeatureSwitchState";
import MapControlButton from "./Base/MapControlButton.svelte" import MapControlButton from "./Base/MapControlButton.svelte";
import ToSvelte from "./Base/ToSvelte.svelte" import ToSvelte from "./Base/ToSvelte.svelte";
import If from "./Base/If.svelte" import If from "./Base/If.svelte";
import { GeolocationControl } from "./BigComponents/GeolocationControl" import { GeolocationControl } from "./BigComponents/GeolocationControl";
import type { Feature } from "geojson" import type { Feature } from "geojson";
import SelectedElementView from "./BigComponents/SelectedElementView.svelte" import SelectedElementView from "./BigComponents/SelectedElementView.svelte";
import LayerConfig from "../Models/ThemeConfig/LayerConfig" import LayerConfig from "../Models/ThemeConfig/LayerConfig";
import Filterview from "./BigComponents/Filterview.svelte" import Filterview from "./BigComponents/Filterview.svelte";
import ThemeViewState from "../Models/ThemeViewState" import ThemeViewState from "../Models/ThemeViewState";
import type { MapProperties } from "../Models/MapProperties" import type { MapProperties } from "../Models/MapProperties";
import Geosearch from "./BigComponents/Geosearch.svelte" import Geosearch from "./BigComponents/Geosearch.svelte";
import Translations from "./i18n/Translations" import Translations from "./i18n/Translations";
import { CogIcon, EyeIcon, HeartIcon, MenuIcon, XCircleIcon } from "@rgossiaux/svelte-heroicons/solid" import { CogIcon, EyeIcon, HeartIcon, MenuIcon, XCircleIcon } from "@rgossiaux/svelte-heroicons/solid";
import Tr from "./Base/Tr.svelte" import Tr from "./Base/Tr.svelte";
import CommunityIndexView from "./BigComponents/CommunityIndexView.svelte" import CommunityIndexView from "./BigComponents/CommunityIndexView.svelte";
import FloatOver from "./Base/FloatOver.svelte" import FloatOver from "./Base/FloatOver.svelte";
import PrivacyPolicy from "./BigComponents/PrivacyPolicy" import PrivacyPolicy from "./BigComponents/PrivacyPolicy";
import Constants from "../Models/Constants" import Constants from "../Models/Constants";
import TabbedGroup from "./Base/TabbedGroup.svelte" import TabbedGroup from "./Base/TabbedGroup.svelte";
import UserRelatedState from "../Logic/State/UserRelatedState" import UserRelatedState from "../Logic/State/UserRelatedState";
import LoginToggle from "./Base/LoginToggle.svelte" import LoginToggle from "./Base/LoginToggle.svelte";
import LoginButton from "./Base/LoginButton.svelte" import LoginButton from "./Base/LoginButton.svelte";
import CopyrightPanel from "./BigComponents/CopyrightPanel" import CopyrightPanel from "./BigComponents/CopyrightPanel";
import DownloadPanel from "./DownloadFlow/DownloadPanel.svelte" import DownloadPanel from "./DownloadFlow/DownloadPanel.svelte";
import ModalRight from "./Base/ModalRight.svelte" import ModalRight from "./Base/ModalRight.svelte";
import { Utils } from "../Utils" import { Utils } from "../Utils";
import Hotkeys from "./Base/Hotkeys" import Hotkeys from "./Base/Hotkeys";
import { VariableUiElement } from "./Base/VariableUIElement" import SvelteUIElement from "./Base/SvelteUIElement";
import SvelteUIElement from "./Base/SvelteUIElement" import OverlayToggle from "./BigComponents/OverlayToggle.svelte";
import OverlayToggle from "./BigComponents/OverlayToggle.svelte" import LevelSelector from "./BigComponents/LevelSelector.svelte";
import LevelSelector from "./BigComponents/LevelSelector.svelte" import ExtraLinkButton from "./BigComponents/ExtraLinkButton";
import ExtraLinkButton from "./BigComponents/ExtraLinkButton" import SelectedElementTitle from "./BigComponents/SelectedElementTitle.svelte";
import SelectedElementTitle from "./BigComponents/SelectedElementTitle.svelte" import ThemeIntroPanel from "./BigComponents/ThemeIntroPanel.svelte";
import ThemeIntroPanel from "./BigComponents/ThemeIntroPanel.svelte" import type { RasterLayerPolygon } from "../Models/RasterLayers";
import type { RasterLayerPolygon } from "../Models/RasterLayers" import { AvailableRasterLayers } from "../Models/RasterLayers";
import { AvailableRasterLayers } from "../Models/RasterLayers" import RasterLayerOverview from "./Map/RasterLayerOverview.svelte";
import RasterLayerOverview from "./Map/RasterLayerOverview.svelte" import IfHidden from "./Base/IfHidden.svelte";
import IfHidden from "./Base/IfHidden.svelte" import { onDestroy } from "svelte";
import { onDestroy } from "svelte" import MapillaryLink from "./BigComponents/MapillaryLink.svelte";
import MapillaryLink from "./BigComponents/MapillaryLink.svelte" import OpenIdEditor from "./BigComponents/OpenIdEditor.svelte";
import OpenIdEditor from "./BigComponents/OpenIdEditor.svelte" import OpenBackgroundSelectorButton from "./BigComponents/OpenBackgroundSelectorButton.svelte";
import OpenBackgroundSelectorButton from "./BigComponents/OpenBackgroundSelectorButton.svelte" import StateIndicator from "./BigComponents/StateIndicator.svelte";
import StateIndicator from "./BigComponents/StateIndicator.svelte" import ShareScreen from "./BigComponents/ShareScreen.svelte";
import ShareScreen from "./BigComponents/ShareScreen.svelte" import UploadingImageCounter from "./Image/UploadingImageCounter.svelte";
import UploadingImageCounter from "./Image/UploadingImageCounter.svelte" import PendingChangesIndicator from "./BigComponents/PendingChangesIndicator.svelte";
import PendingChangesIndicator from "./BigComponents/PendingChangesIndicator.svelte" import Cross from "../assets/svg/Cross.svelte";
import Cross from "../assets/svg/Cross.svelte" import Summary from "./BigComponents/Summary.svelte";
import Summary from "./BigComponents/Summary.svelte" import LanguagePicker from "./InputElement/LanguagePicker.svelte";
import LanguagePicker from "./InputElement/LanguagePicker.svelte" import Mastodon from "../assets/svg/Mastodon.svelte";
import Mastodon from "../assets/svg/Mastodon.svelte" import Bug from "../assets/svg/Bug.svelte";
import Bug from "../assets/svg/Bug.svelte" import Liberapay from "../assets/svg/Liberapay.svelte";
import Liberapay from "../assets/svg/Liberapay.svelte" import OpenJosm from "./Base/OpenJosm.svelte";
import OpenJosm from "./Base/OpenJosm.svelte" import Min from "../assets/svg/Min.svelte";
import Min from "../assets/svg/Min.svelte" import Plus from "../assets/svg/Plus.svelte";
import Plus from "../assets/svg/Plus.svelte" import Filter from "../assets/svg/Filter.svelte";
import Filter from "../assets/svg/Filter.svelte" import Add from "../assets/svg/Add.svelte";
import Add from "../assets/svg/Add.svelte" import Statistics from "../assets/svg/Statistics.svelte";
import Statistics from "../assets/svg/Statistics.svelte" import Community from "../assets/svg/Community.svelte";
import Community from "../assets/svg/Community.svelte" import Download from "../assets/svg/Download.svelte";
import Download from "../assets/svg/Download.svelte" import Share from "../assets/svg/Share.svelte";
import Share from "../assets/svg/Share.svelte" import Favourites from "./Favourites/Favourites.svelte";
import Favourites from "./Favourites/Favourites.svelte" import ImageOperations from "./Image/ImageOperations.svelte";
import ImageOperations from "./Image/ImageOperations.svelte"
export let state: ThemeViewState export let state: ThemeViewState;
let layout = state.layout let layout = state.layout;
let maplibremap: UIEventSource<MlMap> = state.map let maplibremap: UIEventSource<MlMap> = state.map;
let selectedElement: UIEventSource<Feature> = state.selectedElement let selectedElement: UIEventSource<Feature> = new UIEventSource<Feature>(undefined);
let selectedLayer: UIEventSource<LayerConfig> = state.selectedLayer
let currentZoom = state.mapProperties.zoom state.selectedElement.addCallback(selected => {
let showCrosshair = state.userRelatedState.showCrosshair if(!selected){
let arrowKeysWereUsed = state.mapProperties.lastKeyNavigation selectedElement.setData(selected)
let centerFeatures = state.closestFeatures.features return
const selectedElementView = selectedElement.map( }
(selectedElement) => { if(selected !== selectedElement.data){
// Svelte doesn't properly reload some of the legacy UI-elements // We first set the selected element to 'undefined' to force the popup to close...
// As such, we _reconstruct_ the selectedElementView every time a new feature is selected selectedElement.setData(undefined)
// This is a bit wasteful, but until everything is a svelte-component, this should do the trick }
const layer = selectedLayer.data // ... we give svelte some time to update with requestAnimationFrame ...
if (selectedElement === undefined || layer === undefined) { window.requestAnimationFrame(() => {
return undefined // ... and we force a fresh popup window
} selectedElement.setData(selected)
})
if (!(layer.tagRenderings?.length > 0) || layer.title === undefined) { })
return undefined
}
const tags = state.featureProperties.getStore(selectedElement.properties.id) let selectedLayer: UIEventSource<LayerConfig> = state.selectedElement.mapD(element => state.layout.getMatchingLayer(element.properties));
return new SvelteUIElement(SelectedElementView, {
state,
layer,
selectedElement,
tags,
}).SetClass("h-full w-full")
},
[selectedLayer],
)
const selectedElementTitle = selectedElement.map( let currentZoom = state.mapProperties.zoom;
(selectedElement) => { let showCrosshair = state.userRelatedState.showCrosshair;
// Svelte doesn't properly reload some of the legacy UI-elements let arrowKeysWereUsed = state.mapProperties.lastKeyNavigation;
// As such, we _reconstruct_ the selectedElementView every time a new feature is selected let centerFeatures = state.closestFeatures.features;
// This is a bit wasteful, but until everything is a svelte-component, this should do the trick
const layer = selectedLayer.data
if (selectedElement === undefined || layer === undefined) {
return undefined
}
const tags = state.featureProperties.getStore(selectedElement.properties.id)
return new SvelteUIElement(SelectedElementTitle, { state, layer, selectedElement, tags })
},
[selectedLayer],
)
let mapproperties: MapProperties = state.mapProperties let mapproperties: MapProperties = state.mapProperties;
let featureSwitches: FeatureSwitchState = state.featureSwitches let featureSwitches: FeatureSwitchState = state.featureSwitches;
let availableLayers = state.availableLayers let availableLayers = state.availableLayers;
let userdetails = state.osmConnection.userDetails let userdetails = state.osmConnection.userDetails;
let currentViewLayer = layout.layers.find((l) => l.id === "current_view") let currentViewLayer = layout.layers.find((l) => l.id === "current_view");
let rasterLayer: Store<RasterLayerPolygon> = state.mapProperties.rasterLayer let rasterLayer: Store<RasterLayerPolygon> = state.mapProperties.rasterLayer;
let rasterLayerName = let rasterLayerName =
rasterLayer.data?.properties?.name ?? AvailableRasterLayers.maptilerDefaultLayer.properties.name rasterLayer.data?.properties?.name ?? AvailableRasterLayers.maptilerDefaultLayer.properties.name;
onDestroy( onDestroy(
rasterLayer.addCallbackAndRunD((l) => { rasterLayer.addCallbackAndRunD((l) => {
rasterLayerName = l.properties.name rasterLayerName = l.properties.name;
}), })
) );
let previewedImage = state.previewedImage let previewedImage = state.previewedImage;
</script> </script>
<div class="absolute top-0 left-0 h-screen w-screen overflow-hidden"> <div class="absolute top-0 left-0 h-screen w-screen overflow-hidden">
@ -146,7 +124,7 @@
<Geosearch <Geosearch
bounds={state.mapProperties.bounds} bounds={state.mapProperties.bounds}
perLayer={state.perLayer} perLayer={state.perLayer}
{selectedElement} selectedElement={state.selectedElement}
{selectedLayer} {selectedLayer}
/> />
</div> </div>
@ -264,9 +242,7 @@
<If condition={featureSwitches.featureSwitchGeolocation}> <If condition={featureSwitches.featureSwitchGeolocation}>
<MapControlButton> <MapControlButton>
<ToSvelte <ToSvelte
construct={new GeolocationControl(state.geolocation, mapproperties).SetClass( construct={new GeolocationControl(state.geolocation, mapproperties).SetClass("block w-8 h-8")}
"block w-8 h-8"
)}
/> />
</MapControlButton> </MapControlButton>
</If> </If>
@ -285,66 +261,52 @@
</LoginToggle> </LoginToggle>
<If condition={state.previewedImage.map(i => i!==undefined)}> <If condition={state.previewedImage.map(i => i!==undefined)}>
<FloatOver on:close={() => state.previewedImage.setData(undefined)} extraClasses=""> <FloatOver extraClasses="p-1" on:close={() => state.previewedImage.setData(undefined)}>
<div <div
slot="close-button"
class="absolute right-4 top-4 h-8 w-8 cursor-pointer rounded-full hover:bg-white bg-white/50 transition-colors duration-200" class="absolute right-4 top-4 h-8 w-8 cursor-pointer rounded-full hover:bg-white bg-white/50 transition-colors duration-200"
on:click={() => previewedImage.setData(undefined)} on:click={() => previewedImage.setData(undefined)}
slot="close-button"
> >
<XCircleIcon /> <XCircleIcon />
</div> </div>
<ImageOperations image={$previewedImage} /> <ImageOperations clss="focusable" image={$previewedImage} />
</FloatOver> </FloatOver>
</If> </If>
<If {#if $selectedElement !== undefined && $selectedLayer !== undefined && !($selectedLayer.popupInFloatover)}
condition={selectedElementView.map(
(v) =>
v !== undefined && selectedLayer.data !== undefined && !selectedLayer.data.popupInFloatover,
[selectedLayer]
)}
>
<!-- right modal with the selected element view --> <!-- right modal with the selected element view -->
<ModalRight <ModalRight
on:close={() => { on:close={() => {
selectedElement.setData(undefined) selectedElement.setData(undefined)
}} }}
> >
<div slot="close-button"/>
<div class="normal-background absolute flex h-full w-full flex-col"> <div class="normal-background absolute flex h-full w-full flex-col">
<ToSvelte construct={new VariableUiElement(selectedElementTitle)}> <SelectedElementTitle {state} layer={$selectedLayer} selectedElement={$selectedElement} />
<!-- Title --> <SelectedElementView {state} layer={$selectedLayer} selectedElement={$selectedElement} />
</ToSvelte>
<ToSvelte construct={new VariableUiElement(selectedElementView).SetClass("overflow-auto")}>
<!-- Main view -->
</ToSvelte>
</div> </div>
</ModalRight> </ModalRight>
</If> {/if}
<If {#if $selectedElement !== undefined && $selectedLayer !== undefined && $selectedLayer.popupInFloatover}
condition={selectedElementView.map(
(v) =>
v !== undefined && selectedLayer.data !== undefined && selectedLayer.data.popupInFloatover,
[selectedLayer]
)}
>
<!-- Floatover with the selected element, if applicable --> <!-- Floatover with the selected element, if applicable -->
<FloatOver <FloatOver
on:close={() => { on:close={() => {
selectedElement.setData(undefined) selectedElement.setData(undefined)
}} }}
> >
<ToSvelte <div class="h-full w-full flex focusable">
construct={new VariableUiElement(selectedElementView).SetClass("h-full w-full flex")} <SelectedElementView {state} layer={$selectedLayer} selectedElement={$selectedElement} />
/> </div>
</FloatOver> </FloatOver>
</If> {/if}
<If condition={state.guistate.themeIsOpened}> <If condition={state.guistate.themeIsOpened}>
<!-- Theme menu --> <!-- Theme menu -->
<FloatOver on:close={() => state.guistate.themeIsOpened.setData(false)}> <FloatOver on:close={() => state.guistate.themeIsOpened.setData(false)}>
<span slot="close-button"><!-- Disable the close button --></span> <span slot="close-button"><!-- Disable the close button --></span>
<TabbedGroup <TabbedGroup
condition1={state.featureSwitches.featureSwitchFilter} condition1={state.featureSwitches.featureSwitchFilter}
tab={state.guistate.themeViewTabIndex} tab={state.guistate.themeViewTabIndex}
> >
@ -421,7 +383,7 @@
state.guistate.backgroundLayerSelectionIsOpened.setData(false) state.guistate.backgroundLayerSelectionIsOpened.setData(false)
}} }}
> >
<div class="h-full p-2"> <div class="h-full p-2 focusable">
<RasterLayerOverview <RasterLayerOverview
{availableLayers} {availableLayers}
map={state.map} map={state.map}

View file

@ -11,7 +11,7 @@ export class Utils {
public static readonly assets_path = "./assets/svg/" public static readonly assets_path = "./assets/svg/"
public static externalDownloadFunction: ( public static externalDownloadFunction: (
url: string, url: string,
headers?: any, headers?: any
) => Promise<{ content: string } | { redirect: string }> ) => Promise<{ content: string } | { redirect: string }>
public static Special_visualizations_tagsToApplyHelpText = `These can either be a tag to add, such as \`amenity=fast_food\` or can use a substitution, e.g. \`addr:housenumber=$number\`. public static Special_visualizations_tagsToApplyHelpText = `These can either be a tag to add, such as \`amenity=fast_food\` or can use a substitution, e.g. \`addr:housenumber=$number\`.
This new point will then have the tags \`amenity=fast_food\` and \`addr:housenumber\` with the value that was saved in \`number\` in the original feature. This new point will then have the tags \`amenity=fast_food\` and \`addr:housenumber\` with the value that was saved in \`number\` in the original feature.
@ -150,7 +150,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
if (Utils.runningFromConsole) { if (Utils.runningFromConsole) {
return return
} }
DOMPurify.addHook("afterSanitizeAttributes", function(node) { DOMPurify.addHook("afterSanitizeAttributes", function (node) {
// set all elements owning target to target=_blank + add noopener noreferrer // set all elements owning target to target=_blank + add noopener noreferrer
const target = node.getAttribute("target") const target = node.getAttribute("target")
if (target) { if (target) {
@ -172,7 +172,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
*/ */
public static ParseVisArgs( public static ParseVisArgs(
specs: { name: string; defaultValue?: string }[], specs: { name: string; defaultValue?: string }[],
args: string[], args: string[]
): Record<string, string> { ): Record<string, string> {
const parsed: Record<string, string> = {} const parsed: Record<string, string> = {}
if (args.length > specs.length) { if (args.length > specs.length) {
@ -324,7 +324,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
object: any, object: any,
name: string, name: string,
init: () => any, init: () => any,
whenDone?: () => void, whenDone?: () => void
) { ) {
Object.defineProperty(object, name, { Object.defineProperty(object, name, {
enumerable: false, enumerable: false,
@ -347,7 +347,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
object: any, object: any,
name: string, name: string,
init: () => Promise<any>, init: () => Promise<any>,
whenDone?: () => void, whenDone?: () => void
) { ) {
Object.defineProperty(object, name, { Object.defineProperty(object, name, {
enumerable: false, enumerable: false,
@ -487,7 +487,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
public static SubstituteKeys( public static SubstituteKeys(
txt: string | undefined, txt: string | undefined,
tags: Record<string, any> | undefined, tags: Record<string, any> | undefined,
useLang?: string, useLang?: string
): string | undefined { ): string | undefined {
if (txt === undefined) { if (txt === undefined) {
return undefined return undefined
@ -523,7 +523,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
"SubstituteKeys received a BaseUIElement to substitute in - this is probably a bug and will be downcast to a string\nThe key is", "SubstituteKeys received a BaseUIElement to substitute in - this is probably a bug and will be downcast to a string\nThe key is",
key, key,
"\nThe value is", "\nThe value is",
v, v
) )
v = v.InnerConstructElement()?.textContent v = v.InnerConstructElement()?.textContent
} }
@ -636,7 +636,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
if (!Array.isArray(targetV)) { if (!Array.isArray(targetV)) {
throw new Error( throw new Error(
"Cannot concatenate: value to add is not an array: " + "Cannot concatenate: value to add is not an array: " +
JSON.stringify(targetV), JSON.stringify(targetV)
) )
} }
if (Array.isArray(sourceV)) { if (Array.isArray(sourceV)) {
@ -644,9 +644,9 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
} else { } else {
throw new Error( throw new Error(
"Could not merge concatenate " + "Could not merge concatenate " +
JSON.stringify(sourceV) + JSON.stringify(sourceV) +
" and " + " and " +
JSON.stringify(targetV), JSON.stringify(targetV)
) )
} }
} else { } else {
@ -691,7 +691,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
path: string[], path: string[],
object: any, object: any,
replaceLeaf: (leaf: any, travelledPath: string[]) => any, replaceLeaf: (leaf: any, travelledPath: string[]) => any,
travelledPath: string[] = [], travelledPath: string[] = []
): void { ): void {
if (object == null) { if (object == null) {
return return
@ -722,7 +722,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
} }
if (Array.isArray(sub)) { if (Array.isArray(sub)) {
sub.forEach((el, i) => sub.forEach((el, i) =>
Utils.WalkPath(path.slice(1), el, replaceLeaf, [...travelledPath, head, "" + i]), Utils.WalkPath(path.slice(1), el, replaceLeaf, [...travelledPath, head, "" + i])
) )
return return
} }
@ -739,7 +739,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
path: string[], path: string[],
object: any, object: any,
collectedList: { leaf: any; path: string[] }[] = [], collectedList: { leaf: any; path: string[] }[] = [],
travelledPath: string[] = [], travelledPath: string[] = []
): { leaf: any; path: string[] }[] { ): { leaf: any; path: string[] }[] {
if (object === undefined || object === null) { if (object === undefined || object === null) {
return collectedList return collectedList
@ -769,7 +769,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
if (Array.isArray(sub)) { if (Array.isArray(sub)) {
sub.forEach((el, i) => sub.forEach((el, i) =>
Utils.CollectPath(path.slice(1), el, collectedList, [...travelledPath, "" + i]), Utils.CollectPath(path.slice(1), el, collectedList, [...travelledPath, "" + i])
) )
return collectedList return collectedList
} }
@ -813,7 +813,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
json: any, json: any,
f: (v: object | number | string | boolean | undefined, path: string[]) => any, f: (v: object | number | string | boolean | undefined, path: string[]) => any,
isLeaf: (object) => boolean = undefined, isLeaf: (object) => boolean = undefined,
path: string[] = [], path: string[] = []
) { ) {
if (json === undefined || json === null) { if (json === undefined || json === null) {
return f(json, path) return f(json, path)
@ -852,7 +852,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
json: any, json: any,
collect: (v: number | string | boolean | undefined, path: string[]) => any, collect: (v: number | string | boolean | undefined, path: string[]) => any,
isLeaf: (object) => boolean = undefined, isLeaf: (object) => boolean = undefined,
path = [], path = []
): void { ): void {
if (json === undefined) { if (json === undefined) {
return return
@ -927,7 +927,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
continue continue
} }
const i = part.charCodeAt(0) const i = part.charCodeAt(0)
result += "\"" + keys[i] + "\":" + part.substring(1) result += '"' + keys[i] + '":' + part.substring(1)
} }
return result return result
@ -954,7 +954,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
url: string, url: string,
headers?: any, headers?: any,
method: "POST" | "GET" | "PUT" | "UPDATE" | "DELETE" | "OPTIONS" = "GET", method: "POST" | "GET" | "PUT" | "UPDATE" | "DELETE" | "OPTIONS" = "GET",
content?: string, content?: string
): Promise< ): Promise<
| { content: string } | { content: string }
| { redirect: string } | { redirect: string }
@ -1019,7 +1019,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
public static async downloadJsonCached( public static async downloadJsonCached(
url: string, url: string,
maxCacheTimeMs: number, maxCacheTimeMs: number,
headers?: any, headers?: any
): Promise<any> { ): Promise<any> {
const result = await Utils.downloadJsonAdvanced(url, headers) const result = await Utils.downloadJsonAdvanced(url, headers)
if (result["content"]) { if (result["content"]) {
@ -1031,7 +1031,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
public static async downloadJsonCachedAdvanced( public static async downloadJsonCachedAdvanced(
url: string, url: string,
maxCacheTimeMs: number, maxCacheTimeMs: number,
headers?: any, headers?: any
): Promise<{ content: any } | { error: string; url: string; statuscode?: number }> { ): Promise<{ content: any } | { error: string; url: string; statuscode?: number }> {
const cached = Utils._download_cache.get(url) const cached = Utils._download_cache.get(url)
if (cached !== undefined) { if (cached !== undefined) {
@ -1041,9 +1041,9 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
} }
const promise = const promise =
/*NO AWAIT as we work with the promise directly */ Utils.downloadJsonAdvanced( /*NO AWAIT as we work with the promise directly */ Utils.downloadJsonAdvanced(
url, url,
headers, headers
) )
Utils._download_cache.set(url, { promise, timestamp: new Date().getTime() }) Utils._download_cache.set(url, { promise, timestamp: new Date().getTime() })
return await promise return await promise
} }
@ -1058,7 +1058,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
public static async downloadJsonAdvanced( public static async downloadJsonAdvanced(
url: string, url: string,
headers?: any, headers?: any
): Promise<{ content: any } | { error: string; url: string; statuscode?: number }> { ): Promise<{ content: any } | { error: string; url: string; statuscode?: number }> {
const injected = Utils.injectedDownloads[url] const injected = Utils.injectedDownloads[url]
if (injected !== undefined) { if (injected !== undefined) {
@ -1067,7 +1067,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
} }
const result = await Utils.downloadAdvanced( const result = await Utils.downloadAdvanced(
url, url,
Utils.Merge({ accept: "application/json" }, headers ?? {}), Utils.Merge({ accept: "application/json" }, headers ?? {})
) )
if (result["error"] !== undefined) { if (result["error"] !== undefined) {
return <{ error: string; url: string; statuscode?: number }>result return <{ error: string; url: string; statuscode?: number }>result
@ -1087,7 +1087,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
"due to", "due to",
e, e,
"\n", "\n",
e.stack, e.stack
) )
return { error: "malformed", url } return { error: "malformed", url }
} }
@ -1108,7 +1108,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
| "{gpx=application/gpx+xml}" | "{gpx=application/gpx+xml}"
| "application/json" | "application/json"
| "image/png" | "image/png"
}, }
) { ) {
const element = document.createElement("a") const element = document.createElement("a")
let file let file
@ -1212,7 +1212,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
e.preventDefault() e.preventDefault()
return false return false
}, },
false, false
) )
} }
@ -1296,7 +1296,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
public static sortedByLevenshteinDistance<T>( public static sortedByLevenshteinDistance<T>(
reference: string, reference: string,
ts: T[], ts: T[],
getName: (t: T) => string, getName: (t: T) => string
): T[] { ): T[] {
const withDistance: [T, number][] = ts.map((t) => [ const withDistance: [T, number][] = ts.map((t) => [
t, t,
@ -1322,7 +1322,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
track[j][i] = Math.min( track[j][i] = Math.min(
track[j][i - 1] + 1, // deletion track[j][i - 1] + 1, // deletion
track[j - 1][i] + 1, // insertion track[j - 1][i] + 1, // insertion
track[j - 1][i - 1] + indicator, // substitution track[j - 1][i - 1] + indicator // substitution
) )
} }
} }
@ -1331,7 +1331,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
public static MapToObj<V, T>( public static MapToObj<V, T>(
d: Map<string, V>, d: Map<string, V>,
onValue: (t: V, key: string) => T, onValue: (t: V, key: string) => T
): Record<string, T> { ): Record<string, T> {
const o = {} const o = {}
const keys = Array.from(d.keys()) const keys = Array.from(d.keys())
@ -1348,7 +1348,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
* Utils.TransposeMap({"a" : ["b", "c"], "x" : ["b", "y"]}) // => {"b" : ["a", "x"], "c" : ["a"], "y" : ["x"]} * Utils.TransposeMap({"a" : ["b", "c"], "x" : ["b", "y"]}) // => {"b" : ["a", "x"], "c" : ["a"], "y" : ["x"]}
*/ */
public static TransposeMap<K extends string, V extends string>( public static TransposeMap<K extends string, V extends string>(
d: Record<K, V[]>, d: Record<K, V[]>
): Record<V, K[]> { ): Record<V, K[]> {
const newD: Record<V, K[]> = <any>{} const newD: Record<V, K[]> = <any>{}
@ -1422,7 +1422,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
} }
public static asDict( public static asDict(
tags: { key: string; value: string | number }[], tags: { key: string; value: string | number }[]
): Map<string, string | number> { ): Map<string, string | number> {
const d = new Map<string, string | number>() const d = new Map<string, string | number>()
@ -1529,7 +1529,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
* *
*/ */
public static splitIntoSubstitutionParts( public static splitIntoSubstitutionParts(
template: string, template: string
): ({ message: string } | { subs: string })[] { ): ({ message: string } | { subs: string })[] {
const preparts = template.split("{") const preparts = template.split("{")
const spec: ({ message: string } | { subs: string })[] = [] const spec: ({ message: string } | { subs: string })[] = []
@ -1591,7 +1591,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
} }
public static RemoveDiacritics(str?: string): string { public static RemoveDiacritics(str?: string): string {
if(!str){ if (!str) {
return str return str
} }
return str.normalize("NFD").replace(/\p{Diacritic}/gu, "") return str.normalize("NFD").replace(/\p{Diacritic}/gu, "")
@ -1638,6 +1638,41 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
return newObj return newObj
} }
/**
* Searches a child that can be focused on, by first selecting a 'focusable', then a button, then a link
*
* Returns the focussed element
* @param el
*/
public static focusOnFocusableChild(el: HTMLElement): undefined {
if (!el) {
return
}
requestAnimationFrame(() => {
let childs = el.getElementsByClassName("focusable")
if (childs.length == 0) {
childs = el.getElementsByTagName("button")
if (childs.length === 0) {
childs = el.getElementsByTagName("a")
}
}
const child = <HTMLElement>childs.item(0)
if (child === null) {
console.log("Focussing on child element: no child element found for", el)
return undefined
}
if (
child.tagName !== "button" &&
child.tagName !== "a" &&
child.hasAttribute("tabindex")
) {
child.setAttribute("tabindex", "-1")
}
console.log("Focussing on", child)
child?.focus()
})
}
private static findParentWithScrolling(element: HTMLBaseElement): HTMLBaseElement { private static findParentWithScrolling(element: HTMLBaseElement): HTMLBaseElement {
// Check if the element itself has scrolling // Check if the element itself has scrolling
if (element.scrollHeight > element.clientHeight) { if (element.scrollHeight > element.clientHeight) {
@ -1655,7 +1690,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
private static colorDiff( private static colorDiff(
c0: { r: number; g: number; b: number }, c0: { r: number; g: number; b: number },
c1: { r: number; g: number; b: number }, c1: { r: number; g: number; b: number }
) { ) {
return Math.abs(c0.r - c1.r) + Math.abs(c0.g - c1.g) + Math.abs(c0.b - c1.b) return Math.abs(c0.r - c1.r) + Math.abs(c0.g - c1.g) + Math.abs(c0.b - c1.b)
} }

View file

@ -70,6 +70,11 @@ body {
font-family: "Helvetica Neue", Arial, sans-serif; font-family: "Helvetica Neue", Arial, sans-serif;
} }
.focusable {
/* Not a 'real' class, but rather an indication to FloatOver and ModalRight to, when they open, grab the focus */
border: 1px solid red
}
svg, svg,
img { img {
box-sizing: content-box; box-sizing: content-box;
@ -159,7 +164,6 @@ input[type=text] {
border-radius: 0.5rem; border-radius: 0.5rem;
} }
/******************* Styling of input elements **********************/ /******************* Styling of input elements **********************/
/** /**