From 700b48f69315e4e35b3171d42f18781152225ac3 Mon Sep 17 00:00:00 2001
From: pietervdvn
Date: Sat, 22 Jan 2022 02:56:35 +0100
Subject: [PATCH] Fix import flow for more advanced scenarios
---
Customizations/SharedTagRenderings.ts | 3 +
UI/Base/Toggleable.ts | 5 +
UI/BigComponents/Histogram.ts | 11 +-
.../CompareToAlreadyExistingNotes.ts | 72 +-
UI/ImportFlow/FlowStep.ts | 24 +-
UI/ImportFlow/ImportHelperGui.ts | 33 +-
UI/ImportFlow/ImportInspector.ts | 1 +
UI/ImportFlow/ImportUtils.ts | 4 +-
UI/ImportFlow/Introdution.ts | 20 +
UI/ImportFlow/LoginToImport.ts | 52 +
UI/ImportFlow/{DataPanel.ts => MapPreview.ts} | 55 +-
UI/ImportFlow/PreviewPanel.ts | 98 ++
UI/ImportFlow/RequestFile.ts | 46 +-
UI/SpecialVisualizations.ts | 2 +-
langs/en.json | 974 +++++++++---------
package-lock.json | 34 +-
package.json | 4 +-
scripts/generateLayerOverview.ts | 8 +-
18 files changed, 871 insertions(+), 575 deletions(-)
create mode 100644 UI/ImportFlow/ImportInspector.ts
create mode 100644 UI/ImportFlow/Introdution.ts
create mode 100644 UI/ImportFlow/LoginToImport.ts
rename UI/ImportFlow/{DataPanel.ts => MapPreview.ts} (78%)
create mode 100644 UI/ImportFlow/PreviewPanel.ts
diff --git a/Customizations/SharedTagRenderings.ts b/Customizations/SharedTagRenderings.ts
index ad0aa83e2..9901e608e 100644
--- a/Customizations/SharedTagRenderings.ts
+++ b/Customizations/SharedTagRenderings.ts
@@ -39,6 +39,9 @@ export default class SharedTagRenderings {
}
dict.forEach((value, key) => {
+ if(key === "id"){
+ return;
+ }
value.id = value.id ?? key;
})
diff --git a/UI/Base/Toggleable.ts b/UI/Base/Toggleable.ts
index 31c68239b..e7bdccbdc 100644
--- a/UI/Base/Toggleable.ts
+++ b/UI/Base/Toggleable.ts
@@ -62,4 +62,9 @@ export default class Toggleable extends Combine {
})
}
+ public Collapse() : Toggleable{
+ this.isVisible.setData(false)
+ return this;
+ }
+
}
\ No newline at end of file
diff --git a/UI/BigComponents/Histogram.ts b/UI/BigComponents/Histogram.ts
index 2051257ae..919f8f516 100644
--- a/UI/BigComponents/Histogram.ts
+++ b/UI/BigComponents/Histogram.ts
@@ -22,9 +22,12 @@ export default class Histogram extends VariableUiElement {
constructor(values: UIEventSource,
title: string | BaseUIElement,
countTitle: string | BaseUIElement,
- assignColor?: (t0: string) => string
+ options?:{
+ assignColor?: (t0: string) => string,
+ sortMode?: "name" | "name-rev" | "count" | "count-rev"
+ }
) {
- const sortMode = new UIEventSource<"name" | "name-rev" | "count" | "count-rev">("name")
+ const sortMode = new UIEventSource<"name" | "name-rev" | "count" | "count-rev">(options?.sortMode ??"name")
const sortName = new VariableUiElement(sortMode.map(m => {
switch (m) {
case "name":
@@ -107,11 +110,11 @@ export default class Histogram extends VariableUiElement {
return Histogram.defaultPalette[index % Histogram.defaultPalette.length]
};
let actualAssignColor = undefined;
- if (assignColor === undefined) {
+ if (options?.assignColor === undefined) {
actualAssignColor = fallbackColor;
} else {
actualAssignColor = (keyValue: string) => {
- return assignColor(keyValue) ?? fallbackColor(keyValue)
+ return options.assignColor(keyValue) ?? fallbackColor(keyValue)
}
}
diff --git a/UI/ImportFlow/CompareToAlreadyExistingNotes.ts b/UI/ImportFlow/CompareToAlreadyExistingNotes.ts
index 5b48ef6a0..564b62a1e 100644
--- a/UI/ImportFlow/CompareToAlreadyExistingNotes.ts
+++ b/UI/ImportFlow/CompareToAlreadyExistingNotes.ts
@@ -40,7 +40,11 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
tagRenderings: new Map()
}
- const layerConfig = known_layers.filter(l => l.id === params.layer.id)[0]
+
+ const layerConfig = known_layers.layers.filter(l => l.id === params.layer.id)[0]
+ if(layerConfig === undefined){
+ console.error("WEIRD: layer not found in the builtin layer overview")
+ }
const importLayerJson = new CreateNoteImportLayer(365).convertStrict(convertState, layerConfig, "CompareToAlreadyExistingNotes")
const importLayer = new LayerConfig(importLayerJson, "import-layer-dynamic")
const flayer: FilteredLayer = {
@@ -48,8 +52,8 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
isDisplayed: new UIEventSource(true),
layerDef: importLayer
}
- const unfiltered = new GeoJsonSource(flayer, params.bbox.padAbsolute(0.0001))
- unfiltered.features.map(f => MetaTagging.addMetatags(
+ const allNotesWithinBbox = new GeoJsonSource(flayer, params.bbox.padAbsolute(0.0001))
+ allNotesWithinBbox.features.map(f => MetaTagging.addMetatags(
f,
{
memberships: new RelationsTracker(),
@@ -65,30 +69,30 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
}
)
)
- const data = new FilteringFeatureSource(state, undefined, unfiltered)
- data.features.addCallbackD(features => console.log("Loaded and filtered features are", features))
+ const alreadyOpenImportNotes = new FilteringFeatureSource(state, undefined, allNotesWithinBbox)
+ alreadyOpenImportNotes.features.addCallbackD(features => console.log("Loaded and filtered features are", features))
const map = Minimap.createMiniMap()
map.SetClass("w-full").SetStyle("height: 500px")
- const comparison = Minimap.createMiniMap({
+ const comparisonMap = Minimap.createMiniMap({
location: map.location,
})
- comparison.SetClass("w-full").SetStyle("height: 500px")
+ comparisonMap.SetClass("w-full").SetStyle("height: 500px")
new ShowDataLayer({
layerToShow: importLayer,
state,
zoomToFeatures: true,
leafletMap: map.leafletMap,
- features: data,
+ features: alreadyOpenImportNotes,
popup: (tags, layer) => new FeatureInfoBox(tags, layer, state)
})
const maxDistance = new UIEventSource(5)
- const partitionedImportPoints = ImportUtils.partitionFeaturesIfNearby(params.geojson, data.features
+ const partitionedImportPoints = ImportUtils.partitionFeaturesIfNearby(params.geojson, alreadyOpenImportNotes.features
.map(ff => ({features: ff.map(ff => ff.feature)})), maxDistance)
@@ -96,27 +100,34 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
layerToShow: new LayerConfig(import_candidate),
state,
zoomToFeatures: true,
- leafletMap: comparison.leafletMap,
+ leafletMap: comparisonMap.leafletMap,
features: new StaticFeatureSource(partitionedImportPoints.map(p => p.hasNearby), false),
popup: (tags, layer) => new FeatureInfoBox(tags, layer, state)
})
super([
new Title("Compare with already existing 'to-import'-notes"),
- new Toggle(
- new Loading("Fetching notes from OSM"),
- new Combine([
- map,
- "The following (red) elements are elements to import which are nearby a matching element that is already up for import. These won't be imported",
-
- new Toggle(
- new FixedUiElement("All of the proposed points have (or had) an import note already").SetClass("alert w-full block").SetStyle("padding: 0.5rem"),
- new VariableUiElement(partitionedImportPoints.map(({noNearby}) => noNearby.length + " elements can be imported")).SetClass("thanks p-8"),
- partitionedImportPoints.map(({noNearby}) => noNearby.length === 0)
- ).SetClass("w-full"),
- comparison,
- ]).SetClass("flex flex-col"),
- unfiltered.features.map(ff => ff === undefined || ff.length === 0)
+ new VariableUiElement(
+ alreadyOpenImportNotes.features.map(notesWithImport => {
+ if(allNotesWithinBbox.features.data === undefined || allNotesWithinBbox.features.data.length === 0){
+ return new Loading("Fetching notes from OSM")
+ }
+ if(notesWithImport.length === 0){
+ return new FixedUiElement("No previous note to import found").SetClass("thanks")
+ }
+ return new Combine([
+ map,
+ "The following (red) elements are elements to import which are nearby a matching element that is already up for import. These won't be imported",
+
+ new Toggle(
+ new FixedUiElement("All of the proposed points have (or had) an import note already").SetClass("alert w-full block").SetStyle("padding: 0.5rem"),
+ new VariableUiElement(partitionedImportPoints.map(({noNearby}) => noNearby.length + " elements can be imported")).SetClass("thanks p-8"),
+ partitionedImportPoints.map(({noNearby}) => noNearby.length === 0)
+ ).SetClass("w-full"),
+ comparisonMap,
+ ]).SetClass("flex flex-col")
+
+ }, [allNotesWithinBbox.features])
),
@@ -128,7 +139,18 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
layer: params.layer
}))
- this.IsValid = data.features.map(ff => ff.length > 0 && partitionedImportPoints.data.noNearby.length > 0, [partitionedImportPoints])
+ this.IsValid = alreadyOpenImportNotes.features.map(ff => {
+ if(allNotesWithinBbox.features.data.length === 0){
+ // Not yet loaded
+ return false
+ }
+ if(ff.length == 0){
+ // No import notes at all
+ return true;
+ }
+
+ return partitionedImportPoints.data.noNearby.length > 0; // at least _something_ can be imported
+ }, [partitionedImportPoints, allNotesWithinBbox.features])
}
}
\ No newline at end of file
diff --git a/UI/ImportFlow/FlowStep.ts b/UI/ImportFlow/FlowStep.ts
index 62509af15..c8773b1c9 100644
--- a/UI/ImportFlow/FlowStep.ts
+++ b/UI/ImportFlow/FlowStep.ts
@@ -7,6 +7,7 @@ import Translations from "../i18n/Translations";
import {VariableUiElement} from "../Base/VariableUIElement";
import Toggle from "../Input/Toggle";
import {UIElement} from "../UIElement";
+import {FixedUiElement} from "../Base/FixedUiElement";
export interface FlowStep extends BaseUIElement {
readonly IsValid: UIEventSource
@@ -95,10 +96,12 @@ export class FlowPanel extends Toggle {
})
let elements: (BaseUIElement | string)[] = []
+ const isError = new UIEventSource(false)
if (initial !== undefined) {
// Startup the flow
elements = [
initial,
+
new Combine([
backbutton,
new Toggle(
@@ -107,14 +110,25 @@ export class FlowPanel extends Toggle {
Svg.back_svg().SetStyle("transform: rotate(180deg);"),
isConfirm ? t.confirm : t.next
).onClick(() => {
- const v = initial.Value.data;
- nextStep.setData(constructNextstep(v, backButtonForNextStep))
- currentStepActive.setData(false)
+ try {
+
+ const v = initial.Value.data;
+ nextStep.setData(constructNextstep(v, backButtonForNextStep))
+ currentStepActive.setData(false)
+ } catch (e) {
+ console.error(e)
+ isError.setData(true)
+ }
}),
"Select a valid value to continue",
initial.IsValid
- )
- ]).SetClass("flex w-full justify-end space-x-2")
+ ),
+ new Toggle(
+ new FixedUiElement("Something went wrong...").SetClass("alert"),
+ undefined,
+ isError),
+ ]).SetClass("flex w-full justify-end space-x-2"),
+
]
}
diff --git a/UI/ImportFlow/ImportHelperGui.ts b/UI/ImportFlow/ImportHelperGui.ts
index 1faa95ac8..5fbd7f2b4 100644
--- a/UI/ImportFlow/ImportHelperGui.ts
+++ b/UI/ImportFlow/ImportHelperGui.ts
@@ -1,17 +1,14 @@
import Combine from "../Base/Combine";
-import {LoginToggle} from "../Popup/LoginButton";
import Toggle from "../Input/Toggle";
import LanguagePicker from "../LanguagePicker";
import BackToIndex from "../BigComponents/BackToIndex";
import UserRelatedState from "../../Logic/State/UserRelatedState";
import BaseUIElement from "../BaseUIElement";
-import MoreScreen from "../BigComponents/MoreScreen";
import MinimapImplementation from "../Base/MinimapImplementation";
import Translations from "../i18n/Translations";
-import Constants from "../../Models/Constants";
import {FlowPanelFactory} from "./FlowStep";
import {RequestFile} from "./RequestFile";
-import {DataPanel} from "./DataPanel";
+import {PreviewPanel} from "./PreviewPanel";
import ConflationChecker from "./ConflationChecker";
import {AskMetadata} from "./AskMetadata";
import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
@@ -21,8 +18,11 @@ import {FixedUiElement} from "../Base/FixedUiElement";
import {VariableUiElement} from "../Base/VariableUIElement";
import List from "../Base/List";
import {CompareToAlreadyExistingNotes} from "./CompareToAlreadyExistingNotes";
+import Introdution from "./Introdution";
+import LoginToImport from "./LoginToImport";
+import {MapPreview} from "./MapPreview";
-export default class ImportHelperGui extends LoginToggle {
+export default class ImportHelperGui extends Combine {
constructor() {
const t = Translations.t.importHelper;
@@ -33,8 +33,11 @@ export default class ImportHelperGui extends LoginToggle {
const {flow, furthestStep, titles} =
FlowPanelFactory
- .start("Select file", new RequestFile())
- .then("Inspect data", geojson => new DataPanel(state, geojson))
+ .start("Introduction", new Introdution())
+ .then("Login", _ => new LoginToImport(state))
+ .then("Select file", _ => new RequestFile())
+ .then("Inspect attributes", geojson => new PreviewPanel(state, geojson))
+ .then("Inspect data", geojson => new MapPreview(state, geojson))
.then("Compare with open notes", v => new CompareToAlreadyExistingNotes(state, v))
.then("Compare with existing data", v => new ConflationChecker(state, v))
.then("License and community check", v => new ConfirmProcess(v))
@@ -71,24 +74,12 @@ export default class ImportHelperGui extends LoginToggle {
- super(
- new Toggle(
+ super([
new Combine([
leftBar,
flow.SetClass("m-8 w-full mb-24")
- ]).SetClass("h-full block md:flex")
+ ]).SetClass("h-full block md:flex")])
- ,
- new Combine([
- t.lockNotice.Subs(Constants.userJourney),
- MoreScreen.CreateProffessionalSerivesButton()
- ])
-
- ,
- state.osmConnection.userDetails.map(ud => ud.csCount >= Constants.userJourney.importHelperUnlock)),
-
- "Login needed...",
- state)
}
}
diff --git a/UI/ImportFlow/ImportInspector.ts b/UI/ImportFlow/ImportInspector.ts
new file mode 100644
index 000000000..075e2fab2
--- /dev/null
+++ b/UI/ImportFlow/ImportInspector.ts
@@ -0,0 +1 @@
+export default class ImportInspector {}
\ No newline at end of file
diff --git a/UI/ImportFlow/ImportUtils.ts b/UI/ImportFlow/ImportUtils.ts
index 37296dd29..be8957d01 100644
--- a/UI/ImportFlow/ImportUtils.ts
+++ b/UI/ImportFlow/ImportUtils.ts
@@ -7,9 +7,11 @@ export class ImportUtils {
if (osmData?.features === undefined) {
return undefined
}
+ if(osmData.features.length === 0){
+ return {noNearby: toPartitionFeatureCollection.features, hasNearby: []}
+ }
const maxDist = cutoffDistanceInMeters.data
-
const hasNearby = []
const noNearby = []
for (const toImportElement of toPartitionFeatureCollection.features) {
diff --git a/UI/ImportFlow/Introdution.ts b/UI/ImportFlow/Introdution.ts
new file mode 100644
index 000000000..602f87af0
--- /dev/null
+++ b/UI/ImportFlow/Introdution.ts
@@ -0,0 +1,20 @@
+import Combine from "../Base/Combine";
+import {FlowStep} from "./FlowStep";
+import {UIEventSource} from "../../Logic/UIEventSource";
+import Translations from "../i18n/Translations";
+import Title from "../Base/Title";
+
+export default class Introdution extends Combine implements FlowStep {
+ readonly IsValid: UIEventSource = new UIEventSource(true);
+ readonly Value: UIEventSource = new UIEventSource(undefined);
+
+ constructor() {
+ super([
+ new Title( Translations.t.importHelper.title),
+ Translations.t.importHelper.description,
+ Translations.t.importHelper.importFormat,
+ ]);
+ this.SetClass("flex flex-col")
+ }
+
+}
\ No newline at end of file
diff --git a/UI/ImportFlow/LoginToImport.ts b/UI/ImportFlow/LoginToImport.ts
new file mode 100644
index 000000000..141df5735
--- /dev/null
+++ b/UI/ImportFlow/LoginToImport.ts
@@ -0,0 +1,52 @@
+import Combine from "../Base/Combine";
+import {FlowStep} from "./FlowStep";
+import UserRelatedState from "../../Logic/State/UserRelatedState";
+import {UIEventSource} from "../../Logic/UIEventSource";
+import Translations from "../i18n/Translations";
+import Title from "../Base/Title";
+import {VariableUiElement} from "../Base/VariableUIElement";
+import {LoginToggle} from "../Popup/LoginButton";
+import Img from "../Base/Img";
+import Constants from "../../Models/Constants";
+import Toggle from "../Input/Toggle";
+import {SubtleButton} from "../Base/SubtleButton";
+import Svg from "../../Svg";
+import MoreScreen from "../BigComponents/MoreScreen";
+
+export default class LoginToImport extends Combine implements FlowStep {
+ readonly IsValid: UIEventSource;
+ readonly Value: UIEventSource;
+
+ constructor(state: UserRelatedState) {
+ const t = Translations.t.importHelper
+ const isValid = state.osmConnection.userDetails.map(ud => ud.csCount >= Constants.userJourney.importHelperUnlock)
+ super([
+ new Title(t.userAccountTitle),
+ new LoginToggle(
+ new VariableUiElement(state.osmConnection.userDetails.map(ud => {
+ if (ud === undefined) {
+ return undefined
+ }
+ return new Combine([
+ new Img(ud.img ?? "./assets/svgs/help.svg").SetClass("w-16 h-16 rounded-full"),
+ t.loggedInWith.Subs(ud),
+ new SubtleButton(Svg.logout_svg().SetClass("h-8"), Translations.t.general.logout)
+ .onClick(() => state.osmConnection.LogOut())
+ ]);
+ })),
+ new Combine([t.loginRequired,
+ MoreScreen.CreateProffessionalSerivesButton()
+ ]),
+ state
+ ),
+ new Toggle(undefined,
+ new Combine(
+ [t.lockNotice.Subs(Constants.userJourney).SetClass("alert"),
+ MoreScreen.CreateProffessionalSerivesButton()])
+
+ , isValid)
+ ])
+ this.Value = new UIEventSource(state)
+ this.IsValid = isValid;
+ }
+}
\ No newline at end of file
diff --git a/UI/ImportFlow/DataPanel.ts b/UI/ImportFlow/MapPreview.ts
similarity index 78%
rename from UI/ImportFlow/DataPanel.ts
rename to UI/ImportFlow/MapPreview.ts
index 51f7cde23..3e66095d8 100644
--- a/UI/ImportFlow/DataPanel.ts
+++ b/UI/ImportFlow/MapPreview.ts
@@ -17,13 +17,13 @@ import ShowDataMultiLayer from "../ShowDataLayer/ShowDataMultiLayer";
import FilteredLayer, {FilterState} from "../../Models/FilteredLayer";
import StaticFeatureSource from "../../Logic/FeatureSource/Sources/StaticFeatureSource";
import Toggle from "../Input/Toggle";
-import Table from "../Base/Table";
import {VariableUiElement} from "../Base/VariableUIElement";
import {FixedUiElement} from "../Base/FixedUiElement";
import {FlowStep} from "./FlowStep";
import ScrollableFullScreen from "../Base/ScrollableFullScreen";
import {AllTagsPanel} from "../SpecialVisualizations";
import Title from "../Base/Title";
+import CheckBoxes from "../Input/Checkboxes";
class PreviewPanel extends ScrollableFullScreen {
@@ -42,14 +42,14 @@ class PreviewPanel extends ScrollableFullScreen {
/**
* Shows the data to import on a map, asks for the correct layer to be selected
*/
-export class DataPanel extends Combine implements FlowStep<{ bbox: BBox, layer: LayerConfig, geojson: any }>{
+export class MapPreview extends Combine implements FlowStep<{ bbox: BBox, layer: LayerConfig, geojson: any }>{
public readonly IsValid: UIEventSource;
public readonly Value: UIEventSource<{ bbox: BBox, layer: LayerConfig, geojson: any }>
constructor(
state: UserRelatedState,
geojson: { features: { properties: any, geometry: { coordinates: [number, number] } }[] }) {
- const t = Translations.t.importHelper;
+ const t = Translations.t.importHelper.mapPreview;
const propertyKeys = new Set()
for (const f of geojson.features) {
@@ -58,7 +58,7 @@ export class DataPanel extends Combine implements FlowStep<{ bbox: BBox, layer:
const availableLayers = AllKnownLayouts.AllPublicLayers().filter(l => l.name !== undefined && Constants.priviliged_layers.indexOf(l.id) < 0)
- const layerPicker = new DropDown("Which layer does this import match with?",
+ const layerPicker = new DropDown(t.selectLayer,
[{shown: t.selectLayer, value: undefined}].concat(availableLayers.map(l => ({
shown: l.name,
value: l
@@ -126,29 +126,28 @@ export class DataPanel extends Combine implements FlowStep<{ bbox: BBox, layer:
})
var bbox = matching.map(feats => BBox.bboxAroundAll(feats.map(f => new BBox([f.geometry.coordinates]))))
+
+ const mismatchIndicator = new VariableUiElement(matching.map(matching => {
+ if (matching === undefined) {
+ return undefined
+ }
+ const diff = geojson.features.length - matching.length;
+ if (diff === 0) {
+ return undefined
+ }
+ const obligatory = layerPicker.GetValue().data?.source?.osmTags?.asHumanString(false, false, {});
+ return t.mismatch.Subs({count: diff, tags: obligatory}).SetClass("alert")
+ }))
+
+ const confirm = new CheckBoxes([t.confirm]);
super([
- new Title(geojson.features.length + " features to import"),
+ new Title(t.title, 1),
layerPicker,
- new Toggle("Automatically detected layer", undefined, autodetected),
- new Table(["", "Key", "Values", "Unique values seen"],
- Array.from(propertyKeys).map(key => {
- const uniqueValues = Utils.Dedup(Utils.NoNull(geojson.features.map(f => f.properties[key])))
- uniqueValues.sort()
- return [geojson.features.filter(f => f.properties[key] !== undefined).length + "", key, uniqueValues.join(", "), "" + uniqueValues.length]
- })
- ).SetClass("zebra-table table-auto"),
- new VariableUiElement(matching.map(matching => {
- if (matching === undefined) {
- return undefined
- }
- const diff = geojson.features.length - matching.length;
- if (diff === 0) {
- return undefined
- }
- const obligatory = layerPicker.GetValue().data?.source?.osmTags?.asHumanString(false, false, {});
- return new FixedUiElement(`${diff} features will _not_ match this layer. Make sure that all obligatory objects are present: ${obligatory}`).SetClass("alert");
- })),
- map
+ new Toggle(t.autodetected.SetClass("thank"), undefined, autodetected),
+
+ mismatchIndicator ,
+ map,
+ confirm
]);
this.Value = bbox.map(bbox =>
@@ -157,13 +156,17 @@ export class DataPanel extends Combine implements FlowStep<{ bbox: BBox, layer:
geojson,
layer: layerPicker.GetValue().data
}), [layerPicker.GetValue()])
+
this.IsValid = matching.map(matching => {
if (matching === undefined) {
return false
}
+ if(confirm.GetValue().data.length !== 1){
+ return false
+ }
const diff = geojson.features.length - matching.length;
return diff === 0;
- })
+ }, [confirm.GetValue()])
}
}
\ No newline at end of file
diff --git a/UI/ImportFlow/PreviewPanel.ts b/UI/ImportFlow/PreviewPanel.ts
new file mode 100644
index 000000000..352fda8cd
--- /dev/null
+++ b/UI/ImportFlow/PreviewPanel.ts
@@ -0,0 +1,98 @@
+import Combine from "../Base/Combine";
+import {UIEventSource} from "../../Logic/UIEventSource";
+import UserRelatedState from "../../Logic/State/UserRelatedState";
+import Translations from "../i18n/Translations";
+import {Utils} from "../../Utils";
+import {FlowStep} from "./FlowStep";
+import Title from "../Base/Title";
+import BaseUIElement from "../BaseUIElement";
+import Histogram from "../BigComponents/Histogram";
+import Toggleable from "../Base/Toggleable";
+import List from "../Base/List";
+import CheckBoxes from "../Input/Checkboxes";
+
+/**
+ * Shows the data to import on a map, asks for the correct layer to be selected
+ */
+export class PreviewPanel extends Combine implements FlowStep<{ features: { properties: any, geometry: { coordinates: [number, number] } }[] }>{
+ public readonly IsValid: UIEventSource;
+ public readonly Value: UIEventSource< { features: { properties: any, geometry: { coordinates: [number, number] } }[] }>
+
+ constructor(
+ state: UserRelatedState,
+ geojson: { features: { properties: any, geometry: { coordinates: [number, number] } }[] }) {
+ const t = Translations.t.importHelper;
+ console.log("Datapanel received", geojson)
+
+
+ const propertyKeys = new Set()
+ for (const f of geojson.features) {
+ Object.keys(f.properties).forEach(key => propertyKeys.add(key))
+ }
+
+ const attributeOverview : BaseUIElement[] = []
+
+ const n = geojson.features.length;
+ for (const key of Array.from(propertyKeys)) {
+
+ const values = Utils.NoNull(geojson.features.map(f => f.properties[key]))
+ const allSame = !values.some(v => v !== values[0])
+ if(allSame){
+ attributeOverview.push(new Title(key+"="+values[0]))
+ if(values.length === n){
+ attributeOverview.push(t.allAttributesSame)
+ }else{
+ attributeOverview.push(t.someHaveSame.Subs({count: values.length, percentage: Math.floor(100 * values.length / n)}))
+ }
+ continue
+ }
+
+ const uniqueCount = new Set(values).size
+ if(uniqueCount !== values.length){
+ attributeOverview.push()
+ // There are some overlapping values: histogram time!
+ let hist : BaseUIElement = new Histogram(
+ new UIEventSource(values),
+ "Value",
+ "Occurence",
+ {
+ sortMode: "count-rev"
+ }
+
+ )
+
+ const title = new Title(key+"=*")
+ if(uniqueCount > 15){
+ hist = new Toggleable(title,
+ hist.SetClass("block")
+ ).Collapse()
+
+ }else{
+ attributeOverview.push(title)
+ }
+
+ attributeOverview.push(hist)
+ continue
+ }
+
+ // All values are different, we add a boring (but collapsable) list
+ attributeOverview.push(new Toggleable(
+ new Title(key+"=*"),
+ new List(values)
+ ) )
+
+ }
+
+ const confirm = new CheckBoxes([t.inspectLooksCorrect])
+
+ super([
+ new Title(t.inspectDataTitle.Subs({count:geojson.features.length })),
+ ...attributeOverview,
+ confirm
+ ]);
+
+ this.Value = new UIEventSource<{features: {properties: any; geometry: {coordinates: [number, number]}}[]}>(geojson)
+ this.IsValid = confirm.GetValue().map(selected => selected.length == 1)
+
+ }
+}
\ No newline at end of file
diff --git a/UI/ImportFlow/RequestFile.ts b/UI/ImportFlow/RequestFile.ts
index 2f8ea2861..58bfbe4e5 100644
--- a/UI/ImportFlow/RequestFile.ts
+++ b/UI/ImportFlow/RequestFile.ts
@@ -9,6 +9,7 @@ import InputElementMap from "../Input/InputElementMap";
import BaseUIElement from "../BaseUIElement";
import FileSelectorButton from "../Input/FileSelectorButton";
import {FlowStep} from "./FlowStep";
+import { parse } from "papaparse";
class FileSelector extends InputElementMap }> {
constructor(label: BaseUIElement) {
@@ -42,8 +43,8 @@ export class RequestFile extends Combine implements FlowStep {
public readonly Value: UIEventSource
constructor() {
- const t = Translations.t.importHelper;
- const csvSelector = new FileSelector(new SubtleButton(undefined, t.selectFile))
+ const t = Translations.t.importHelper.selectFile;
+ const csvSelector = new FileSelector(new SubtleButton(undefined, t.description))
const loadedFiles = new VariableUiElement(csvSelector.GetValue().map(file => {
if (file === undefined) {
return t.noFilesLoaded.SetClass("alert")
@@ -59,39 +60,53 @@ export class RequestFile extends Combine implements FlowStep {
return UIEventSource.FromPromise(v.contents)
}))
- const asGeoJson: UIEventSource = text.map(src => {
+ const asGeoJson: UIEventSource = text.map(src => {
if (src === undefined) {
return undefined
}
try {
const parsed = JSON.parse(src)
if (parsed["type"] !== "FeatureCollection") {
- return {error: "The loaded JSON-file is not a geojson-featurecollection"}
+ return {error: t.errNotFeatureCollection}
}
if (parsed.features.some(f => f.geometry.type != "Point")) {
- return {error: "The loaded JSON-file should only contain points"}
+ return {error: t.errPointsOnly}
}
return parsed;
} catch (e) {
// Loading as CSV
- const lines = src.split("\n")
- const header = lines[0].split(",")
+ var lines : string[][] = parse(src).data;
+ const header = lines[0]
lines.splice(0, 1)
if (header.indexOf("lat") < 0 || header.indexOf("lon") < 0) {
- return {error: "The header does not contain `lat` or `lon`"}
+ return {error: t.errNoLatOrLon}
}
+ if (header.some(h => h.trim() == "")) {
+ return {error:t.errNoName}
+ }
+
+
+ if (new Set(header).size !== header.length) {
+ return {error:t.errDuplicate}
+ }
+
+
const features = []
for (let i = 0; i < lines.length; i++) {
- const line = lines[i];
- if (line.trim() === "") {
+ const attrs = lines[i];
+ if(attrs.length == 0 || (attrs.length == 1 && attrs[0] == "")){
+ // empty line
continue
}
- const attrs = line.split(",")
const properties = {}
for (let i = 0; i < header.length; i++) {
- properties[header[i]] = attrs[i];
+ const v = attrs[i]
+ if(v === undefined || v === ""){
+ continue
+ }
+ properties[header[i]] = v;
}
const coordinates = [Number(properties["lon"]), Number(properties["lat"])]
delete properties["lat"]
@@ -125,18 +140,21 @@ export class RequestFile extends Combine implements FlowStep {
if (v?.error === undefined) {
return undefined;
}
- return new FixedUiElement(v?.error).SetClass("alert");
+ return v.error.Clone().SetClass("alert");
}))
super([
new Title(t.title, 1),
- t.description,
+ t.fileFormatDescription,
+ t.fileFormatDescriptionCsv,
+ t.fileFormatDescriptionGeoJson,
csvSelector,
loadedFiles,
errorIndicator
]);
+ this.SetClass("flex flex-col wi")
this.IsValid = asGeoJson.map(geojson => geojson !== undefined && geojson["error"] === undefined)
this.Value = asGeoJson
}
diff --git a/UI/SpecialVisualizations.ts b/UI/SpecialVisualizations.ts
index d7d83b060..1307d9212 100644
--- a/UI/SpecialVisualizations.ts
+++ b/UI/SpecialVisualizations.ts
@@ -455,7 +455,7 @@ export default class SpecialVisualizations {
return undefined;
}
})
- return new Histogram(listSource, args[1], args[2], assignColors)
+ return new Histogram(listSource, args[1], args[2], {assignColor: assignColors})
}
},
{
diff --git a/langs/en.json b/langs/en.json
index a5a111ed1..f98dfa695 100644
--- a/langs/en.json
+++ b/langs/en.json
@@ -1,483 +1,513 @@
{
- "image": {
- "addPicture": "Add picture",
- "uploadingPicture": "Uploading your picture…",
- "uploadingMultiple": "Uploading {count} pictures…",
- "pleaseLogin": "Please log in to add a picture",
- "willBePublished": "Your picture will be published ",
- "cco": "in the public domain",
- "ccbs": "under the CC-BY-SA-license",
- "ccb": "under the CC-BY-license",
- "ccoExplanation": "Adding a picture in the public domain implies that anyone can do anything with your picture",
- "ccbsExplanation": "The CC-BY-SA license implies that anyone may use your picture for any purpose, but they have to attribute you and remixes of the picture have to be republished under the same license",
- "ccbExplanation": "The CC-BY license implies that anyone may use your picture for any purpose, but they have to attribute you",
- "uploadFailed": "Could not upload your picture. Are you connected to the Internet, and allow third party API's? The Brave browser or the uMatrix plugin might block them.",
- "respectPrivacy": "Do not photograph people nor license plates. Do not upload Google Maps, Google Streetview or other copyrighted sources.",
- "uploadDone": "Your picture has been added. Thanks for helping out!",
- "uploadMultipleDone": "{count} pictures have been added. Thanks for helping out!",
- "dontDelete": "Cancel",
- "doDelete": "Remove image",
- "isDeleted": "Deleted",
- "toBig": "Your image is too large as it is {actual_size}. Please use images of at most {max_size}"
+ "image": {
+ "addPicture": "Add picture",
+ "uploadingPicture": "Uploading your picture…",
+ "uploadingMultiple": "Uploading {count} pictures…",
+ "pleaseLogin": "Please log in to add a picture",
+ "willBePublished": "Your picture will be published ",
+ "cco": "in the public domain",
+ "ccbs": "under the CC-BY-SA-license",
+ "ccb": "under the CC-BY-license",
+ "ccoExplanation": "Adding a picture in the public domain implies that anyone can do anything with your picture",
+ "ccbsExplanation": "The CC-BY-SA license implies that anyone may use your picture for any purpose, but they have to attribute you and remixes of the picture have to be republished under the same license",
+ "ccbExplanation": "The CC-BY license implies that anyone may use your picture for any purpose, but they have to attribute you",
+ "uploadFailed": "Could not upload your picture. Are you connected to the Internet, and allow third party API's? The Brave browser or the uMatrix plugin might block them.",
+ "respectPrivacy": "Do not photograph people nor license plates. Do not upload Google Maps, Google Streetview or other copyrighted sources.",
+ "uploadDone": "Your picture has been added. Thanks for helping out!",
+ "uploadMultipleDone": "{count} pictures have been added. Thanks for helping out!",
+ "dontDelete": "Cancel",
+ "doDelete": "Remove image",
+ "isDeleted": "Deleted",
+ "toBig": "Your image is too large as it is {actual_size}. Please use images of at most {max_size}"
+ },
+ "centerMessage": {
+ "loadingData": "Loading data…",
+ "zoomIn": "Zoom in to view or edit the data",
+ "ready": "Done!",
+ "retrying": "Loading data failed. Trying again in {count} seconds…"
+ },
+ "index": {
+ "#": "These texts are shown above the theme buttons when no theme is loaded",
+ "title": "Welcome to MapComplete",
+ "featuredThemeTitle": "Featured this week",
+ "intro": "MapComplete is an OpenStreetMap-viewer and editor, which shows you information about features of a specific theme and allows to update it.",
+ "pickTheme": "Pick a theme below to get started.",
+ "logIn": "Log in to see other themes you previously visited"
+ },
+ "split": {
+ "split": "Split",
+ "cancel": "Cancel",
+ "inviteToSplit": "Split this road in smaller segments. This allows to give different properties to parts of the road.",
+ "loginToSplit": "You must be logged in to split a road",
+ "splitTitle": "Choose on the map where to split this road",
+ "hasBeenSplit": "This way has been split"
+ },
+ "delete": {
+ "delete": "Delete",
+ "cancel": "Cancel",
+ "isDeleted": "This feature is deleted",
+ "cannotBeDeleted": "This feature can not be deleted",
+ "loginToDelete": "You must be logged in to delete a point",
+ "safeDelete": "This point can be safely deleted.",
+ "isntAPoint": "Only points can be deleted, the selected feature is a way, area or relation.",
+ "onlyEditedByLoggedInUser": "This point has only be edited by yourself, you can safely delete it.",
+ "notEnoughExperience": "This point was made by someone else.",
+ "useSomethingElse": "Use another OpenStreetMap-editor to delete it instead",
+ "partOfOthers": "This point is part of some way or relation and can not be deleted directly.",
+ "loading": "Inspecting properties to check if this feature can be deleted.",
+ "whyDelete": "Why should this point be deleted?",
+ "reasons": {
+ "test": "This was a testing point - the feature was never actually there",
+ "disused": "This feature is disused or removed",
+ "notFound": "This feature couldn't be found",
+ "duplicate": "This point is a duplicate of another feature"
},
- "centerMessage": {
- "loadingData": "Loading data…",
- "zoomIn": "Zoom in to view or edit the data",
- "ready": "Done!",
- "retrying": "Loading data failed. Trying again in {count} seconds…"
+ "explanations": {
+ "selectReason": "Please, select why this feature should be deleted",
+ "hardDelete": "This point will be deleted in OpenStreetMap. It can be recovered by an experienced contributor",
+ "softDelete": "This feature will be updated and hidden from this application. {reason}"
},
- "index": {
- "#": "These texts are shown above the theme buttons when no theme is loaded",
- "title": "Welcome to MapComplete",
- "featuredThemeTitle": "Featured this week",
- "intro": "MapComplete is an OpenStreetMap-viewer and editor, which shows you information about features of a specific theme and allows to update it.",
- "pickTheme": "Pick a theme below to get started.",
- "logIn": "Log in to see other themes you previously visited"
+ "readMessages": "You have unread messages. Read these before deleting a point - someone might have feedback"
+ },
+ "general": {
+ "logout": "Log out",
+ "next": "Next",
+ "confirm": "Confirm",
+ "back": "Back",
+ "backToMapcomplete": "Back to the theme overview",
+ "loading": "Loading...",
+ "pdf": {
+ "generatedWith": "Generated with MapComplete.osm.be",
+ "attr": "Map data © OpenStreetMap Contributors, reusable under ODbL",
+ "attrBackground": "Background layer: {background}",
+ "versionInfo": "v{version} - generated on {date}"
},
- "split": {
- "split": "Split",
- "cancel": "Cancel",
- "inviteToSplit": "Split this road in smaller segments. This allows to give different properties to parts of the road.",
- "loginToSplit": "You must be logged in to split a road",
- "splitTitle": "Choose on the map where to split this road",
- "hasBeenSplit": "This way has been split"
+ "loginWithOpenStreetMap": "Login with OpenStreetMap",
+ "welcomeBack": "You are logged in, welcome back!",
+ "loginToStart": "Log in to answer this question",
+ "openStreetMapIntro": "An Open Map
One that everyone can use and edit freely. A single place to store all geo-info. Different, small, incompatible and outdated maps are not needed anywhere.
OpenStreetMap is not the enemy map. The map data can be used freely (with attribution and publication of changes to that data). Everyone can add new data and fix errors. This website uses OpenStreetMap. All the data is from there, and your answers and corrections are used all over.
Many people and apps already use OpenStreetMap: Organic Maps, OsmAnd, but also the maps at Facebook, Instagram, Apple-maps and Bing-maps are (partly) powered by OpenStreetMap.
",
+ "search": {
+ "search": "Search a location",
+ "searching": "Searching…",
+ "nothing": "Nothing found…",
+ "error": "Something went wrong…"
},
- "delete": {
- "delete": "Delete",
- "cancel": "Cancel",
- "isDeleted": "This feature is deleted",
- "cannotBeDeleted": "This feature can not be deleted",
- "loginToDelete": "You must be logged in to delete a point",
- "safeDelete": "This point can be safely deleted.",
- "isntAPoint": "Only points can be deleted, the selected feature is a way, area or relation.",
- "onlyEditedByLoggedInUser": "This point has only be edited by yourself, you can safely delete it.",
- "notEnoughExperience": "This point was made by someone else.",
- "useSomethingElse": "Use another OpenStreetMap-editor to delete it instead",
- "partOfOthers": "This point is part of some way or relation and can not be deleted directly.",
- "loading": "Inspecting properties to check if this feature can be deleted.",
- "whyDelete": "Why should this point be deleted?",
- "reasons": {
- "test": "This was a testing point - the feature was never actually there",
- "disused": "This feature is disused or removed",
- "notFound": "This feature couldn't be found",
- "duplicate": "This point is a duplicate of another feature"
- },
- "explanations": {
- "selectReason": "Please, select why this feature should be deleted",
- "hardDelete": "This point will be deleted in OpenStreetMap. It can be recovered by an experienced contributor",
- "softDelete": "This feature will be updated and hidden from this application. {reason}"
- },
- "readMessages": "You have unread messages. Read these before deleting a point - someone might have feedback"
+ "returnToTheMap": "Return to the map",
+ "save": "Save",
+ "cancel": "Cancel",
+ "skip": "Skip this question",
+ "oneSkippedQuestion": "One question is skipped",
+ "skippedQuestions": "Some questions are skipped",
+ "number": "number",
+ "osmLinkTooltip": "Browse this object on OpenStreetMap for history and more editing options",
+ "add": {
+ "addNewMapLabel": "Click here to add a new item",
+ "disableFiltersExplanation": "Some features might be hidden by a filter",
+ "disableFilters": "Disable all filters",
+ "addNew": "Add a new {category} here",
+ "presetInfo": "The new POI will have {tags}",
+ "warnVisibleForEveryone": "Your addition will be visible for everyone",
+ "title": "Add a new point?",
+ "intro": "You clicked somewhere where no data is known yet.
",
+ "pleaseLogin": "Please log in to add a new point",
+ "zoomInFurther": "Zoom in further to add a point.",
+ "stillLoading": "The data is still loading. Please wait a bit before you add a new point.",
+ "confirmIntro": "Add a {title} here?
The point you create here will be visible for everyone. Please, only add things on to the map if they truly exist. A lot of applications use this data.",
+ "confirmButton": "Add a {category} here.
Your addition is visible for everyone
",
+ "openLayerControl": "Open the layer control box",
+ "layerNotEnabled": "The layer {layer} is not enabled. Enable this layer to add a point",
+ "hasBeenImported": "This point has already been imported",
+ "importTags": "The element will receive {tags}",
+ "zoomInMore": "Zoom in more to import this feature",
+ "wrongType": "This element is not a point or a way and can not be imported",
+ "import": {
+ "officialThemesOnly": "The import button is disabled for unofficial themes to prevent accidents",
+ "howToTest": "To test, add test=true or backend=osm-test to the URL. The changeset will be printed in the console. Please open a PR to officialize this theme to actually enable the import button.",
+ "hasBeenImported": "This object has been imported",
+ "importTags": "The element will receive {tags}",
+ "zoomInMore": "Zoom in more to import this feature",
+ "wrongType": "This element is not a point or a way and can not be imported"
+ }
},
- "general": {
- "next": "Next",
- "confirm": "Confirm",
- "back": "Back",
- "backToMapcomplete": "Back to the theme overview",
- "loading": "Loading...",
- "pdf": {
- "generatedWith": "Generated with MapComplete.osm.be",
- "attr": "Map data © OpenStreetMap Contributors, reusable under ODbL",
- "attrBackground": "Background layer: {background}",
- "versionInfo": "v{version} - generated on {date}"
+ "pickLanguage": "Choose a language: ",
+ "about": "Easily edit and add OpenStreetMap for a certain theme",
+ "nameInlineQuestion": "The name of this {category} is $$$",
+ "noNameCategory": "{category} without a name",
+ "questions": {
+ "phoneNumberOf": "What is the phone number of {category}?",
+ "phoneNumberIs": "The phone number of this {category} is {phone}",
+ "websiteOf": "What is the website of {category}?",
+ "websiteIs": "Website: {website}",
+ "emailOf": "What is the email address of {category}?",
+ "emailIs": "The email address of this {category} is {email}"
+ },
+ "morescreen": {
+ "intro": "More thematic maps?
Do you enjoy collecting geodata?
There are more themes available.",
+ "requestATheme": "If you want a custom-built theme, request it in the issue tracker",
+ "streetcomplete": "Another, similar application is StreetComplete.",
+ "createYourOwnTheme": "Create your own MapComplete theme from scratch",
+ "previouslyHiddenTitle": "Previously visited hidden themes",
+ "hiddenExplanation": "These themes are only accessible to those with the link. You have discovered {hidden_discovered} of {total_hidden} hidden themes."
+ },
+ "sharescreen": {
+ "intro": "Share this map
Share this map by copying the link below and sending it to friends and family:",
+ "addToHomeScreen": "Add to your home screen
You can easily add this website to your smartphone home screen for a native feel. Click the 'Add to home screen' button in the URL bar to do this.",
+ "embedIntro": "Embed on your website
Please, embed this map into your website.
We encourage you to do it - you don't even have to ask permission.
It is free, and always will be. The more people are using this, the more valuable it becomes.",
+ "copiedToClipboard": "Link copied to clipboard",
+ "thanksForSharing": "Thanks for sharing!",
+ "editThisTheme": "Edit this theme",
+ "editThemeDescription": "Add or change questions to this map theme",
+ "fsUserbadge": "Enable the login button",
+ "fsSearch": "Enable the search bar",
+ "fsWelcomeMessage": "Show the welcome message popup and associated tabs",
+ "fsLayers": "Enable the layer control",
+ "fsLayerControlToggle": "Start with the layer control expanded",
+ "fsAddNew": "Enable the 'add new POI' button",
+ "fsGeolocation": "Enable the 'geolocate-me' button (mobile only)",
+ "fsIncludeCurrentBackgroundMap": "Include the current background choice {name}",
+ "fsIncludeCurrentLayers": "Include the current layer choices",
+ "fsIncludeCurrentLocation": "Include current location"
+ },
+ "attribution": {
+ "attributionTitle": "Attribution notice",
+ "attributionContent": "All data is provided by OpenStreetMap, freely reusable under the Open DataBase License.
",
+ "themeBy": "Theme maintained by {author}",
+ "iconAttribution": {
+ "title": "Used icons"
+ },
+ "mapContributionsBy": "The current visible data has edits made by {contributors}",
+ "mapContributionsByAndHidden": "The current visible data has edits made by {contributors} and {hiddenCount} more contributors",
+ "codeContributionsBy": "MapComplete has been built by {contributors} and {hiddenCount} more contributors",
+ "openOsmcha": "See latest edits made with {theme}",
+ "openMapillary": "Open Mapillary here",
+ "openIssueTracker": "File a bug",
+ "josmOpened": "JOSM is opened",
+ "josmNotOpened": "JOSM could not be reached. Make sure it is opened and remote control is enabled",
+ "editJosm": "Edit here with JOSM",
+ "editId": "Open the OpenStreetMap online editor here",
+ "donate": "Support MapComplete financially"
+ },
+ "readYourMessages": "Please, read all your OpenStreetMap-messages before adding a new point.",
+ "fewChangesBefore": "Please, answer a few questions of existing points before adding a new point.",
+ "goToInbox": "Open inbox",
+ "removeLocationHistory": "Delete the location history",
+ "getStartedLogin": "Log in with OpenStreetMap to get started",
+ "getStartedNewAccount": " or create a new account",
+ "noTagsSelected": "No tags selected",
+ "testing": "Testing - changes won't be saved",
+ "customThemeIntro": "Custom themes
These are previously visited user-generated themes.",
+ "aboutMapcomplete": "About MapComplete
Use it to add OpenStreetMap info on a single theme. Answer questions, and within minutes your contributions are available everywhere. The theme maintainer defines elements, questions and languages for it.
Find out more
MapComplete always offers the next step to learn more about OpenStreetMap.
- When embedded in a website, the iframe links to a full-screen MapComplete
- The fullscreen version offers info about OpenStreetMap
- Viewing works without login, but editing requires an OSM account.
- If you are not logged in, you are asked to do so
- Once you answered a single question, you can add new points to the map
- After a while, actual OSM-tags are shown, later linking to the wiki
Did you notice an issue? Do you have a feature request? Want to help translate? Head over to the source code or issue tracker.
Want to see your progress? Follow the edit count on OsmCha.
",
+ "backgroundMap": "Background map",
+ "openTheMap": "Open the map",
+ "loginOnlyNeededToEdit": "if you want to edit the map",
+ "layerSelection": {
+ "zoomInToSeeThisLayer": "Zoom in to see this layer",
+ "title": "Select layers"
+ },
+ "download": {
+ "title": "Download visible data",
+ "downloadAsPdf": "Download a PDF of the current map",
+ "downloadAsPdfHelper": "Ideal to print the current map",
+ "downloadGeojson": "Download visible data as GeoJSON",
+ "downloadGpx": "Download as GPX-file",
+ "downloadGpxHelper": "A GPX-file can be used with most navigation devices and applications",
+ "uploadGpx": "Upload your track to OpenStreetMap",
+ "exporting": "Exporting…",
+ "downloadGeoJsonHelper": "Compatible with QGIS, ArcGIS, ESRI, …",
+ "downloadCSV": "Download visible data as CSV",
+ "downloadCSVHelper": "Compatible with LibreOffice Calc, Excel, …",
+ "includeMetaData": "Include metadata (last editor, calculated values, …)",
+ "licenseInfo": "Copyright notice
The provided data is available under ODbL. Reusing it is gratis for any purpose, but - the attribution © OpenStreetMap contributors is required
- Any change must be use the license
Please read the full copyright notice for details.",
+ "noDataLoaded": "No data is loaded yet. Download will be available soon",
+ "downloadFeatureAsGpx": "Download as GPX-file",
+ "downloadFeatureAsGeojson": "Download as GeoJson-file"
+ },
+ "weekdays": {
+ "abbreviations": {
+ "monday": "Mon",
+ "tuesday": "Tue",
+ "wednesday": "Wed",
+ "thursday": "Thu",
+ "friday": "Fri",
+ "saturday": "Sat",
+ "sunday": "Sun"
+ },
+ "monday": "Monday",
+ "tuesday": "Tuesday",
+ "wednesday": "Wednesday",
+ "thursday": "Thursday",
+ "friday": "Friday",
+ "saturday": "Saturday",
+ "sunday": "Sunday"
+ },
+ "opening_hours": {
+ "error_loading": "Error: could not visualize these opening hours.",
+ "open_during_ph": "During a public holiday, this is",
+ "opensAt": "from",
+ "openTill": "till",
+ "not_all_rules_parsed": "These opening hours are complicated. The following rules are ignored in the input element:",
+ "closed_until": "Closed until {date}",
+ "closed_permanently": "Closed for an unkown duration",
+ "open_24_7": "Opened around the clock",
+ "ph_not_known": " ",
+ "ph_closed": "closed",
+ "ph_open": "opened",
+ "ph_open_as_usual": "opened as usual",
+ "loadingCountry": "Determining country…"
+ },
+ "histogram": {
+ "error_loading": "Could not load the histogram"
+ },
+ "wikipedia": {
+ "wikipediaboxTitle": "Wikipedia",
+ "failed": "Loading the Wikipedia entry failed",
+ "loading": "Loading Wikipedia...",
+ "noWikipediaPage": "This Wikidata item has no corresponding Wikipedia page yet.",
+ "searchWikidata": "Search on Wikidata",
+ "noResults": "Nothing found for {search}",
+ "doSearch": "Search above to see results",
+ "createNewWikidata": "Create a new Wikidata item"
+ },
+ "apply_button": {
+ "isApplied": "The changes are applied",
+ "appliedOnAnotherObject": "The object {id} will receive {tags}"
+ }
+ },
+ "favourite": {
+ "panelIntro": "Your personal theme
Activate your favourite layers from all the official themes",
+ "loginNeeded": "Log in
A personal layout is only available for OpenStreetMap users",
+ "reload": "Reload the data"
+ },
+ "reviews": {
+ "title": "{count} reviews",
+ "title_singular": "One review",
+ "name_required": "A name is required in order to display and create reviews",
+ "no_reviews_yet": "There are no reviews yet. Be the first to write one and help open data and the business!",
+ "write_a_comment": "Leave a review…",
+ "no_rating": "No rating given",
+ "posting_as": "Posting as",
+ "i_am_affiliated": "I am affiliated with this object
Check if you are an owner, creator, employee, …",
+ "affiliated_reviewer_warning": "(Affiliated review)",
+ "saving_review": "Saving…",
+ "saved": "Review saved. Thanks for sharing!",
+ "tos": "If you create a review, you agree to the TOS and privacy policy of Mangrove.reviews",
+ "attribution": "Reviews are powered by Mangrove Reviews and are available under CC-BY 4.0.",
+ "plz_login": "Log in to leave a review"
+ },
+ "multi_apply": {
+ "autoApply": "When changing the attributes {attr_names}, these attributes will automatically be changed on {count} other objects too"
+ },
+ "move": {
+ "loginToMove": "You must be logged in to move a point",
+ "inviteToMoveAgain": "Move this point again",
+ "moveTitle": "Move this point",
+ "whyMove": "Why do you want to move this point?",
+ "confirmMove": "Move here",
+ "pointIsMoved": "The point has been moved",
+ "zoomInFurther": "Zoom in further to confirm this move",
+ "selectReason": "Why do you move this object?",
+ "reasons": {
+ "reasonRelocation": "The object has been relocated to a totally different location",
+ "reasonInaccurate": "The location of this object is inaccurate and should be moved a few meter"
+ },
+ "inviteToMove": {
+ "generic": "Move this point",
+ "reasonInaccurate": "Improve the accuracy of this point",
+ "reasonRelocation": "Move this object to a another place because it has relocated"
+ },
+ "cannotBeMoved": "This feature cannot be moved.",
+ "isWay": "This feature is a way. Use another OpenStreetMap editor to move it.",
+ "isRelation": "This feature is a relation and can not be moved",
+ "partOfAWay": "This feature is part of another way. Use another editor to move it.",
+ "partOfRelation": "This feature is part of a relation. Use another editor to move it.",
+ "cancel": "Cancel move"
+ },
+ "privacy": {
+ "title": "Privacy policy",
+ "intro": "Privacy is important - for both the individual and for society. MapComplete tries to respect your privacy as much as possible - up to the point no annoying cookie banner is needed. However, we still would like to inform you which information is gathered and shared, under which circumstances and why these trade-offs are made.",
+ "trackingTitle": "Statistical data",
+ "tracking": "To gather some insight in whom visits our website, some technical information is collected. This is included the country you visited the webpage from, which website referred you to MapComplete, the type of your device and the screensize. A cookie is placed on your device to indicate that you visited MapComplete earlier today. This data is not detailed enough to personally identify you. These statistics are only available to anyone in aggregate and are publicly available to anyone",
+ "geodataTitle": "Your geolocation",
+ "geodata": "When MapComplete gets your geolocation, your geolocation and previously visited locations stay on your device. Your location data is never automatically sent to anywhere else - unless some functionality clearly states otherwise.",
+ "editingTitle": "When making changes",
+ "editing": "When you make a change to the map, this change is recorded on OpenStreetMap and is publicly available to anyone. A changeset made with MapComplete includes the following data: - The changes you made
- Your username
- When this change is made
- The theme you used while making the change
- The language of the user interface
- An indication of how close you were to changed objects. Other mappers can use this information to determine if a change was made based on survey or on remote research
Please refer to the privacy policy on OpenStreetMap.org for detailed information. We'd like to remind you that you can use a fictional name when signing up.",
+ "miscCookiesTitle": "Other cookies",
+ "miscCookies": "MapComplete integrates with various other services, especially to load images of features. Images are hosted on various third-party servers, which might set cookies on their own.",
+ "whileYoureHere": "Do you care about privacy?",
+ "surveillance": "As you are reading the privacy policy, you probably care about privacy - so do we! We even made a theme showing surveillance cameras. Feel free to map them all!"
+ },
+ "professional": {
+ "indexPage": {
+ "hook": "Need professional support?",
+ "hookMore": "We can help with setting up surveys, data imports and OpenStreetMap-consultancy",
+ "button": "Discover our services"
+ },
+ "title": "Professional support with MapComplete",
+ "intro": "The developer of MapComplete offers professional support. This document outlines some of the possibilities, common questions and the boundaries of MapComplete",
+ "osmTitle": "What can OpenStreetMap and MapComplete do for your organisation?",
+ "text0": "Maintaining a set of up-to-date geodata is hard, error prone and expensive.
To add insult to injury, many organizations end up collecting the same data independently - resulting in duplicated efforts, non-standardized data formats and many incomplete, unmaintained datasets.
At the same time, there is a huge community which gathers a lot of geodata into one shared, global and standardized database - namely OpenStreetMap.org.
",
+ "text1": "MapComplete is the editor to make contributing data to OpenStreetMap easy.
",
+ "aboutOsm": {
+ "aboutOsm": {
+ "title": "What is OpenStreetMap?",
+ "intro": "OpenStreetMap is a shared, global database, built by volunteers. All geodata can be contributed to OpenStreetMap, as long as it can be verified on the ground.
OpenStreetMap has grown to be a very broad and deep dataset as it contains data over thousands of categories of objects.An individual object might also have a ton of attributes, bringing a lot of nuance, e.g.:",
+ "li0": "Streets have geometry, but might also have information about the maxspeed, surface, wether they are lit, their name, a link to Wikipedia, a link to what they are named after, which hiking-, cycle- and busroutes run over theme",
+ "li1": "Shops and other amenities might have opening hours, a phone number, a link to the website, which payment methods are supported, what they sell, which services they offer, …",
+ "li2": "Toilets might have information about wheelchair accessibility, a changing table, if payment is needed, …",
+ "li3": "and much, much more…"
+ },
+ "benefits": {
+ "title": "Benefits of the OSM-ecosystem",
+ "intro": "It can be very hard to leave your own dataset behind, as building this dataset often took a lot of time and effort.
However, the benefits of switching over to OSM are huge:",
+ "li0": "You are not alone anymore to gather and maintain this dataset - a whole community is at your side",
+ "li1": "Your data will reach a bigger audience then ever via Bing Maps, Apple Maps, Facebook, Instagram, Pokemon Go, OsmAnd, Organic Maps, Maps.me, Mapbox, Komoot, nearly all cycle-applications, …",
+ "li2": "Many governement organisations and municipalities use OpenStreetMap on their websites too"
+ },
+ "license": {
+ "title": "The license",
+ "intro": "OpenStreetMap is licensed under the Open Database License. The full copyright text can be summarized as following:",
+ "li0": "A product using OpenStreetMap data must give attribution.",
+ "li1": "OpenStreetMap-data must remain open. This means that data of a map containing OpenStreetMap data can be copied again.",
+ "outro": "The license has a few implications - these are explained below."
+ },
+ "vandalism": {
+ "title": "What about vandalism?",
+ "intro": "As anyone can edit the data, it is indeed possible that a malicious change is made. However, this is extremely rare for a few reasons:",
+ "li0": "the technical barrier to make changes is high",
+ "li1": "a small malicious change has low impact, thus little reward for a vandal",
+ "li2": "a high impact change is quickly noticed and reverted since so many people use this data",
+ "li3": "all changes are tracked and tied to a single user. A repeating offender is quickly banned",
+ "li4": "In Belgium (and some other countries), the first edit by a new contributor is systematically checked and corrected if needed."
+ }
+ },
+ "aboutMc": {
+ "title": "Using MapComplete in your organization",
+ "text0": "If an existing MapComplete theme is what you, feel free to use it or embed it on your website. Embedding the public themes is free and always will be.",
+ "text1": "Do you need some other data, but does the theme not exist yet? The MapComplete-developers can build it for you on a decent budget. Get in touch via email, github or send a message via osm.org",
+ "text2": "If you still feel unsure, the possibilities are outlined below. Additionally, some common questions are answered",
+ "layers": {
+ "title": "What data can be shown with MapComplete?",
+ "intro": "MapComplete has a powerful templating system, which allows to quickly create a map showing precisely those features that you need and showing relevant attributes in the popups.
This data can be fetched from OpenStreetMap directly, but MapComplete can also use external datasets - e.g. to compare OpenStreetMap with another dataset or to show data that is not suited for OpenStreetMap (planned activities, statistics, ...)"
+ },
+ "survey": {
+ "title": "Survey possibilities",
+ "intro": "
MapComplete is an easy to use survey tool. It is ideal to collect the necessary in a few clicks, both on desktop and on mobile. This data is contributed directly into OpenStreetMap.
We can setup a custom survey tool, asking precisely the data you need in a future-proof way.
Do you have a dataset that has to be (re)surveyed? This is the perfect moment to make the switch to OpenStreetMap.MapComplete can show your dataset and OpenStreetMap at the same time, making it easier to visit all the locations and to see what the community already contributed.
\n"
+ },
+ "internalUse": {
+ "title": "Using the data in internal processes",
+ "intro": "Once the data is in OpenStreetMap, you'll probably want to use the data as well. Your MapComplete theme can have a convenient export-button, offering to download the data in many open formats usable in QGis, ArcGis, Excel, LibreOffice-calc, ...
Someone with basic spreadsheet-skills can thus easily create graphs and insights about the data, whereas the GIS-experts within your organisation can easily work with this data in their preferred application.
If an automated setup is needed, a free-to-use, community-run API is available.
"
+ }
+ },
+ "services": {
+ "title": "MapComplete services",
+ "intro": "The developer of MapComplete can help you with the following services:",
+ "li0": "Setting up a theme tailored for your need",
+ "li1": "Help with setting up the internal data flow to integrate OpenStreetMap",
+ "li2": "Training on how to contribute data with MapComplete",
+ "li3": "Advanced training (e.g. for the GIS-team) on how to add advanced data to OpenStreetMap",
+ "li4": "Training on how to download filtered data from OpenStreetMap",
+ "outro": "These services are offered at competitive prices. A simple theme without extra support can be setup for as little €2000, and a small additional yearly hosting cost."
+ },
+ "drawbacks": {
+ "title": "A few drawbacks to keep in mind",
+ "intro": "While joining this community has tremendous benefits, there are a few topics to carefully consider.",
+ "unsuitedData": {
+ "title": "Data not suited for OpenStreetMap",
+ "intro": "The basic rule for OpenStreetMap is that all data must be verifiable on the ground and are somewhat permanent. This implies that some data cannot be sent to OpenStreetMap directly - but some workarounds exist.",
+ "li0": "Subjective data (such as reviews) are not suited for OpenStreetMap. However, MapComplete has an integration with Mangrove.reviews, an openly licensed review website",
+ "li1": "Events of a few days, road works that are planned next month are thus not recorded, neither are road works which only last a few days.",
+ "li2": "Temporal data (e.g. statistics of air quality, traffic intensity, ...) can not stored on OpenStreetMap as they are hard to verify by a volunteer. Note that, if this data is available elsewhere, it can still be visualized within MapComplete as extra layer."
+ },
+ "licenseNuances": {
+ "title": "Implications of ODbL: some use cases",
+ "intro": "OpenStreetMap is licensed unter the Open Database License which states that:",
+ "li0": "All data can be reused for any purpose - including commercial purposes",
+ "li1": "Applications or products using OpenStreetMap should give a clear copyright notice",
+ "li2": "Any dataset or product which contains OpenStreetMap-data must be republished under ODbL too, including modifications to this dataset and in a usable format.",
+ "outro": "This has a few implications which should be considered for some usecases, as explained below",
+ "usecaseMapDifferentSources": {
+ "title": "Creating a map from different sources",
+ "intro": "For example, one could make a map with all benches in some city, based on the benches known by OpenStreetMap. This printed map needs a clear statement that the map data is based on OpenStreetMap. Selling these maps is permitted.If the mapmaker notices that the benches are missing in some area and adds them on the printed map, the data on the missing benches are automatically open data too. This means that an OpenStreetMap-contributor is allowed to take the paper map and use it to add the missing benches back into OpenStreetMap.
This contributor also has the right to ask for the dataset of the missing benches, which should be provided too.
If the mapmaker notices that the benches are missing in some area and adds them on the printed map, the data on the missing benches are automatically open data too. This means that an OpenStreetMap-contributor is allowed to take the paper map and use it to add the missing benches back into OpenStreetMap. This contributor also has the right to ask for the dataset of the missing benches, which should be provided too.
Of course, a map with only benches can be boring. The mapmaker might also decide to add in a layer with shops, possibly sourced from another geodata provider under another license. This is permitted to, if the map clearly states that the benches are sourced from OSM (under ODBL) and the shops have a different source (eventually with an all rights reserved).
However, mixing two datasets into one undistinguishible layer might not be permitted. For example, the mapmaker migth find that OSM has excellent data on benches in one part of the city and the closed-source provider might have excellent data on benches in another part of the city, merging these datasets into one could be problematic:
",
+ "li0": "the open license would require the modifications to be openly republished...",
+ "li1": "...whereas the all-rights-reserved license would prohibit this.",
+ "outro": "As a result, this kind of mixing is not allowed"
},
- "loginWithOpenStreetMap": "Login with OpenStreetMap",
- "welcomeBack": "You are logged in, welcome back!",
- "loginToStart": "Log in to answer this question",
- "openStreetMapIntro": "An Open Map
One that everyone can use and edit freely. A single place to store all geo-info. Different, small, incompatible and outdated maps are not needed anywhere.
OpenStreetMap is not the enemy map. The map data can be used freely (with attribution and publication of changes to that data). Everyone can add new data and fix errors. This website uses OpenStreetMap. All the data is from there, and your answers and corrections are used all over.
Many people and apps already use OpenStreetMap: Organic Maps, OsmAnd, but also the maps at Facebook, Instagram, Apple-maps and Bing-maps are (partly) powered by OpenStreetMap.
",
- "search": {
- "search": "Search a location",
- "searching": "Searching…",
- "nothing": "Nothing found…",
- "error": "Something went wrong…"
- },
- "returnToTheMap": "Return to the map",
- "save": "Save",
- "cancel": "Cancel",
- "skip": "Skip this question",
- "oneSkippedQuestion": "One question is skipped",
- "skippedQuestions": "Some questions are skipped",
- "number": "number",
- "osmLinkTooltip": "Browse this object on OpenStreetMap for history and more editing options",
- "add": {
- "addNewMapLabel": "Click here to add a new item",
- "disableFiltersExplanation": "Some features might be hidden by a filter",
- "disableFilters": "Disable all filters",
- "addNew": "Add a new {category} here",
- "presetInfo": "The new POI will have {tags}",
- "warnVisibleForEveryone": "Your addition will be visible for everyone",
- "title": "Add a new point?",
- "intro": "You clicked somewhere where no data is known yet.
",
- "pleaseLogin": "Please log in to add a new point",
- "zoomInFurther": "Zoom in further to add a point.",
- "stillLoading": "The data is still loading. Please wait a bit before you add a new point.",
- "confirmIntro": "Add a {title} here?
The point you create here will be visible for everyone. Please, only add things on to the map if they truly exist. A lot of applications use this data.",
- "confirmButton": "Add a {category} here.
Your addition is visible for everyone
",
- "openLayerControl": "Open the layer control box",
- "layerNotEnabled": "The layer {layer} is not enabled. Enable this layer to add a point",
- "hasBeenImported": "This point has already been imported",
- "importTags": "The element will receive {tags}",
- "zoomInMore": "Zoom in more to import this feature",
- "wrongType": "This element is not a point or a way and can not be imported",
- "import": {
- "officialThemesOnly": "The import button is disabled for unofficial themes to prevent accidents",
- "howToTest": "To test, add test=true or backend=osm-test to the URL. The changeset will be printed in the console. Please open a PR to officialize this theme to actually enable the import button.",
- "hasBeenImported": "This object has been imported",
- "importTags": "The element will receive {tags}",
- "zoomInMore": "Zoom in more to import this feature",
- "wrongType": "This element is not a point or a way and can not be imported"
- }
- },
- "pickLanguage": "Choose a language: ",
- "about": "Easily edit and add OpenStreetMap for a certain theme",
- "nameInlineQuestion": "The name of this {category} is $$$",
- "noNameCategory": "{category} without a name",
- "questions": {
- "phoneNumberOf": "What is the phone number of {category}?",
- "phoneNumberIs": "The phone number of this {category} is {phone}",
- "websiteOf": "What is the website of {category}?",
- "websiteIs": "Website: {website}",
- "emailOf": "What is the email address of {category}?",
- "emailIs": "The email address of this {category} is {email}"
- },
- "morescreen": {
- "intro": "More thematic maps?
Do you enjoy collecting geodata?
There are more themes available.",
- "requestATheme": "If you want a custom-built theme, request it in the issue tracker",
- "streetcomplete": "Another, similar application is StreetComplete.",
- "createYourOwnTheme": "Create your own MapComplete theme from scratch",
- "previouslyHiddenTitle": "Previously visited hidden themes",
- "hiddenExplanation": "These themes are only accessible to those with the link. You have discovered {hidden_discovered} of {total_hidden} hidden themes."
- },
- "sharescreen": {
- "intro": "Share this map
Share this map by copying the link below and sending it to friends and family:",
- "addToHomeScreen": "Add to your home screen
You can easily add this website to your smartphone home screen for a native feel. Click the 'Add to home screen' button in the URL bar to do this.",
- "embedIntro": "Embed on your website
Please, embed this map into your website.
We encourage you to do it - you don't even have to ask permission.
It is free, and always will be. The more people are using this, the more valuable it becomes.",
- "copiedToClipboard": "Link copied to clipboard",
- "thanksForSharing": "Thanks for sharing!",
- "editThisTheme": "Edit this theme",
- "editThemeDescription": "Add or change questions to this map theme",
- "fsUserbadge": "Enable the login button",
- "fsSearch": "Enable the search bar",
- "fsWelcomeMessage": "Show the welcome message popup and associated tabs",
- "fsLayers": "Enable the layer control",
- "fsLayerControlToggle": "Start with the layer control expanded",
- "fsAddNew": "Enable the 'add new POI' button",
- "fsGeolocation": "Enable the 'geolocate-me' button (mobile only)",
- "fsIncludeCurrentBackgroundMap": "Include the current background choice {name}",
- "fsIncludeCurrentLayers": "Include the current layer choices",
- "fsIncludeCurrentLocation": "Include current location"
- },
- "attribution": {
- "attributionTitle": "Attribution notice",
- "attributionContent": "All data is provided by OpenStreetMap, freely reusable under the Open DataBase License.
",
- "themeBy": "Theme maintained by {author}",
- "iconAttribution": {
- "title": "Used icons"
- },
- "mapContributionsBy": "The current visible data has edits made by {contributors}",
- "mapContributionsByAndHidden": "The current visible data has edits made by {contributors} and {hiddenCount} more contributors",
- "codeContributionsBy": "MapComplete has been built by {contributors} and {hiddenCount} more contributors",
- "openOsmcha": "See latest edits made with {theme}",
- "openMapillary": "Open Mapillary here",
- "openIssueTracker": "File a bug",
- "josmOpened": "JOSM is opened",
- "josmNotOpened": "JOSM could not be reached. Make sure it is opened and remote control is enabled",
- "editJosm": "Edit here with JOSM",
- "editId": "Open the OpenStreetMap online editor here",
- "donate": "Support MapComplete financially"
- },
- "readYourMessages": "Please, read all your OpenStreetMap-messages before adding a new point.",
- "fewChangesBefore": "Please, answer a few questions of existing points before adding a new point.",
- "goToInbox": "Open inbox",
- "removeLocationHistory": "Delete the location history",
- "getStartedLogin": "Log in with OpenStreetMap to get started",
- "getStartedNewAccount": " or create a new account",
- "noTagsSelected": "No tags selected",
- "testing": "Testing - changes won't be saved",
- "customThemeIntro": "Custom themes
These are previously visited user-generated themes.",
- "aboutMapcomplete": "About MapComplete
Use it to add OpenStreetMap info on a single theme. Answer questions, and within minutes your contributions are available everywhere. The theme maintainer defines elements, questions and languages for it.
Find out more
MapComplete always offers the next step to learn more about OpenStreetMap.
- When embedded in a website, the iframe links to a full-screen MapComplete
- The fullscreen version offers info about OpenStreetMap
- Viewing works without login, but editing requires an OSM account.
- If you are not logged in, you are asked to do so
- Once you answered a single question, you can add new points to the map
- After a while, actual OSM-tags are shown, later linking to the wiki
Did you notice an issue? Do you have a feature request? Want to help translate? Head over to the source code or issue tracker.
Want to see your progress? Follow the edit count on OsmCha.
",
- "backgroundMap": "Background map",
- "openTheMap": "Open the map",
- "loginOnlyNeededToEdit": "if you want to edit the map",
- "layerSelection": {
- "zoomInToSeeThisLayer": "Zoom in to see this layer",
- "title": "Select layers"
- },
- "download": {
- "title": "Download visible data",
- "downloadAsPdf": "Download a PDF of the current map",
- "downloadAsPdfHelper": "Ideal to print the current map",
- "downloadGeojson": "Download visible data as GeoJSON",
- "downloadGpx": "Download as GPX-file",
- "downloadGpxHelper": "A GPX-file can be used with most navigation devices and applications",
- "uploadGpx": "Upload your track to OpenStreetMap",
- "exporting": "Exporting…",
- "downloadGeoJsonHelper": "Compatible with QGIS, ArcGIS, ESRI, …",
- "downloadCSV": "Download visible data as CSV",
- "downloadCSVHelper": "Compatible with LibreOffice Calc, Excel, …",
- "includeMetaData": "Include metadata (last editor, calculated values, …)",
- "licenseInfo": "Copyright notice
The provided data is available under ODbL. Reusing it is gratis for any purpose, but - the attribution © OpenStreetMap contributors is required
- Any change must be use the license
Please read the full copyright notice for details.",
- "noDataLoaded": "No data is loaded yet. Download will be available soon",
- "downloadFeatureAsGpx": "Download as GPX-file",
- "downloadFeatureAsGeojson": "Download as GeoJson-file"
- },
- "weekdays": {
- "abbreviations": {
- "monday": "Mon",
- "tuesday": "Tue",
- "wednesday": "Wed",
- "thursday": "Thu",
- "friday": "Fri",
- "saturday": "Sat",
- "sunday": "Sun"
- },
- "monday": "Monday",
- "tuesday": "Tuesday",
- "wednesday": "Wednesday",
- "thursday": "Thursday",
- "friday": "Friday",
- "saturday": "Saturday",
- "sunday": "Sunday"
- },
- "opening_hours": {
- "error_loading": "Error: could not visualize these opening hours.",
- "open_during_ph": "During a public holiday, this is",
- "opensAt": "from",
- "openTill": "till",
- "not_all_rules_parsed": "These opening hours are complicated. The following rules are ignored in the input element:",
- "closed_until": "Closed until {date}",
- "closed_permanently": "Closed for an unkown duration",
- "open_24_7": "Opened around the clock",
- "ph_not_known": " ",
- "ph_closed": "closed",
- "ph_open": "opened",
- "ph_open_as_usual": "opened as usual",
- "loadingCountry": "Determining country…"
- },
- "histogram": {
- "error_loading": "Could not load the histogram"
- },
- "wikipedia": {
- "wikipediaboxTitle": "Wikipedia",
- "failed": "Loading the Wikipedia entry failed",
- "loading": "Loading Wikipedia...",
- "noWikipediaPage": "This Wikidata item has no corresponding Wikipedia page yet.",
- "searchWikidata": "Search on Wikidata",
- "noResults": "Nothing found for {search}",
- "doSearch": "Search above to see results",
- "createNewWikidata": "Create a new Wikidata item"
- },
- "apply_button": {
- "isApplied": "The changes are applied",
- "appliedOnAnotherObject": "The object {id} will receive {tags}"
+ "usecaseGatheringOpenData": {
+ "title": "Gathering open data",
+ "intro": "MapComplete is an excellent way to create Open Data, also for governments. By default, this data will be freely redistributable under the ODbL. However, if there is a requirement to publish the gathered data under a public domain-license (where all rights are granted to the public and no attribution is required), the ODbL is too restrictive. In this case, one can ask the contributors to add data as Public Domain (e.g. by informing them in the mapcomplete theme). "
}
+ }
+ }
+ },
+ "notes": {
+ "isClosed": "This note is resolved",
+ "addCommentPlaceholder": "Add a comment...",
+ "addComment": "Add comment",
+ "addCommentAndClose": "Add comment and close",
+ "closeNote": "Close note",
+ "reopenNote": "Reopen note",
+ "reopenNoteAndComment": "Reopen note and comment",
+ "anonymous": "Anonymous user",
+ "loginToAddComment": "Login to add a comment",
+ "loginToAddPicture": "Login to add a picture",
+ "loginToClose": "Login to close this note",
+ "createNoteTitle": "Create a new note here",
+ "createNote": "Create a new note",
+ "noteIsPublic": "This will be visible to everyone",
+ "createNoteIntro": "Is something wrong or missing on the map? Create a note here. These will be checked by volunteers",
+ "warnAnonymous": "You are not logged in. We won't be able to contact you to resolve your issue.",
+ "notesLayerMustBeEnabled": "The 'notes'-layer is disabled. Enable it to add a note",
+ "isCreated": "Your note has been created!",
+ "noteLayerNotEnabled": "The layer showing notes is not enabled. This layer must be enabled to add a new note",
+ "noteLayerHasFilters": "Some notes might be hidden by a filter",
+ "disableAllNoteFilters": "Disable all filters",
+ "noteLayerDoEnable": "Enable the layer showing notes"
+ },
+ "importLayer": {
+ "layerName": "Possible {title}",
+ "description": "A layer which imports entries for {title}",
+ "popupTitle": "Possible {title}",
+ "importButton": "import_button({layerId}, _tags, I have found a {title} here - add it to the map,./assets/svg/addSmall.svg,,,id)",
+ "notFound": "I could not find {title} - remove it",
+ "alreadyMapped": "There already is another {title} on the map - this point is a duplicate",
+ "importHandled": "This feature has been handled! Thanks for your effort
"
+ },
+ "importHelper": {
+ "title": "Import helper",
+ "description": "The import helper converts an external dataset to notes. The external dataset must match one of the existing MapComplete layers. For every item you put in the importer, a single note will be created. These notes will be shown together with the relevant features in these maps to easily add them.",
+ "importFormat": "A text in a note should have the following format in order to be picked up:
[A bit of introduction]
https://mapcomplete.osm.be/[themename].html?[parameters such as lat and lon]#import
[all tags of the feature]
",
+ "userAccountTitle": "Select user account",
+ "loggedInWith": "You are currently logged in as {name} and have made {csCount} changesets",
+ "loginRequired": "You have to be logged in to continue",
+ "locked": "You need at least {importHelperUnlock} to use the import helper",
+ "lockNotice": "This page is locked. You need {importHelperUnlock} changesets before you can access here.",
+ "selectLayer": "Select a layer...",
+ "selectFile": {
+ "title": "Select file",
+ "description": "Select a .csv or .geojson file to get started",
+ "fileFormatDescription": "Select a .csv or a .geojson file",
+ "fileFormatDescriptionCsv": "In the CSV-file, there should be a column lat and lon with the coordinates in WGS84. There should be an additional column for every attribute.",
+ "fileFormatDescriptionGeoJson": "In the geojson file, only points should be present. The properties should be exactly those properties that should go into OpenStreetMap",
+ "errNoName": "Some columns don't have a name",
+ "noFilesLoaded": "No file is currently loaded",
+ "errDuplicate": "Some columns have the same name",
+ "loadedFilesAre": "Currently loaded file is {file}",
+ "errNoLatOrLon":"The header does not contain `lat` or `lon`",
+ "errPointsOnly": "The loaded JSON-file should only contain points",
+ "errNotFeatureCollection": "The loaded JSON-file is not a geojson-featurecollection"
},
- "favourite": {
- "panelIntro": "Your personal theme
Activate your favourite layers from all the official themes",
- "loginNeeded": "Log in
A personal layout is only available for OpenStreetMap users",
- "reload": "Reload the data"
+ "mapPreview": {
+ "title": "Map preview",
+ "autodetected": "The layer was automatically deducted based on the properties",
+ "selectLayer": "Which layer does this import match with?",
+ "mismatch": "{count} features did not match the selected layer. Make sure that the tags to indicate the type are present, namely {tags}",
+ "confirm": "The features are on the right location on the map"
},
- "reviews": {
- "title": "{count} reviews",
- "title_singular": "One review",
- "name_required": "A name is required in order to display and create reviews",
- "no_reviews_yet": "There are no reviews yet. Be the first to write one and help open data and the business!",
- "write_a_comment": "Leave a review…",
- "no_rating": "No rating given",
- "posting_as": "Posting as",
- "i_am_affiliated": "I am affiliated with this object
Check if you are an owner, creator, employee, …",
- "affiliated_reviewer_warning": "(Affiliated review)",
- "saving_review": "Saving…",
- "saved": "Review saved. Thanks for sharing!",
- "tos": "If you create a review, you agree to the TOS and privacy policy of Mangrove.reviews",
- "attribution": "Reviews are powered by Mangrove Reviews and are available under CC-BY 4.0.",
- "plz_login": "Log in to leave a review"
- },
- "multi_apply": {
- "autoApply": "When changing the attributes {attr_names}, these attributes will automatically be changed on {count} other objects too"
- },
- "move": {
- "loginToMove": "You must be logged in to move a point",
- "inviteToMoveAgain": "Move this point again",
- "moveTitle": "Move this point",
- "whyMove": "Why do you want to move this point?",
- "confirmMove": "Move here",
- "pointIsMoved": "The point has been moved",
- "zoomInFurther": "Zoom in further to confirm this move",
- "selectReason": "Why do you move this object?",
- "reasons": {
- "reasonRelocation": "The object has been relocated to a totally different location",
- "reasonInaccurate": "The location of this object is inaccurate and should be moved a few meter"
- },
- "inviteToMove": {
- "generic": "Move this point",
- "reasonInaccurate": "Improve the accuracy of this point",
- "reasonRelocation": "Move this object to a another place because it has relocated"
- },
- "cannotBeMoved": "This feature cannot be moved.",
- "isWay": "This feature is a way. Use another OpenStreetMap editor to move it.",
- "isRelation": "This feature is a relation and can not be moved",
- "partOfAWay": "This feature is part of another way. Use another editor to move it.",
- "partOfRelation": "This feature is part of a relation. Use another editor to move it.",
- "cancel": "Cancel move"
- },
- "privacy": {
- "title": "Privacy policy",
- "intro": "Privacy is important - for both the individual and for society. MapComplete tries to respect your privacy as much as possible - up to the point no annoying cookie banner is needed. However, we still would like to inform you which information is gathered and shared, under which circumstances and why these trade-offs are made.",
- "trackingTitle": "Statistical data",
- "tracking": "To gather some insight in whom visits our website, some technical information is collected. This is included the country you visited the webpage from, which website referred you to MapComplete, the type of your device and the screensize. A cookie is placed on your device to indicate that you visited MapComplete earlier today. This data is not detailed enough to personally identify you. These statistics are only available to anyone in aggregate and are publicly available to anyone",
- "geodataTitle": "Your geolocation",
- "geodata": "When MapComplete gets your geolocation, your geolocation and previously visited locations stay on your device. Your location data is never automatically sent to anywhere else - unless some functionality clearly states otherwise.",
- "editingTitle": "When making changes",
- "editing": "When you make a change to the map, this change is recorded on OpenStreetMap and is publicly available to anyone. A changeset made with MapComplete includes the following data: - The changes you made
- Your username
- When this change is made
- The theme you used while making the change
- The language of the user interface
- An indication of how close you were to changed objects. Other mappers can use this information to determine if a change was made based on survey or on remote research
Please refer to the privacy policy on OpenStreetMap.org for detailed information. We'd like to remind you that you can use a fictional name when signing up.",
- "miscCookiesTitle": "Other cookies",
- "miscCookies": "MapComplete integrates with various other services, especially to load images of features. Images are hosted on various third-party servers, which might set cookies on their own.",
- "whileYoureHere": "Do you care about privacy?",
- "surveillance": "As you are reading the privacy policy, you probably care about privacy - so do we! We even made a theme showing surveillance cameras. Feel free to map them all!"
- },
- "professional": {
- "indexPage": {
- "hook": "Need professional support?",
- "hookMore": "We can help with setting up surveys, data imports and OpenStreetMap-consultancy",
- "button": "Discover our services"
- },
- "title": "Professional support with MapComplete",
- "intro": "The developer of MapComplete offers professional support. This document outlines some of the possibilities, common questions and the boundaries of MapComplete",
- "osmTitle": "What can OpenStreetMap and MapComplete do for your organisation?",
- "text0": "Maintaining a set of up-to-date geodata is hard, error prone and expensive.
To add insult to injury, many organizations end up collecting the same data independently - resulting in duplicated efforts, non-standardized data formats and many incomplete, unmaintained datasets.
At the same time, there is a huge community which gathers a lot of geodata into one shared, global and standardized database - namely OpenStreetMap.org.
",
- "text1": "MapComplete is the editor to make contributing data to OpenStreetMap easy.
",
- "aboutOsm": {
- "aboutOsm": {
- "title": "What is OpenStreetMap?",
- "intro": "OpenStreetMap is a shared, global database, built by volunteers. All geodata can be contributed to OpenStreetMap, as long as it can be verified on the ground.
OpenStreetMap has grown to be a very broad and deep dataset as it contains data over thousands of categories of objects.An individual object might also have a ton of attributes, bringing a lot of nuance, e.g.:",
- "li0": "Streets have geometry, but might also have information about the maxspeed, surface, wether they are lit, their name, a link to Wikipedia, a link to what they are named after, which hiking-, cycle- and busroutes run over theme",
- "li1": "Shops and other amenities might have opening hours, a phone number, a link to the website, which payment methods are supported, what they sell, which services they offer, …",
- "li2": "Toilets might have information about wheelchair accessibility, a changing table, if payment is needed, …",
- "li3": "and much, much more…"
- },
- "benefits": {
- "title": "Benefits of the OSM-ecosystem",
- "intro": "It can be very hard to leave your own dataset behind, as building this dataset often took a lot of time and effort.
However, the benefits of switching over to OSM are huge:",
- "li0": "You are not alone anymore to gather and maintain this dataset - a whole community is at your side",
- "li1": "Your data will reach a bigger audience then ever via Bing Maps, Apple Maps, Facebook, Instagram, Pokemon Go, OsmAnd, Organic Maps, Maps.me, Mapbox, Komoot, nearly all cycle-applications, …",
- "li2": "Many governement organisations and municipalities use OpenStreetMap on their websites too"
- },
- "license": {
- "title": "The license",
- "intro": "OpenStreetMap is licensed under the Open Database License. The full copyright text can be summarized as following:",
- "li0": "A product using OpenStreetMap data must give attribution.",
- "li1": "OpenStreetMap-data must remain open. This means that data of a map containing OpenStreetMap data can be copied again.",
- "outro": "The license has a few implications - these are explained below."
- },
- "vandalism": {
- "title": "What about vandalism?",
- "intro": "As anyone can edit the data, it is indeed possible that a malicious change is made. However, this is extremely rare for a few reasons:",
- "li0": "the technical barrier to make changes is high",
- "li1": "a small malicious change has low impact, thus little reward for a vandal",
- "li2": "a high impact change is quickly noticed and reverted since so many people use this data",
- "li3": "all changes are tracked and tied to a single user. A repeating offender is quickly banned",
- "li4": "In Belgium (and some other countries), the first edit by a new contributor is systematically checked and corrected if needed."
- }
- },
- "aboutMc": {
- "title": "Using MapComplete in your organization",
- "text0": "If an existing MapComplete theme is what you, feel free to use it or embed it on your website. Embedding the public themes is free and always will be.",
- "text1": "Do you need some other data, but does the theme not exist yet? The MapComplete-developers can build it for you on a decent budget. Get in touch via email, github or send a message via osm.org",
- "text2": "If you still feel unsure, the possibilities are outlined below. Additionally, some common questions are answered",
- "layers": {
- "title": "What data can be shown with MapComplete?",
- "intro": "MapComplete has a powerful templating system, which allows to quickly create a map showing precisely those features that you need and showing relevant attributes in the popups.
This data can be fetched from OpenStreetMap directly, but MapComplete can also use external datasets - e.g. to compare OpenStreetMap with another dataset or to show data that is not suited for OpenStreetMap (planned activities, statistics, ...)"
- },
- "survey": {
- "title": "Survey possibilities",
- "intro": "
MapComplete is an easy to use survey tool. It is ideal to collect the necessary in a few clicks, both on desktop and on mobile. This data is contributed directly into OpenStreetMap.
We can setup a custom survey tool, asking precisely the data you need in a future-proof way.
Do you have a dataset that has to be (re)surveyed? This is the perfect moment to make the switch to OpenStreetMap.MapComplete can show your dataset and OpenStreetMap at the same time, making it easier to visit all the locations and to see what the community already contributed.
\n"
- },
- "internalUse": {
- "title": "Using the data in internal processes",
- "intro": "Once the data is in OpenStreetMap, you'll probably want to use the data as well. Your MapComplete theme can have a convenient export-button, offering to download the data in many open formats usable in QGis, ArcGis, Excel, LibreOffice-calc, ...
Someone with basic spreadsheet-skills can thus easily create graphs and insights about the data, whereas the GIS-experts within your organisation can easily work with this data in their preferred application.
If an automated setup is needed, a free-to-use, community-run API is available.
"
- }
- },
- "services": {
- "title": "MapComplete services",
- "intro": "The developer of MapComplete can help you with the following services:",
- "li0": "Setting up a theme tailored for your need",
- "li1": "Help with setting up the internal data flow to integrate OpenStreetMap",
- "li2": "Training on how to contribute data with MapComplete",
- "li3": "Advanced training (e.g. for the GIS-team) on how to add advanced data to OpenStreetMap",
- "li4": "Training on how to download filtered data from OpenStreetMap",
- "outro": "These services are offered at competitive prices. A simple theme without extra support can be setup for as little €2000, and a small additional yearly hosting cost."
- },
- "drawbacks": {
- "title": "A few drawbacks to keep in mind",
- "intro": "While joining this community has tremendous benefits, there are a few topics to carefully consider.",
- "unsuitedData": {
- "title": "Data not suited for OpenStreetMap",
- "intro": "The basic rule for OpenStreetMap is that all data must be verifiable on the ground and are somewhat permanent. This implies that some data cannot be sent to OpenStreetMap directly - but some workarounds exist.",
- "li0": "Subjective data (such as reviews) are not suited for OpenStreetMap. However, MapComplete has an integration with Mangrove.reviews, an openly licensed review website",
- "li1": "Events of a few days, road works that are planned next month are thus not recorded, neither are road works which only last a few days.",
- "li2": "Temporal data (e.g. statistics of air quality, traffic intensity, ...) can not stored on OpenStreetMap as they are hard to verify by a volunteer. Note that, if this data is available elsewhere, it can still be visualized within MapComplete as extra layer."
- },
- "licenseNuances": {
- "title": "Implications of ODbL: some use cases",
- "intro": "OpenStreetMap is licensed unter the Open Database License which states that:",
- "li0": "All data can be reused for any purpose - including commercial purposes",
- "li1": "Applications or products using OpenStreetMap should give a clear copyright notice",
- "li2": "Any dataset or product which contains OpenStreetMap-data must be republished under ODbL too, including modifications to this dataset and in a usable format.",
- "outro": "This has a few implications which should be considered for some usecases, as explained below",
- "usecaseMapDifferentSources": {
- "title": "Creating a map from different sources",
- "intro": "For example, one could make a map with all benches in some city, based on the benches known by OpenStreetMap. This printed map needs a clear statement that the map data is based on OpenStreetMap. Selling these maps is permitted.If the mapmaker notices that the benches are missing in some area and adds them on the printed map, the data on the missing benches are automatically open data too. This means that an OpenStreetMap-contributor is allowed to take the paper map and use it to add the missing benches back into OpenStreetMap.
This contributor also has the right to ask for the dataset of the missing benches, which should be provided too.
If the mapmaker notices that the benches are missing in some area and adds them on the printed map, the data on the missing benches are automatically open data too. This means that an OpenStreetMap-contributor is allowed to take the paper map and use it to add the missing benches back into OpenStreetMap. This contributor also has the right to ask for the dataset of the missing benches, which should be provided too.
Of course, a map with only benches can be boring. The mapmaker might also decide to add in a layer with shops, possibly sourced from another geodata provider under another license. This is permitted to, if the map clearly states that the benches are sourced from OSM (under ODBL) and the shops have a different source (eventually with an all rights reserved).
However, mixing two datasets into one undistinguishible layer might not be permitted. For example, the mapmaker migth find that OSM has excellent data on benches in one part of the city and the closed-source provider might have excellent data on benches in another part of the city, merging these datasets into one could be problematic:
",
- "li0": "the open license would require the modifications to be openly republished...",
- "li1": "...whereas the all-rights-reserved license would prohibit this.",
- "outro": "As a result, this kind of mixing is not allowed"
- },
- "usecaseGatheringOpenData": {
- "title": "Gathering open data",
- "intro": "MapComplete is an excellent way to create Open Data, also for governments. By default, this data will be freely redistributable under the ODbL. However, if there is a requirement to publish the gathered data under a public domain-license (where all rights are granted to the public and no attribution is required), the ODbL is too restrictive. In this case, one can ask the contributors to add data as Public Domain (e.g. by informing them in the mapcomplete theme). "
- }
- }
- }
- },
- "notes": {
- "isClosed": "This note is resolved",
- "addCommentPlaceholder": "Add a comment...",
- "addComment": "Add comment",
- "addCommentAndClose": "Add comment and close",
- "closeNote": "Close note",
- "reopenNote": "Reopen note",
- "reopenNoteAndComment": "Reopen note and comment",
- "anonymous": "Anonymous user",
- "loginToAddComment": "Login to add a comment",
- "loginToAddPicture": "Login to add a picture",
- "loginToClose": "Login to close this note",
- "createNoteTitle": "Create a new note here",
- "createNote": "Create a new note",
- "noteIsPublic": "This will be visible to everyone",
- "createNoteIntro": "Is something wrong or missing on the map? Create a note here. These will be checked by volunteers",
- "warnAnonymous": "You are not logged in. We won't be able to contact you to resolve your issue.",
- "notesLayerMustBeEnabled": "The 'notes'-layer is disabled. Enable it to add a note",
- "isCreated": "Your note has been created!",
- "noteLayerNotEnabled": "The layer showing notes is not enabled. This layer must be enabled to add a new note",
- "noteLayerHasFilters": "Some notes might be hidden by a filter",
- "disableAllNoteFilters": "Disable all filters",
- "noteLayerDoEnable": "Enable the layer showing notes"
- },
- "importLayer": {
- "layerName": "Possible {title}",
- "description": "A layer which imports entries for {title}",
- "popupTitle": "Possible {title}",
- "importButton": "import_button({layerId}, _tags, There might be a {title} here,./assets/svg/addSmall.svg,,,id)",
- "importHandled": "This feature has been handled! Thanks for your effort
"
- },
- "importHelper": {
- "title": "Import helper",
- "description": "The import helper converts an external dataset to notes",
- "lockNotice": "This page is locked. You need {importHelperUnlock} changesets before you can access here.",
- "selectFile": "Select a .csv or .geojson file to get started",
- "loadedFilesAre": "Currently loaded file is {file}",
- "noFilesLoaded": "No file is currently loaded",
- "selectLayer": "Select a layer...",
- "selectFileTitle": "Select file",
- "validateDataTitle": "Validate data"
- }
+ "validateDataTitle": "Validate data",
+ "allAttributesSame": "All features to import have this tag",
+ "someHaveSame": "{count} features to import have this tag, this is {percentage}% of the total",
+ "inspectDataTitle": "Inspect data of {count} features to import",
+ "inspectDidAutoDected": "Layer was chosen automatically",
+ "inspectLooksCorrect": "These values look correct"
+ }
}
diff --git a/package-lock.json b/package-lock.json
index 3498ed170..bf2e78f06 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -20,6 +20,7 @@
"@types/leaflet-providers": "^1.2.0",
"@types/leaflet.markercluster": "^1.4.3",
"@types/lz-string": "^1.3.34",
+ "@types/papaparse": "^5.3.1",
"@types/prompt-sync": "^4.1.0",
"@types/wikidata-sdk": "^6.1.0",
"country-language": "^0.1.7",
@@ -44,6 +45,7 @@
"opening_hours": "^3.6.0",
"osm-auth": "^1.0.2",
"osmtogeojson": "^3.0.0-beta.4",
+ "papaparse": "^5.3.1",
"parcel": "^1.2.4",
"prompt-sync": "^4.2.0",
"svg-resizer": "github:vieron/svg-resizer",
@@ -3254,8 +3256,15 @@
"node_modules/@types/node": {
"version": "7.10.14",
"resolved": "https://registry.npmjs.org/@types/node/-/node-7.10.14.tgz",
- "integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA==",
- "dev": true
+ "integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA=="
+ },
+ "node_modules/@types/papaparse": {
+ "version": "5.3.1",
+ "resolved": "https://registry.npmjs.org/@types/papaparse/-/papaparse-5.3.1.tgz",
+ "integrity": "sha512-1lbngk9wty2kCyQB42LjqSa12SEop3t9wcEC7/xYr3ujTSTmv7HWKjKYXly0GkMfQ42PRb2lFPFEibDOiMXS0g==",
+ "dependencies": {
+ "@types/node": "*"
+ }
},
"node_modules/@types/parse-json": {
"version": "4.0.0",
@@ -10090,6 +10099,11 @@
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
"integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU="
},
+ "node_modules/papaparse": {
+ "version": "5.3.1",
+ "resolved": "https://registry.npmjs.org/papaparse/-/papaparse-5.3.1.tgz",
+ "integrity": "sha512-Dbt2yjLJrCwH2sRqKFFJaN5XgIASO9YOFeFP8rIBRG2Ain8mqk5r1M6DkfvqEVozVcz3r3HaUGw253hA1nLIcA=="
+ },
"node_modules/parcel": {
"version": "1.12.4",
"resolved": "https://registry.npmjs.org/parcel/-/parcel-1.12.4.tgz",
@@ -19227,8 +19241,15 @@
"@types/node": {
"version": "7.10.14",
"resolved": "https://registry.npmjs.org/@types/node/-/node-7.10.14.tgz",
- "integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA==",
- "dev": true
+ "integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA=="
+ },
+ "@types/papaparse": {
+ "version": "5.3.1",
+ "resolved": "https://registry.npmjs.org/@types/papaparse/-/papaparse-5.3.1.tgz",
+ "integrity": "sha512-1lbngk9wty2kCyQB42LjqSa12SEop3t9wcEC7/xYr3ujTSTmv7HWKjKYXly0GkMfQ42PRb2lFPFEibDOiMXS0g==",
+ "requires": {
+ "@types/node": "*"
+ }
},
"@types/parse-json": {
"version": "4.0.0",
@@ -24635,6 +24656,11 @@
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
"integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU="
},
+ "papaparse": {
+ "version": "5.3.1",
+ "resolved": "https://registry.npmjs.org/papaparse/-/papaparse-5.3.1.tgz",
+ "integrity": "sha512-Dbt2yjLJrCwH2sRqKFFJaN5XgIASO9YOFeFP8rIBRG2Ain8mqk5r1M6DkfvqEVozVcz3r3HaUGw253hA1nLIcA=="
+ },
"parcel": {
"version": "1.12.4",
"resolved": "https://registry.npmjs.org/parcel/-/parcel-1.12.4.tgz",
diff --git a/package.json b/package.json
index 0aa0b4c39..1818e3965 100644
--- a/package.json
+++ b/package.json
@@ -44,7 +44,7 @@
"lint": "tslint --project . -c tslint.json '**.ts' ",
"clean": "rm -rf .cache/ && (find *.html | grep -v \"\\(404\\|index\\|land\\|test\\|preferences\\|customGenerator\\|professional\\|automaton\\|import_helper\\|theme\\).html\" | xargs rm) && (ls | grep \"^index_[a-zA-Z_]\\+\\.ts$\" | xargs rm) && (ls | grep \".*.webmanifest$\" | xargs rm)",
"generate:dependency-graph": "node_modules/.bin/depcruise --exclude \"^node_modules\" --output-type dot Logic/State/MapState.ts > dependencies.dot && dot dependencies.dot -T svg -o dependencies.svg && rm dependencies.dot",
- "bicycle_rental": "ts-node ./scripts/extractBikeRental.ts"
+ "bicycle_rental": "ts-node ./scripts/extractBikeRental.ts"
},
"keywords": [
"OpenStreetMap",
@@ -64,6 +64,7 @@
"@types/leaflet-providers": "^1.2.0",
"@types/leaflet.markercluster": "^1.4.3",
"@types/lz-string": "^1.3.34",
+ "@types/papaparse": "^5.3.1",
"@types/prompt-sync": "^4.1.0",
"@types/wikidata-sdk": "^6.1.0",
"country-language": "^0.1.7",
@@ -88,6 +89,7 @@
"opening_hours": "^3.6.0",
"osm-auth": "^1.0.2",
"osmtogeojson": "^3.0.0-beta.4",
+ "papaparse": "^5.3.1",
"parcel": "^1.2.4",
"prompt-sync": "^4.2.0",
"svg-resizer": "github:vieron/svg-resizer",
diff --git a/scripts/generateLayerOverview.ts b/scripts/generateLayerOverview.ts
index 052d73399..84c9c2935 100644
--- a/scripts/generateLayerOverview.ts
+++ b/scripts/generateLayerOverview.ts
@@ -72,10 +72,16 @@ class LayerOverviewUtils {
const dict = new Map();
for (const key in questions["default"]) {
+ if(key==="id"){
+ continue
+ }
questions[key].id = key;
dict.set(key, questions[key])
}
for (const key in icons["default"]) {
+ if(key==="id"){
+ continue
+ }
if (typeof icons[key] !== "object") {
continue
}
@@ -105,7 +111,7 @@ class LayerOverviewUtils {
"themes": Array.from(sharedThemes.values())
}))
- writeFileSync("./assets/generated/known_layers.json", JSON.stringify(Array.from(sharedLayers.values())))
+ writeFileSync("./assets/generated/known_layers.json", JSON.stringify({layers: Array.from(sharedLayers.values())}))
{