Merge branch 'master' into develop

This commit is contained in:
Pieter Vander Vennet 2024-08-23 11:44:46 +02:00
commit fbf5ce6fec
26 changed files with 1225 additions and 847 deletions

View file

@ -90,11 +90,11 @@
"minzoom": 6,
"title": {
"render": {
"en": "Cycle highway",
"en": "cycle highway",
"de": "Radschnellweg",
"ca": "Via ciclista",
"ca": "via ciclista",
"fr": "Aménagement cyclable",
"nl": "Fietssnelweg",
"nl": "fietssnelweg",
"es": "autovía ciclista",
"nb_NO": "sykkelmotorvei",
"da": "cykelmotorvej",

View file

@ -276,5 +276,8 @@
]
}
],
"deletion": {
"softDeletionTags": "disused:tourism:={tourism}"
},
"allowMove": true
}

View file

@ -374,11 +374,15 @@
"fr": "Quel fond souhaitez-vous utiliser par défaut?",
"da": "Hvilket baggrundslag skal vises som standard?"
},
"questionHint":{
"en": "To set a specific background as default, select it in the background menu first after which it will appear here."
},
"condition": "_theme:backgroundLayer=",
"mappings": [
{
"if": "mapcomplete-preferred-background-layer=default",
"alsoShowIf": "mapcomplete-preferred-background-layer=",
"icon": "./assets/svg/generic_map.svg",
"then": {
"en": "Use the default background layer",
"ca": "Utilitzeu la capa de fons predeterminada",
@ -391,6 +395,9 @@
},
{
"if": "mapcomplete-preferred-background-layer=osm",
"icon": {
"path":"./assets/svg/osm-logo.svg"
},
"then": {
"en": "Use OpenStreetMap-carto as default layer",
"ca": "Utilitzeu OpenStreetMap-carto com a capa predeterminada",
@ -403,6 +410,7 @@
},
{
"if": "mapcomplete-preferred-background-layer=photo",
"icon": "\uD83D\uDEF0\uFE0F",
"then": {
"en": "Use aerial imagery as default background",
"ca": "Utilitzeu imatges aèries com a fons predeterminat",
@ -415,6 +423,7 @@
},
{
"if": "mapcomplete-preferred-background-layer=map",
"icon": "./assets/svg/generic_map.svg",
"then": {
"en": "Use a non-openstreetmap based map as default background",
"ca": "Utilitzeu un mapa que no sigui openstreetmap com a fons predeterminat",
@ -725,7 +734,8 @@
},
"mappings": [
{
"if": "mapcomplete-fixate-north=",
"if": "mapcomplete-fixate-north=no",
"alsoShowIf": "mapcomplete-fixate-north=",
"icon": "./assets/svg/compass.svg",
"then": {
"en": "Allow to rotate the map",

File diff suppressed because one or more lines are too long

Before

Width:  |  Height:  |  Size: 135 KiB

After

Width:  |  Height:  |  Size: 128 KiB

View file

@ -74,7 +74,6 @@
]
}
},
"cycle_highways"
],
"overpassTimeout": 60,

View file

@ -20,6 +20,7 @@
"cancel": "Zrušit",
"cannotBeDeleted": "Tuto funkci nelze odstranit",
"delete": "Smazat",
"deletedTitle": "Smazaná funkce",
"explanations": {
"hardDelete": "Tato funkce bude v OpenStreetMap odstraněna. Zkušený přispěvatel ji může obnovit",
"retagNoOtherThemes": "Tato funkce bude překlasifikována a skryta z této aplikace",
@ -27,6 +28,7 @@
"selectReason": "Vyberte, proč by tato funkce měla být odstraněna",
"softDelete": "Tato funkce bude v této aplikaci aktualizována a skryta. <span class='subtle'>{reason}</span>"
},
"isChanged": "Tato funkce byla změněna a již neodpovídá této vrstvě",
"isDeleted": "Tato funkce je smazána",
"isntAPoint": "Odstranit lze pouze uzly, vybraným prvkem je cesta, oblast nebo relace.",
"loading": "Kontrola vlastností, zda lze tuto funkci odstranit.",
@ -45,15 +47,55 @@
"useSomethingElse": "Místo toho použijte jiný editor OpenStreetMap a odstraňte jej",
"whyDelete": "Proč by měla být tato funkce odstraněna?"
},
"external": {
"allAreApplied": "Všechny chybějící externí hodnoty byly zkopírovány do OpenStreetMap",
"allIncluded": "Data načtená ze {source} jsou obsažena v OpenStreetMap",
"apply": "Použít",
"applyAll": "Použít všechny chybějící hodnoty",
"conflicting": {
"intro": "U následujících hodnot má OpenStreetMap jinou hodnotu než zdrojová webová stránka.",
"title": "Konfliktní položky"
},
"currentInOsmIs": "V současné době má OpenStreetMap zaznamenány následující hodnoty:",
"done": "Hotovo",
"error": "Nelze načíst propojená data z webové stránky",
"lastModified": "Externí data byla naposledy upravena dne {date}",
"loadedFrom": "Následující data jsou načtena z <a href={url}>{source}</a> pomocí vloženého JSON-LD",
"missing": {
"intro": "OpenStreetMap nemá žádné informace o následujících atributech",
"title": "Chybějící položky"
},
"noDataLoaded": "Externí webová stránka neobsahuje žádná propojená data, která by bylo možné načíst",
"overwrite": "Přepsání v OpenStreetMap",
"title": "Strukturovaná data načtená z externí webové stránky"
},
"favourite": {
"loginNeeded": "<h3>Přihlášení</h3>Osobní rozvržení je k dispozici pouze pro uživatele OpenStreetMap",
"panelIntro": "<h3>Vaše osobní téma</h3>Aktivujte si své oblíbené vrstvy ze všech oficiálních témat",
"reload": "Znovu načíst data"
},
"favouritePoi": {
"button": {
"isFavourite": "Toto místo je v současné době označeno jako oblíbené a bude se zobrazovat na všech tematických mapách MapComplete, které navštívíte.",
"isMarkedShort": "Označeno jako oblíbené místo",
"isNotMarkedShort": "Neoznačeno jako oblíbené",
"markAsFavouriteTitle": "Označit toto místo jako oblíbené",
"markDescription": "Přidat toto místo do osobního seznamu oblíbených",
"unmark": "Odebrat z osobního seznamu oblíbených",
"unmarkNotDeleted": "Tento bod nebude smazán a bude stále viditelný na příslušné mapě pro vás i ostatní"
},
"downloadGeojson": "Stáhněte si oblíbené položky jako geojson",
"downloadGpx": "Stáhnout oblíbené položky jako GPX",
"intro": "Označili jste {length} místa jako oblíbené místo.",
"introPrivacy": "Tento seznam je viditelný pouze pro vás",
"loginToSeeList": "Přihlášení pro zobrazení seznamu míst, která jste označili jako oblíbená",
"tab": "Vaše oblíbené položky a recenze",
"title": "Vaše oblíbená místa"
},
"flyer": {
"aerial": "Tato mapa používá jiné pozadí, konkrétně letecké snímky od Agency Information Flanders",
"callToAction": "Vyzkoušet na mapcomplete.org",
"cyclofix": "Cyklistické pumpy, opravárenské stanice, prodejny pitné vody a cykloprodejny jsou na CycloFix",
"aerial": "Tato mapa používá jiný podklad, a to letecké snímky od Agentschap Informatie Vlaanderen",
"callToAction": "Otestovat na mapcomplete.org",
"cyclofix": "Pumpy na kola, opravny, prodejny pitné vody a cyklistické obchody jsou na webu CycloFix",
"description": "Leták formátu A4 na propagaci MapComplete",
"editing": {
"ex": "Zjednodušený příklad, jak to vypadá v případě přírodní rezervace, je uveden níže.",
@ -104,6 +146,7 @@
"confirmLocation": "Potvrďte toto umístění",
"confirmTitle": "Přidat {title}?",
"confirmWarning": "Funkce, kterou zde vytvoříte, bude <b>viditelná pro všechny</b>. Prosím, přidávejte věci na mapu pouze tehdy, pokud skutečně existují. Tato data využívá mnoho aplikací.",
"creating": "Tvorba nového bodu...",
"disableFilters": "Vypnout všechny filtry",
"disableFiltersExplanation": "Některé funkce mohou být filtrem skryty",
"enableLayer": "Povolit vrstvu {name}",
@ -121,13 +164,13 @@
"intro": "Klikli jste někam, kde zatím nejsou známy žádné údaje.<br/>",
"layerNotEnabled": "Vrstva {layer} není povolena. Chcete-li přidat funkci, povolte tuto vrstvu",
"openLayerControl": "Otevřete ovládací okno vrstvy",
"pleaseLogin": "Chcete-li přidat novou funkci, přihlaste se",
"pleaseLogin": "Chcete-li přidat novou funkci, přihlaste se prosím do aplikace OpenStreetMap",
"presetInfo": "Nový bod zájmu bude mít {tags}",
"stillLoading": "Data se stále načítají. Před přidáním nové funkce prosím chvíli počkejte.",
"title": "Přidat novou funkci",
"warnVisibleForEveryone": "Váš příspěvek bude viditelný pro všechny",
"wrongType": "Tato funkce není uzel ani cesta a nelze ji importovat",
"zoomInFurther": "Další přiblížení pro přidání funkce.",
"zoomInFurther": "Další přiblížení a přidání funkce",
"zoomInMore": "Pro import této funkce si ji více přibližte"
},
"apply_button": {
@ -137,13 +180,14 @@
"attribution": {
"attributionBackgroundLayer": "Aktuální vrstva pozadí je {name}",
"attributionBackgroundLayerWithCopyright": "Aktuální vrstva pozadí je {name}: {copyright}",
"attributionContent": "<p>Všechna data jsou poskytována službou <a href='https://osm.org' target='_blank'>OpenStreetMap</a>, volně opakovaně použitelná pod <a href='https://osm.org/copyright' target='_blank'>licencí Open DataBase</a>.</p>",
"attributionContent": "<p>Všechna data poskytuje <a href='https://www.openstreetmap.org' target='_blank'>OpenStreetMap</a>, volně znovu použitelná pod <a href='https://www.openstreetmap .org/copyright' target='_blank'>licence pro otevřenou databázi</a>.</p>",
"attributionTitle": "Oznámení o atribuci",
"codeContributionsBy": "MapComplete vytvořili {contributors} a <a href='https://github.com/pietervdvn/MapComplete/graphs/contributors' target='_blank'>{hiddenCount} dalších přispěvatelů</a>",
"donate": "Podpořte MapComplete finančně",
"editId": "Zde otevřít online editor OpenStreetMap",
"editJosm": "Upravit zde pomocí JOSM",
"followOnMastodon": "Sledujte MapComplete na Mastodonu",
"gotoSourceCode": "Zobrazit zdrojový kód",
"iconAttribution": {
"title": "Použité ikony"
},
@ -156,24 +200,39 @@
"openIssueTracker": "Nahlaste chybu",
"openMapillary": "Zde otevřít Mapillary",
"openOsmcha": "Podívejte se na nejnovější úpravy provedené v tématu {theme}",
"openOsmchaLastWeek": "Zobrazit úpravy za posledních 7 dní",
"openThemeDocumentation": "Otevřít dokumentaci k tematické mapě {name}",
"seeOnMapillary": "Podívejte se na tento obrázek na Mapillary",
"themeBy": "Téma spravuje {author}",
"title": "Autorská práva a uvedení zdroje",
"translatedBy": "MapComplete přeložili {contributors} a <a href='https://github.com/pietervdvn/MapComplete/graphs/contributors' target='_blank'>{hiddenCount} dalších přispěvatelů</a>"
},
"back": "Zpět",
"backToIndex": "Návrat na přehled všech tematických map",
"backToMap": "Zpět na mapu",
"backgroundMap": "Vyberte vrstvu pozadí",
"backgroundSwitch": "Přepnout pozadí",
"cancel": "Zrušit",
"clearPendingChanges": "Vymazat nevyřízené změny",
"confirm": "Potvrdit",
"customThemeIntro": "<h3>Vlastní motivy</h3>Tyto jsou dříve navštívené motivy vytvořené uživateli.",
"customThemeIntro": "Jedná se o dříve navštívená témata vytvořená uživateli.",
"customThemeTitle": "Vlastní témata",
"download": {
"custom": {
"download": "Stáhnout PNG o {width} mm na šířku a {height} mm na výšku",
"downloadHelper": "Toto je určeno pro tisk",
"height": "Výška obrázku (v mm):",
"title": "Stáhněte si obrázek s vlastní šířkou a výškou",
"width": "Šířka obrázku (v mm): "
},
"downloadAsPdf": "Stáhněte si aktuální mapu ve formátu PDF",
"downloadAsPdfHelper": "Ideální pro tisk aktuální mapy",
"downloadAsPng": "Stáhnout jako obrázek",
"downloadAsPngHelper": "Ideální pro zahrnutí do zpráv",
"downloadAsSvg": "Stáhnout SVG aktuální mapy",
"downloadAsSvgHelper": "Kompatibilní s Inkscape nebo Adobe Illustrator; bude nutné další zpracování",
"downloadAsSvgHelper": "Kompatibilní s Inkscape nebo Adobe Illustrator; bude potřebovat další zpracování",
"downloadAsSvgLinesOnly": "Stáhněte si SVG aktuální mapy obsahující pouze čáry",
"downloadAsSvgLinesOnlyHelper": "Samo se protínající čáry jsou rozděleny, lze je použít s některým 3D softwarem",
"downloadCSV": "Stáhnout viditelná data jako CSV",
"downloadCSVHelper": "Kompatibilní s aplikacemi LibreOffice Calc, Excel, …",
"downloadFeatureAsGeojson": "Stáhnout jako soubor GeoJSON",
@ -186,13 +245,24 @@
"includeMetaData": "Zahrnout metadata (poslední editor, vypočtené hodnoty, ...)",
"licenseInfo": "<h3>Upozornění na autorská práva</h3>Poskytovaná data jsou dostupná pod ODbL. Opětovné použití je zdarma pro jakýkoli účel, ale <ul><li>musí být uveden zdroj <b>© přispěvatelé OpenStreetMap</b><li><li>Jakákoli změna musí být publikována pod stejnou licencí</li>< /ul> Podrobnosti naleznete v úplném <a href='https://www.openstreetmap.org/copyright' target='_blank'>upozornění na autorská práva</a>.",
"noDataLoaded": "Zatím nejsou načtena žádná data. Brzy bude k dispozici ke stažení",
"pdf": {
"current_view_generic": "Exportujte PDF aktuálního zobrazení pro {paper_size} v orientaci {orientation}"
},
"title": "Stáhnout",
"toMuch": "Existuje mnoho funkcí, které lze stáhnout všechny",
"uploadGpx": "Nahrajte svou trasu do OpenStreetMap"
},
"enableGeolocationForSafari": "Nezobrazilo se vám vyskakovací okno s žádostí o geografické povolení?",
"enableGeolocationForSafariLink": "Přečtěte si, jak povolit oprávnění ke geolokaci v nastavení",
"eraseValue": "Vymažte tuto hodnotu",
"error": "Něco se pokazilo",
"example": "Příklad",
"examples": "Příklady",
"fewChangesBefore": "Před přidáním nové funkce prosím odpovězte na několik otázek týkajících se stávajících funkcí.",
"filterPanel": {
"disableAll": "Zakázat vše",
"enableAll": "Povolit vše"
},
"geopermissionDenied": "Použití geolokace bylo zamítnuto",
"getStartedLogin": "Přihlaste se pomocí OpenStreetMap a začněte",
"getStartedNewAccount": " nebo <a href='https://www.openstreetmap.org/user/new' target='_blank'>vytvořte nový účet</a>",
@ -200,6 +270,15 @@
"histogram": {
"error_loading": "Nelze načíst histogram"
},
"labels": {
"background": "Změnit pozadí",
"filter": "Filtrovat data",
"jumpToLocation": "Přejděte na svou aktuální polohu",
"locationNotAvailable": "GPS poloha není k dispozici. Má toto zařízení polohu nebo jste v tunelu?",
"menu": "Menu",
"zoomIn": "Přiblížit",
"zoomOut": "Oddálit"
},
"layerSelection": {
"title": "Výběr vrstev",
"zoomInToSeeThisLayer": "Přibližte si tuto vrstvu, abyste ji viděli"
@ -218,6 +297,7 @@
"loginToStart": "Přihlaste se a odpovězte na tuto otázku",
"loginWithOpenStreetMap": "Přihlaste se pomocí OpenStreetMap",
"logout": "Odhlásit se",
"mappingsAreHidden": "Některé možnosti jsou skryté. Pomocí vyhledávání zobrazíte další možnosti.",
"menu": {
"aboutMapComplete": "O MapComplete",
"filter": "Filtrování dat"
@ -244,20 +324,35 @@
"openStreetMapIntro": "<h3>Otevřená mapa</h3><p>Takovou, kterou může každý volně používat a upravovat. Jediné místo pro uložení všech geoinformací. Různé, malé, nekompatibilní a zastaralé mapy nejsou nikde potřeba.</p><p><b><a href='https://OpenStreetMap.org' target='_blank'>OpenStreetMap</a></b > není nepřátelská mapa. Mapová data lze volně používat (s <a href='https://osm.org/copyright' target='_blank'>přiřazením a zveřejněním změn těchto dat</a>). Každý může přidávat nová data a opravovat chyby. Tento web používá OpenStreetMap. Všechna data jsou odtud a vaše odpovědi a opravy se používají všude.</p><p>Mnoho lidí a aplikací již OpenStreetMap používá: <a href='https://organicmaps.app/' target='_blank '>Organické mapy</a>, <a href='https://osmAnd.net' target='_blank'>OsmAnd</a>, ale také mapy na Facebooku, Instagramu, Apple-maps a Bing-maps jsou (částečně) poháněny OpenStreetMap.</p>",
"openTheMap": "Otevřít mapu",
"openTheMapAtGeolocation": "Přiblížení k vaší poloze",
"openTheMapReason": "pro zobrazení, úpravu a přidání informací",
"opening_hours": {
"closed_permanently": "Uzavřeno na neznámou dobu",
"closed_until": "Uzavřeno do {date}",
"all_days_from": "Otevřeno každý den {ranges}",
"closed_permanently": "Na neznámou dobu zavřeno",
"closed_until": "Otevírá v {date}",
"error": "Nepodařilo se analyzovat otevírací dobu",
"error_loading": "Chyba: tyto otevírací hodiny nelze zobrazit.",
"friday": "V pátek {ranges}",
"loadingCountry": "Určení země…",
"monday": "V pondělí {ranges}",
"not_all_rules_parsed": "Tato otevírací doba je komplikovaná. Následující pravidla jsou ve vstupním prvku ignorována:",
"on_weekdays": "Otevřeno ve všední dny {ranges}",
"on_weekends": "Otevřeno o víkendech {ranges}",
"openTill": "do",
"open_24_7": "Otevřeno nepřetržitě",
"open_during_ph": "Během státního svátku je místo",
"open_during_ph": "Během státního svátku je",
"opensAt": "od",
"ph_closed": "zavřené",
"ph_not_known": " ",
"ph_open": "otevřené",
"ph_open_as_usual": "otevřené, jako obvykle"
"ph_open_as_usual": "otevřené, jako obvykle",
"ranges": "od {starttime} do {endtime}",
"rangescombined": "{range0} a {range1}",
"saturday": "V sobotu {ranges}",
"sunday": "V neděli {ranges}",
"thursday": "Ve čtvrtek {ranges}",
"tuesday": "V úterý {ranges}",
"unknown": "Otevírací doba není známa",
"wednesday": "Ve středu {ranges}"
},
"osmLinkTooltip": "Prohlédněte si tento objekt na OpenStreetMap pro historii a další možnosti úprav",
"pdf": {
@ -266,7 +361,8 @@
"generatedWith": "Generováno pomocí mapcomplete.org/{layoutid}",
"versionInfo": "v{version} - vygenerováno {date}"
},
"pickLanguage": "Vyberte si jazyk: ",
"pickLanguage": "Vyberte jazyk",
"poweredByMapComplete": "Provozováno na MapComplete crowdsourcované, tematické mapy s OpenStreetMap",
"poweredByOsm": "Běží na OpenStreetMap",
"questionBox": {
"answeredMultiple": "Odpověděli jste na {answered} otázky",
@ -290,9 +386,10 @@
},
"readYourMessages": "Před přidáním nové funkce si prosím přečtěte všechny zprávy OpenStreetMap.",
"removeLocationHistory": "Odstranit historii polohy",
"retry": "Zkusit znovu",
"returnToTheMap": "Návrat na mapu",
"save": "Uložit",
"screenToSmall": "Otevřít {theme} v novém okně",
"screenToSmall": "Otevřít <i>{theme}</i> v novém okně",
"search": {
"error": "Něco se pokazilo…",
"nothing": "Nic nenalezeno…",
@ -300,13 +397,22 @@
"searchShort": "Vyhledávání…",
"searching": "Hledání…"
},
"searchAnswer": "Vyhledat možnost…",
"seeIndex": "Podívejte se na přehled se všemi tematickými mapami",
"share": "Sdílet",
"sharescreen": {
"copiedToClipboard": "Odkaz zkopírovaný do schránky",
"documentation": "Další informace o dostupných parametrech URL naleznete <a href='https://github.com/pietervdvn/MapComplete/blob/develop/Docs/URL_Parameters.md' target='_blank'>v dokumentaci</a>",
"embedIntro": "<h3>Vložte mapu na své webové stránky</h3>Prosíme, vložte tuto mapu na své webové stránky. <br/>Doporučujeme vám to udělat - nemusíte ani žádat o povolení. <br/> Je a vždy to bude zdarma. Čím více lidí bude projekt používat, tím bude cennější.",
"fsUserbadge": "Povolit tlačítko přihlášení",
"fsBackground": "Povolit přepínání pozadí",
"fsFilter": "Povolit možnost přepínání vrstev a filtrů",
"fsGeolocation": "Povolit geolokaci",
"fsUserbadge": "Povolit přihlašovací tlačítko a tím i možnost provádět změny",
"fsWelcomeMessage": "Zobrazit vyskakovací okno s uvítací zprávou a související karty",
"intro": "<h3>Sdílejte tuto mapu</h3> Sdílejte tuto mapu zkopírováním níže uvedeného odkazu a jeho zasláním přátelům a rodině:",
"openLayers": "Otevřete nabídku vrstev a filtrů",
"options": "Možnosti sdílení",
"stateIsIncluded": "Aktuální stav vrstev a filtrů je zahrnut ve sdíleném odkazu a prvku iframe.",
"thanksForSharing": "Díky za sdílení!",
"title": "Sdílet tuto mapu"
},
@ -314,7 +420,7 @@
"testing": "Testování - změny se neuloží",
"uploadError": "Chyba při nahrávání změn: {error}",
"uploadGpx": {
"choosePermission": "Níže vyberte, zda má být vaše stopa sdílena:",
"choosePermission": "Níže vyberte, jak má být vaše trasa sdílena:",
"confirm": "Potvrdit nahrávání",
"gpxServiceOffline": "Služba GPX je v současné době offline - nahrávání není v současné době možné. Zkuste to později.",
"intro0": "Nahráním stopy si OpenStreetMap.org ponechá její úplnou kopii.",
@ -343,7 +449,41 @@
"uploadPending": "Počet nevyřízených změn: {count}",
"uploadPendingSingle": "Čeká se na jednu změnu",
"uploadingChanges": "Nahrávání změn…",
"useSearch": "Pro zobrazení předvoleb použijte vyhledávání výše",
"useSearch": "Pomocí vyhledávání výše zobrazíte další možnosti",
"visualFeedback": {
"closestFeaturesAre": "{n} objektů ve výřezu.",
"directionsAbsolute": {
"E": "východní",
"N": "severní",
"NE": "severovýchod",
"NW": "severozápadní",
"S": "jižní",
"SE": "jihovýchodní",
"SW": "jihozápadní",
"W": "západ"
},
"directionsRelative": {
"behind": "za vašimi zády",
"left": "vlevo",
"right": "vpravo",
"sharp_left": "ostře vlevo",
"sharp_right": "ostře doprava",
"slight_left": "mírně vlevo",
"slight_right": "mírně vpravo",
"straight": "přímo vpřed"
},
"fromGps": "{distance} {direction} vaší polohy",
"fromMapCenter": "{distance} {direction} středu mapy",
"in": "Přiblížení na úroveň {z}",
"islocked": "Pohled uzamčen na vaši polohu GPS, pohyb zakázán. Pro odemknutí stiskněte tlačítko geolokace.",
"locked": "Zobrazení je nyní uzamčeno na vaši polohu GPS, pohyb je zakázán.",
"navigation": "Pomocí kláves se šipkami posouvejte mapu, stisknutím mezerníku vyberte nejbližší objekt. Stisknutím čísla vyberte místa dále.",
"noCloseFeatures": "Nejsou zobrazeny žádné funkce.",
"oneFeatureInView": "Jedna funkce ve výřezu.",
"out": "Oddálení na úroveň {z}",
"viewportCenterCloseToGps": "Mapa je vycentrována kolem vaší polohy.",
"viewportCenterDetails": "Střed výřezu je vzdálen {distance} a {bearing} od vaší polohy."
},
"waitingForGeopermission": "Čekáme na vaše povolení používat geolokaci…",
"waitingForLocation": "Vyhledávání vaší aktuální polohy…",
"weekdays": {
@ -366,7 +506,7 @@
},
"welcomeBack": "Vítejte zpět!",
"welcomeExplanation": {
"addNew": "Klepněte na mapu pro přidání nového bodu zájmu.",
"addNew": "Chybí nějaká položka? Pro přidání nového bodu zájmu použijte tlačítko vlevo dole.",
"general": "Na této mapě si můžete zobrazit, upravovat a přidávat <i>body zájmu</i>. Přibližováním si zobrazte body zájmu a klepnutím na jeden z nich si zobrazte nebo upravte informace. Všechna data pocházejí z OpenStreetMap a jsou v ní uložena, takže je lze volně používat."
},
"wikipedia": {
@ -394,7 +534,9 @@
"geolocate": "Posune mapu na aktuální polohu nebo ji přiblíží k aktuální poloze. Vyžaduje geooprávnění",
"intro": "MapComplete podporuje následující klávesy:",
"key": "Kombinace kláves",
"openLayersPanel": "Otevře panel vrstev a filtrů",
"openFilterPanel": "Otevře vrstvy POI a panel filtrů",
"openLayersPanel": "Otevírá panel Vrstvy pozadí",
"queryCurrentLocation": "Zobrazte adresu, která je nejblíže středu mapy",
"selectAerial": "Nastavte pozadí na letecké nebo satelitní snímky. Přepíná mezi dvěma nejlepšími dostupnými vrstvami",
"selectMap": "Nastavení pozadí na mapu z externích zdrojů. Přepíná mezi dvěma nejlepšími dostupnými vrstvami",
"selectMapnik": "Nastavení vrstvy pozadí na OpenStreetMap-carto",

View file

@ -847,6 +847,7 @@
},
"tooLong": "The text is too long, at most 255 characters are allowed. You have {count} characters now.",
"url": {
"aggregator": "{host} is a third-party aggregator website. If possible, search the official website.",
"description": "link to a website",
"feedback": "This is not a valid web address"
},

View file

@ -7322,9 +7322,6 @@
},
"9": {
"then": "Die Oberfläche besteht aus Gummi, z. B. aus Gummifliesen, Gummimulch oder einer großen Gummifläche"
},
"10": {
"then": "Die Oberfläche ist <b>feiner Kies</b> (weniger als 2 cm pro Stein)"
}
},
"question": "Welchen Bodenbelag hat dieser Spielplatz?",
@ -10525,6 +10522,34 @@
}
},
"tourism_accomodation": {
"description": "Verschiedene Arten von Unterkünften",
"filter": {
"0": {
"options": {
"0": {
"question": "Alle Arten"
},
"1": {
"question": "Hotels"
},
"2": {
"question": "Hostels / Herbergen"
},
"3": {
"question": "Gasthäuser und Bed & Breakfasts"
},
"4": {
"question": "Motels"
},
"5": {
"question": "Ferienhaus"
},
"6": {
"question": "Ferienwohnung"
}
}
}
},
"name": "Touristische Unterkunft",
"presets": {
"0": {
@ -10534,6 +10559,25 @@
"1": {
"description": "Ein Hostel ist eine Art von Touristenherberge, in der man in einem Zimmer schlafen kann, das man mit Fremden teilt",
"title": "Ein Hostel"
},
"2": {
"description": "Ein möbliertes Appartement oder eine Wohnung mit Kochgelegenheit und Bad in einem Gemeinschaftsgebäude, das für Ferienzwecke gemietet werden kann, normalerweise ohne Frühstück oder Rezeption",
"title": "eine Ferienwohnung"
},
"3": {
"description": "Ein Ferienhaus oder eine Ferienwohnung mit Kochgelegenheit und Bad, das bzw. die für Ferienaufenthalte gemietet werden kann, in der Regel ohne Frühstück oder Rezeption",
"title": "ein Ferienchalet"
},
"4": {
"description": "Ein Motel ist eine Einrichtung, die eine bezahlte Unterkunft anbietet, in der Regel für kurze Zeit, mit bequemen Parkplätzen für Kraftfahrzeuge am oder in der Nähe des Zimmers. Sie sind in der Regel billiger als ein Hotel",
"title": "ein Motel"
},
"5": {
"description": "Ein Bed-and-Breakfast in einem Gästehaus ist ein kleiner Beherbergungsbetrieb. Oft handelt es sich um einige Zimmer in einem Privathaus, in dem die Eigentümer auch die Gäste beherbergen. Es gibt keine Rezeption und es ist auch nicht ständig Personal anwesend. In einigen Fällen erfolgt das Einchecken aus der Ferne, indem ein Zugangscode mitgeteilt wird.",
"title": "eine Frühstückspension"
},
"6": {
"title": "eine kleine Beherbergungseinrichtung, die in der Regel vom Eigentümer betrieben wird"
}
},
"tagRenderings": {
@ -10549,6 +10593,21 @@
},
"1": {
"then": "Hostel {name}"
},
"2": {
"then": "B&B {name}"
},
"3": {
"then": "Ferienhaus {name}"
},
"4": {
"then": "Motel {name}"
},
"5": {
"then": "Ferienwohnung {name}"
},
"6": {
"then": "Ferienhaus {name}"
}
},
"render": "Touristenunterkunft {name}"

View file

@ -27,6 +27,13 @@
"advertising": {
"name": "Reclame",
"presets": {
"10": {
"description": "Een stuk groot, weerbestendig textiel met opgedrukte reclameboodschap die permanent aan de muur hangt",
"title": "een spandoek"
},
"14": {
"title": "een muurschildering"
},
"4": {
"description": "Een klein uithangbord voor buurtadvertenties, meestal gericht op voetgangers",
"title": "een uithangbord"
@ -43,13 +50,6 @@
},
"8": {
"title": "een scherm op een muur"
},
"10": {
"description": "Een stuk groot, weerbestendig textiel met opgedrukte reclameboodschap die permanent aan de muur hangt",
"title": "een spandoek"
},
"14": {
"title": "een muurschildering"
}
},
"tagRenderings": {
@ -107,6 +107,9 @@
},
"title": {
"mappings": {
"10": {
"then": "Muurschildering"
},
"3": {
"then": "Aanplakzuil"
},
@ -124,9 +127,6 @@
},
"9": {
"then": "Aanplakzuil"
},
"10": {
"then": "Muurschildering"
}
}
}
@ -208,6 +208,15 @@
"1": {
"then": "Muurschildering"
},
"10": {
"then": "Azulejo (Spaanse siertegels)"
},
"11": {
"then": "Tegelwerk"
},
"12": {
"then": "Houtsculptuur"
},
"2": {
"then": "Schilderij"
},
@ -231,15 +240,6 @@
},
"9": {
"then": "Reliëf"
},
"10": {
"then": "Azulejo (Spaanse siertegels)"
},
"11": {
"then": "Tegelwerk"
},
"12": {
"then": "Houtsculptuur"
}
},
"question": "Wat voor soort kunstwerk is dit?",
@ -1656,30 +1656,6 @@
"1": {
"question": "Heeft een <div style='display: inline-block'><b><b>Schuko stekker</b> zonder aardingspin (CEE7/4 type F)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/CEE7_4F.svg'/></div>"
},
"2": {
"question": "Heeft een <div style='display: inline-block'><b><b>Europese stekker</b> met aardingspin (CEE7/4 type E)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/TypeE.svg'/></div>"
},
"3": {
"question": "Heeft een <div style='display: inline-block'><b><b>Chademo</b></b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Chademo_type4.svg'/></div>"
},
"4": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 1 met kabel</b> (J1772)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type1_J1772.svg'/></div>"
},
"5": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 1 <i>zonder</i> kabel</b> (J1772)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type1_J1772.svg'/></div>"
},
"6": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 1 CCS</b> (ook gekend als Type 1 Combo)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type1-ccs.svg'/></div>"
},
"7": {
"question": "Heeft een <div style='display: inline-block'><b><b>Tesla Supercharger</b></b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Tesla-hpwc-model-s.svg'/></div>"
},
"8": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 2</b> (mennekes)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type2_socket.svg'/></div>"
},
"9": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 2 CCS</b> (mennekes)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type2_CCS.svg'/></div>"
},
"10": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 2 met kabel</b> (J1772)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type2_tethered.svg'/></div>"
},
@ -1710,11 +1686,35 @@
"19": {
"question": "Heeft een <div style='display: inline-block'><b><b>SEV 1011 T23</b> (Zwitserse 3-pin)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/typej.svg'/></div>"
},
"2": {
"question": "Heeft een <div style='display: inline-block'><b><b>Europese stekker</b> met aardingspin (CEE7/4 type E)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/TypeE.svg'/></div>"
},
"20": {
"question": "Heeft een <div style='display: inline-block'><b><b>AS3112</b> (Australische 3-pin)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/as3112.svg'/></div>"
},
"21": {
"question": "Heeft een <div style='display: inline-block'><b><b>NEMA 5-20</b> (VS 3-pin)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/nema-5-20.svg'/></div>"
},
"3": {
"question": "Heeft een <div style='display: inline-block'><b><b>Chademo</b></b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Chademo_type4.svg'/></div>"
},
"4": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 1 met kabel</b> (J1772)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type1_J1772.svg'/></div>"
},
"5": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 1 <i>zonder</i> kabel</b> (J1772)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type1_J1772.svg'/></div>"
},
"6": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 1 CCS</b> (ook gekend als Type 1 Combo)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type1-ccs.svg'/></div>"
},
"7": {
"question": "Heeft een <div style='display: inline-block'><b><b>Tesla Supercharger</b></b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Tesla-hpwc-model-s.svg'/></div>"
},
"8": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 2</b> (mennekes)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type2_socket.svg'/></div>"
},
"9": {
"question": "Heeft een <div style='display: inline-block'><b><b>Type 2 CCS</b> (mennekes)</b> <img style='width:1rem; display: inline-block' src='./assets/layers/charging_station/Type2_CCS.svg'/></div>"
}
}
}
@ -1770,30 +1770,6 @@
"1": {
"then": "<b>Schuko stekker</b> zonder aardingspin (CEE7/4 type F)"
},
"2": {
"then": "<b>Europese stekker</b> met aardingspin (CEE7/4 type E)"
},
"3": {
"then": "<b>Europese stekker</b> met aardingspin (CEE7/4 type E)"
},
"4": {
"then": "<b>Chademo</b>"
},
"5": {
"then": "<b>Chademo</b>"
},
"6": {
"then": "<b>Type 1 met kabel</b> (J1772)"
},
"7": {
"then": "<b>Type 1 met kabel</b> (J1772)"
},
"8": {
"then": "<b>Type 1 <i>zonder</i> kabel</b> (J1772)"
},
"9": {
"then": "<b>Type 1 <i>zonder</i> kabel</b> (J1772)"
},
"10": {
"then": "<b>Type 1 CCS</b> (ook gekend als Type 1 Combo)"
},
@ -1824,6 +1800,9 @@
"19": {
"then": "<b>Type 2 met kabel</b> (J1772)"
},
"2": {
"then": "<b>Europese stekker</b> met aardingspin (CEE7/4 type E)"
},
"20": {
"then": "<b>Tesla Supercharger CCS</b> (een type2 CCS met Tesla-logo)"
},
@ -1854,6 +1833,9 @@
"29": {
"then": "<b>Bosch Active Connect met 3 pinnen</b> aan een kabel"
},
"3": {
"then": "<b>Europese stekker</b> met aardingspin (CEE7/4 type E)"
},
"30": {
"then": "<b>Bosch Active Connect met 5 pinnen</b> aan een kabel"
},
@ -1884,11 +1866,29 @@
"39": {
"then": "<b>AS3112</b> (Australische 3-pin)"
},
"4": {
"then": "<b>Chademo</b>"
},
"40": {
"then": "<b>NEMA 5-20</b> (VS 3-pin)"
},
"41": {
"then": "<b>NEMA 5-20</b> (VS 3-pin)"
},
"5": {
"then": "<b>Chademo</b>"
},
"6": {
"then": "<b>Type 1 met kabel</b> (J1772)"
},
"7": {
"then": "<b>Type 1 met kabel</b> (J1772)"
},
"8": {
"then": "<b>Type 1 <i>zonder</i> kabel</b> (J1772)"
},
"9": {
"then": "<b>Type 1 <i>zonder</i> kabel</b> (J1772)"
}
},
"question": "Welke aansluitingen zijn hier beschikbaar?"
@ -2082,30 +2082,6 @@
"1": {
"2": "<b>Europese stekker</b> met aardingspin (CEE7/4 type E)"
},
"2": {
"2": "<b>Chademo</b>"
},
"3": {
"2": "<b>Type 1 met kabel</b> (J1772)"
},
"4": {
"2": "<b>Type 1 <i>zonder</i> kabel</b> (J1772)"
},
"5": {
"2": "<b>Type 1 CCS</b> (ook gekend als Type 1 Combo)"
},
"6": {
"2": "<b>Tesla Supercharger</b>"
},
"7": {
"2": "<b>Type 2</b> (mennekes)"
},
"8": {
"2": "<b>Type 2 CCS</b> (mennekes)"
},
"9": {
"2": "<b>Type 2 met kabel</b> (J1772)"
},
"10": {
"2": "<b>Tesla Supercharger CCS</b> (een type2 CCS met Tesla-logo)"
},
@ -2136,8 +2112,32 @@
"19": {
"2": "<b>AS3112</b> (Australische 3-pin)"
},
"2": {
"2": "<b>Chademo</b>"
},
"20": {
"2": "<b>NEMA 5-20</b> (VS 3-pin)"
},
"3": {
"2": "<b>Type 1 met kabel</b> (J1772)"
},
"4": {
"2": "<b>Type 1 <i>zonder</i> kabel</b> (J1772)"
},
"5": {
"2": "<b>Type 1 CCS</b> (ook gekend als Type 1 Combo)"
},
"6": {
"2": "<b>Tesla Supercharger</b>"
},
"7": {
"2": "<b>Type 2</b> (mennekes)"
},
"8": {
"2": "<b>Type 2 CCS</b> (mennekes)"
},
"9": {
"2": "<b>Type 2 met kabel</b> (J1772)"
}
}
}
@ -2909,6 +2909,15 @@
"1": {
"then": "Dit fietspad is geplaveid"
},
"10": {
"then": "Dit fietspad is gemaakt van fijn grind"
},
"11": {
"then": "Dit fietspad is gemaakt van kiezelsteentjes"
},
"12": {
"then": "Dit fietspad is gemaakt van aarde"
},
"2": {
"then": "Dit fietspad is gemaakt van asfalt"
},
@ -2932,15 +2941,6 @@
},
"9": {
"then": "Dit fietspad is gemaakt van grind"
},
"10": {
"then": "Dit fietspad is gemaakt van fijn grind"
},
"11": {
"then": "Dit fietspad is gemaakt van kiezelsteentjes"
},
"12": {
"then": "Dit fietspad is gemaakt van aarde"
}
},
"question": "Waaruit is het oppervlak van het fietspad van gemaakt?",
@ -2989,6 +2989,15 @@
"1": {
"then": "Dit fietspad is geplaveid"
},
"10": {
"then": "Dit fietspad is gemaakt van fijn grind"
},
"11": {
"then": "Dit fietspad is gemaakt van kiezelsteentjes"
},
"12": {
"then": "Dit fietspad is gemaakt van aarde"
},
"2": {
"then": "Dit fietspad is gemaakt van asfalt"
},
@ -3012,15 +3021,6 @@
},
"9": {
"then": "Dit fietspad is gemaakt van grind"
},
"10": {
"then": "Dit fietspad is gemaakt van fijn grind"
},
"11": {
"then": "Dit fietspad is gemaakt van kiezelsteentjes"
},
"12": {
"then": "Dit fietspad is gemaakt van aarde"
}
},
"question": "Waaruit is het oppervlak van de straat gemaakt?",
@ -4076,6 +4076,21 @@
"1": {
"then": "Dit is een frituur"
},
"10": {
"then": "Dit is een Chinees restaurant"
},
"11": {
"then": "Dit is een Grieks restaurant"
},
"12": {
"then": "Dit is een Indisch restaurant"
},
"13": {
"then": "Dit is een Turks restaurant (dat meer dan enkel kebab verkoopt)"
},
"14": {
"then": "Dit is een Thaïs restaurant"
},
"2": {
"then": "Dit is een pastazaak"
},
@ -4099,21 +4114,6 @@
},
"9": {
"then": "Dit is een Frans restaurant"
},
"10": {
"then": "Dit is een Chinees restaurant"
},
"11": {
"then": "Dit is een Grieks restaurant"
},
"12": {
"then": "Dit is een Indisch restaurant"
},
"13": {
"then": "Dit is een Turks restaurant (dat meer dan enkel kebab verkoopt)"
},
"14": {
"then": "Dit is een Thaïs restaurant"
}
},
"question": "Welk soort gerechten worden hier geserveerd?",
@ -5144,6 +5144,12 @@
"0": {
"then": "Dit is een standbeeld"
},
"10": {
"then": "Dit is een kruis"
},
"12": {
"then": "Dit is een historische tank, permanent in de publieke ruimte geplaatst als gedenkteken"
},
"2": {
"then": "Dit is een zitbank die ook als herdenking dienst doet"
},
@ -5155,12 +5161,6 @@
},
"8": {
"then": "Dit is een sculptuur"
},
"10": {
"then": "Dit is een kruis"
},
"12": {
"then": "Dit is een historische tank, permanent in de publieke ruimte geplaatst als gedenkteken"
}
}
},
@ -5312,6 +5312,19 @@
}
}
},
"10": {
"options": {
"0": {
"question": "Alle Notes"
},
"1": {
"question": "Verberg import Notes"
},
"2": {
"question": "Toon enkel import Notes"
}
}
},
"2": {
"options": {
"0": {
@ -5367,19 +5380,6 @@
"question": "Toon enkel open Notes"
}
}
},
"10": {
"options": {
"0": {
"question": "Alle Notes"
},
"1": {
"question": "Verberg import Notes"
},
"2": {
"question": "Toon enkel import Notes"
}
}
}
},
"name": "OpenStreetMap Notes",
@ -5657,6 +5657,18 @@
"1": {
"then": "Dit is een normale parkeerplek."
},
"10": {
"then": "Deze parkeerplek is gereserveerd voor personeel."
},
"11": {
"then": "Deze parkeerplek is gereserveerd voor taxis."
},
"12": {
"then": "Deze parkeerplek is gereserveerd voor voertuigen met een aanhanger."
},
"13": {
"then": "Deze parkeerplek is gereserveerd voor autodelen."
},
"2": {
"then": "Dit is een gehandicaptenparkeerplaats."
},
@ -5680,18 +5692,6 @@
},
"9": {
"then": "Deze parkeerplek is gereserveerd voor ouders met kinderen."
},
"10": {
"then": "Deze parkeerplek is gereserveerd voor personeel."
},
"11": {
"then": "Deze parkeerplek is gereserveerd voor taxis."
},
"12": {
"then": "Deze parkeerplek is gereserveerd voor voertuigen met een aanhanger."
},
"13": {
"then": "Deze parkeerplek is gereserveerd voor autodelen."
}
},
"question": "Wat voor parkeerplek is dit?"
@ -5960,6 +5960,9 @@
"1": {
"then": "De ondergrond is <b>zand</b>"
},
"10": {
"then": "De ondergrond bestaat uit <b>kleine grindsteentjes</b> (steentjes kleiner dan 2 cm)"
},
"2": {
"then": "De ondergrond bestaat uit <b>houtsnippers</b>"
},
@ -5983,9 +5986,6 @@
},
"9": {
"then": "De ondergrond bestaat uit rubber, zoals rubberen tegels, rubber snippers of een groot rubberen oppervlak"
},
"10": {
"then": "De ondergrond bestaat uit <b>kleine grindsteentjes</b> (steentjes kleiner dan 2 cm)"
}
},
"question": "Wat is de ondergrond van deze speeltuin?",
@ -6009,6 +6009,9 @@
"0": {
"then": "Dit is een schommel"
},
"11": {
"then": "Dit is een rekstok"
},
"3": {
"then": "Dit is een zandbak"
},
@ -6020,9 +6023,6 @@
},
"6": {
"then": "Dit is een wipwap"
},
"11": {
"then": "Dit is een rekstok"
}
},
"question": "Wat voor speeltoestel is dit?"
@ -6311,6 +6311,21 @@
"1": {
"then": "Munten van 2 cent worden geaccepteerd"
},
"10": {
"then": "Munten van 20 rappen worden geaccepteerd"
},
"11": {
"then": "Munten van ½ frank worden geaccepteerd"
},
"12": {
"then": "Munten van 1 frank worden geaccepteerd"
},
"13": {
"then": "Munten van 2 frank worden geaccepteerd"
},
"14": {
"then": "Munten van 5 frank worden geaccepteerd"
},
"2": {
"then": "Munten van 5 cent worden geaccepteerd"
},
@ -6334,21 +6349,6 @@
},
"9": {
"then": "Munten van 10 rappen worden geaccepteerd"
},
"10": {
"then": "Munten van 20 rappen worden geaccepteerd"
},
"11": {
"then": "Munten van ½ frank worden geaccepteerd"
},
"12": {
"then": "Munten van 1 frank worden geaccepteerd"
},
"13": {
"then": "Munten van 2 frank worden geaccepteerd"
},
"14": {
"then": "Munten van 5 frank worden geaccepteerd"
}
},
"question": "Met welke munten kan je hier betalen?"
@ -6361,6 +6361,15 @@
"1": {
"then": "Biljetten van 10 euro worden geaccepteerd"
},
"10": {
"then": "Biljetten van 100 frank worden geaccepteerd"
},
"11": {
"then": "Biljetten van 200 frank worden geaccepteerd"
},
"12": {
"then": "Biljetten van 1000 frank worden geaccepteerd"
},
"2": {
"then": "Biljetten van 20 euro worden geaccepteerd"
},
@ -6384,15 +6393,6 @@
},
"9": {
"then": "Biljetten van 50 frank worden geaccepteerd"
},
"10": {
"then": "Biljetten van 100 frank worden geaccepteerd"
},
"11": {
"then": "Biljetten van 200 frank worden geaccepteerd"
},
"12": {
"then": "Biljetten van 1000 frank worden geaccepteerd"
}
},
"question": "Met welke bankbiljetten kan je hier betalen?"
@ -6737,30 +6737,6 @@
"1": {
"question": "Recycling van batterijen"
},
"2": {
"question": "Recycling van drankpakken"
},
"3": {
"question": "Recycling van blikken"
},
"4": {
"question": "Recycling van kleding"
},
"5": {
"question": "Recycling van frituurvet"
},
"6": {
"question": "Recycling van motorolie"
},
"7": {
"question": "Recycling van tl-buizen"
},
"8": {
"question": "Recycling van groen afval"
},
"9": {
"question": "Recycling van glazen flessen"
},
"10": {
"question": "Recycling van glas"
},
@ -6791,6 +6767,9 @@
"19": {
"question": "Recycling van restafval"
},
"2": {
"question": "Recycling van drankpakken"
},
"20": {
"question": "Recycling van inktpatronen"
},
@ -6799,6 +6778,27 @@
},
"22": {
"question": "Recycling van plastic verkpakkingen, metalen verkpakkingen en drankpakken (PMD)"
},
"3": {
"question": "Recycling van blikken"
},
"4": {
"question": "Recycling van kleding"
},
"5": {
"question": "Recycling van frituurvet"
},
"6": {
"question": "Recycling van motorolie"
},
"7": {
"question": "Recycling van tl-buizen"
},
"8": {
"question": "Recycling van groen afval"
},
"9": {
"question": "Recycling van glazen flessen"
}
}
},
@ -6866,30 +6866,6 @@
"1": {
"then": "Drankpakken kunnen hier gerecycled worden"
},
"2": {
"then": "Blikken kunnen hier gerecycled worden"
},
"3": {
"then": "Kleren kunnen hier gerecycled worden"
},
"4": {
"then": "Frituurvet kan hier gerecycled worden"
},
"5": {
"then": "Motorolie kan hier gerecycled worden"
},
"6": {
"then": "TL-buizen kunnen hier gerecycled worden"
},
"7": {
"then": "Groen afval kan hier gerecycled worden"
},
"8": {
"then": "Organisch afval kan hier gerecycled worden"
},
"9": {
"then": "Glazen flessen kunnen hier gerecycled worden"
},
"10": {
"then": "Glas kan hier gerecycled worden"
},
@ -6920,6 +6896,9 @@
"19": {
"then": "Oud metaal kan hier gerecycled worden"
},
"2": {
"then": "Blikken kunnen hier gerecycled worden"
},
"20": {
"then": "Schoenen kunnen hier gerecycled worden"
},
@ -6937,6 +6916,27 @@
},
"25": {
"then": "Fietsen (en fietswrakken) kunnen hier gerecycled worden"
},
"3": {
"then": "Kleren kunnen hier gerecycled worden"
},
"4": {
"then": "Frituurvet kan hier gerecycled worden"
},
"5": {
"then": "Motorolie kan hier gerecycled worden"
},
"6": {
"then": "TL-buizen kunnen hier gerecycled worden"
},
"7": {
"then": "Groen afval kan hier gerecycled worden"
},
"8": {
"then": "Organisch afval kan hier gerecycled worden"
},
"9": {
"then": "Glazen flessen kunnen hier gerecycled worden"
}
},
"question": "Wat kan hier gerecycled worden?"
@ -7769,6 +7769,12 @@
"1": {
"then": "Deze lantaarn gebruikt LEDs"
},
"10": {
"then": "Deze lantaarn gebruikt hogedruknatriumlampen (oranje met wit)"
},
"11": {
"then": "Deze lantaarn wordt verlicht met gas"
},
"2": {
"then": "Deze lantaarn gebruikt gloeilampen"
},
@ -7792,12 +7798,6 @@
},
"9": {
"then": "Deze lantaarn gebruikt lagedruknatriumlampen (monochroom oranje)"
},
"10": {
"then": "Deze lantaarn gebruikt hogedruknatriumlampen (oranje met wit)"
},
"11": {
"then": "Deze lantaarn wordt verlicht met gas"
}
},
"question": "Wat voor verlichting gebruikt deze lantaarn?"
@ -8348,15 +8348,68 @@
}
},
"tourism_accomodation": {
"description": "Verschillende soorten overnachtingsmogelijkheden voor toeristen",
"filter": {
"0": {
"options": {
"0": {
"question": "Alle types"
},
"1": {
"question": "Hotels"
},
"2": {
"question": "Hostels"
},
"3": {
"question": "Gastenkamers en bed-en-breakfasts"
},
"4": {
"question": "Motels"
},
"5": {
"question": "Vakantiehuisje"
},
"6": {
"question": "Vakantie-appartement"
}
}
}
},
"name": "Accomodatie voor toeristen",
"presets": {
"0": {
"description": "En hotel is een gebouw waar je tegen betaling kan overnachten voor een korte periode. Je krijgt je eigen kamer.",
"title": "een hotel"
},
"1": {
"description": "Een herberg is een gebouw waar je enkele dagen kan blijven. Je deelt een kamer met onbekenden.",
"title": "een herberg"
},
"2": {
"description": "Een bemeubeld apparement met kookgelegenheid en een badkamer in een groter gebouw. Het appartement kan gehuurd worden voor vakanties. Er is geen receptie of ontbijt voorzien.",
"title": "een vakantie-appartement"
},
"4": {
"title": "een motel"
}
},
"tagRenderings": {
"name": {
"question": "Wat is de naam van deze {title()}",
"render": "{name}"
}
},
"title": {
"mappings": {
"0": {
"then": "Hotel {name}"
},
"1": {
"then": "Herberg {name}"
},
"2": {
"then": "B&B {name}"
}
}
}
@ -9076,30 +9129,6 @@
"1": {
"question": "Verkoop van dranken"
},
"2": {
"question": "Verkoop van snoep"
},
"3": {
"question": "Verkoop van eten"
},
"4": {
"question": "Verkoop van sigaretten"
},
"5": {
"question": "Verkoop van condooms"
},
"6": {
"question": "Verkoop van koffie"
},
"7": {
"question": "Verkoop van water"
},
"8": {
"question": "Verkoop van kranten"
},
"9": {
"question": "Verkoop van fietsbinnenbanden"
},
"10": {
"question": "Verkoop van melk"
},
@ -9130,6 +9159,9 @@
"19": {
"question": "Verkoop van bloemen"
},
"2": {
"question": "Verkoop van snoep"
},
"23": {
"question": "Verkoop van fietslampjes"
},
@ -9144,6 +9176,27 @@
},
"27": {
"question": "Verkoop van fietssloten"
},
"3": {
"question": "Verkoop van eten"
},
"4": {
"question": "Verkoop van sigaretten"
},
"5": {
"question": "Verkoop van condooms"
},
"6": {
"question": "Verkoop van koffie"
},
"7": {
"question": "Verkoop van water"
},
"8": {
"question": "Verkoop van kranten"
},
"9": {
"question": "Verkoop van fietsbinnenbanden"
}
}
}
@ -9234,30 +9287,6 @@
"1": {
"then": "Snoep wordt verkocht"
},
"2": {
"then": "Eten wordt verkocht"
},
"3": {
"then": "Sigaretten worden verkocht"
},
"4": {
"then": "Condooms worden verkocht"
},
"5": {
"then": "Koffie wordt verkocht"
},
"6": {
"then": "Drinkwater wordt verkocht"
},
"7": {
"then": "Kranten worden verkocht"
},
"8": {
"then": "Binnenbanden voor fietsen worden verkocht"
},
"9": {
"then": "Melk wordt verkocht"
},
"10": {
"then": "Brood wordt verkocht"
},
@ -9288,6 +9317,9 @@
"19": {
"then": "Parkeerkaarten worden verkocht"
},
"2": {
"then": "Eten wordt verkocht"
},
"21": {
"then": "Openbaar vervoerkaartjes worden verkocht"
},
@ -9305,6 +9337,27 @@
},
"26": {
"then": "Fietssloten worden verkocht"
},
"3": {
"then": "Sigaretten worden verkocht"
},
"4": {
"then": "Condooms worden verkocht"
},
"5": {
"then": "Koffie wordt verkocht"
},
"6": {
"then": "Drinkwater wordt verkocht"
},
"7": {
"then": "Kranten worden verkocht"
},
"8": {
"then": "Binnenbanden voor fietsen worden verkocht"
},
"9": {
"then": "Melk wordt verkocht"
}
},
"question": "Wat verkoopt deze verkoopautomaat?",
@ -9606,4 +9659,4 @@
"render": "windturbine"
}
}
}
}

View file

@ -703,6 +703,7 @@
},
"tooLong": "Deze tekst is te lang. De tekst heeft {count} lettertekens, er mogen maximaal 255 letters zijn",
"url": {
"aggregator": "{host} is een aggregator-website. Gebruik de officiele website indien mogelijk.",
"description": "een link naar een webpagina",
"feedback": "Dit is geen geldige link"
},

View file

@ -87,7 +87,7 @@
"generate:contributor-list": "vite-node scripts/generateContributors.ts",
"generate:service-worker": "tsc src/service-worker.ts --outFile public/service-worker.js && git_hash=$(git rev-parse HEAD) && sed -i.bak \"s/GITHUB-COMMIT/$git_hash/\" public/service-worker.js && rm public/service-worker.js.bak",
"reset:layeroverview": "npm run prep:layeroverview && npm run generate:layeroverview && npm run refresh:layeroverview",
"prep:layeroverview": "mkdir -p ./src/assets/generated/layers; echo {\\\"themes\\\":[]} > ./src/assets/generated/known_themes.json && echo {\\\"layers\\\": []} > ./src/assets/generated/known_layers.json && rm -f ./src/assets/generated/layers/*.json && rm -f ./src/assets/generated/themes/*.json && cp ./assets/layers/usersettings/usersettings.json ./src/assets/generated/layers/usersettings.json && echo '{}' > ./src/assets/generated/layers/favourite.json && echo '{}' > ./src/assets/generated/layers/summary.json && echo '{}' > ./src/assets/generated/layers/last_click.json",
"prep:layeroverview": "mkdir -p ./src/assets/generated/layers; echo {\\\"themes\\\":[]} > ./src/assets/generated/known_themes.json && echo {\\\"layers\\\": []} > ./src/assets/generated/known_layers.json && rm -f ./src/assets/generated/layers/*.json && rm -f ./src/assets/generated/themes/*.json && cp ./assets/layers/usersettings/usersettings.json ./src/assets/generated/layers/usersettings.json && echo '{}' > ./src/assets/generated/layers/favourite.json && echo '{}' > ./src/assets/generated/layers/summary.json && echo '{}' > ./src/assets/generated/layers/last_click.json && echo '[]' > ./src/assets/generated/theme_overview.json",
"generate": "npm run generate:licenses && npm run generate:images && npm run generate:charging-stations && npm run generate:translations && npm run refresh:layeroverview && npm run generate:service-worker",
"generate:charging-stations": "cd ./assets/layers/charging_station && vite-node csvToJson.ts && cd -",
"clean:tests": "find . -type f -name \"*.doctest.ts\" | xargs -r rm",

View file

@ -39,10 +39,10 @@
.mapping-icon-small {
/* A mapping icon type */
width: 1.5rem;
width: 2rem;
height: fit-content;
max-height: 1.5rem;
max-height: 2rem;
margin-right: 0.5rem;
}

View file

@ -19,13 +19,14 @@ export default class BackgroundLayerResetter {
return
}
currentBackgroundLayer.addCallbackAndRunD((l) => {
currentBackgroundLayer.addCallbackAndRunD(async (l) => {
if (
l.geometry !== undefined &&
AvailableRasterLayers.globalLayers.find(
(global) => global.properties.id !== l.properties.id
)
) {
await AvailableRasterLayers.editorLayerIndex()
BackgroundLayerResetter.installHandler(
currentBackgroundLayer,
availableLayers.store

View file

@ -61,9 +61,9 @@ export class PreferredRasterLayerSelector {
* Returns 'true' if the target layer is set or is the current layer
* @private
*/
private updateLayer() {
private async updateLayer() {
// What is the ID of the layer we have to (try to) load?
const targetLayerId = this._queryParameter.data ?? this._preferredBackgroundLayer.data
const targetLayerId = (this._queryParameter.data ?? this._preferredBackgroundLayer.data)?.toLowerCase()
if (targetLayerId === undefined || targetLayerId === "default") {
return
}
@ -74,12 +74,13 @@ export class PreferredRasterLayerSelector {
this._rasterLayerSetting.setData(global)
return
}
await AvailableRasterLayers.editorLayerIndex()
const isCategory =
targetLayerId === "photo" || targetLayerId === "osmbasedmap" || targetLayerId === "map"
const available = this._availableLayers.store.data
const foundLayer = isCategory
? available.find((l) => l.properties.category === targetLayerId)
: available.find((l) => l.properties.id === targetLayerId)
: available.find((l) => l.properties.id.toLowerCase() === targetLayerId)
console.debug("Updating background layer to", foundLayer?.id, {
targetLayerId,
queryParam: this._queryParameter?.data,

View file

@ -14,6 +14,10 @@ export class And extends TagsFilter {
constructor(and: ReadonlyArray<TagsFilter>) {
super()
this.and = and
if(and.some(p => typeof p === "string")){
console.error("Assertion failed: invalid subtags:", and)
throw "Assertion failed: invalid subtags found"
}
}
public static construct(and: ReadonlyArray<TagsFilter>): TagsFilter

View file

@ -28,7 +28,7 @@ export class AvailableRasterLayers {
return this._editorLayerIndex
}
public static globalLayers: RasterLayerPolygon[] = globallayers.layers
public static globalLayers: ReadonlyArray<RasterLayerPolygon> = globallayers.layers
.filter(
(properties) =>
properties.id !== "osm.carto" && properties.id !== "Bing" /*Added separately*/
@ -140,28 +140,24 @@ export class RasterLayerUtils {
* @param available
* @param preferredCategory
* @param ignoreLayer
* @param skipLayers Skip the first N layers
*/
public static SelectBestLayerAccordingTo(
available: RasterLayerPolygon[],
preferredCategory: string,
ignoreLayer?: RasterLayerPolygon
ignoreLayer?: RasterLayerPolygon,
skipLayers: number = 0
): RasterLayerPolygon {
let secondBest: RasterLayerPolygon = undefined
for (const rasterLayer of available) {
if (rasterLayer === ignoreLayer) {
continue
}
const p = rasterLayer.properties
if (p.category === preferredCategory) {
if (p.best) {
return rasterLayer
}
if (!secondBest) {
secondBest = rasterLayer
}
}
const inCategory = available.filter(l => l.properties.category === preferredCategory)
const best : RasterLayerPolygon[] = inCategory.filter(l => l.properties.best)
const others : RasterLayerPolygon[] = inCategory.filter(l => !l.properties.best)
let all = best.concat(others)
console.log("Selected layers are:", all.map(l => l.properties.id))
if(others.length > skipLayers){
all = all.slice(skipLayers)
}
return secondBest
return all.find(l => l !== ignoreLayer)
}
}

View file

@ -973,12 +973,11 @@ export class TagRenderingConfigUtils {
}
const clone: TagRenderingConfig = Object.create(config)
// The original mappings get "priorityIf" set
const oldMappingsCloned =
clone.mappings?.map((m) => ({
...m,
addExtraTags: "nobrand=",
priorityIf: m.priorityIf ?? TagUtils.Tag("id~*"),
})) ?? []
const oldMappingsCloned = clone.mappings?.map((m) => (<Mapping> {
...m,
addExtraTags: [new Tag("nobrand","")],
priorityIf: m.priorityIf ?? TagUtils.Tag("id~*"),
})) ?? [];
clone.mappings = [...oldMappingsCloned, ...extraMappings]
return clone
})

View file

@ -161,7 +161,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
this.featureSwitches = new FeatureSwitchState(layout)
this.guistate = new MenuState(
this.featureSwitches.featureSwitchWelcomeMessage.data,
layout.id
layout.id,
)
this.map = new UIEventSource<MlMap>(undefined)
const geolocationState = new GeoLocationState()
@ -177,14 +177,14 @@ export default class ThemeViewState implements SpecialVisualizationState {
oauth_token: QueryParameters.GetQueryParameter(
"oauth_token",
undefined,
"Used to complete the login"
"Used to complete the login",
),
})
this.userRelatedState = new UserRelatedState(
this.osmConnection,
layout,
this.featureSwitches,
this.mapProperties
this.mapProperties,
)
this.userRelatedState.fixateNorth.addCallbackAndRunD((fixated) => {
this.mapProperties.allowRotating.setData(fixated !== "yes")
@ -195,20 +195,20 @@ export default class ThemeViewState implements SpecialVisualizationState {
geolocationState,
this.selectedElement,
this.mapProperties,
this.userRelatedState.gpsLocationHistoryRetentionTime
this.userRelatedState.gpsLocationHistoryRetentionTime,
)
this.geolocationControl = new GeolocationControlState(this.geolocation, this.mapProperties)
this.availableLayers = AvailableRasterLayers.layersAvailableAt(
this.mapProperties.location,
this.osmConnection.isLoggedIn
this.osmConnection.isLoggedIn,
)
this.layerState = new LayerState(
this.osmConnection,
layout.layers,
layout.id,
this.featureSwitches.featureSwitchLayerDefault
this.featureSwitches.featureSwitchLayerDefault,
)
{
@ -217,7 +217,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
const isDisplayed = QueryParameters.GetBooleanQueryParameter(
"overlay-" + rasterInfo.id,
rasterInfo.defaultState ?? true,
"Whether or not overlay layer " + rasterInfo.id + " is shown"
"Whether or not overlay layer " + rasterInfo.id + " is shown",
)
const state = { isDisplayed }
overlayLayerStates.set(rasterInfo.id, state)
@ -242,7 +242,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
this.osmConnection.Backend(),
(id) => this.layerState.filteredLayers.get(id).isDisplayed,
mvtAvailableLayers,
this.fullNodeDatabase
this.fullNodeDatabase,
)
let currentViewIndex = 0
@ -260,7 +260,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
id: "current_view_" + currentViewIndex,
}),
]
})
}),
)
this.featuresInView = new BBoxFeatureSource(layoutSource, this.mapProperties.bounds)
@ -278,19 +278,19 @@ export default class ThemeViewState implements SpecialVisualizationState {
featureSwitches: this.featureSwitches,
},
layout?.isLeftRightSensitive() ?? false,
(e) => this.reportError(e)
(e) => this.reportError(e),
)
this.historicalUserLocations = this.geolocation.historicalUserLocations
this.newFeatures = new NewGeometryFromChangesFeatureSource(
this.changes,
layoutSource,
this.featureProperties
this.featureProperties,
)
layoutSource.addSource(this.newFeatures)
const perLayer = new PerLayerFeatureSourceSplitter(
Array.from(this.layerState.filteredLayers.values()).filter(
(l) => l.layerDef?.source !== null
(l) => l.layerDef?.source !== null,
),
new ChangeGeometryApplicator(this.indexedFeatures, this.changes),
{
@ -301,10 +301,10 @@ export default class ThemeViewState implements SpecialVisualizationState {
"Got ",
features.length,
"leftover features, such as",
features[0].properties
features[0].properties,
)
},
}
},
)
this.perLayer = perLayer.perLayer
}
@ -344,12 +344,12 @@ export default class ThemeViewState implements SpecialVisualizationState {
this.lastClickObject = new LastClickFeatureSource(
this.layout,
this.mapProperties.lastClickLocation,
this.userRelatedState.addNewFeatureMode
this.userRelatedState.addNewFeatureMode,
)
this.osmObjectDownloader = new OsmObjectDownloader(
this.osmConnection.Backend(),
this.changes
this.changes,
)
this.perLayerFiltered = this.showNormalDataOn(this.map)
@ -360,7 +360,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
currentZoom: this.mapProperties.zoom,
layerState: this.layerState,
bounds: this.visualFeedbackViewportBounds,
}
},
)
this.hasDataInView = new NoElementsInViewDetector(this).hasFeatureInView
this.imageUploadManager = new ImageUploadManager(
@ -368,7 +368,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
Imgur.singleton,
this.featureProperties,
this.osmConnection,
this.changes
this.changes,
)
this.favourites = new FavouritesFeatureSource(this)
const longAgo = new Date()
@ -414,7 +414,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
LayoutSource.fromCacheZoomLevel,
fs,
this.featureProperties,
fs.layer.layerDef.maxAgeOfCache
fs.layer.layerDef.maxAgeOfCache,
)
toLocalStorage.set(layerId, storage)
})
@ -427,7 +427,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
const doShowLayer = this.mapProperties.zoom.map(
(z) =>
(fs.layer.isDisplayed?.data ?? true) && z >= (fs.layer.layerDef?.minzoom ?? 0),
[fs.layer.isDisplayed]
[fs.layer.isDisplayed],
)
if (!doShowLayer.data && this.featureSwitches.featureSwitchFilter.data === false) {
@ -444,7 +444,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
fs.layer,
fs,
(id) => this.featureProperties.getStore(id),
this.layerState.globalFilters
this.layerState.globalFilters,
)
filteringFeatureSource.set(layerName, filtered)
@ -588,7 +588,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
return
}
this.selectClosestAtCenter(0)
}
},
)
for (let i = 1; i < 9; i++) {
@ -606,7 +606,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
onUp: true,
},
doc,
() => this.selectClosestAtCenter(i - 1)
() => this.selectClosestAtCenter(i - 1),
)
}
@ -623,7 +623,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
if (this.featureSwitches.featureSwitchBackgroundSelection.data) {
this.guistate.backgroundLayerSelectionIsOpened.setData(true)
}
}
},
)
Hotkeys.RegisterHotkey(
{
@ -635,18 +635,11 @@ export default class ThemeViewState implements SpecialVisualizationState {
if (this.featureSwitches.featureSwitchFilter.data) {
this.guistate.openFilterView()
}
}
},
)
Hotkeys.RegisterHotkey(
{ shift: "O" },
Translations.t.hotkeyDocumentation.selectMapnik,
() => {
this.mapProperties.rasterLayer.setData(AvailableRasterLayers.osmCarto)
}
)
const setLayerCategory = (category: EliCategory) => {
const setLayerCategory = (category: EliCategory, skipLayers: number = 0) => {
const timeOfCall = new Date()
const available = this.availableLayers.store.addCallbackAndRunD((available) => {
this.availableLayers.store.addCallbackAndRunD((available) => {
const now = new Date()
const timeDiff = (now.getTime() - timeOfCall.getTime()) / 1000
if (timeDiff > 3) {
@ -656,9 +649,13 @@ export default class ThemeViewState implements SpecialVisualizationState {
const best = RasterLayerUtils.SelectBestLayerAccordingTo(
available,
category,
current.data
current.data,
skipLayers
)
console.log("Best layer for category", category, "is", best.properties.id)
if(!best){
return
}
console.log("Best layer for category", category, "is", best?.properties?.id)
current.setData(best)
})
}
@ -666,26 +663,43 @@ export default class ThemeViewState implements SpecialVisualizationState {
Hotkeys.RegisterHotkey(
{ nomod: "O" },
Translations.t.hotkeyDocumentation.selectOsmbasedmap,
() => setLayerCategory("osmbasedmap")
() => setLayerCategory("osmbasedmap"),
)
Hotkeys.RegisterHotkey(
{ nomod: "M" },
Translations.t.hotkeyDocumentation.selectMap,
() => setLayerCategory("map")
() => setLayerCategory("map"),
)
Hotkeys.RegisterHotkey(
{ nomod: "P" },
Translations.t.hotkeyDocumentation.selectAerial,
() => setLayerCategory("photo")
() => setLayerCategory("photo"),
)
Hotkeys.RegisterHotkey(
{ shift: "O" },
Translations.t.hotkeyDocumentation.selectOsmbasedmap,
() => setLayerCategory("osmbasedmap",2),
)
Hotkeys.RegisterHotkey(
{ shift: "M" },
Translations.t.hotkeyDocumentation.selectMap,
() => setLayerCategory("map",2),
)
Hotkeys.RegisterHotkey(
{ shift: "P" },
Translations.t.hotkeyDocumentation.selectAerial,
() => setLayerCategory("photo",2),
)
Hotkeys.RegisterHotkey(
{ nomod: "L" },
Translations.t.hotkeyDocumentation.geolocate,
() => {
this.geolocationControl.handleClick()
}
},
)
return true
})
@ -697,7 +711,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
Translations.t.hotkeyDocumentation.translationMode,
() => {
Locale.showLinkToWeblate.setData(!Locale.showLinkToWeblate.data)
}
},
)
}
@ -708,7 +722,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
const normalLayers = this.layout.layers.filter(
(l) =>
Constants.priviliged_layers.indexOf(<any>l.id) < 0 &&
!l.id.startsWith("note_import")
!l.id.startsWith("note_import"),
)
const maxzoom = Math.min(...normalLayers.map((l) => l.minzoom))
@ -716,7 +730,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
(l) =>
Constants.priviliged_layers.indexOf(<any>l.id) < 0 &&
l.source.geojsonSource === undefined &&
l.doCount
l.doCount,
)
const summaryTileSource = new SummaryTileSource(
Constants.SummaryServer,
@ -725,7 +739,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
this.mapProperties,
{
isActive: this.mapProperties.zoom.map((z) => z < maxzoom),
}
},
)
return new SummaryTileSourceRewriter(summaryTileSource, this.layerState.filteredLayers)
@ -746,12 +760,12 @@ export default class ThemeViewState implements SpecialVisualizationState {
gps_location_history: this.geolocation.historicalUserLocations,
gps_track: this.geolocation.historicalUserLocationsTrack,
selected_element: new StaticFeatureSource(
this.selectedElement.map((f) => (f === undefined ? empty : [f]))
this.selectedElement.map((f) => (f === undefined ? empty : [f])),
),
range: new StaticFeatureSource(
this.mapProperties.maxbounds.map((bbox) =>
bbox === undefined ? empty : <Feature[]>[bbox.asGeoJson({ id: "range" })]
)
bbox === undefined ? empty : <Feature[]>[bbox.asGeoJson({ id: "range" })],
),
),
current_view: this.currentView,
favourite: this.favourites,
@ -766,7 +780,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
ShowDataLayer.showRange(
this.map,
new StaticFeatureSource([bbox.asGeoJson({ id: "range" })]),
this.featureSwitches.featureSwitchIsTesting
this.featureSwitches.featureSwitchIsTesting,
)
}
const currentViewLayer = this.layout.layers.find((l) => l.id === "current_view")
@ -780,7 +794,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
currentViewLayer,
this.layout,
this.osmObjectDownloader,
this.featureProperties
this.featureProperties,
)
})
}
@ -824,20 +838,20 @@ export default class ThemeViewState implements SpecialVisualizationState {
const lastClickLayerConfig = new LayerConfig(
<LayerConfigJson>last_click_layerconfig,
"last_click"
"last_click",
)
const lastClickFiltered =
lastClickLayerConfig.isShown === undefined
? specialLayers.last_click
: specialLayers.last_click.features.mapD((fs) =>
fs.filter((f) => {
const matches = lastClickLayerConfig.isShown.matchesProperties(
f.properties
)
console.debug("LastClick ", f, "matches", matches)
return matches
})
)
fs.filter((f) => {
const matches = lastClickLayerConfig.isShown.matchesProperties(
f.properties,
)
console.debug("LastClick ", f, "matches", matches)
return matches
}),
)
new ShowDataLayer(this.map, {
features: new StaticFeatureSource(lastClickFiltered),
layer: lastClickLayerConfig,
@ -884,7 +898,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
this.mapProperties.rasterLayer,
this.availableLayers,
this.featureSwitches.backgroundLayerId,
this.userRelatedState.preferredBackgroundLayer
this.userRelatedState.preferredBackgroundLayer,
)
}
@ -900,7 +914,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
? ">>> _Not_ reporting error to report server as testmode is on"
: ">>> Reporting error to",
Constants.ErrorReportServer,
message
message,
)
if (isTesting) {
return

View file

@ -85,13 +85,12 @@
feedback?.setData(undefined)
return
}
feedback?.setData(validator?.getFeedback(v, getCountry))
if (!validator?.isValid(v, getCountry)) {
feedback?.setData(validator?.getFeedback(v, getCountry))
value.setData(undefined)
return
}
feedback?.setData(undefined)
if (selectedUnit.data) {
value.setData(unit.toOsm(v, selectedUnit.data))
} else {

View file

@ -1,16 +1,30 @@
import { Validator } from "../Validator"
import { Translation } from "../../i18n/Translation"
import Translations from "../../i18n/Translations"
export default class UrlValidator extends Validator {
private readonly _forceHttps: boolean
private static readonly aggregatorWebsites = new Set<string>([
"booking.com",
"hotel-details-guide.com", "tripingguide.com",
"tripadvisor.com", "tripadvisor.co.uk", "tripadvisor.com.au",
])
constructor(name?: string, explanation?: string, forceHttps?: boolean) {
super(
name ?? "url",
explanation ??
"The validatedTextField will format URLs to always be valid and have a https://-header (even though the 'https'-part will be hidden from the user. Furthermore, some tracking parameters will be removed",
"url"
"The validatedTextField will format URLs to always be valid and have a https://-header (even though the 'https'-part will be hidden from the user. Furthermore, some tracking parameters will be removed",
"url",
)
this._forceHttps = forceHttps ?? false
}
/**
*
* new UrlValidator().reformat("https://example.com/page?fbclid=123456&utm_source=mastodon") // => "https://example.com/page"
*/
reformat(str: string): string {
try {
let url: URL
@ -63,6 +77,33 @@ export default class UrlValidator extends Validator {
}
}
/**
*
* const v = new UrlValidator()
* v.getFeedback("example.").textFor("en") // => "This is not a valid web address"
* v.getFeedback("https://booking.com/some-hotel.html").textFor("en").indexOf("search the official website") > 0 // => true
*
*/
getFeedback(s: string, getCountry?: () => string): Translation | undefined {
const upstream = super.getFeedback(s, getCountry)
if (upstream) {
return upstream
}
/*
Upstream calls 'isValid', which checks if it is an actual URL.
If we reach this point, we can safely assume 'new URL' will work
*/
const url = new URL(s)
let host = url.host.toLowerCase()
if (host.startsWith("www.")) {
host = host.slice(4)
}
if (UrlValidator.aggregatorWebsites.has(host)) {
return Translations.t.validation.url.aggregator.Subs({ host })
}
return undefined
}
isValid(str: string): boolean {
try {
if (

View file

@ -50,7 +50,7 @@
export let color: string | undefined = undefined
export let clss: string | undefined = ""
clss ??= ""
export let emojiHeight = 40
export let emojiHeight = "40px"
</script>
{#if icon}
@ -147,7 +147,7 @@
{:else if icon === "user_circle"}
<UserCircleIcon class={clss} {color} />
{:else if Utils.isEmoji(icon)}
<span style={`font-size: ${emojiHeight}px; line-height: ${emojiHeight}px`}>
<span style= {`font-size: ${emojiHeight}; line-height: ${emojiHeight}`}>
{icon}
</span>
{:else}

View file

@ -30,6 +30,7 @@
* Class which is applied onto the individual icons
*/
export let clss = ""
export let emojiHeight : string = "40px"
/**
* Class applied onto the entire element
@ -41,7 +42,7 @@
<div class={twMerge("relative", size)}>
{#each iconsParsed as icon}
<div class="absolute top-0 left-0 flex h-full w-full items-center">
<Icon icon={icon.icon} color={icon.color} {clss} />
<Icon icon={icon.icon} color={icon.color} {clss} {emojiHeight} />
</div>
{/each}
</div>

View file

@ -23,7 +23,7 @@
let rasterLayerId = rasterLayer.sync(
(l) => l?.properties?.id,
[],
(id) => availableLayers.find((l) => l.properties.id === id)
(id) => availableLayers.find((l) => l.properties.id === id),
)
rasterLayer.setData(availableLayers[0])
$: rasterLayer.setData(availableLayers[0])
@ -36,13 +36,13 @@
return
}
rasterLayer.setData(fav)
})
}),
)
onDestroy(
rasterLayer.addCallbackAndRunD((selected) => {
favourite?.setData(selected.properties.id)
})
}),
)
}
@ -56,7 +56,7 @@
} else {
rasterLayerOnMap.setData(undefined)
}
})
}),
)
}
@ -93,6 +93,15 @@
{#each availableLayers as availableLayer}
<option value={availableLayer.properties.id}>
{availableLayer.properties.name}
{#if availableLayer.properties.category.startsWith("historic")}
⏱️
{/if}
{#if availableLayer.properties.category.endsWith("elevation")}
{/if}
{#if availableLayer.properties.best}
{/if}
</option>
{/each}
</select>

View file

@ -30,6 +30,12 @@
| "large-height"
| string
}
const emojiHeights = {
"small":"2rem",
"medium":"3rem",
"large":"5rem"
}
</script>
{#if mapping.icon !== undefined}
@ -42,6 +48,9 @@
}-width`,
"shrink-0"
)}
emojiHeight={ emojiHeights[mapping.iconClass] ?? "2rem"}
clss={`mapping-icon-${mapping.iconClass ?? "small"}`}
/>
<SpecialTranslation t={mapping.then} {tags} {state} {layer} feature={selectedElement} {clss} />

View file

@ -1727,11 +1727,12 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
}
}
private static emojiRegex = /^\p{Extended_Pictographic}+$/u
private static emojiRegex = /[\p{Extended_Pictographic}🛰]$/u
/**
* Returns 'true' if the given string contains at least one and only emoji characters
* @param string
*
* Utils.isEmoji("⛰\uFE0F") // => true
*/
public static isEmoji(string: string) {
return Utils.emojiRegex.test(string)

View file

@ -1,5 +1,21 @@
{
"layers": [
{
"url": "pmtiles://https://api.protomaps.com/tiles/v3.json?key=2af8b969a9e8b692",
"style": "assets/sunny.json",
"connect-src": [
"https://protomaps.github.io"
],
"best": true,
"id": "protomaps.sunny",
"name": "Protomaps Sunny",
"type": "vector",
"category": "osmbasedmap",
"attribution": {
"text": "Protomaps",
"url": "https://protomaps.com/"
}
},
{
"name": "OpenStreetMap Carto",
"url": "https://tile.openstreetmap.org/{z}/{x}/{y}.png",
@ -87,21 +103,6 @@
"url": "https://protomaps.com/"
}
},
{
"url": "pmtiles://https://api.protomaps.com/tiles/v3.json?key=2af8b969a9e8b692",
"style": "assets/sunny.json",
"connect-src": [
"https://protomaps.github.io"
],
"id": "protomaps.sunny",
"name": "Protomaps Sunny",
"type": "vector",
"category": "osmbasedmap",
"attribution": {
"text": "Protomaps",
"url": "https://protomaps.com/"
}
},
{
"url": "pmtiles://https://api.protomaps.com/tiles/v3.json?key=2af8b969a9e8b692",
"style": "assets/sunny-unlabeled.json",