forked from MapComplete/MapComplete
Fix import flow for more advanced scenarios
This commit is contained in:
parent
913dc07eea
commit
700b48f693
18 changed files with 871 additions and 575 deletions
|
@ -39,6 +39,9 @@ export default class SharedTagRenderings {
|
||||||
}
|
}
|
||||||
|
|
||||||
dict.forEach((value, key) => {
|
dict.forEach((value, key) => {
|
||||||
|
if(key === "id"){
|
||||||
|
return;
|
||||||
|
}
|
||||||
value.id = value.id ?? key;
|
value.id = value.id ?? key;
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -62,4 +62,9 @@ export default class Toggleable extends Combine {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Collapse() : Toggleable{
|
||||||
|
this.isVisible.setData(false)
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -22,9 +22,12 @@ export default class Histogram<T> extends VariableUiElement {
|
||||||
constructor(values: UIEventSource<string[]>,
|
constructor(values: UIEventSource<string[]>,
|
||||||
title: string | BaseUIElement,
|
title: string | BaseUIElement,
|
||||||
countTitle: 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 => {
|
const sortName = new VariableUiElement(sortMode.map(m => {
|
||||||
switch (m) {
|
switch (m) {
|
||||||
case "name":
|
case "name":
|
||||||
|
@ -107,11 +110,11 @@ export default class Histogram<T> extends VariableUiElement {
|
||||||
return Histogram.defaultPalette[index % Histogram.defaultPalette.length]
|
return Histogram.defaultPalette[index % Histogram.defaultPalette.length]
|
||||||
};
|
};
|
||||||
let actualAssignColor = undefined;
|
let actualAssignColor = undefined;
|
||||||
if (assignColor === undefined) {
|
if (options?.assignColor === undefined) {
|
||||||
actualAssignColor = fallbackColor;
|
actualAssignColor = fallbackColor;
|
||||||
} else {
|
} else {
|
||||||
actualAssignColor = (keyValue: string) => {
|
actualAssignColor = (keyValue: string) => {
|
||||||
return assignColor(keyValue) ?? fallbackColor(keyValue)
|
return options.assignColor(keyValue) ?? fallbackColor(keyValue)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,11 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
|
||||||
tagRenderings: new Map()
|
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, <LayerConfigJson> layerConfig, "CompareToAlreadyExistingNotes")
|
const importLayerJson = new CreateNoteImportLayer(365).convertStrict(convertState, <LayerConfigJson> layerConfig, "CompareToAlreadyExistingNotes")
|
||||||
const importLayer = new LayerConfig(importLayerJson, "import-layer-dynamic")
|
const importLayer = new LayerConfig(importLayerJson, "import-layer-dynamic")
|
||||||
const flayer: FilteredLayer = {
|
const flayer: FilteredLayer = {
|
||||||
|
@ -48,8 +52,8 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
|
||||||
isDisplayed: new UIEventSource<boolean>(true),
|
isDisplayed: new UIEventSource<boolean>(true),
|
||||||
layerDef: importLayer
|
layerDef: importLayer
|
||||||
}
|
}
|
||||||
const unfiltered = new GeoJsonSource(flayer, params.bbox.padAbsolute(0.0001))
|
const allNotesWithinBbox = new GeoJsonSource(flayer, params.bbox.padAbsolute(0.0001))
|
||||||
unfiltered.features.map(f => MetaTagging.addMetatags(
|
allNotesWithinBbox.features.map(f => MetaTagging.addMetatags(
|
||||||
f,
|
f,
|
||||||
{
|
{
|
||||||
memberships: new RelationsTracker(),
|
memberships: new RelationsTracker(),
|
||||||
|
@ -65,30 +69,30 @@ export class CompareToAlreadyExistingNotes extends Combine implements FlowStep<{
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
const data = new FilteringFeatureSource(state, undefined, unfiltered)
|
const alreadyOpenImportNotes = new FilteringFeatureSource(state, undefined, allNotesWithinBbox)
|
||||||
data.features.addCallbackD(features => console.log("Loaded and filtered features are", features))
|
alreadyOpenImportNotes.features.addCallbackD(features => console.log("Loaded and filtered features are", features))
|
||||||
const map = Minimap.createMiniMap()
|
const map = Minimap.createMiniMap()
|
||||||
map.SetClass("w-full").SetStyle("height: 500px")
|
map.SetClass("w-full").SetStyle("height: 500px")
|
||||||
|
|
||||||
const comparison = Minimap.createMiniMap({
|
const comparisonMap = Minimap.createMiniMap({
|
||||||
location: map.location,
|
location: map.location,
|
||||||
|
|
||||||
})
|
})
|
||||||
comparison.SetClass("w-full").SetStyle("height: 500px")
|
comparisonMap.SetClass("w-full").SetStyle("height: 500px")
|
||||||
|
|
||||||
new ShowDataLayer({
|
new ShowDataLayer({
|
||||||
layerToShow: importLayer,
|
layerToShow: importLayer,
|
||||||
state,
|
state,
|
||||||
zoomToFeatures: true,
|
zoomToFeatures: true,
|
||||||
leafletMap: map.leafletMap,
|
leafletMap: map.leafletMap,
|
||||||
features: data,
|
features: alreadyOpenImportNotes,
|
||||||
popup: (tags, layer) => new FeatureInfoBox(tags, layer, state)
|
popup: (tags, layer) => new FeatureInfoBox(tags, layer, state)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
const maxDistance = new UIEventSource<number>(5)
|
const maxDistance = new UIEventSource<number>(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)
|
.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),
|
layerToShow: new LayerConfig(import_candidate),
|
||||||
state,
|
state,
|
||||||
zoomToFeatures: true,
|
zoomToFeatures: true,
|
||||||
leafletMap: comparison.leafletMap,
|
leafletMap: comparisonMap.leafletMap,
|
||||||
features: new StaticFeatureSource(partitionedImportPoints.map(p => p.hasNearby), false),
|
features: new StaticFeatureSource(partitionedImportPoints.map(p => p.hasNearby), false),
|
||||||
popup: (tags, layer) => new FeatureInfoBox(tags, layer, state)
|
popup: (tags, layer) => new FeatureInfoBox(tags, layer, state)
|
||||||
})
|
})
|
||||||
|
|
||||||
super([
|
super([
|
||||||
new Title("Compare with already existing 'to-import'-notes"),
|
new Title("Compare with already existing 'to-import'-notes"),
|
||||||
new Toggle(
|
new VariableUiElement(
|
||||||
new Loading("Fetching notes from OSM"),
|
alreadyOpenImportNotes.features.map(notesWithImport => {
|
||||||
new Combine([
|
if(allNotesWithinBbox.features.data === undefined || allNotesWithinBbox.features.data.length === 0){
|
||||||
map,
|
return new Loading("Fetching notes from OSM")
|
||||||
"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",
|
}
|
||||||
|
if(notesWithImport.length === 0){
|
||||||
new Toggle(
|
return new FixedUiElement("No previous note to import found").SetClass("thanks")
|
||||||
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"),
|
return new Combine([
|
||||||
partitionedImportPoints.map(({noNearby}) => noNearby.length === 0)
|
map,
|
||||||
).SetClass("w-full"),
|
"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",
|
||||||
comparison,
|
|
||||||
]).SetClass("flex flex-col"),
|
new Toggle(
|
||||||
unfiltered.features.map(ff => ff === undefined || ff.length === 0)
|
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
|
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])
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -7,6 +7,7 @@ import Translations from "../i18n/Translations";
|
||||||
import {VariableUiElement} from "../Base/VariableUIElement";
|
import {VariableUiElement} from "../Base/VariableUIElement";
|
||||||
import Toggle from "../Input/Toggle";
|
import Toggle from "../Input/Toggle";
|
||||||
import {UIElement} from "../UIElement";
|
import {UIElement} from "../UIElement";
|
||||||
|
import {FixedUiElement} from "../Base/FixedUiElement";
|
||||||
|
|
||||||
export interface FlowStep<T> extends BaseUIElement {
|
export interface FlowStep<T> extends BaseUIElement {
|
||||||
readonly IsValid: UIEventSource<boolean>
|
readonly IsValid: UIEventSource<boolean>
|
||||||
|
@ -95,10 +96,12 @@ export class FlowPanel<T> extends Toggle {
|
||||||
})
|
})
|
||||||
|
|
||||||
let elements: (BaseUIElement | string)[] = []
|
let elements: (BaseUIElement | string)[] = []
|
||||||
|
const isError = new UIEventSource(false)
|
||||||
if (initial !== undefined) {
|
if (initial !== undefined) {
|
||||||
// Startup the flow
|
// Startup the flow
|
||||||
elements = [
|
elements = [
|
||||||
initial,
|
initial,
|
||||||
|
|
||||||
new Combine([
|
new Combine([
|
||||||
backbutton,
|
backbutton,
|
||||||
new Toggle(
|
new Toggle(
|
||||||
|
@ -107,14 +110,25 @@ export class FlowPanel<T> extends Toggle {
|
||||||
Svg.back_svg().SetStyle("transform: rotate(180deg);"),
|
Svg.back_svg().SetStyle("transform: rotate(180deg);"),
|
||||||
isConfirm ? t.confirm : t.next
|
isConfirm ? t.confirm : t.next
|
||||||
).onClick(() => {
|
).onClick(() => {
|
||||||
const v = initial.Value.data;
|
try {
|
||||||
nextStep.setData(constructNextstep(v, backButtonForNextStep))
|
|
||||||
currentStepActive.setData(false)
|
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",
|
"Select a valid value to continue",
|
||||||
initial.IsValid
|
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"),
|
||||||
|
|
||||||
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,17 +1,14 @@
|
||||||
import Combine from "../Base/Combine";
|
import Combine from "../Base/Combine";
|
||||||
import {LoginToggle} from "../Popup/LoginButton";
|
|
||||||
import Toggle from "../Input/Toggle";
|
import Toggle from "../Input/Toggle";
|
||||||
import LanguagePicker from "../LanguagePicker";
|
import LanguagePicker from "../LanguagePicker";
|
||||||
import BackToIndex from "../BigComponents/BackToIndex";
|
import BackToIndex from "../BigComponents/BackToIndex";
|
||||||
import UserRelatedState from "../../Logic/State/UserRelatedState";
|
import UserRelatedState from "../../Logic/State/UserRelatedState";
|
||||||
import BaseUIElement from "../BaseUIElement";
|
import BaseUIElement from "../BaseUIElement";
|
||||||
import MoreScreen from "../BigComponents/MoreScreen";
|
|
||||||
import MinimapImplementation from "../Base/MinimapImplementation";
|
import MinimapImplementation from "../Base/MinimapImplementation";
|
||||||
import Translations from "../i18n/Translations";
|
import Translations from "../i18n/Translations";
|
||||||
import Constants from "../../Models/Constants";
|
|
||||||
import {FlowPanelFactory} from "./FlowStep";
|
import {FlowPanelFactory} from "./FlowStep";
|
||||||
import {RequestFile} from "./RequestFile";
|
import {RequestFile} from "./RequestFile";
|
||||||
import {DataPanel} from "./DataPanel";
|
import {PreviewPanel} from "./PreviewPanel";
|
||||||
import ConflationChecker from "./ConflationChecker";
|
import ConflationChecker from "./ConflationChecker";
|
||||||
import {AskMetadata} from "./AskMetadata";
|
import {AskMetadata} from "./AskMetadata";
|
||||||
import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
|
import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
|
||||||
|
@ -21,8 +18,11 @@ import {FixedUiElement} from "../Base/FixedUiElement";
|
||||||
import {VariableUiElement} from "../Base/VariableUIElement";
|
import {VariableUiElement} from "../Base/VariableUIElement";
|
||||||
import List from "../Base/List";
|
import List from "../Base/List";
|
||||||
import {CompareToAlreadyExistingNotes} from "./CompareToAlreadyExistingNotes";
|
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() {
|
constructor() {
|
||||||
const t = Translations.t.importHelper;
|
const t = Translations.t.importHelper;
|
||||||
|
|
||||||
|
@ -33,8 +33,11 @@ export default class ImportHelperGui extends LoginToggle {
|
||||||
|
|
||||||
const {flow, furthestStep, titles} =
|
const {flow, furthestStep, titles} =
|
||||||
FlowPanelFactory
|
FlowPanelFactory
|
||||||
.start("Select file", new RequestFile())
|
.start("Introduction", new Introdution())
|
||||||
.then("Inspect data", geojson => new DataPanel(state, geojson))
|
.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 open notes", v => new CompareToAlreadyExistingNotes(state, v))
|
||||||
.then("Compare with existing data", v => new ConflationChecker(state, v))
|
.then("Compare with existing data", v => new ConflationChecker(state, v))
|
||||||
.then("License and community check", v => new ConfirmProcess(v))
|
.then("License and community check", v => new ConfirmProcess(v))
|
||||||
|
@ -71,24 +74,12 @@ export default class ImportHelperGui extends LoginToggle {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
super(
|
super([
|
||||||
new Toggle(
|
|
||||||
new Combine([
|
new Combine([
|
||||||
leftBar,
|
leftBar,
|
||||||
flow.SetClass("m-8 w-full mb-24")
|
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)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
1
UI/ImportFlow/ImportInspector.ts
Normal file
1
UI/ImportFlow/ImportInspector.ts
Normal file
|
@ -0,0 +1 @@
|
||||||
|
export default class ImportInspector {}
|
|
@ -7,9 +7,11 @@ export class ImportUtils {
|
||||||
if (osmData?.features === undefined) {
|
if (osmData?.features === undefined) {
|
||||||
return undefined
|
return undefined
|
||||||
}
|
}
|
||||||
|
if(osmData.features.length === 0){
|
||||||
|
return {noNearby: toPartitionFeatureCollection.features, hasNearby: []}
|
||||||
|
}
|
||||||
const maxDist = cutoffDistanceInMeters.data
|
const maxDist = cutoffDistanceInMeters.data
|
||||||
|
|
||||||
|
|
||||||
const hasNearby = []
|
const hasNearby = []
|
||||||
const noNearby = []
|
const noNearby = []
|
||||||
for (const toImportElement of toPartitionFeatureCollection.features) {
|
for (const toImportElement of toPartitionFeatureCollection.features) {
|
||||||
|
|
20
UI/ImportFlow/Introdution.ts
Normal file
20
UI/ImportFlow/Introdution.ts
Normal file
|
@ -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<void> {
|
||||||
|
readonly IsValid: UIEventSource<boolean> = new UIEventSource<boolean>(true);
|
||||||
|
readonly Value: UIEventSource<void> = new UIEventSource<void>(undefined);
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super([
|
||||||
|
new Title( Translations.t.importHelper.title),
|
||||||
|
Translations.t.importHelper.description,
|
||||||
|
Translations.t.importHelper.importFormat,
|
||||||
|
]);
|
||||||
|
this.SetClass("flex flex-col")
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
52
UI/ImportFlow/LoginToImport.ts
Normal file
52
UI/ImportFlow/LoginToImport.ts
Normal file
|
@ -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<UserRelatedState> {
|
||||||
|
readonly IsValid: UIEventSource<boolean>;
|
||||||
|
readonly Value: UIEventSource<UserRelatedState>;
|
||||||
|
|
||||||
|
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<UserRelatedState>(state)
|
||||||
|
this.IsValid = isValid;
|
||||||
|
}
|
||||||
|
}
|
|
@ -17,13 +17,13 @@ import ShowDataMultiLayer from "../ShowDataLayer/ShowDataMultiLayer";
|
||||||
import FilteredLayer, {FilterState} from "../../Models/FilteredLayer";
|
import FilteredLayer, {FilterState} from "../../Models/FilteredLayer";
|
||||||
import StaticFeatureSource from "../../Logic/FeatureSource/Sources/StaticFeatureSource";
|
import StaticFeatureSource from "../../Logic/FeatureSource/Sources/StaticFeatureSource";
|
||||||
import Toggle from "../Input/Toggle";
|
import Toggle from "../Input/Toggle";
|
||||||
import Table from "../Base/Table";
|
|
||||||
import {VariableUiElement} from "../Base/VariableUIElement";
|
import {VariableUiElement} from "../Base/VariableUIElement";
|
||||||
import {FixedUiElement} from "../Base/FixedUiElement";
|
import {FixedUiElement} from "../Base/FixedUiElement";
|
||||||
import {FlowStep} from "./FlowStep";
|
import {FlowStep} from "./FlowStep";
|
||||||
import ScrollableFullScreen from "../Base/ScrollableFullScreen";
|
import ScrollableFullScreen from "../Base/ScrollableFullScreen";
|
||||||
import {AllTagsPanel} from "../SpecialVisualizations";
|
import {AllTagsPanel} from "../SpecialVisualizations";
|
||||||
import Title from "../Base/Title";
|
import Title from "../Base/Title";
|
||||||
|
import CheckBoxes from "../Input/Checkboxes";
|
||||||
|
|
||||||
class PreviewPanel extends ScrollableFullScreen {
|
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
|
* 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<boolean>;
|
public readonly IsValid: UIEventSource<boolean>;
|
||||||
public readonly Value: UIEventSource<{ bbox: BBox, layer: LayerConfig, geojson: any }>
|
public readonly Value: UIEventSource<{ bbox: BBox, layer: LayerConfig, geojson: any }>
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
state: UserRelatedState,
|
state: UserRelatedState,
|
||||||
geojson: { features: { properties: any, geometry: { coordinates: [number, number] } }[] }) {
|
geojson: { features: { properties: any, geometry: { coordinates: [number, number] } }[] }) {
|
||||||
const t = Translations.t.importHelper;
|
const t = Translations.t.importHelper.mapPreview;
|
||||||
|
|
||||||
const propertyKeys = new Set<string>()
|
const propertyKeys = new Set<string>()
|
||||||
for (const f of geojson.features) {
|
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 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: t.selectLayer, value: undefined}].concat(availableLayers.map(l => ({
|
||||||
shown: l.name,
|
shown: l.name,
|
||||||
value: l
|
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]))))
|
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([
|
super([
|
||||||
new Title(geojson.features.length + " features to import"),
|
new Title(t.title, 1),
|
||||||
layerPicker,
|
layerPicker,
|
||||||
new Toggle("Automatically detected layer", undefined, autodetected),
|
new Toggle(t.autodetected.SetClass("thank"), undefined, autodetected),
|
||||||
new Table(["", "Key", "Values", "Unique values seen"],
|
|
||||||
Array.from(propertyKeys).map(key => {
|
mismatchIndicator ,
|
||||||
const uniqueValues = Utils.Dedup(Utils.NoNull(geojson.features.map(f => f.properties[key])))
|
map,
|
||||||
uniqueValues.sort()
|
confirm
|
||||||
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
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
this.Value = bbox.map(bbox =>
|
this.Value = bbox.map(bbox =>
|
||||||
|
@ -157,13 +156,17 @@ export class DataPanel extends Combine implements FlowStep<{ bbox: BBox, layer:
|
||||||
geojson,
|
geojson,
|
||||||
layer: layerPicker.GetValue().data
|
layer: layerPicker.GetValue().data
|
||||||
}), [layerPicker.GetValue()])
|
}), [layerPicker.GetValue()])
|
||||||
|
|
||||||
this.IsValid = matching.map(matching => {
|
this.IsValid = matching.map(matching => {
|
||||||
if (matching === undefined) {
|
if (matching === undefined) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
if(confirm.GetValue().data.length !== 1){
|
||||||
|
return false
|
||||||
|
}
|
||||||
const diff = geojson.features.length - matching.length;
|
const diff = geojson.features.length - matching.length;
|
||||||
return diff === 0;
|
return diff === 0;
|
||||||
})
|
}, [confirm.GetValue()])
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
98
UI/ImportFlow/PreviewPanel.ts
Normal file
98
UI/ImportFlow/PreviewPanel.ts
Normal file
|
@ -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<boolean>;
|
||||||
|
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<string>()
|
||||||
|
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<string[]>(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)
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -9,6 +9,7 @@ import InputElementMap from "../Input/InputElementMap";
|
||||||
import BaseUIElement from "../BaseUIElement";
|
import BaseUIElement from "../BaseUIElement";
|
||||||
import FileSelectorButton from "../Input/FileSelectorButton";
|
import FileSelectorButton from "../Input/FileSelectorButton";
|
||||||
import {FlowStep} from "./FlowStep";
|
import {FlowStep} from "./FlowStep";
|
||||||
|
import { parse } from "papaparse";
|
||||||
|
|
||||||
class FileSelector extends InputElementMap<FileList, { name: string, contents: Promise<string> }> {
|
class FileSelector extends InputElementMap<FileList, { name: string, contents: Promise<string> }> {
|
||||||
constructor(label: BaseUIElement) {
|
constructor(label: BaseUIElement) {
|
||||||
|
@ -42,8 +43,8 @@ export class RequestFile extends Combine implements FlowStep<any> {
|
||||||
public readonly Value: UIEventSource<any>
|
public readonly Value: UIEventSource<any>
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
const t = Translations.t.importHelper;
|
const t = Translations.t.importHelper.selectFile;
|
||||||
const csvSelector = new FileSelector(new SubtleButton(undefined, t.selectFile))
|
const csvSelector = new FileSelector(new SubtleButton(undefined, t.description))
|
||||||
const loadedFiles = new VariableUiElement(csvSelector.GetValue().map(file => {
|
const loadedFiles = new VariableUiElement(csvSelector.GetValue().map(file => {
|
||||||
if (file === undefined) {
|
if (file === undefined) {
|
||||||
return t.noFilesLoaded.SetClass("alert")
|
return t.noFilesLoaded.SetClass("alert")
|
||||||
|
@ -59,39 +60,53 @@ export class RequestFile extends Combine implements FlowStep<any> {
|
||||||
return UIEventSource.FromPromise(v.contents)
|
return UIEventSource.FromPromise(v.contents)
|
||||||
}))
|
}))
|
||||||
|
|
||||||
const asGeoJson: UIEventSource<any | { error: string }> = text.map(src => {
|
const asGeoJson: UIEventSource<any | { error: string | BaseUIElement }> = text.map(src => {
|
||||||
if (src === undefined) {
|
if (src === undefined) {
|
||||||
return undefined
|
return undefined
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const parsed = JSON.parse(src)
|
const parsed = JSON.parse(src)
|
||||||
if (parsed["type"] !== "FeatureCollection") {
|
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")) {
|
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;
|
return parsed;
|
||||||
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// Loading as CSV
|
// Loading as CSV
|
||||||
const lines = src.split("\n")
|
var lines : string[][] = <any> parse(src).data;
|
||||||
const header = lines[0].split(",")
|
const header = lines[0]
|
||||||
lines.splice(0, 1)
|
lines.splice(0, 1)
|
||||||
if (header.indexOf("lat") < 0 || header.indexOf("lon") < 0) {
|
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 = []
|
const features = []
|
||||||
for (let i = 0; i < lines.length; i++) {
|
for (let i = 0; i < lines.length; i++) {
|
||||||
const line = lines[i];
|
const attrs = lines[i];
|
||||||
if (line.trim() === "") {
|
if(attrs.length == 0 || (attrs.length == 1 && attrs[0] == "")){
|
||||||
|
// empty line
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
const attrs = line.split(",")
|
|
||||||
const properties = {}
|
const properties = {}
|
||||||
for (let i = 0; i < header.length; i++) {
|
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"])]
|
const coordinates = [Number(properties["lon"]), Number(properties["lat"])]
|
||||||
delete properties["lat"]
|
delete properties["lat"]
|
||||||
|
@ -125,18 +140,21 @@ export class RequestFile extends Combine implements FlowStep<any> {
|
||||||
if (v?.error === undefined) {
|
if (v?.error === undefined) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
return new FixedUiElement(v?.error).SetClass("alert");
|
return v.error.Clone().SetClass("alert");
|
||||||
}))
|
}))
|
||||||
|
|
||||||
super([
|
super([
|
||||||
|
|
||||||
new Title(t.title, 1),
|
new Title(t.title, 1),
|
||||||
t.description,
|
t.fileFormatDescription,
|
||||||
|
t.fileFormatDescriptionCsv,
|
||||||
|
t.fileFormatDescriptionGeoJson,
|
||||||
csvSelector,
|
csvSelector,
|
||||||
loadedFiles,
|
loadedFiles,
|
||||||
errorIndicator
|
errorIndicator
|
||||||
|
|
||||||
]);
|
]);
|
||||||
|
this.SetClass("flex flex-col wi")
|
||||||
this.IsValid = asGeoJson.map(geojson => geojson !== undefined && geojson["error"] === undefined)
|
this.IsValid = asGeoJson.map(geojson => geojson !== undefined && geojson["error"] === undefined)
|
||||||
this.Value = asGeoJson
|
this.Value = asGeoJson
|
||||||
}
|
}
|
||||||
|
|
|
@ -455,7 +455,7 @@ export default class SpecialVisualizations {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return new Histogram(listSource, args[1], args[2], assignColors)
|
return new Histogram(listSource, args[1], args[2], {assignColor: assignColors})
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
974
langs/en.json
974
langs/en.json
|
@ -1,483 +1,513 @@
|
||||||
{
|
{
|
||||||
"image": {
|
"image": {
|
||||||
"addPicture": "Add picture",
|
"addPicture": "Add picture",
|
||||||
"uploadingPicture": "Uploading your picture…",
|
"uploadingPicture": "Uploading your picture…",
|
||||||
"uploadingMultiple": "Uploading {count} pictures…",
|
"uploadingMultiple": "Uploading {count} pictures…",
|
||||||
"pleaseLogin": "Please log in to add a picture",
|
"pleaseLogin": "Please log in to add a picture",
|
||||||
"willBePublished": "Your picture will be published ",
|
"willBePublished": "Your picture will be published ",
|
||||||
"cco": "in the public domain",
|
"cco": "in the public domain",
|
||||||
"ccbs": "under the CC-BY-SA-license",
|
"ccbs": "under the CC-BY-SA-license",
|
||||||
"ccb": "under the CC-BY-license",
|
"ccb": "under the CC-BY-license",
|
||||||
"ccoExplanation": "Adding a picture in the public domain implies that anyone can do anything with your picture",
|
"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",
|
"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",
|
"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.",
|
"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.",
|
"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!",
|
"uploadDone": "Your picture has been added. Thanks for helping out!",
|
||||||
"uploadMultipleDone": "{count} pictures have been added. Thanks for helping out!",
|
"uploadMultipleDone": "{count} pictures have been added. Thanks for helping out!",
|
||||||
"dontDelete": "Cancel",
|
"dontDelete": "Cancel",
|
||||||
"doDelete": "Remove image",
|
"doDelete": "Remove image",
|
||||||
"isDeleted": "Deleted",
|
"isDeleted": "Deleted",
|
||||||
"toBig": "Your image is too large as it is {actual_size}. Please use images of at most {max_size}"
|
"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": {
|
"explanations": {
|
||||||
"loadingData": "Loading data…",
|
"selectReason": "Please, select why this feature should be deleted",
|
||||||
"zoomIn": "Zoom in to view or edit the data",
|
"hardDelete": "This point will be deleted in OpenStreetMap. It can be recovered by an experienced contributor",
|
||||||
"ready": "Done!",
|
"softDelete": "This feature will be updated and hidden from this application. <span class='subtle'>{reason}</span>"
|
||||||
"retrying": "Loading data failed. Trying again in {count} seconds…"
|
|
||||||
},
|
},
|
||||||
"index": {
|
"readMessages": "You have unread messages. Read these before deleting a point - someone might have feedback"
|
||||||
"#": "These texts are shown above the theme buttons when no theme is loaded",
|
},
|
||||||
"title": "Welcome to MapComplete",
|
"general": {
|
||||||
"featuredThemeTitle": "Featured this week",
|
"logout": "Log out",
|
||||||
"intro": "MapComplete is an OpenStreetMap-viewer and editor, which shows you information about features of a specific theme and allows to update it.",
|
"next": "Next",
|
||||||
"pickTheme": "Pick a theme below to get started.",
|
"confirm": "Confirm",
|
||||||
"logIn": "Log in to see other themes you previously visited"
|
"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": {
|
"loginWithOpenStreetMap": "Login with OpenStreetMap",
|
||||||
"split": "Split",
|
"welcomeBack": "You are logged in, welcome back!",
|
||||||
"cancel": "Cancel",
|
"loginToStart": "Log in to answer this question",
|
||||||
"inviteToSplit": "Split this road in smaller segments. This allows to give different properties to parts of the road.",
|
"openStreetMapIntro": "<h3>An Open Map</h3><p>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.</p><p><b><a href='https://OpenStreetMap.org' target='_blank'>OpenStreetMap</a></b> is not the enemy map. The map data can be used freely (with <a href='https://osm.org/copyright' target='_blank'>attribution and publication of changes to that data</a>). 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.</p><p>Many people and apps already use OpenStreetMap: <a href='https://organicmaps.app/' target='_blank'>Organic Maps</a>, <a href='https://osmAnd.net' target='_blank'>OsmAnd</a>, but also the maps at Facebook, Instagram, Apple-maps and Bing-maps are (partly) powered by OpenStreetMap.</p>",
|
||||||
"loginToSplit": "You must be logged in to split a road",
|
"search": {
|
||||||
"splitTitle": "Choose on the map where to split this road",
|
"search": "Search a location",
|
||||||
"hasBeenSplit": "This way has been split"
|
"searching": "Searching…",
|
||||||
|
"nothing": "Nothing found…",
|
||||||
|
"error": "Something went wrong…"
|
||||||
},
|
},
|
||||||
"delete": {
|
"returnToTheMap": "Return to the map",
|
||||||
"delete": "Delete",
|
"save": "Save",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"isDeleted": "This feature is deleted",
|
"skip": "Skip this question",
|
||||||
"cannotBeDeleted": "This feature can not be deleted",
|
"oneSkippedQuestion": "One question is skipped",
|
||||||
"loginToDelete": "You must be logged in to delete a point",
|
"skippedQuestions": "Some questions are skipped",
|
||||||
"safeDelete": "This point can be safely deleted.",
|
"number": "number",
|
||||||
"isntAPoint": "Only points can be deleted, the selected feature is a way, area or relation.",
|
"osmLinkTooltip": "Browse this object on OpenStreetMap for history and more editing options",
|
||||||
"onlyEditedByLoggedInUser": "This point has only be edited by yourself, you can safely delete it.",
|
"add": {
|
||||||
"notEnoughExperience": "This point was made by someone else.",
|
"addNewMapLabel": "Click here to add a new item",
|
||||||
"useSomethingElse": "Use another OpenStreetMap-editor to delete it instead",
|
"disableFiltersExplanation": "Some features might be hidden by a filter",
|
||||||
"partOfOthers": "This point is part of some way or relation and can not be deleted directly.",
|
"disableFilters": "Disable all filters",
|
||||||
"loading": "Inspecting properties to check if this feature can be deleted.",
|
"addNew": "Add a new {category} here",
|
||||||
"whyDelete": "Why should this point be deleted?",
|
"presetInfo": "The new POI will have {tags}",
|
||||||
"reasons": {
|
"warnVisibleForEveryone": "Your addition will be visible for everyone",
|
||||||
"test": "This was a testing point - the feature was never actually there",
|
"title": "Add a new point?",
|
||||||
"disused": "This feature is disused or removed",
|
"intro": "You clicked somewhere where no data is known yet.<br/>",
|
||||||
"notFound": "This feature couldn't be found",
|
"pleaseLogin": "<a class='activate-osm-authentication'>Please log in to add a new point</a>",
|
||||||
"duplicate": "This point is a duplicate of another feature"
|
"zoomInFurther": "Zoom in further to add a point.",
|
||||||
},
|
"stillLoading": "The data is still loading. Please wait a bit before you add a new point.",
|
||||||
"explanations": {
|
"confirmIntro": "<h3>Add a {title} here?</h3>The point you create here will be <b>visible for everyone</b>. Please, only add things on to the map if they truly exist. A lot of applications use this data.",
|
||||||
"selectReason": "Please, select why this feature should be deleted",
|
"confirmButton": "Add a {category} here.<br/><div class='alert'>Your addition is visible for everyone</div>",
|
||||||
"hardDelete": "This point will be deleted in OpenStreetMap. It can be recovered by an experienced contributor",
|
"openLayerControl": "Open the layer control box",
|
||||||
"softDelete": "This feature will be updated and hidden from this application. <span class='subtle'>{reason}</span>"
|
"layerNotEnabled": "The layer {layer} is not enabled. Enable this layer to add a point",
|
||||||
},
|
"hasBeenImported": "This point has already been imported",
|
||||||
"readMessages": "You have unread messages. Read these before deleting a point - someone might have feedback"
|
"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 <b>test=true</b> or <b>backend=osm-test</b> 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": {
|
"pickLanguage": "Choose a language: ",
|
||||||
"next": "Next",
|
"about": "Easily edit and add OpenStreetMap for a certain theme",
|
||||||
"confirm": "Confirm",
|
"nameInlineQuestion": "The name of this {category} is $$$",
|
||||||
"back": "Back",
|
"noNameCategory": "{category} without a name",
|
||||||
"backToMapcomplete": "Back to the theme overview",
|
"questions": {
|
||||||
"loading": "Loading...",
|
"phoneNumberOf": "What is the phone number of {category}?",
|
||||||
"pdf": {
|
"phoneNumberIs": "The phone number of this {category} is <a href='tel:{phone}' target='_blank'>{phone}</a>",
|
||||||
"generatedWith": "Generated with MapComplete.osm.be",
|
"websiteOf": "What is the website of {category}?",
|
||||||
"attr": "Map data © OpenStreetMap Contributors, reusable under ODbL",
|
"websiteIs": "Website: <a href='{website}' target='_blank'>{website}</a>",
|
||||||
"attrBackground": "Background layer: {background}",
|
"emailOf": "What is the email address of {category}?",
|
||||||
"versionInfo": "v{version} - generated on {date}"
|
"emailIs": "The email address of this {category} is <a href='mailto:{email}' target='_blank'>{email}</a>"
|
||||||
|
},
|
||||||
|
"morescreen": {
|
||||||
|
"intro": "<h3>More thematic maps?</h3>Do you enjoy collecting geodata? <br/>There are more themes available.",
|
||||||
|
"requestATheme": "If you want a custom-built theme, request it in the issue tracker",
|
||||||
|
"streetcomplete": "Another, similar application is <a href='https://play.google.com/store/apps/details?id=de.westnordost.streetcomplete' class='underline hover:text-blue-800' class='underline hover:text-blue-800' target='_blank'>StreetComplete</a>.",
|
||||||
|
"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": "<h3>Share this map</h3> Share this map by copying the link below and sending it to friends and family:",
|
||||||
|
"addToHomeScreen": "<h3>Add to your home screen</h3>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": "<h3>Embed on your website</h3>Please, embed this map into your website. <br/>We encourage you to do it - you don't even have to ask permission. <br/> 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 <b>{name}</b>",
|
||||||
|
"fsIncludeCurrentLayers": "Include the current layer choices",
|
||||||
|
"fsIncludeCurrentLocation": "Include current location"
|
||||||
|
},
|
||||||
|
"attribution": {
|
||||||
|
"attributionTitle": "Attribution notice",
|
||||||
|
"attributionContent": "<p>All data is provided by <a href='https://osm.org' target='_blank'>OpenStreetMap</a>, freely reusable under <a href='https://osm.org/copyright' target='_blank'>the Open DataBase License</a>.</p>",
|
||||||
|
"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 <a href='https://github.com/pietervdvn/MapComplete/graphs/contributors' target='_blank'>{hiddenCount} more contributors</a>",
|
||||||
|
"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 <a href='https://www.openstreetmap.org/user/new' target='_blank'>create a new account</a>",
|
||||||
|
"noTagsSelected": "No tags selected",
|
||||||
|
"testing": "Testing - changes won't be saved",
|
||||||
|
"customThemeIntro": "<h3>Custom themes</h3>These are previously visited user-generated themes.",
|
||||||
|
"aboutMapcomplete": "<h3>About MapComplete</h3><p>Use it to add OpenStreetMap info on a <b>single theme.</b> Answer questions, and within minutes your contributions are available everywhere. The <b>theme maintainer</b> defines elements, questions and languages for it.</p><h3>Find out more</h3><p>MapComplete always <b>offers the next step</b> to learn more about OpenStreetMap.<ul><li>When embedded in a website, the iframe links to a full-screen MapComplete</li><li>The fullscreen version offers info about OpenStreetMap</li><li>Viewing works without login, but editing requires an OSM account.</li><li>If you are not logged in, you are asked to do so</li><li>Once you answered a single question, you can add new points to the map</li><li>After a while, actual OSM-tags are shown, later linking to the wiki</li></ul></p><br/><p>Did you notice <b>an issue</b>? Do you have a <b>feature request</b>? Want to <b>help translate</b>? Head over to <a href='https://github.com/pietervdvn/MapComplete' target='_blank'>the source code</a> or <a href='https://github.com/pietervdvn/MapComplete/issues' target='_blank'>issue tracker.</a> </p><p> Want to see <b>your progress</b>? Follow the edit count on <a href='{osmcha_link}' target='_blank' >OsmCha</a>.</p>",
|
||||||
|
"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": "<h3>Copyright notice</h3>The provided data is available under ODbL. Reusing it is gratis for any purpose, but <ul><li>the attribution <b>© OpenStreetMap contributors</b> is required</li><li>Any change must be use the license</li></ul> Please read the full <a href='https://www.openstreetmap.org/copyright' target='_blank'>copyright notice</a> 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 <i>{search}</i>",
|
||||||
|
"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": "<h3>Your personal theme</h3>Activate your favourite layers from all the official themes",
|
||||||
|
"loginNeeded": "<h3>Log in</h3>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": "<span>I am affiliated with this object</span><br/><span class='subtle'>Check if you are an owner, creator, employee, …</span>",
|
||||||
|
"affiliated_reviewer_warning": "(Affiliated review)",
|
||||||
|
"saving_review": "Saving…",
|
||||||
|
"saved": "<span class='thanks'>Review saved. Thanks for sharing!</span>",
|
||||||
|
"tos": "If you create a review, you agree to <a href='https://mangrove.reviews/terms' target='_blank'>the TOS and privacy policy of Mangrove.reviews</a>",
|
||||||
|
"attribution": "Reviews are powered by <a href='https://mangrove.reviews/' target='_blank'>Mangrove Reviews</a> and are available under <a href='https://mangrove.reviews/terms#8-licensing-of-content' target='_blank'>CC-BY 4.0</a>.",
|
||||||
|
"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 <a href='https://pietervdvn.goatcounter.com' target='_blank'>publicly available to anyone</a>",
|
||||||
|
"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: <ul><li>The changes you made</li><li>Your username</li><li>When this change is made</li><li>The theme you used while making the change</li><li>The language of the user interface</li><li>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</li></ul> Please refer to <a href='https://wiki.osmfoundation.org/wiki/Privacy_Policy' target='_blank'>the privacy policy on OpenStreetMap.org</a> 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 href='https://mapcomplete.osm.be/surveillance'>a theme showing surveillance cameras.</a> 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": "<p>Maintaining a set of up-to-date geodata is hard, error prone and expensive.<br/>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.</p><p>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.</p>",
|
||||||
|
"text1": "<p>MapComplete is the editor to make contributing data to OpenStreetMap easy.</p>",
|
||||||
|
"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 <b>it can be verified on the ground</b>.<br/> 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.<br/>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 <a href='https://osm.org/copyright' target='_blank'>full copyright text</a> can be summarized as following:",
|
||||||
|
"li0": "A product using OpenStreetMap data must give attribution.",
|
||||||
|
"li1": "OpenStreetMap-data must remain <i>open</i>. 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 <b>build it for you</b> on a decent budget. Get in touch via <a href='mailto:pietervdvn@posteo.net'>email</a>, <a href='https://github.com/pietervdvn/MapComplete/issues'>github</a> or <a href'https://www.openstreetmap.org/message/new/Pieter%20Vander%20Vennet'>send a message via osm.org</a>",
|
||||||
|
"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": "<p>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.</p><p>This data can be fetched from <b>OpenStreetMap</b> directly, but MapComplete can also use <b>external datasets</b> - 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": "<p>MapComplete is an easy to use <i>survey</i> 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.</p><p>We can setup a <b>custom survey tool</b>, asking precisely the data you need in a future-proof way.</p><p>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.</p>\n"
|
||||||
|
},
|
||||||
|
"internalUse": {
|
||||||
|
"title": "Using the data in internal processes",
|
||||||
|
"intro": "<p>Once the data is in OpenStreetMap, you'll probably want to use the data as well. Your MapComplete theme can have a convenient <i>export</i>-button, offering to download the data in many open formats usable in QGis, ArcGis, Excel, LibreOffice-calc, ...</p><p>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.</p><p>If an automated setup is needed, a free-to-use, community-run API is available.</p>"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"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 <b>verifiable on the ground</b> and are somewhat permanent. This implies that some data <i>cannot</i> 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 <a href='https://mangrove.reviews/'>Mangrove.reviews</a>, an openly licensed review website",
|
||||||
|
"li1": "Events of a few days, road works that are planned next month are thus <i>not</i> 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": "<p>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.</p><p>This contributor also has the right to ask for the dataset of the missing benches, which should be provided too.</p><p>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.</p><p>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).</p><p>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: </p>",
|
||||||
|
"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",
|
"usecaseGatheringOpenData": {
|
||||||
"welcomeBack": "You are logged in, welcome back!",
|
"title": "Gathering open data",
|
||||||
"loginToStart": "Log in to answer this question",
|
"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 <i>public domain</i>-license (where <b>all</b> 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). "
|
||||||
"openStreetMapIntro": "<h3>An Open Map</h3><p>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.</p><p><b><a href='https://OpenStreetMap.org' target='_blank'>OpenStreetMap</a></b> is not the enemy map. The map data can be used freely (with <a href='https://osm.org/copyright' target='_blank'>attribution and publication of changes to that data</a>). 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.</p><p>Many people and apps already use OpenStreetMap: <a href='https://organicmaps.app/' target='_blank'>Organic Maps</a>, <a href='https://osmAnd.net' target='_blank'>OsmAnd</a>, but also the maps at Facebook, Instagram, Apple-maps and Bing-maps are (partly) powered by OpenStreetMap.</p>",
|
|
||||||
"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.<br/>",
|
|
||||||
"pleaseLogin": "<a class='activate-osm-authentication'>Please log in to add a new point</a>",
|
|
||||||
"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": "<h3>Add a {title} here?</h3>The point you create here will be <b>visible for everyone</b>. Please, only add things on to the map if they truly exist. A lot of applications use this data.",
|
|
||||||
"confirmButton": "Add a {category} here.<br/><div class='alert'>Your addition is visible for everyone</div>",
|
|
||||||
"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 <b>test=true</b> or <b>backend=osm-test</b> 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 <a href='tel:{phone}' target='_blank'>{phone}</a>",
|
|
||||||
"websiteOf": "What is the website of {category}?",
|
|
||||||
"websiteIs": "Website: <a href='{website}' target='_blank'>{website}</a>",
|
|
||||||
"emailOf": "What is the email address of {category}?",
|
|
||||||
"emailIs": "The email address of this {category} is <a href='mailto:{email}' target='_blank'>{email}</a>"
|
|
||||||
},
|
|
||||||
"morescreen": {
|
|
||||||
"intro": "<h3>More thematic maps?</h3>Do you enjoy collecting geodata? <br/>There are more themes available.",
|
|
||||||
"requestATheme": "If you want a custom-built theme, request it in the issue tracker",
|
|
||||||
"streetcomplete": "Another, similar application is <a href='https://play.google.com/store/apps/details?id=de.westnordost.streetcomplete' class='underline hover:text-blue-800' class='underline hover:text-blue-800' target='_blank'>StreetComplete</a>.",
|
|
||||||
"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": "<h3>Share this map</h3> Share this map by copying the link below and sending it to friends and family:",
|
|
||||||
"addToHomeScreen": "<h3>Add to your home screen</h3>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": "<h3>Embed on your website</h3>Please, embed this map into your website. <br/>We encourage you to do it - you don't even have to ask permission. <br/> 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 <b>{name}</b>",
|
|
||||||
"fsIncludeCurrentLayers": "Include the current layer choices",
|
|
||||||
"fsIncludeCurrentLocation": "Include current location"
|
|
||||||
},
|
|
||||||
"attribution": {
|
|
||||||
"attributionTitle": "Attribution notice",
|
|
||||||
"attributionContent": "<p>All data is provided by <a href='https://osm.org' target='_blank'>OpenStreetMap</a>, freely reusable under <a href='https://osm.org/copyright' target='_blank'>the Open DataBase License</a>.</p>",
|
|
||||||
"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 <a href='https://github.com/pietervdvn/MapComplete/graphs/contributors' target='_blank'>{hiddenCount} more contributors</a>",
|
|
||||||
"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 <a href='https://www.openstreetmap.org/user/new' target='_blank'>create a new account</a>",
|
|
||||||
"noTagsSelected": "No tags selected",
|
|
||||||
"testing": "Testing - changes won't be saved",
|
|
||||||
"customThemeIntro": "<h3>Custom themes</h3>These are previously visited user-generated themes.",
|
|
||||||
"aboutMapcomplete": "<h3>About MapComplete</h3><p>Use it to add OpenStreetMap info on a <b>single theme.</b> Answer questions, and within minutes your contributions are available everywhere. The <b>theme maintainer</b> defines elements, questions and languages for it.</p><h3>Find out more</h3><p>MapComplete always <b>offers the next step</b> to learn more about OpenStreetMap.<ul><li>When embedded in a website, the iframe links to a full-screen MapComplete</li><li>The fullscreen version offers info about OpenStreetMap</li><li>Viewing works without login, but editing requires an OSM account.</li><li>If you are not logged in, you are asked to do so</li><li>Once you answered a single question, you can add new points to the map</li><li>After a while, actual OSM-tags are shown, later linking to the wiki</li></ul></p><br/><p>Did you notice <b>an issue</b>? Do you have a <b>feature request</b>? Want to <b>help translate</b>? Head over to <a href='https://github.com/pietervdvn/MapComplete' target='_blank'>the source code</a> or <a href='https://github.com/pietervdvn/MapComplete/issues' target='_blank'>issue tracker.</a> </p><p> Want to see <b>your progress</b>? Follow the edit count on <a href='{osmcha_link}' target='_blank' >OsmCha</a>.</p>",
|
|
||||||
"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": "<h3>Copyright notice</h3>The provided data is available under ODbL. Reusing it is gratis for any purpose, but <ul><li>the attribution <b>© OpenStreetMap contributors</b> is required</li><li>Any change must be use the license</li></ul> Please read the full <a href='https://www.openstreetmap.org/copyright' target='_blank'>copyright notice</a> 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 <i>{search}</i>",
|
|
||||||
"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}"
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"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": "<div class='thanks'>This feature has been handled! Thanks for your effort</div>"
|
||||||
|
},
|
||||||
|
"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: <br/><div class='literal-code'>[A bit of introduction]<br/>https://mapcomplete.osm.be/[themename].html?[parameters such as lat and lon]#import<br/>[all tags of the feature] </div>",
|
||||||
|
"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 <b class='code'>.csv</b> or a <b class='code'>.geojson</b> file",
|
||||||
|
"fileFormatDescriptionCsv": "In the CSV-file, there should be a column <span class='literal-code'>lat</span> and <span class='literal-code'>lon</span> 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": {
|
"mapPreview": {
|
||||||
"panelIntro": "<h3>Your personal theme</h3>Activate your favourite layers from all the official themes",
|
"title": "Map preview",
|
||||||
"loginNeeded": "<h3>Log in</h3>A personal layout is only available for OpenStreetMap users",
|
"autodetected": "The layer was automatically deducted based on the properties",
|
||||||
"reload": "Reload the data"
|
"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": {
|
"validateDataTitle": "Validate data",
|
||||||
"title": "{count} reviews",
|
"allAttributesSame": "All features to import have this tag",
|
||||||
"title_singular": "One review",
|
"someHaveSame": "{count} features to import have this tag, this is {percentage}% of the total",
|
||||||
"name_required": "A name is required in order to display and create reviews",
|
"inspectDataTitle": "Inspect data of {count} features to import",
|
||||||
"no_reviews_yet": "There are no reviews yet. Be the first to write one and help open data and the business!",
|
"inspectDidAutoDected": "Layer was chosen automatically",
|
||||||
"write_a_comment": "Leave a review…",
|
"inspectLooksCorrect": "These values look correct"
|
||||||
"no_rating": "No rating given",
|
}
|
||||||
"posting_as": "Posting as",
|
|
||||||
"i_am_affiliated": "<span>I am affiliated with this object</span><br/><span class='subtle'>Check if you are an owner, creator, employee, …</span>",
|
|
||||||
"affiliated_reviewer_warning": "(Affiliated review)",
|
|
||||||
"saving_review": "Saving…",
|
|
||||||
"saved": "<span class='thanks'>Review saved. Thanks for sharing!</span>",
|
|
||||||
"tos": "If you create a review, you agree to <a href='https://mangrove.reviews/terms' target='_blank'>the TOS and privacy policy of Mangrove.reviews</a>",
|
|
||||||
"attribution": "Reviews are powered by <a href='https://mangrove.reviews/' target='_blank'>Mangrove Reviews</a> and are available under <a href='https://mangrove.reviews/terms#8-licensing-of-content' target='_blank'>CC-BY 4.0</a>.",
|
|
||||||
"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 <a href='https://pietervdvn.goatcounter.com' target='_blank'>publicly available to anyone</a>",
|
|
||||||
"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: <ul><li>The changes you made</li><li>Your username</li><li>When this change is made</li><li>The theme you used while making the change</li><li>The language of the user interface</li><li>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</li></ul> Please refer to <a href='https://wiki.osmfoundation.org/wiki/Privacy_Policy' target='_blank'>the privacy policy on OpenStreetMap.org</a> 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 href='https://mapcomplete.osm.be/surveillance'>a theme showing surveillance cameras.</a> 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": "<p>Maintaining a set of up-to-date geodata is hard, error prone and expensive.<br/>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.</p><p>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.</p>",
|
|
||||||
"text1": "<p>MapComplete is the editor to make contributing data to OpenStreetMap easy.</p>",
|
|
||||||
"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 <b>it can be verified on the ground</b>.<br/> 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.<br/>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 <a href='https://osm.org/copyright' target='_blank'>full copyright text</a> can be summarized as following:",
|
|
||||||
"li0": "A product using OpenStreetMap data must give attribution.",
|
|
||||||
"li1": "OpenStreetMap-data must remain <i>open</i>. 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 <b>build it for you</b> on a decent budget. Get in touch via <a href='mailto:pietervdvn@posteo.net'>email</a>, <a href='https://github.com/pietervdvn/MapComplete/issues'>github</a> or <a href'https://www.openstreetmap.org/message/new/Pieter%20Vander%20Vennet'>send a message via osm.org</a>",
|
|
||||||
"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": "<p>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.</p><p>This data can be fetched from <b>OpenStreetMap</b> directly, but MapComplete can also use <b>external datasets</b> - 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": "<p>MapComplete is an easy to use <i>survey</i> 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.</p><p>We can setup a <b>custom survey tool</b>, asking precisely the data you need in a future-proof way.</p><p>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.</p>\n"
|
|
||||||
},
|
|
||||||
"internalUse": {
|
|
||||||
"title": "Using the data in internal processes",
|
|
||||||
"intro": "<p>Once the data is in OpenStreetMap, you'll probably want to use the data as well. Your MapComplete theme can have a convenient <i>export</i>-button, offering to download the data in many open formats usable in QGis, ArcGis, Excel, LibreOffice-calc, ...</p><p>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.</p><p>If an automated setup is needed, a free-to-use, community-run API is available.</p>"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"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 <b>verifiable on the ground</b> and are somewhat permanent. This implies that some data <i>cannot</i> 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 <a href='https://mangrove.reviews/'>Mangrove.reviews</a>, an openly licensed review website",
|
|
||||||
"li1": "Events of a few days, road works that are planned next month are thus <i>not</i> 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": "<p>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.</p><p>This contributor also has the right to ask for the dataset of the missing benches, which should be provided too.</p><p>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.</p><p>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).</p><p>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: </p>",
|
|
||||||
"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 <i>public domain</i>-license (where <b>all</b> 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": "<div class='thanks'>This feature has been handled! Thanks for your effort</div>"
|
|
||||||
},
|
|
||||||
"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"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
34
package-lock.json
generated
34
package-lock.json
generated
|
@ -20,6 +20,7 @@
|
||||||
"@types/leaflet-providers": "^1.2.0",
|
"@types/leaflet-providers": "^1.2.0",
|
||||||
"@types/leaflet.markercluster": "^1.4.3",
|
"@types/leaflet.markercluster": "^1.4.3",
|
||||||
"@types/lz-string": "^1.3.34",
|
"@types/lz-string": "^1.3.34",
|
||||||
|
"@types/papaparse": "^5.3.1",
|
||||||
"@types/prompt-sync": "^4.1.0",
|
"@types/prompt-sync": "^4.1.0",
|
||||||
"@types/wikidata-sdk": "^6.1.0",
|
"@types/wikidata-sdk": "^6.1.0",
|
||||||
"country-language": "^0.1.7",
|
"country-language": "^0.1.7",
|
||||||
|
@ -44,6 +45,7 @@
|
||||||
"opening_hours": "^3.6.0",
|
"opening_hours": "^3.6.0",
|
||||||
"osm-auth": "^1.0.2",
|
"osm-auth": "^1.0.2",
|
||||||
"osmtogeojson": "^3.0.0-beta.4",
|
"osmtogeojson": "^3.0.0-beta.4",
|
||||||
|
"papaparse": "^5.3.1",
|
||||||
"parcel": "^1.2.4",
|
"parcel": "^1.2.4",
|
||||||
"prompt-sync": "^4.2.0",
|
"prompt-sync": "^4.2.0",
|
||||||
"svg-resizer": "github:vieron/svg-resizer",
|
"svg-resizer": "github:vieron/svg-resizer",
|
||||||
|
@ -3254,8 +3256,15 @@
|
||||||
"node_modules/@types/node": {
|
"node_modules/@types/node": {
|
||||||
"version": "7.10.14",
|
"version": "7.10.14",
|
||||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-7.10.14.tgz",
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-7.10.14.tgz",
|
||||||
"integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA==",
|
"integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA=="
|
||||||
"dev": true
|
},
|
||||||
|
"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": {
|
"node_modules/@types/parse-json": {
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
|
@ -10090,6 +10099,11 @@
|
||||||
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
|
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
|
||||||
"integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU="
|
"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": {
|
"node_modules/parcel": {
|
||||||
"version": "1.12.4",
|
"version": "1.12.4",
|
||||||
"resolved": "https://registry.npmjs.org/parcel/-/parcel-1.12.4.tgz",
|
"resolved": "https://registry.npmjs.org/parcel/-/parcel-1.12.4.tgz",
|
||||||
|
@ -19227,8 +19241,15 @@
|
||||||
"@types/node": {
|
"@types/node": {
|
||||||
"version": "7.10.14",
|
"version": "7.10.14",
|
||||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-7.10.14.tgz",
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-7.10.14.tgz",
|
||||||
"integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA==",
|
"integrity": "sha512-29GS75BE8asnTno3yB6ubOJOO0FboExEqNJy4bpz0GSmW/8wPTNL4h9h63c6s1uTrOopCmJYe/4yJLh5r92ZUA=="
|
||||||
"dev": true
|
},
|
||||||
|
"@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": {
|
"@types/parse-json": {
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
|
@ -24635,6 +24656,11 @@
|
||||||
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
|
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
|
||||||
"integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU="
|
"integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU="
|
||||||
},
|
},
|
||||||
|
"papaparse": {
|
||||||
|
"version": "5.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/papaparse/-/papaparse-5.3.1.tgz",
|
||||||
|
"integrity": "sha512-Dbt2yjLJrCwH2sRqKFFJaN5XgIASO9YOFeFP8rIBRG2Ain8mqk5r1M6DkfvqEVozVcz3r3HaUGw253hA1nLIcA=="
|
||||||
|
},
|
||||||
"parcel": {
|
"parcel": {
|
||||||
"version": "1.12.4",
|
"version": "1.12.4",
|
||||||
"resolved": "https://registry.npmjs.org/parcel/-/parcel-1.12.4.tgz",
|
"resolved": "https://registry.npmjs.org/parcel/-/parcel-1.12.4.tgz",
|
||||||
|
|
|
@ -44,7 +44,7 @@
|
||||||
"lint": "tslint --project . -c tslint.json '**.ts' ",
|
"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)",
|
"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",
|
"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": [
|
"keywords": [
|
||||||
"OpenStreetMap",
|
"OpenStreetMap",
|
||||||
|
@ -64,6 +64,7 @@
|
||||||
"@types/leaflet-providers": "^1.2.0",
|
"@types/leaflet-providers": "^1.2.0",
|
||||||
"@types/leaflet.markercluster": "^1.4.3",
|
"@types/leaflet.markercluster": "^1.4.3",
|
||||||
"@types/lz-string": "^1.3.34",
|
"@types/lz-string": "^1.3.34",
|
||||||
|
"@types/papaparse": "^5.3.1",
|
||||||
"@types/prompt-sync": "^4.1.0",
|
"@types/prompt-sync": "^4.1.0",
|
||||||
"@types/wikidata-sdk": "^6.1.0",
|
"@types/wikidata-sdk": "^6.1.0",
|
||||||
"country-language": "^0.1.7",
|
"country-language": "^0.1.7",
|
||||||
|
@ -88,6 +89,7 @@
|
||||||
"opening_hours": "^3.6.0",
|
"opening_hours": "^3.6.0",
|
||||||
"osm-auth": "^1.0.2",
|
"osm-auth": "^1.0.2",
|
||||||
"osmtogeojson": "^3.0.0-beta.4",
|
"osmtogeojson": "^3.0.0-beta.4",
|
||||||
|
"papaparse": "^5.3.1",
|
||||||
"parcel": "^1.2.4",
|
"parcel": "^1.2.4",
|
||||||
"prompt-sync": "^4.2.0",
|
"prompt-sync": "^4.2.0",
|
||||||
"svg-resizer": "github:vieron/svg-resizer",
|
"svg-resizer": "github:vieron/svg-resizer",
|
||||||
|
|
|
@ -72,10 +72,16 @@ class LayerOverviewUtils {
|
||||||
const dict = new Map<string, TagRenderingConfigJson>();
|
const dict = new Map<string, TagRenderingConfigJson>();
|
||||||
|
|
||||||
for (const key in questions["default"]) {
|
for (const key in questions["default"]) {
|
||||||
|
if(key==="id"){
|
||||||
|
continue
|
||||||
|
}
|
||||||
questions[key].id = key;
|
questions[key].id = key;
|
||||||
dict.set(key, <TagRenderingConfigJson>questions[key])
|
dict.set(key, <TagRenderingConfigJson>questions[key])
|
||||||
}
|
}
|
||||||
for (const key in icons["default"]) {
|
for (const key in icons["default"]) {
|
||||||
|
if(key==="id"){
|
||||||
|
continue
|
||||||
|
}
|
||||||
if (typeof icons[key] !== "object") {
|
if (typeof icons[key] !== "object") {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -105,7 +111,7 @@ class LayerOverviewUtils {
|
||||||
"themes": Array.from(sharedThemes.values())
|
"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())}))
|
||||||
|
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue