diff --git a/404.html b/404.html
new file mode 100644
index 0000000000..fa1f722f3a
--- /dev/null
+++ b/404.html
@@ -0,0 +1,56 @@
+
+
+
+
+
+
+
+
+
+
+ MapComplete - page not found
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Not found...
+
+
+
+
+
+
+
diff --git a/Customizations/AllKnownLayers.ts b/Customizations/AllKnownLayers.ts
deleted file mode 100644
index 0f32405e1a..0000000000
--- a/Customizations/AllKnownLayers.ts
+++ /dev/null
@@ -1,131 +0,0 @@
-import * as known_layers from "../assets/generated/known_layers_and_themes.json"
-import {Utils} from "../Utils";
-import LayerConfig from "../Models/ThemeConfig/LayerConfig";
-import {TagRenderingConfigJson} from "../Models/ThemeConfig/Json/TagRenderingConfigJson";
-import SharedTagRenderings from "./SharedTagRenderings";
-import {LayerConfigJson} from "../Models/ThemeConfig/Json/LayerConfigJson";
-import WithContextLoader from "../Models/ThemeConfig/WithContextLoader";
-
-export default class AllKnownLayers {
-
- public static inited = (_ => {
- WithContextLoader.getKnownTagRenderings = (id => AllKnownLayers.getTagRendering(id))
- return true
- })()
-
- public static runningGenerateScript = false;
-
- // Must be below the list...
- public static sharedLayers: Map = AllKnownLayers.getSharedLayers();
- public static sharedLayersJson: Map = AllKnownLayers.getSharedLayersJson();
-
-
- public static added_by_default: string[] = ["gps_location", "gps_location_history", "home_location", "gps_track"]
- public static no_include: string[] = ["conflation", "left_right_style", "split_point","current_view","matchpoint"]
- /**
- * Layer IDs of layers which have special properties through built-in hooks
- */
- public static priviliged_layers: string[] = [...AllKnownLayers.added_by_default, "type_node", ...AllKnownLayers.no_include]
-
- /**
- * Gets the appropriate tagRenderingJSON
- * Allows to steal them from other layers.
- * This will add the tags of the layer to the configuration though!
- * @param renderingId
- */
- static getTagRendering(renderingId: string): TagRenderingConfigJson[] {
- if (renderingId.indexOf(".") < 0) {
- const found = SharedTagRenderings.SharedTagRenderingJson.get(renderingId)
- if(found === undefined){
- return []
- }
- return [found]
- }
-
- const [layerId, id] = renderingId.split(".")
- const layer = AllKnownLayers.getSharedLayersJson().get(layerId)
- if (layer === undefined) {
- if (AllKnownLayers.runningGenerateScript) {
- // Probably generating the layer overview
- return [{
- id: "dummy"
- }]
- }
- throw "Builtin layer " + layerId + " not found"
- }
-
- const renderings = layer?.tagRenderings ?? []
- if (id === "*") {
- return JSON.parse(JSON.stringify(renderings))
- }
-
- const selectByGroup = id.startsWith("*")
- const expectedGroupName = id.substring(1)
-
- const allValidValues = []
- for (const rendering of renderings) {
- if ((!selectByGroup && rendering["id"] === id) || (selectByGroup && rendering["group"] === expectedGroupName)) {
- const found = JSON.parse(JSON.stringify(rendering))
- if (found.condition === undefined) {
- found.condition = layer.source.osmTags
- } else {
- found.condition = {and: [found.condition, layer.source.osmTags]}
- }
- allValidValues.push(found)
- }
- }
- if(allValidValues.length === 0){
-
- throw `The rendering with id ${id} was not found in the builtin layer ${layerId}. Try one of ${Utils.NoNull(renderings.map(r => r["id"])).join(", ")}`
- }
- return allValidValues
- }
-
- private static getSharedLayers(): Map {
- const sharedLayers = new Map();
- for (const layer of known_layers.layers) {
- try {
- // @ts-ignore
- const parsed = new LayerConfig(layer, "shared_layers")
- sharedLayers.set(layer.id, parsed);
- } catch (e) {
- if (!Utils.runningFromConsole) {
- console.error("CRITICAL: Could not parse a layer configuration!", layer.id, " due to", e)
- }
- }
- }
-
- for (const layout of known_layers.themes) {
- for (const layer of layout.layers) {
- if (typeof layer === "string") {
- continue;
- }
- if (layer.builtin !== undefined) {
- // This is a builtin layer of which stuff is overridden - skip
- continue;
- }
- try {
- const parsed = new LayerConfig(layer, "shared_layer_in_theme")
- sharedLayers.set(layer.id, parsed);
- sharedLayers[layer.id] = parsed;
- } catch (e) {
- if (!Utils.runningFromConsole) {
- console.error("Could not parse a layer in theme ", layout.id, "due to", e)
- }
- }
- }
- }
-
- return sharedLayers;
- }
-
- private static getSharedLayersJson(): Map {
- const sharedLayers = new Map();
- for (const layer of known_layers.layers) {
- sharedLayers.set(layer.id, layer);
- sharedLayers[layer.id] = layer;
- }
- return sharedLayers;
- }
-
-}
diff --git a/Customizations/AllKnownLayouts.ts b/Customizations/AllKnownLayouts.ts
index 22964023cc..381069908e 100644
--- a/Customizations/AllKnownLayouts.ts
+++ b/Customizations/AllKnownLayouts.ts
@@ -1,4 +1,3 @@
-import AllKnownLayers from "./AllKnownLayers";
import * as known_themes from "../assets/generated/known_layers_and_themes.json"
import LayoutConfig from "../Models/ThemeConfig/LayoutConfig";
import LayerConfig from "../Models/ThemeConfig/LayerConfig";
@@ -7,9 +6,29 @@ import Combine from "../UI/Base/Combine";
import Title from "../UI/Base/Title";
import List from "../UI/Base/List";
import DependencyCalculator from "../Models/ThemeConfig/DependencyCalculator";
+import Constants from "../Models/Constants";
+import {Utils} from "../Utils";
export class AllKnownLayouts {
+ // Must be below the list...
+ private static sharedLayers: Map = AllKnownLayouts.getSharedLayers();
+ private static getSharedLayers(): Map {
+ const sharedLayers = new Map();
+ for (const layer of known_themes.layers) {
+ try {
+ // @ts-ignore
+ const parsed = new LayerConfig(layer, "shared_layers")
+ sharedLayers.set(layer.id, parsed);
+ } catch (e) {
+ if (!Utils.runningFromConsole) {
+ console.error("CRITICAL: Could not parse a layer configuration!", layer.id, " due to", e)
+ }
+ }
+ }
+
+ return sharedLayers;
+ }
public static allKnownLayouts: Map = AllKnownLayouts.AllLayouts();
public static layoutsList: LayoutConfig[] = AllKnownLayouts.GenerateOrderedList(AllKnownLayouts.allKnownLayouts);
@@ -35,14 +54,14 @@ export class AllKnownLayouts {
}
public static GenLayerOverviewText(): BaseUIElement {
- for (const id of AllKnownLayers.priviliged_layers) {
- if (!AllKnownLayers.sharedLayers.has(id)) {
+ for (const id of Constants.priviliged_layers) {
+ if (!AllKnownLayouts.sharedLayers.has(id)) {
throw "Priviliged layer definition not found: " + id
}
}
- const allLayers: LayerConfig[] = Array.from(AllKnownLayers.sharedLayers.values())
- .filter(layer => AllKnownLayers.priviliged_layers.indexOf(layer.id) < 0)
+ const allLayers: LayerConfig[] = Array.from(AllKnownLayouts.sharedLayers.values())
+ .filter(layer => Constants.priviliged_layers.indexOf(layer.id) < 0)
const builtinLayerIds: Set = new Set()
allLayers.forEach(l => builtinLayerIds.add(l.id))
@@ -89,10 +108,10 @@ export class AllKnownLayouts {
new Title("Special and other useful layers", 1),
"MapComplete has a few data layers available in the theme which have special properties through builtin-hooks. Furthermore, there are some normal layers (which are built from normal Theme-config files) but are so general that they get a mention here.",
new Title("Priviliged layers", 1),
- new List(AllKnownLayers.priviliged_layers.map(id => "[" + id + "](#" + id + ")")),
- ...AllKnownLayers.priviliged_layers
- .map(id => AllKnownLayers.sharedLayers.get(id))
- .map((l) => l.GenerateDocumentation(themesPerLayer.get(l.id), layerIsNeededBy, DependencyCalculator.getLayerDependencies(l),AllKnownLayers.added_by_default.indexOf(l.id) >= 0, AllKnownLayers.no_include.indexOf(l.id) < 0)),
+ new List(Constants.priviliged_layers.map(id => "[" + id + "](#" + id + ")")),
+ ...Constants.priviliged_layers
+ .map(id => AllKnownLayouts.sharedLayers.get(id))
+ .map((l) => l.GenerateDocumentation(themesPerLayer.get(l.id), layerIsNeededBy, DependencyCalculator.getLayerDependencies(l),Constants.added_by_default.indexOf(l.id) >= 0, Constants.no_include.indexOf(l.id) < 0)),
new Title("Normal layers", 1),
"The following layers are included in MapComplete",
new Title("Frequently reused layers", 2),
@@ -108,53 +127,26 @@ export class AllKnownLayouts {
}
private static GenerateOrderedList(allKnownLayouts: Map): LayoutConfig[] {
- const keys = ["personal", "cyclofix", "hailhydrant", "bookcases", "toilets", "aed"]
const list = []
- for (const key of keys) {
- list.push(allKnownLayouts.get(key))
- }
allKnownLayouts.forEach((layout, key) => {
- if (keys.indexOf(key) < 0) {
- list.push(layout)
- }
+ list.push(layout)
})
return list;
}
- private static AddGhostBikes(layout: LayoutConfig): LayoutConfig {
- const now = new Date();
- const m = now.getMonth() + 1;
- const day = new Date().getDate() + 1;
- const date = day + "/" + m;
- if (date === "31/10" || date === "1/11" || date === "2/11") {
- console.log("The current date is ", date, ", which means we remember our dead")
- // Around Halloween/Fiesta de muerte/Allerzielen, we remember the dead
- layout.layers.push(
- AllKnownLayers.sharedLayers.get("ghost_bike")
- );
-
- }
- return layout;
-
- }
-
private static AllLayouts(): Map {
const dict: Map = new Map();
for (const layoutConfigJson of known_themes.themes) {
// @ts-ignore
const layout = new LayoutConfig(layoutConfigJson, true)
-
- if (layout.id === "cyclofix") {
- AllKnownLayouts.AddGhostBikes(layout)
- }
dict.set(layout.id, layout)
for (let i = 0; i < layout.layers.length; i++) {
let layer = layout.layers[i];
if (typeof (layer) === "string") {
- layer = layout.layers[i] = AllKnownLayers.sharedLayers.get(layer);
+ layer = layout.layers[i] = AllKnownLayouts.sharedLayers.get(layer);
if (layer === undefined) {
- console.log("Defined layers are ", AllKnownLayers.sharedLayers.keys())
+ console.log("Defined layers are ", AllKnownLayouts.sharedLayers.keys())
throw `Layer ${layer} was not found or defined - probably a type was made`
}
}
diff --git a/Customizations/SharedTagRenderings.ts b/Customizations/SharedTagRenderings.ts
index 4344b6e893..ad0aa83e2a 100644
--- a/Customizations/SharedTagRenderings.ts
+++ b/Customizations/SharedTagRenderings.ts
@@ -37,8 +37,10 @@ export default class SharedTagRenderings {
for (const key in icons) {
dict.set(key, icons[key])
}
-
- dict.forEach((value, key) => value.id = key)
+
+ dict.forEach((value, key) => {
+ value.id = value.id ?? key;
+ })
return dict;
}
diff --git a/Docs/BuiltinLayers.md b/Docs/BuiltinLayers.md
index 7e61e352d0..659005e751 100644
--- a/Docs/BuiltinLayers.md
+++ b/Docs/BuiltinLayers.md
@@ -29,8 +29,6 @@
* [Themes using this layer](#themes-using-this-layer)
+ [walls_and_buildings](#walls_and_buildings)
* [Themes using this layer](#themes-using-this-layer)
- + [all_streets](#all_streets)
- * [Themes using this layer](#themes-using-this-layer)
+ [ambulancestation](#ambulancestation)
* [Themes using this layer](#themes-using-this-layer)
+ [artwork](#artwork)
@@ -113,36 +111,6 @@
* [Themes using this layer](#themes-using-this-layer)
+ [waste_basket](#waste_basket)
* [Themes using this layer](#themes-using-this-layer)
- + [caravansites](#caravansites)
- * [Themes using this layer](#themes-using-this-layer)
- + [dumpstations](#dumpstations)
- * [Themes using this layer](#themes-using-this-layer)
- + [climbing_club](#climbing_club)
- * [Themes using this layer](#themes-using-this-layer)
- + [climbing_gym](#climbing_gym)
- * [Themes using this layer](#themes-using-this-layer)
- + [climbing_route](#climbing_route)
- * [Themes using this layer](#themes-using-this-layer)
- + [climbing](#climbing)
- * [Themes using this layer](#themes-using-this-layer)
- + [maybe_climbing](#maybe_climbing)
- * [Themes using this layer](#themes-using-this-layer)
- + [fietsstraat](#fietsstraat)
- * [Themes using this layer](#themes-using-this-layer)
- + [toekomstige_fietsstraat](#toekomstige_fietsstraat)
- * [Themes using this layer](#themes-using-this-layer)
- + [facadegardens](#facadegardens)
- * [Themes using this layer](#themes-using-this-layer)
- + [hackerspaces](#hackerspaces)
- * [Themes using this layer](#themes-using-this-layer)
- + [windturbine](#windturbine)
- * [Themes using this layer](#themes-using-this-layer)
- + [postboxes](#postboxes)
- * [Themes using this layer](#themes-using-this-layer)
- + [postoffices](#postoffices)
- * [Themes using this layer](#themes-using-this-layer)
- + [lit_streets](#lit_streets)
- * [Themes using this layer](#themes-using-this-layer)
MapComplete has a few data layers available in the theme which have special properties through builtin-hooks. Furthermore, there are some normal layers (which are built from normal Theme-config files) but are so general that they get a mention here.
@@ -235,7 +203,6 @@ This is a priviliged meta_layer which exports _every_ point in OSM. This only wo
- Not rendered on the map by default. If you want to rendering this on the map, override `mapRenderings`
- - This layer is needed as dependency for layer [GRB](#GRB)
### conflation
@@ -320,7 +287,6 @@ The default rendering for a locationInput which snaps onto another object
- [food](#food)
- [map](#map)
- [walls_and_buildings](#walls_and_buildings)
- - [all_streets](#all_streets)
### bicycle_library
@@ -456,29 +422,6 @@ Special builtin layer providing all walls and buildings. This layer is useful in
- [surveillance](https://mapcomplete.osm.be/surveillance)
-### all_streets
-
-
-
-[Go to the source code](../assets/layers/all_streets/all_streets.json)
-
-
-
- - Not rendered on the map by default. If you want to rendering this on the map, override `mapRenderings`
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [cyclestreets](https://mapcomplete.osm.be/cyclestreets)
- - [street_lighting](https://mapcomplete.osm.be/street_lighting)
-
-
- [ambulancestation](#ambulancestation)
- [artwork](#artwork)
- [barrier](#barrier)
@@ -520,21 +463,6 @@ Special builtin layer providing all walls and buildings. This layer is useful in
- [toilet](#toilet)
- [tree_node](#tree_node)
- [waste_basket](#waste_basket)
- - [caravansites](#caravansites)
- - [dumpstations](#dumpstations)
- - [climbing_club](#climbing_club)
- - [climbing_gym](#climbing_gym)
- - [climbing_route](#climbing_route)
- - [climbing](#climbing)
- - [maybe_climbing](#maybe_climbing)
- - [fietsstraat](#fietsstraat)
- - [toekomstige_fietsstraat](#toekomstige_fietsstraat)
- - [facadegardens](#facadegardens)
- - [hackerspaces](#hackerspaces)
- - [windturbine](#windturbine)
- - [postboxes](#postboxes)
- - [postoffices](#postoffices)
- - [lit_streets](#lit_streets)
### ambulancestation
@@ -985,7 +913,6 @@ A layer showing defibrillators which can be used in case of emergency. This cont
- This layer will automatically load [walls_and_buildings](#walls_and_buildings) into the layout as it depends on it: a preset snaps to this layer (presets[1])
- - This layer is needed as dependency for layer [Brugge](#Brugge)
@@ -1418,7 +1345,7 @@ A layer showing street lights
- - This layer is needed as dependency for layer [Assen](#Assen)
+
@@ -1528,358 +1455,4 @@ This is a public waste basket, thrash can, where you can throw away your thrash.
- [waste_basket](https://mapcomplete.osm.be/waste_basket)
-### caravansites
-
-
-
-camper sites
-
-[Go to the source code](../assets/layers/caravansites/caravansites.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [campersite](https://mapcomplete.osm.be/campersite)
-
-
-### dumpstations
-
-
-
-Sanitary dump stations
-
-[Go to the source code](../assets/layers/dumpstations/dumpstations.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [campersite](https://mapcomplete.osm.be/campersite)
-
-
-### climbing_club
-
-
-
-A climbing club or organisations
-
-[Go to the source code](../assets/layers/climbing_club/climbing_club.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [climbing](https://mapcomplete.osm.be/climbing)
-
-
-### climbing_gym
-
-
-
-A climbing gym
-
-[Go to the source code](../assets/layers/climbing_gym/climbing_gym.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [climbing](https://mapcomplete.osm.be/climbing)
-
-
-### climbing_route
-
-
-
-[Go to the source code](../assets/layers/climbing_route/climbing_route.json)
-
-
-
- - This layer is needed as dependency for layer [climbing](#climbing)
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [climbing](https://mapcomplete.osm.be/climbing)
-
-
-### climbing
-
-
-
-A climbing opportunity
-
-[Go to the source code](../assets/layers/climbing/climbing.json)
-
-
-
- - This layer will automatically load [climbing_route](#climbing_route) into the layout as it depends on it: A calculated tag loads features from this layer (calculatedTag[0] which calculates the value for _contained_climbing_routes_properties)
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [climbing](https://mapcomplete.osm.be/climbing)
-
-
-### maybe_climbing
-
-
-
-A climbing opportunity?
-
-[Go to the source code](../assets/layers/maybe_climbing/maybe_climbing.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [climbing](https://mapcomplete.osm.be/climbing)
-
-
-### fietsstraat
-
-
-
-A cyclestreet is a street where motorized traffic is not allowed to overtake a cyclist
-
-[Go to the source code](../assets/layers/fietsstraat/fietsstraat.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [cyclestreets](https://mapcomplete.osm.be/cyclestreets)
-
-
-### toekomstige_fietsstraat
-
-
-
-This street will become a cyclestreet soon
-
-[Go to the source code](../assets/layers/toekomstige_fietsstraat/toekomstige_fietsstraat.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [cyclestreets](https://mapcomplete.osm.be/cyclestreets)
-
-
-### facadegardens
-
-
-
-Facade gardens
-
-[Go to the source code](../assets/layers/facadegardens/facadegardens.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [facadegardens](https://mapcomplete.osm.be/facadegardens)
-
-
-### hackerspaces
-
-
-
-Hackerspace
-
-[Go to the source code](../assets/layers/hackerspaces/hackerspaces.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [hackerspaces](https://mapcomplete.osm.be/hackerspaces)
-
-
-### windturbine
-
-
-
-[Go to the source code](../assets/layers/windturbine/windturbine.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [openwindpowermap](https://mapcomplete.osm.be/openwindpowermap)
-
-
-### postboxes
-
-
-
-The layer showing postboxes.
-
-[Go to the source code](../assets/layers/postboxes/postboxes.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [postboxes](https://mapcomplete.osm.be/postboxes)
-
-
-### postoffices
-
-
-
-A layer showing post offices.
-
-[Go to the source code](../assets/layers/postoffices/postoffices.json)
-
-
-
-
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [postboxes](https://mapcomplete.osm.be/postboxes)
-
-
-### lit_streets
-
-
-
-[Go to the source code](../assets/layers/lit_streets/lit_streets.json)
-
-
-
- - Not rendered on the map by default. If you want to rendering this on the map, override `mapRenderings`
-
-
-
-
-#### Themes using this layer
-
-
-
-
-
- - [street_lighting](https://mapcomplete.osm.be/street_lighting)
-
-
This document is autogenerated from AllKnownLayers.ts
\ No newline at end of file
diff --git a/Logic/Actors/InstalledThemes.ts b/Logic/Actors/InstalledThemes.ts
deleted file mode 100644
index b57468524d..0000000000
--- a/Logic/Actors/InstalledThemes.ts
+++ /dev/null
@@ -1,61 +0,0 @@
-import {UIEventSource} from "../UIEventSource";
-import {OsmConnection} from "../Osm/OsmConnection";
-import {Utils} from "../../Utils";
-import LZString from "lz-string";
-import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig";
-
-export default class InstalledThemes {
- public installedThemes: UIEventSource<{ layout: LayoutConfig; definition: string }[]>;
-
- constructor(osmConnection: OsmConnection) {
- this.installedThemes = osmConnection.preferencesHandler.preferences.map<{ layout: LayoutConfig, definition: string }[]>(allPreferences => {
- const installedThemes: { layout: LayoutConfig, definition: string }[] = [];
- if (allPreferences === undefined) {
- console.log("All prefs is undefined");
- return installedThemes;
- }
- const invalidThemes = []
- for (const allPreferencesKey in allPreferences) {
- const themename = allPreferencesKey.match(/^mapcomplete-installed-theme-(.*)-combined-length$/);
- if (themename && themename[1] !== "") {
- const customLayout = osmConnection.GetLongPreference("installed-theme-" + themename[1]);
- if (customLayout.data === undefined) {
- console.log("No data defined for ", themename[1]);
- continue;
- }
- try {
- let layoutJson;
- try {
- layoutJson = JSON.parse(atob(customLayout.data))
- } catch (e) {
- layoutJson = JSON.parse(Utils.UnMinify(LZString.decompressFromBase64(customLayout.data)))
- }
- const layout = new LayoutConfig(layoutJson, false);
- installedThemes.push({
- layout: layout,
- definition: customLayout.data
- });
- } catch (e) {
- console.warn("Could not parse custom layout from preferences - deleting: ", allPreferencesKey, e, customLayout.data);
- invalidThemes.push(themename[1])
- }
- }
- }
-
- InstalledThemes.DeleteInvalid(osmConnection, invalidThemes);
-
- return installedThemes;
-
- });
- }
-
- private static DeleteInvalid(osmConnection: OsmConnection, invalidThemes: any[]) {
- for (const invalid of invalidThemes) {
- console.error("Attempting to remove ", invalid)
- osmConnection.GetLongPreference(
- "installed-theme-" + invalid
- ).setData(null);
- }
- }
-
-}
\ No newline at end of file
diff --git a/Logic/Actors/OverpassFeatureSource.ts b/Logic/Actors/OverpassFeatureSource.ts
index a630d98808..4510fd7336 100644
--- a/Logic/Actors/OverpassFeatureSource.ts
+++ b/Logic/Actors/OverpassFeatureSource.ts
@@ -10,7 +10,7 @@ import RelationsTracker from "../Osm/RelationsTracker";
import {BBox} from "../BBox";
import Loc from "../../Models/Loc";
import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
-import AllKnownLayers from "../../Customizations/AllKnownLayers";
+import Constants from "../../Models/Constants";
export default class OverpassFeatureSource implements FeatureSource {
@@ -122,7 +122,7 @@ export default class OverpassFeatureSource implements FeatureSource {
if (typeof (layer) === "string") {
throw "A layer was not expanded!"
}
- if(AllKnownLayers.priviliged_layers.indexOf(layer.id) >= 0){
+ if(Constants.priviliged_layers.indexOf(layer.id) >= 0){
continue
}
if (this.state.locationControl.data.zoom < layer.minzoom) {
diff --git a/Logic/Actors/TitleHandler.ts b/Logic/Actors/TitleHandler.ts
index a3c4b50813..d20bea7b9e 100644
--- a/Logic/Actors/TitleHandler.ts
+++ b/Logic/Actors/TitleHandler.ts
@@ -29,7 +29,7 @@ export default class TitleHandler {
}
if (layer.source.osmTags.matchesProperties(tags)) {
const tagsSource = state.allElements.getEventSourceById(tags.id) ?? new UIEventSource(tags)
- const title = new TagRenderingAnswer(tagsSource, layer.title)
+ const title = new TagRenderingAnswer(tagsSource, layer.title, {})
return new Combine([defaultTitle, " | ", title]).ConstructElement()?.innerText ?? defaultTitle;
}
}
diff --git a/Logic/DetermineLayout.ts b/Logic/DetermineLayout.ts
index b077c50427..5f9c1f0e6a 100644
--- a/Logic/DetermineLayout.ts
+++ b/Logic/DetermineLayout.ts
@@ -10,30 +10,29 @@ import {UIEventSource} from "./UIEventSource";
import {LocalStorageSource} from "./Web/LocalStorageSource";
import LZString from "lz-string";
import * as personal from "../assets/themes/personal/personal.json";
-import LegacyJsonConvert from "../Models/ThemeConfig/LegacyJsonConvert";
+import {FixLegacyTheme, PrepareTheme} from "../Models/ThemeConfig/LegacyJsonConvert";
+import {LayerConfigJson} from "../Models/ThemeConfig/Json/LayerConfigJson";
+import SharedTagRenderings from "../Customizations/SharedTagRenderings";
+import * as known_layers from "../assets/generated/known_layers.json"
+import {LayoutConfigJson} from "../Models/ThemeConfig/Json/LayoutConfigJson";
export default class DetermineLayout {
/**
* Gets the correct layout for this website
*/
- public static async GetLayout(): Promise<[LayoutConfig, string]> {
+ public static async GetLayout(): Promise {
const loadCustomThemeParam = QueryParameters.GetQueryParameter("userlayout", "false", "If not 'false', a custom (non-official) theme is loaded. This custom layout can be done in multiple ways: \n\n- The hash of the URL contains a base64-encoded .json-file containing the theme definition\n- The hash of the URL contains a lz-compressed .json-file, as generated by the custom theme generator\n- The parameter itself is an URL, in which case that URL will be downloaded. It should point to a .json of a theme")
const layoutFromBase64 = decodeURIComponent(loadCustomThemeParam.data);
if (layoutFromBase64.startsWith("http")) {
- const layout = await DetermineLayout.LoadRemoteTheme(layoutFromBase64)
- return [layout, undefined]
+ return await DetermineLayout.LoadRemoteTheme(layoutFromBase64)
}
if (layoutFromBase64 !== "false") {
// We have to load something from the hash (or from disk)
- let loaded = DetermineLayout.LoadLayoutFromHash(loadCustomThemeParam);
- if (loaded === null) {
- return [null, undefined]
- }
- return loaded
+ return DetermineLayout.LoadLayoutFromHash(loadCustomThemeParam)
}
let layoutId: string = undefined
@@ -43,7 +42,7 @@ export default class DetermineLayout {
const path = window.location.pathname.split("/").slice(-1)[0];
- if (path !== "index.html" && path !== "") {
+ if (path !== "theme.html" && path !== "") {
layoutId = path;
if (path.endsWith(".html")) {
layoutId = path.substr(0, path.length - 5);
@@ -61,12 +60,28 @@ export default class DetermineLayout {
}
}
- return [layoutToUse, undefined]
+ return layoutToUse
}
+ private static prepCustomTheme(json: any): LayoutConfigJson{
+ const knownLayersDict = new Map()
+ for (const key in known_layers["default"]) {
+ const layer = known_layers["default"][key]
+ knownLayersDict.set(layer.id, layer)
+ }
+ const converState = {
+ tagRenderings: SharedTagRenderings.SharedTagRenderingJson,
+ sharedLayers: knownLayersDict
+ }
+ json = new FixLegacyTheme().convertStrict(converState, json, "While loading a dynamic theme")
+ json = new PrepareTheme().convertStrict(converState, json, "While preparing a dynamic theme")
+ console.log("The layoutconfig is ", json)
+ return json
+ }
+
public static LoadLayoutFromHash(
userLayoutParam: UIEventSource
- ): [LayoutConfig, string] | null {
+ ): LayoutConfig | null {
let hash = location.hash.substr(1);
try {
// layoutFromBase64 contains the name of the theme. This is partly to do tracking with goat counter
@@ -104,10 +119,9 @@ export default class DetermineLayout {
}
}
- LegacyJsonConvert.fixThemeConfig(json)
- const layoutToUse = new LayoutConfig(json, false);
+ const layoutToUse = DetermineLayout.prepCustomTheme(json)
userLayoutParam.setData(layoutToUse.id);
- return [layoutToUse, btoa(Utils.MinifyJSON(JSON.stringify(json)))];
+ return new LayoutConfig(layoutToUse, false);
} catch (e) {
console.error(e)
if (hash === undefined || hash.length < 10) {
@@ -141,12 +155,20 @@ export default class DetermineLayout {
try {
- const parsed = await Utils.downloadJson(link)
+ let parsed = await Utils.downloadJson(link)
console.log("Got ", parsed)
- LegacyJsonConvert.fixThemeConfig(parsed)
- try {
+ parsed = new FixLegacyTheme().convertStrict({
+ tagRenderings: SharedTagRenderings.SharedTagRenderingJson,
+ sharedLayers: new Map() // FIXME: actually add the layers
+ }, parsed, "While loading a dynamic theme")
+
+
parsed.id = link;
- return new LayoutConfig(parsed, false).patchImages(link, JSON.stringify(parsed));
+
+
+ try {
+ const layoutToUse = DetermineLayout.prepCustomTheme(parsed)
+ return new LayoutConfig(layoutToUse,false).patchImages(link, JSON.stringify(layoutToUse));
} catch (e) {
console.error(e)
DetermineLayout.ShowErrorOnCustomTheme(
diff --git a/Logic/State/ElementsState.ts b/Logic/State/ElementsState.ts
index 90abe4d037..8167f7cb2b 100644
--- a/Logic/State/ElementsState.ts
+++ b/Logic/State/ElementsState.ts
@@ -10,7 +10,6 @@ import {LocalStorageSource} from "../Web/LocalStorageSource";
import {Utils} from "../../Utils";
import ChangeToElementsActor from "../Actors/ChangeToElementsActor";
import PendingChangesUploader from "../Actors/PendingChangesUploader";
-import TitleHandler from "../Actors/TitleHandler";
/**
* The part of the state keeping track of where the elements, loading them, configuring the feature pipeline etc
@@ -90,7 +89,6 @@ export default class ElementsState extends FeatureSwitchState {
new ChangeToElementsActor(this.changes, this.allElements)
new PendingChangesUploader(this.changes, this.selectedElement);
- new TitleHandler(this);
}
}
\ No newline at end of file
diff --git a/Logic/State/MapState.ts b/Logic/State/MapState.ts
index a5b13e0987..07d3450dca 100644
--- a/Logic/State/MapState.ts
+++ b/Logic/State/MapState.ts
@@ -17,6 +17,7 @@ import {FeatureSourceForLayer, Tiled} from "../FeatureSource/FeatureSource";
import SimpleFeatureSource from "../FeatureSource/Sources/SimpleFeatureSource";
import {LocalStorageSource} from "../Web/LocalStorageSource";
import {GeoOperations} from "../GeoOperations";
+import TitleHandler from "../Actors/TitleHandler";
/**
* Contains all the leaflet-map related state
@@ -130,6 +131,8 @@ export default class MapState extends UserRelatedState {
this.initGpsLocation()
this.initUserLocationTrail()
this.initCurrentView()
+
+ new TitleHandler(this);
}
public AddAllOverlaysToMap(leafletMap: UIEventSource) {
diff --git a/Logic/State/UserRelatedState.ts b/Logic/State/UserRelatedState.ts
index ca69006ac4..37deb04dc2 100644
--- a/Logic/State/UserRelatedState.ts
+++ b/Logic/State/UserRelatedState.ts
@@ -3,12 +3,12 @@ import {OsmConnection} from "../Osm/OsmConnection";
import {MangroveIdentity} from "../Web/MangroveReviews";
import {UIEventSource} from "../UIEventSource";
import {QueryParameters} from "../Web/QueryParameters";
-import InstalledThemes from "../Actors/InstalledThemes";
import {LocalStorageSource} from "../Web/LocalStorageSource";
import {Utils} from "../../Utils";
import Locale from "../../UI/i18n/Locale";
import ElementsState from "./ElementsState";
import SelectedElementTagsUpdater from "../Actors/SelectedElementTagsUpdater";
+import {log} from "util";
/**
* The part of the state which keeps track of user-related stuff, e.g. the OSM-connection,
@@ -33,7 +33,10 @@ export default class UserRelatedState extends ElementsState {
/**
* WHich other themes the user previously visited
*/
- public installedThemes: UIEventSource<{ layout: LayoutConfig; definition: string }[]>;
+ public installedThemes: UIEventSource<{ id: string, // The id doubles as the URL
+ icon: string,
+ title: any,
+ shortDescription: any}[]>;
constructor(layoutToUse: LayoutConfig, options?:{attemptLogin : true | boolean}) {
@@ -69,9 +72,47 @@ export default class UserRelatedState extends ElementsState {
})
}
- this.installedThemes = new InstalledThemes(
- this.osmConnection
- ).installedThemes;
+ this.installedThemes = this.osmConnection.GetLongPreference("installed-themes").map(
+ str => {
+ if(str === undefined || str === ""){
+ return []
+ }
+ try{
+ return JSON.parse(str)
+ }catch(e){
+ console.warn("Could not parse preference with installed themes due to ", e,"\nThe offending string is",str)
+ return []
+ }
+ }, [],(installed => JSON.stringify(installed))
+ )
+
+
+ const self = this;
+ this.osmConnection.isLoggedIn.addCallbackAndRunD(loggedIn => {
+ if(!loggedIn){
+ return
+ }
+
+ if(this.layoutToUse.id.startsWith("http")){
+ if(!this.installedThemes.data.some(installed => installed.id === this.layoutToUse.id)){
+
+ this.installedThemes.data.push({
+ id: this.layoutToUse.id,
+ icon: this.layoutToUse.icon,
+ title: this.layoutToUse.title.translations,
+ shortDescription: this.layoutToUse.shortDescription.translations
+ })
+ }
+ this.installedThemes.ping()
+ console.log("Registered "+this.layoutToUse.id+" as installed themes")
+ }
+
+
+
+
+ return true;
+ })
+
// Important: the favourite layers are initialized _after_ the installed themes, as these might contain an installedTheme
this.favouriteLayers = LocalStorageSource.Get("favouriteLayers")
@@ -81,8 +122,7 @@ export default class UserRelatedState extends ElementsState {
[],
(layers) => Utils.Dedup(layers)?.join(";")
);
-
-
+
this.InitializeLanguage();
new SelectedElementTagsUpdater(this)
diff --git a/Models/Constants.ts b/Models/Constants.ts
index eaa6c0eab5..22e85ed969 100644
--- a/Models/Constants.ts
+++ b/Models/Constants.ts
@@ -2,7 +2,7 @@ import {Utils} from "../Utils";
export default class Constants {
- public static vNumber = "0.13.0";
+ public static vNumber = "0.14.0-alpha-1";
public static ImgurApiKey = '7070e7167f0a25a'
public static readonly mapillary_client_token_v4 = "MLY|4441509239301885|b40ad2d3ea105435bd40c7e76993ae85"
@@ -17,6 +17,14 @@ export default class Constants {
]
+ public static readonly added_by_default: string[] = ["gps_location", "gps_location_history", "home_location", "gps_track"]
+ public static readonly no_include: string[] = ["conflation", "left_right_style", "split_point","current_view","matchpoint"]
+ /**
+ * Layer IDs of layers which have special properties through built-in hooks
+ */
+ public static readonly priviliged_layers: string[] = [...Constants.added_by_default, "type_node", ...Constants.no_include]
+
+
// The user journey states thresholds when a new feature gets unlocked
public static userJourney = {
moreScreenUnlock: 1,
@@ -51,6 +59,7 @@ export default class Constants {
* For every bin, the totals are uploaded as metadata
*/
static distanceToChangeObjectBins = [25,50,100,500,1000,5000, Number.MAX_VALUE]
+ static themeOrder = ["personal", "cyclofix", "hailhydrant", "bookcases", "toilets", "aed"];
private static isRetina(): boolean {
if (Utils.runningFromConsole) {
diff --git a/Models/ThemeConfig/Json/LineRenderingConfigJson.ts b/Models/ThemeConfig/Json/LineRenderingConfigJson.ts
index 2b2606473c..b29f9a9919 100644
--- a/Models/ThemeConfig/Json/LineRenderingConfigJson.ts
+++ b/Models/ThemeConfig/Json/LineRenderingConfigJson.ts
@@ -18,7 +18,7 @@ export default interface LineRenderingConfigJson {
/**
* The stroke-width for way-elements
*/
- width?: string | TagRenderingConfigJson;
+ width?: string | number | TagRenderingConfigJson;
/**
* A dasharray, e.g. "5 6"
diff --git a/Models/ThemeConfig/LayerConfig.ts b/Models/ThemeConfig/LayerConfig.ts
index 091422bb66..b2c52c74d1 100644
--- a/Models/ThemeConfig/LayerConfig.ts
+++ b/Models/ThemeConfig/LayerConfig.ts
@@ -22,7 +22,7 @@ import Title from "../../UI/Base/Title";
import List from "../../UI/Base/List";
import Link from "../../UI/Base/Link";
import {Utils} from "../../Utils";
-import * as icons from "../../assets/tagRenderings/icons.json"
+import {tag} from "@turf/turf";
export default class LayerConfig extends WithContextLoader {
@@ -236,26 +236,14 @@ export default class LayerConfig extends WithContextLoader {
throw "Missing ids in tagrenderings"
}
- this.tagRenderings = this.ExtractLayerTagRenderings(json, official)
- if (official) {
-
- const emptyIds = this.tagRenderings.filter(tr => tr.id === "");
- if (emptyIds.length > 0) {
- throw `Some tagrendering-ids are empty or have an emtpy string; this is not allowed (at ${context})`
- }
-
- const duplicateIds = Utils.Dupicates(this.tagRenderings.map(f => f.id).filter(id => id !== "questions"))
- if (duplicateIds.length > 0) {
- throw `Some tagRenderings have a duplicate id: ${duplicateIds} (at ${context}.tagRenderings)`
- }
- }
+ this.tagRenderings = (json.tagRenderings ?? []).map((tr, i) => new TagRenderingConfig(tr, this.id + ".tagRenderings[" + i + "]"))
this.filters = (json.filter ?? []).map((option, i) => {
return new FilterConfig(option, `${context}.filter-[${i}]`)
});
{
- const duplicateIds = Utils.Dupicates(this.filters.map(f => f.id))
+ const duplicateIds = Utils.Dupiclates(this.filters.map(f => f.id))
if (duplicateIds.length > 0) {
throw `Some filters have a duplicate id: ${duplicateIds} (at ${context}.filters)`
}
@@ -265,17 +253,8 @@ export default class LayerConfig extends WithContextLoader {
throw "Error in " + context + ": use 'filter' instead of 'filters'"
}
- const titleIcons = [];
- const defaultIcons = icons.defaultIcons;
- for (const icon of json.titleIcons ?? defaultIcons) {
- if (icon === "defaults") {
- titleIcons.push(...defaultIcons);
- } else {
- titleIcons.push(icon);
- }
- }
- this.titleIcons = this.ParseTagRenderings(titleIcons, {
+ this.titleIcons = this.ParseTagRenderings(( json.titleIcons), {
readOnlyMode: true
});
@@ -320,109 +299,6 @@ export default class LayerConfig extends WithContextLoader {
const defaultTags = new UIEventSource(TagUtils.changeAsProperties(this.source.osmTags.asChange({id: "node/-1"})))
return mapRendering.GenerateLeafletStyle(defaultTags, false, {noSize: true}).html
}
-
- public ExtractLayerTagRenderings(json: LayerConfigJson, official: boolean): TagRenderingConfig[] {
-
- if (json.tagRenderings === undefined) {
- return []
- }
-
- const normalTagRenderings: (string | { builtin: string, override: any } | TagRenderingConfigJson)[] = []
-
-
- const renderingsToRewrite: ({
- rewrite: {
- sourceString: string,
- into: string[]
- }, renderings: (string | { builtin: string, override: any } | TagRenderingConfigJson)[]
- })[] = []
- for (let i = 0; i < json.tagRenderings.length; i++) {
- const tr = json.tagRenderings[i];
- const rewriteDefined = tr["rewrite"] !== undefined
- const renderingsDefined = tr["renderings"]
-
- if (!rewriteDefined && !renderingsDefined) {
- // @ts-ignore
- normalTagRenderings.push(tr)
- continue
- }
- if (rewriteDefined && renderingsDefined) {
- // @ts-ignore
- renderingsToRewrite.push(tr)
- continue
- }
- throw `Error in ${this._context}.tagrenderings[${i}]: got a value which defines either \`rewrite\` or \`renderings\`, but not both. Either define both or move the \`renderings\` out of this scope`
- }
-
- const allRenderings = this.ParseTagRenderings(normalTagRenderings,
- {
- requiresId: official
- });
-
- if (renderingsToRewrite.length === 0) {
- return allRenderings
- }
-
- /* Used for left|right group creation and replacement */
- function prepConfig(keyToRewrite: string, target: string, tr: TagRenderingConfigJson) {
-
- function replaceRecursive(transl: string | any) {
- if (typeof transl === "string") {
- return transl.replace(keyToRewrite, target)
- }
- if (transl.map !== undefined) {
- return transl.map(o => replaceRecursive(o))
- }
- transl = {...transl}
- for (const key in transl) {
- transl[key] = replaceRecursive(transl[key])
- }
- return transl
- }
-
- const orig = tr;
- tr = replaceRecursive(tr)
-
- tr.id = target + "-" + orig.id
- tr.group = target
- return tr
- }
-
- const rewriteGroups: Map = new Map()
- for (const rewriteGroup of renderingsToRewrite) {
-
- const tagRenderings = rewriteGroup.renderings
- const textToReplace = rewriteGroup.rewrite.sourceString
- const targets = rewriteGroup.rewrite.into
- for (const target of targets) {
- const parsedRenderings = this.ParseTagRenderings(tagRenderings, {
- prepConfig: tr => prepConfig(textToReplace, target, tr)
- })
-
- if (!rewriteGroups.has(target)) {
- rewriteGroups.set(target, [])
- }
- rewriteGroups.get(target).push(...parsedRenderings)
- }
- }
-
-
- rewriteGroups.forEach((group, groupName) => {
- group.push(new TagRenderingConfig({
- id: "questions",
- group: groupName
- }))
- })
-
- rewriteGroups.forEach(group => {
- allRenderings.push(...group)
- })
-
-
- return allRenderings;
-
- }
-
public GenerateDocumentation(usedInThemes: string[], layerIsNeededBy: Map, dependencies: {
context?: string;
reason: string;
@@ -512,5 +388,4 @@ export default class LayerConfig extends WithContextLoader {
public isLeftRightSensitive(): boolean {
return this.lineRendering.some(lr => lr.leftRightSensitive)
}
-
}
\ No newline at end of file
diff --git a/Models/ThemeConfig/LayoutConfig.ts b/Models/ThemeConfig/LayoutConfig.ts
index e18cd7d8fe..b047fe3630 100644
--- a/Models/ThemeConfig/LayoutConfig.ts
+++ b/Models/ThemeConfig/LayoutConfig.ts
@@ -1,12 +1,9 @@
import {Translation} from "../../UI/i18n/Translation";
import {LayoutConfigJson} from "./Json/LayoutConfigJson";
-import AllKnownLayers from "../../Customizations/AllKnownLayers";
-import {Utils} from "../../Utils";
import LayerConfig from "./LayerConfig";
import {LayerConfigJson} from "./Json/LayerConfigJson";
import Constants from "../Constants";
import TilesourceConfig from "./TilesourceConfig";
-import DependencyCalculator from "./DependencyCalculator";
export default class LayoutConfig {
public readonly id: string;
@@ -15,7 +12,7 @@ export default class LayoutConfig {
public readonly version: string;
public readonly language: string[];
public readonly title: Translation;
- public readonly shortDescription?: Translation;
+ public readonly shortDescription: Translation;
public readonly description: Translation;
public readonly descriptionTail?: Translation;
public readonly icon: string;
@@ -53,7 +50,6 @@ export default class LayoutConfig {
public readonly overpassMaxZoom: number
public readonly osmApiTileSize: number
public readonly official: boolean;
- public readonly trackAllNodes: boolean;
constructor(json: LayoutConfigJson, official = true, context?: string) {
this.official = official;
@@ -75,14 +71,28 @@ export default class LayoutConfig {
} else {
this.language = json.language;
}
- if (this.language.length == 0) {
- throw `No languages defined. Define at least one language. (${context}.languages)`
- }
- if (json.title === undefined) {
- throw "Title not defined in " + this.id;
- }
- if (json.description === undefined) {
- throw "Description not defined in " + this.id;
+ {
+ if (this.language.length == 0) {
+ throw `No languages defined. Define at least one language. (${context}.languages)`
+ }
+ if (json.title === undefined) {
+ throw "Title not defined in " + this.id;
+ }
+ if (json.description === undefined) {
+ throw "Description not defined in " + this.id;
+ }
+ if (json.widenFactor <= 0) {
+ throw "Widenfactor too small, shoud be > 0"
+ }
+ if (json.widenFactor > 20) {
+ throw "Widenfactor is very big, use a value between 1 and 5 (current value is " + json.widenFactor + ") at " + context
+ }
+ if (json["hideInOverview"]) {
+ throw "The json for " + this.id + " contains a 'hideInOverview'. Did you mean hideFromOverview instead?"
+ }
+ if (json.layers === undefined) {
+ throw "Got undefined layers for " + json.id + " at " + context
+ }
}
this.title = new Translation(json.title, context + ".title");
this.description = new Translation(json.description, context + ".description");
@@ -93,19 +103,12 @@ export default class LayoutConfig {
this.startZoom = json.startZoom;
this.startLat = json.startLat;
this.startLon = json.startLon;
- if (json.widenFactor <= 0) {
- throw "Widenfactor too small, shoud be > 0"
- }
- if (json.widenFactor > 20) {
- throw "Widenfactor is very big, use a value between 1 and 5 (current value is " + json.widenFactor + ") at " + context
- }
-
this.widenFactor = json.widenFactor ?? 1.5;
this.defaultBackgroundId = json.defaultBackgroundId;
this.tileLayerSources = (json.tileLayerSources ?? []).map((config, i) => new TilesourceConfig(config, `${this.id}.tileLayerSources[${i}]`))
- const layerInfo = LayoutConfig.ExtractLayers(json, official, context);
- this.layers = layerInfo.layers
+ // At this point, layers should be expanded and validated either by the generateScript or the LegacyJsonConvert
+ this.layers = json.layers.map(lyrJson => new LayerConfig(lyrJson, json.id + ".layers." + lyrJson["id"], official));
this.clustering = {
@@ -125,10 +128,6 @@ export default class LayoutConfig {
}
this.hideFromOverview = json.hideFromOverview ?? false;
- // @ts-ignore
- if (json.hideInOverview) {
- throw "The json for " + this.id + " contains a 'hideInOverview'. Did you mean hideFromOverview instead?"
- }
this.lockLocation = <[[number, number], [number, number]]>json.lockLocation ?? undefined;
this.enableUserBadge = json.enableUserBadge ?? true;
this.enableShareScreen = json.enableShareScreen ?? true;
@@ -157,120 +156,6 @@ export default class LayoutConfig {
}
- private static ExtractLayers(json: LayoutConfigJson, official: boolean, context: string): { layers: LayerConfig[], extractAllNodes: boolean } {
- const result: LayerConfig[] = []
- let exportAllNodes = false
- if(json.layers === undefined){
- throw "Got undefined layers for "+json.id+" at "+context
- }
- json.layers.forEach((layer, i) => {
-
- if (typeof layer === "string") {
- if (AllKnownLayers.sharedLayersJson.get(layer) !== undefined) {
- if (json.overrideAll !== undefined) {
- let lyr = JSON.parse(JSON.stringify(AllKnownLayers.sharedLayersJson.get(layer)));
- const newLayer = new LayerConfig(Utils.Merge(json.overrideAll, lyr), `${json.id}+overrideAll.layers[${i}]`, official)
- result.push(newLayer)
- return
- } else {
- const shared = AllKnownLayers.sharedLayers.get(layer)
- if (shared === undefined) {
- throw `Shared layer ${layer} not found (at ${context}.layers[${i}])`
- }
- result.push(shared)
- return
- }
- } else {
- console.log("Layer ", layer, " not kown, try one of", Array.from(AllKnownLayers.sharedLayers.keys()).join(", "))
- throw `Unknown builtin layer ${layer} at ${context}.layers[${i}]`;
- }
- }
-
- if (layer["builtin"] === undefined) {
- if (json.overrideAll !== undefined) {
- layer = Utils.Merge(json.overrideAll, layer);
- }
- // @ts-ignore
- result.push(new LayerConfig(layer, `${json.id}.layers[${i}]`, official))
- return
- }
-
- // @ts-ignore
- let names = layer.builtin;
- if (typeof names === "string") {
- names = [names]
- }
-
- // This is a very special layer which triggers special behaviour
- exportAllNodes = names.some(name => name === "type_node");
-
- names.forEach(name => {
- const shared = AllKnownLayers.sharedLayersJson.get(name);
- if (shared === undefined) {
- throw `Unknown shared/builtin layer ${name} at ${context}.layers[${i}]. Available layers are ${Array.from(AllKnownLayers.sharedLayersJson.keys()).join(", ")}`;
- }
- let newLayer: LayerConfigJson = Utils.Merge(layer["override"], JSON.parse(JSON.stringify(shared))); // We make a deep copy of the shared layer, in order to protect it from changes
- if (json.overrideAll !== undefined) {
- newLayer = Utils.Merge(json.overrideAll, newLayer);
- }
- result.push(new LayerConfig(newLayer, `${json.id}.layers[${i}]`, official))
- return
- })
-
- });
-
- // Some special layers which are always included by default
- for (const defaultLayer of AllKnownLayers.added_by_default) {
- if (result.some(l => l?.id === defaultLayer)) {
- continue; // Already added
- }
- const sharedLayer = AllKnownLayers.sharedLayers.get(defaultLayer)
- if (sharedLayer !== undefined) {
- result.push(sharedLayer)
- }else if(!AllKnownLayers.runningGenerateScript){
- throw "SharedLayer "+defaultLayer+" not found"
- }
- }
-
- if(AllKnownLayers.runningGenerateScript){
- return {layers: result, extractAllNodes: exportAllNodes}
- }
- // Verify cross-dependencies
- let unmetDependencies: { neededLayer: string, neededBy: string, reason: string, context?: string }[] = []
- do {
- const dependencies: { neededLayer: string, reason: string, context?: string, neededBy: string }[] = []
-
- for (const layerConfig of result) {
- const layerDeps = DependencyCalculator.getLayerDependencies(layerConfig)
- dependencies.push(...layerDeps)
- }
-
- const loadedLayers = new Set(result.map(r => r.id))
- // During the generate script, builtin layers are verified but not loaded - so we have to add them manually here
- // Their existance is checked elsewhere, so this is fine
- unmetDependencies = dependencies.filter(dep => !loadedLayers.has(dep.neededLayer))
- for (const unmetDependency of unmetDependencies) {
- const dep = AllKnownLayers.sharedLayers.get(unmetDependency.neededLayer)
- if (dep === undefined) {
-
- const message =
- ["Loading a dependency failed: layer "+unmetDependency.neededLayer+" is not found, neither as layer of "+json.id+" nor as builtin layer.",
- "This layer is needed by "+unmetDependency.neededBy,
- unmetDependency.reason+" (at "+unmetDependency.context+")",
- "Loaded layers are: "+result.map(l => l.id).join(",")
-
- ]
- throw message.join("\n\t");
- }
- result.unshift(dep)
- unmetDependencies = unmetDependencies.filter(d => d.neededLayer !== unmetDependency.neededLayer)
- }
-
- } while (unmetDependencies.length > 0)
-
- return {layers: result, extractAllNodes: exportAllNodes}
- }
-
public CustomCodeSnippets(): string[] {
if (this.official) {
return [];
diff --git a/Models/ThemeConfig/LegacyJsonConvert.ts b/Models/ThemeConfig/LegacyJsonConvert.ts
index 8189814360..7e0397137a 100644
--- a/Models/ThemeConfig/LegacyJsonConvert.ts
+++ b/Models/ThemeConfig/LegacyJsonConvert.ts
@@ -1,14 +1,404 @@
import LineRenderingConfigJson from "./Json/LineRenderingConfigJson";
-import PointRenderingConfig from "./PointRenderingConfig";
+import LayerConfig from "./LayerConfig";
+import Constants from "../Constants";
+import {LayoutConfigJson} from "./Json/LayoutConfigJson";
+import {LayerConfigJson} from "./Json/LayerConfigJson";
+import DependencyCalculator from "./DependencyCalculator";
+import {TagRenderingConfigJson} from "./Json/TagRenderingConfigJson";
+import {Utils} from "../../Utils";
+import LayoutConfig from "./LayoutConfig";
+import {Translation} from "../../UI/i18n/Translation";
-export default class LegacyJsonConvert {
+export interface DesugaringContext {
+ tagRenderings: Map
+ sharedLayers: Map
+}
+
+abstract class Conversion {
+ protected readonly doc: string;
+ public readonly modifiedAttributes: string[];
+
+ constructor(doc: string, modifiedAttributes: string[] = []) {
+ this.modifiedAttributes = modifiedAttributes;
+ this.doc = doc + "\n\nModified attributes are\n" + modifiedAttributes.join(", ");
+ }
+
+ public convertStrict(state: DesugaringContext, json: TIn, context: string): TOut {
+ const fixed = this.convert(state, json, context)
+ return DesugaringStep.strict(fixed)
+ }
+
+ public static strict(fixed: { errors: string[], warnings: string[], result?: T }): T {
+ if (fixed.errors?.length > 0) {
+ throw fixed.errors.join("\n");
+ }
+ fixed.warnings?.forEach(w => console.warn(w))
+ return fixed.result;
+ }
+
+ abstract convert(state: DesugaringContext, json: TIn, context: string): { result: TOut, errors: string[], warnings: string[] }
+
+ public convertAll(state: DesugaringContext, jsons: TIn[], context: string): { result: TOut[], errors: string[], warnings: string[] } {
+ const result = []
+ const errors = []
+ const warnings = []
+ for (let i = 0; i < jsons.length; i++) {
+ const json = jsons[i];
+ const r = this.convert(state, json, context + "[" + i + "]")
+ result.push(r.result)
+ errors.push(...r.errors)
+ warnings.push(...r.warnings)
+ }
+ return {
+ result,
+ errors,
+ warnings
+ }
+ }
+
+}
+
+abstract class DesugaringStep extends Conversion {
+}
+
+class OnEvery extends DesugaringStep {
+ private readonly key: string;
+ private readonly step: DesugaringStep;
+
+ constructor(key: string, step: DesugaringStep) {
+ super("Applies " + step.constructor.name + " onto every object of the list `key`", [key]);
+ this.step = step;
+ this.key = key;
+ }
+
+ convert(state: DesugaringContext, json: T, context: string): { result: T; errors: string[]; warnings: string[] } {
+ json = {...json}
+ const step = this.step
+ const key = this.key;
+ const r = step.convertAll(state, (json[key]), context + "." + key)
+ json[key] = r.result
+ return {
+ result: json,
+ errors: r.errors,
+ warnings: r.warnings
+ };
+ }
+}
+
+class OnEveryConcat extends DesugaringStep {
+ private readonly key: string;
+ private readonly step: Conversion;
+
+ constructor(key: string, step: Conversion) {
+ super(`Applies ${step.constructor.name} onto every object of the list \`${key}\``, [key]);
+ this.step = step;
+ this.key = key;
+ }
+
+ convert(state: DesugaringContext, json: T, context: string): { result: T; errors: string[]; warnings: string[] } {
+ json = {...json}
+ const step = this.step
+ const key = this.key;
+ const values = json[key]
+ if (values === undefined) {
+ // Move on - nothing to see here!
+ return {
+ result: json,
+ errors: [],
+ warnings: []
+ }
+ }
+ const r = step.convertAll(state, (values), context + "." + key)
+ const vals: X[][] = r.result
+ json[key] = [].concat(...vals)
+ return {
+ result: json,
+ errors: r.errors,
+ warnings: r.warnings
+ };
+
+ }
+}
+
+class Fuse extends DesugaringStep {
+ private readonly steps: DesugaringStep[];
+
+ constructor(doc: string, ...steps: DesugaringStep[]) {
+ super((doc ?? "") + "This fused pipeline of the following steps: " + steps.map(s => s.constructor.name).join(", "),
+ Utils.Dedup([].concat(...steps.map(step => step.modifiedAttributes)))
+ );
+ this.steps = steps;
+ }
+
+ convert(state: DesugaringContext, json: T, context: string): { result: T; errors: string[]; warnings: string[] } {
+ const errors = []
+ const warnings = []
+ for (let i = 0; i < this.steps.length; i++){
+ const step = this.steps[i];
+ let r = step.convert(state, json, context + "(fusion "+this.constructor.name+"."+i+")")
+ errors.push(...r.errors)
+ warnings.push(...r.warnings)
+ json = r.result
+ if (errors.length > 0) {
+ break;
+ }
+ }
+ return {
+ result: json,
+ errors,
+ warnings
+ };
+ }
+
+}
+
+class ExpandTagRendering extends Conversion {
+ constructor() {
+ super("Converts a tagRenderingSpec into the full tagRendering", []);
+ }
+
+ private lookup(state: DesugaringContext, name: string): TagRenderingConfigJson[] {
+ if (state.tagRenderings.has(name)) {
+ return [state.tagRenderings.get(name)]
+ }
+ if (name.indexOf(".") >= 0) {
+ const spl = name.split(".");
+ const layer = state.sharedLayers.get(spl[0])
+ if (spl.length === 2 && layer !== undefined) {
+ const id = spl[1];
+
+ const layerTrs = layer.tagRenderings.filter(tr => tr["id"] !== undefined)
+ let matchingTrs: TagRenderingConfigJson[]
+ if (id === "*") {
+ matchingTrs = layerTrs
+ } else if (id.startsWith("*")) {
+ const id_ = id.substring(1)
+ matchingTrs = layerTrs.filter(tr => tr.group === id_)
+ } else {
+ matchingTrs = layerTrs.filter(tr => tr.id === id)
+ }
+
+
+ for (let i = 0; i < matchingTrs.length; i++) {
+ // The matched tagRenderings are 'stolen' from another layer. This means that they must match the layer condition before being shown
+ const found = Utils.Clone(matchingTrs[i]);
+ if (found.condition === undefined) {
+ found.condition = layer.source.osmTags
+ } else {
+ found.condition = {and: [found.condition, layer.source.osmTags]}
+ }
+ matchingTrs[i] = found
+ }
+
+ if (matchingTrs.length !== 0) {
+ return matchingTrs
+ }
+ }
+ }
+ return undefined;
+ }
+
+ private convertOnce(state: DesugaringContext, tr: string | any, warnings: string[], errors: string[], ctx: string): TagRenderingConfigJson[] {
+ if (tr === "questions") {
+ return [{
+ id: "questions"
+ }]
+ }
+
+
+ if (typeof tr === "string") {
+ const lookup = this.lookup(state, tr);
+ if (lookup !== undefined) {
+ return lookup
+ }
+ warnings.push(ctx + "A literal rendering was detected: " + tr)
+ return [{
+ render: tr,
+ id: tr.replace(/![a-zA-Z0-9]/g, "")
+ }]
+ }
+
+ if (tr["builtin"] !== undefined) {
+ let names = tr["builtin"]
+ if (typeof names === "string") {
+ names = [names]
+ }
+ const trs: TagRenderingConfigJson[] = []
+ for (const name of names) {
+ const lookup = this.lookup(state, name)
+ if (lookup === undefined) {
+ errors.push(ctx + ": The tagRendering with identifier " + name + " was not found.\n\tDid you mean one of " + Array.from(state.tagRenderings.keys()).join(", ") + "?")
+ continue
+ }
+ for (let tr of lookup) {
+ tr = Utils.Clone(tr)
+ Utils.Merge(tr["override"] ?? {}, tr)
+ trs.push(tr)
+ }
+ }
+ return trs;
+ }
+
+ return [tr]
+ }
+
+ private convertUntilStable(state: DesugaringContext, spec: string | any, warnings: string[], errors: string[], ctx: string): TagRenderingConfigJson[] {
+ const trs = this.convertOnce(state, spec, warnings, errors, ctx);
+
+ const result = []
+ for (const tr of trs) {
+ if (tr["builtin"] !== undefined) {
+ const stable = this.convertUntilStable(state, tr, warnings, errors, ctx + "(RECURSIVE RESOLVE)")
+ result.push(...stable)
+ } else {
+ result.push(tr)
+ }
+ }
+
+ return result;
+ }
+
+
+ convert(state: DesugaringContext, json: string | TagRenderingConfigJson | { builtin: string | string[]; override: any }, context: string): { result: TagRenderingConfigJson[]; errors: string[]; warnings: string[] } {
+ const errors = []
+ const warnings = []
+
+ return {
+ result: this.convertUntilStable(state, json, warnings, errors, context),
+ errors, warnings
+ };
+ }
+}
+
+class ExpandGroupRewrite extends Conversion<{
+ rewrite: {
+ sourceString: string,
+ into: string[]
+ }[],
+ renderings: (string | { builtin: string, override: any } | TagRenderingConfigJson)[]
+} | TagRenderingConfigJson, TagRenderingConfigJson[]> {
+
+
+ private static expandSubTagRenderings = new ExpandTagRendering()
+
+ constructor() {
+ super(
+ "Converts a rewrite config for tagRenderings into the expanded form"
+ );
+ }
+
+ /* Used for left|right group creation and replacement */
+ private prepConfig(keyToRewrite: string, target: string, tr: TagRenderingConfigJson) {
+
+ function replaceRecursive(transl: string | any) {
+ if (typeof transl === "string") {
+ return transl.replace(keyToRewrite, target)
+ }
+ if (transl.map !== undefined) {
+ return transl.map(o => replaceRecursive(o))
+ }
+ transl = {...transl}
+ for (const key in transl) {
+ transl[key] = replaceRecursive(transl[key])
+ }
+ return transl
+ }
+
+ const orig = tr;
+ tr = replaceRecursive(tr)
+
+ tr.id = target + "-" + orig.id
+ tr.group = target
+ return tr
+ }
+
+ convert(state: DesugaringContext, json:
+ {
+ rewrite:
+ { sourceString: string; into: string[] }[]; renderings: (string | { builtin: string; override: any } | TagRenderingConfigJson)[]
+ } | TagRenderingConfigJson, context: string): { result: TagRenderingConfigJson[]; errors: string[]; warnings: string[] } {
+
+ if (json["rewrite"] === undefined) {
+ return {result: [json], errors: [], warnings: []}
+ }
+ let config = <{
+ rewrite:
+ { sourceString: string; into: string[] }[];
+ renderings: (string | { builtin: string; override: any } | TagRenderingConfigJson)[]
+ }>json;
+
+
+ const subRenderingsRes = ExpandGroupRewrite.expandSubTagRenderings.convertAll(state, config.renderings, context);
+ const subRenderings: TagRenderingConfigJson[] = [].concat(subRenderingsRes.result);
+ const errors = subRenderingsRes.errors;
+ const warnings = subRenderingsRes.warnings;
+
+
+ const rewrittenPerGroup = new Map()
+
+ // The actual rewriting
+ for (const rewrite of config.rewrite) {
+ const source = rewrite.sourceString;
+ for (const target of rewrite.into) {
+ const groupName = target;
+ const trs: TagRenderingConfigJson[] = []
+
+ for (const tr of subRenderings) {
+ trs.push( this.prepConfig(source, target, tr))
+ }
+ if(rewrittenPerGroup.has(groupName)){
+ rewrittenPerGroup.get(groupName).push(...trs)
+
+ }else{
+ rewrittenPerGroup.set(groupName, trs)
+
+ }
+ }
+ }
+
+ // Add questions box for this category
+ rewrittenPerGroup.forEach((group, groupName) => {
+ group.push({
+ id: "questions",
+ group: groupName
+ })
+ })
+
+
+ rewrittenPerGroup.forEach((group, groupName) => {
+ group.forEach(tr => {
+ if(tr.id === undefined || tr.id === ""){
+ errors.push("A tagrendering has an empty ID after expanding the tag")
+ }
+ })
+ })
+
+ return {
+ result: [].concat(...Array.from(rewrittenPerGroup.values())),
+ errors, warnings
+ };
+ }
+}
+
+
+export class UpdateLegacyLayer extends DesugaringStep {
+
+ constructor() {
+ super("Updates various attributes from the old data format to the new to provide backwards compatibility with the formats",
+ ["overpassTags", "source.osmtags", "tagRenderings[*].id", "mapRendering"]);
+ }
+
+ convert(state: {}, json: LayerConfigJson, context: string): { result: LayerConfigJson; errors: string[]; warnings: string[] } {
+ const warnings = []
+ if (typeof json === "string") {
+ return json
+ }
+ if (json["builtin"] !== undefined) {
+ // @ts-ignore
+ return json;
+ }
+ let config: any = {...json};
- /**
- * Updates the config file in-place
- * @param config
- * @private
- */
- public static fixLayerConfig(config: any): void {
if (config["overpassTags"]) {
config.source = config.source ?? {}
config.source.osmTags = config["overpassTags"]
@@ -16,7 +406,12 @@ export default class LegacyJsonConvert {
}
if (config.tagRenderings !== undefined) {
+ let i =0;
for (const tagRendering of config.tagRenderings) {
+ i++;
+ if(typeof tagRendering === "string" || tagRendering["builtin"] !== undefined){
+ continue
+ }
if (tagRendering["id"] === undefined) {
if (tagRendering["#"] !== undefined) {
@@ -24,11 +419,14 @@ export default class LegacyJsonConvert {
delete tagRendering["#"]
} else if (tagRendering["freeform"]?.key !== undefined) {
tagRendering["id"] = config.id + "-" + tagRendering["freeform"]["key"]
+ }else{
+ tagRendering["id"] = "tr-"+i
}
}
}
}
+
if (config.mapRendering === undefined) {
config.mapRendering = []
// This is a legacy format, lets create a pointRendering
@@ -37,19 +435,18 @@ export default class LegacyJsonConvert {
if (wayHandling !== 0) {
location = ["point", "centroid"]
}
- if(config["icon"] ?? config["label"] !== undefined){
-
- const pointConfig = {
- icon: config["icon"],
- iconBadges: config["iconOverlays"],
- label: config["label"],
- iconSize: config["iconSize"],
- location,
- rotation: config["rotation"]
+ if (config["icon"] ?? config["label"] !== undefined) {
+
+ const pointConfig = {
+ icon: config["icon"],
+ iconBadges: config["iconOverlays"],
+ label: config["label"],
+ iconSize: config["iconSize"],
+ location,
+ rotation: config["rotation"]
+ }
+ config.mapRendering.push(pointConfig)
}
- config.mapRendering.push(pointConfig)
- }
-
if (wayHandling !== 1) {
const lineRenderConfig = {
@@ -61,12 +458,13 @@ export default class LegacyJsonConvert {
config.mapRendering.push(lineRenderConfig)
}
}
- if(config.mapRendering.length === 0){
- throw "Could not convert the legacy theme into a new theme: no renderings defined for layer "+config.id
+ if (config.mapRendering.length === 0) {
+ throw "Could not convert the legacy theme into a new theme: no renderings defined for layer " + config.id
}
}
+
delete config["color"]
delete config["width"]
delete config["dashArray"]
@@ -78,39 +476,470 @@ export default class LegacyJsonConvert {
delete config["rotation"]
delete config["wayHandling"]
delete config["hideUnderlayingFeaturesMinPercentage"]
-
+
for (const mapRenderingElement of config.mapRendering) {
if (mapRenderingElement["iconOverlays"] !== undefined) {
mapRenderingElement["iconBadges"] = mapRenderingElement["iconOverlays"]
}
for (const overlay of mapRenderingElement["iconBadges"] ?? []) {
if (overlay["badge"] !== true) {
- console.log("Warning: non-overlay element for ", config.id)
+ warnings.push("Warning: non-overlay element for ", config.id)
}
delete overlay["badge"]
}
}
+ return {
+ result: config,
+ errors: [],
+ warnings
+ };
}
+}
- /**
- * Given an old (parsed) JSON-config, will (in place) fix some issues
- * @param oldThemeConfig: the config to update to the latest format
- */
- public static fixThemeConfig(oldThemeConfig: any): void {
- for (const layerConfig of oldThemeConfig.layers ?? []) {
- if (typeof layerConfig === "string" || layerConfig["builtin"] !== undefined) {
- continue
+class UpdateLegacyTheme extends DesugaringStep {
+ constructor() {
+ super("Small fixes in the theme config", ["roamingRenderings"]);
+ }
+
+ convert(state: DesugaringContext, json: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors: string[]; warnings: string[] } {
+ const oldThemeConfig = {...json}
+ if (oldThemeConfig["roamingRenderings"] !== undefined) {
+
+ if (oldThemeConfig["roamingRenderings"].length == 0) {
+ delete oldThemeConfig["roamingRenderings"]
+ } else {
+ return {
+ result: null,
+ errors: [context + ": The theme contains roamingRenderings. These are not supported anymore"],
+ warnings: []
+ }
}
- // @ts-ignore
- LegacyJsonConvert.fixLayerConfig(layerConfig)
}
-
- if (oldThemeConfig["roamingRenderings"] !== undefined && oldThemeConfig["roamingRenderings"].length == 0) {
- delete oldThemeConfig["roamingRenderings"]
+ return {
+ errors: [],
+ warnings: [],
+ result: oldThemeConfig
}
}
+}
+
+export class FixLegacyTheme extends Fuse {
+ constructor() {
+ super(
+ "Fixes a legacy theme to the modern JSON format geared to humans. Syntactic sugars are kept (i.e. no tagRenderings are expandend, no dependencies are automatically gathered)",
+ new UpdateLegacyTheme(),
+ new OnEvery("layers", new UpdateLegacyLayer())
+ );
+ }
+}
+
+export class ValidateLayer extends DesugaringStep {
+ /**
+ * The paths where this layer is originally saved. Triggers some extra checks
+ * @private
+ */
+ private readonly _path?: string;
+ private readonly knownImagePaths?: Set;
+ private readonly _isBuiltin: boolean;
+
+ constructor(knownImagePaths: Set, path: string, isBuiltin: boolean) {
+ super("Doesn't change anything, but emits warnings and errors", []);
+ this.knownImagePaths = knownImagePaths;
+ this._path = path;
+ this._isBuiltin = isBuiltin;
+ }
+
+ convert(state: DesugaringContext, json: LayerConfigJson, context: string): { result: LayerConfigJson; errors: string[]; warnings: string[] } {
+ const errors = []
+ const warnings = []
+
+ if (typeof json === "string") {
+ errors.push(context + ": This layer hasn't been expanded: " + json)
+ return {
+ result: null,
+ warnings: [],
+ errors
+ }
+ }
+
+ if (json["builtin"] !== undefined) {
+ errors.push(context + ": This layer hasn't been expanded: " + json)
+ return {
+ result: null,
+ warnings: [],
+ errors
+ }
+ }
+
+ try {
+ {
+ // Some checks for legacy elements
+
+ if (json["overpassTags"] !== undefined) {
+ errors.push("Layer " + json.id + "still uses the old 'overpassTags'-format. Please use \"source\": {\"osmTags\": }' instead of \"overpassTags\": (note: this isn't your fault, the custom theme generator still spits out the old format)")
+ }
+ const forbiddenTopLevel = ["icon", "wayHandling", "roamingRenderings", "roamingRendering", "label", "width", "color", "colour", "iconOverlays"]
+ for (const forbiddenKey of forbiddenTopLevel) {
+ if (json[forbiddenKey] !== undefined)
+ errors.push(context + ": layer " + json.id + " still has a forbidden key " + forbiddenKey)
+ }
+ if (json["hideUnderlayingFeaturesMinPercentage"] !== undefined) {
+ errors.push(context + ": layer " + json.id + " contains an old 'hideUnderlayingFeaturesMinPercentage'")
+ }
+ }
+ {
+ const layer = new LayerConfig(json, "test", true)
+ const images = Array.from(layer.ExtractImages())
+ const remoteImages = images.filter(img => img.indexOf("http") == 0)
+ for (const remoteImage of remoteImages) {
+ errors.push("Found a remote image: " + remoteImage + " in layer " + layer.id + ", please download it. You can use the fixTheme script to automate this")
+ }
+ for (const image of images) {
+ if (image.indexOf("{") >= 0) {
+ warnings.push("Ignoring image with { in the path: ", image)
+ continue
+ }
+
+ if (this.knownImagePaths !== undefined && !this.knownImagePaths.has(image)) {
+ const ctx = context === undefined ? "" : ` in a layer defined in the theme ${context}`
+ errors.push(`Image with path ${image} not found or not attributed; it is used in ${layer.id}${ctx}`)
+ }
+ }
+
+ }
+ {
+ // CHeck location
+ const expected: string = `assets/layers/${json.id}/${json.id}.json`
+ if (this._path != undefined && this._path.indexOf(expected) < 0) {
+ errors.push("Layer is in an incorrect place. The path is " + this._path + ", but expected " + expected)
+ }
+ }
+ if (this._isBuiltin ) {
+ if (json.tagRenderings?.some(tr => tr["id"] === "")) {
+ const emptyIndexes : number[] = []
+ for (let i = 0; i < json.tagRenderings.length; i++){
+ const tagRendering = json.tagRenderings[i];
+ if(tagRendering["id"] === ""){
+ emptyIndexes.push(i)
+ }
+ }
+ errors.push(`Some tagrendering-ids are empty or have an emtpy string; this is not allowed (at ${context}.tagRenderings.[${emptyIndexes.join(",")}])`)
+ }
+
+ const duplicateIds = Utils.Dupiclates((json.tagRenderings ?? [])?.map(f => f["id"]).filter(id => id !== "questions"))
+ if (duplicateIds.length > 0 && !Utils.runningFromConsole) {
+ errors.push(`Some tagRenderings have a duplicate id: ${duplicateIds} (at ${context}.tagRenderings)`)
+ }
+
+
+ if(json.description === undefined){
+
+ if (Constants.priviliged_layers.indexOf(json.id) >= 0) {
+ errors.push(
+ context + ": A priviliged layer must have a description"
+ )
+ } else {
+ warnings.push(
+ context + ": A builtin layer should have a description"
+ )
+ }}
+ }
+ } catch (e) {
+ errors.push(e)
+ }
+ return {
+ result: undefined,
+ errors,
+ warnings
+ };
+ }
+}
+
+class ValidateLanguageCompleteness extends DesugaringStep {
+ private readonly _languages: string[];
+
+ constructor(...languages: string[]) {
+ super("Checks that the given object is fully translated in the specified languages", []);
+ this._languages = languages;
+ }
+
+ convert(state: DesugaringContext, obj: any, context: string): { result: LayerConfig; errors: string[]; warnings: string[] } {
+ const errors = []
+ const translations = Translation.ExtractAllTranslationsFrom(
+ obj
+ )
+ for (const neededLanguage of this._languages) {
+ translations
+ .filter(t => t.tr.translations[neededLanguage] === undefined && t.tr.translations["*"] === undefined)
+ .forEach(missing => {
+ errors.push(context + "A theme should be translation-complete for " + neededLanguage + ", but it lacks a translation for " + missing.context + ".\n\tThe english translation is " + missing.tr.textFor('en'))
+ })
+ }
+
+ return {
+ result: obj,
+ warnings: [], errors
+ };
+ }
+}
+
+class ValidateTheme extends DesugaringStep {
+ /**
+ * The paths where this layer is originally saved. Triggers some extra checks
+ * @private
+ */
+ private readonly _path?: string;
+ private readonly knownImagePaths: Set;
+ private readonly _isBuiltin: boolean;
+
+ constructor(knownImagePaths: Set, path: string, isBuiltin: boolean) {
+ super("Doesn't change anything, but emits warnings and errors", []);
+ this.knownImagePaths = knownImagePaths;
+ this._path = path;
+ this._isBuiltin = isBuiltin;
+ }
+
+ convert(state: DesugaringContext, json: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors: string[]; warnings: string[] } {
+ const errors = []
+ const warnings = []
+ {
+ // Legacy format checks
+ if (this._isBuiltin) {
+ if (typeof json.language === "string") {
+ errors.push("The theme " + json.id + " has a string as language. Please use a list of strings")
+ }
+ if (json["units"] !== undefined) {
+ errors.push("The theme " + json.id + " has units defined - these should be defined on the layer instead. (Hint: use overrideAll: { '+units': ... }) ")
+ }
+ if (json["roamingRenderings"] !== undefined) {
+ errors.push("Theme " + json.id + " contains an old 'roamingRenderings'. Use an 'overrideAll' instead")
+ }
+ }
+ }
+
+ try {
+ const theme = new LayoutConfig(json, true, "test")
+ if (theme.id !== theme.id.toLowerCase()) {
+ errors.push("Theme ids should be in lowercase, but it is " + theme.id)
+ }
+
+ const filename = this._path.substring(this._path.lastIndexOf("/") + 1, this._path.length - 5)
+ if (theme.id !== filename) {
+ errors.push("Theme ids should be the same as the name.json, but we got id: " + theme.id + " and filename " + filename + " (" + this._path + ")")
+ }
+ if (!this.knownImagePaths.has(theme.icon)) {
+ errors.push("The theme image " + theme.icon + " is not attributed or not saved locally")
+ }
+ const dups = Utils.Dupiclates(json.layers.map(layer => layer["id"]))
+ if (dups.length > 0) {
+ errors.push(`The theme ${json.id} defines multiple layers with id ${dups.join(", ")}`)
+ }
+ if (json["mustHaveLanguage"] !== undefined) {
+ const checked = new ValidateLanguageCompleteness(...json["mustHaveLanguage"])
+ .convert(state, theme, theme.id)
+ errors.push(...checked.errors)
+ warnings.push(...checked.warnings)
+ }
+
+ } catch (e) {
+ errors.push(e)
+ }
+
+ return {
+ result: json,
+ errors,
+ warnings
+ };
+ }
+}
+
+export class ValidateThemeAndLayers extends Fuse {
+ constructor(knownImagePaths: Set, path: string, isBuiltin: boolean) {
+ super("Validates a theme and the contained layers",
+ new ValidateTheme(knownImagePaths, path, isBuiltin),
+ new OnEvery("layers", new ValidateLayer(knownImagePaths, undefined, false))
+ );
+ }
+}
+
+class AddDependencyLayersToTheme extends DesugaringStep {
+ constructor() {
+ super("If a layer has a dependency on another layer, these layers are added automatically on the theme. (For example: defibrillator depends on 'walls_and_buildings' to snap onto. This layer is added automatically)", ["layers"]);
+ }
+
+ private static CalculateDependencies(alreadyLoaded: LayerConfigJson[], allKnownLayers: Map, themeId: string): LayerConfigJson[] {
+ const dependenciesToAdd: LayerConfigJson[] = []
+ const loadedLayerIds: Set = new Set(alreadyLoaded.map(l => l.id));
+
+ // Verify cross-dependencies
+ let unmetDependencies: { neededLayer: string, neededBy: string, reason: string, context?: string }[] = []
+ do {
+ const dependencies: { neededLayer: string, reason: string, context?: string, neededBy: string }[] = []
+
+ for (const layerConfig of alreadyLoaded) {
+ const layerDeps = DependencyCalculator.getLayerDependencies(new LayerConfig(layerConfig))
+ dependencies.push(...layerDeps)
+ }
+
+ // During the generate script, builtin layers are verified but not loaded - so we have to add them manually here
+ // Their existance is checked elsewhere, so this is fine
+ unmetDependencies = dependencies.filter(dep => !loadedLayerIds.has(dep.neededLayer))
+ for (const unmetDependency of unmetDependencies) {
+ if(loadedLayerIds.has(unmetDependency.neededLayer)){
+ continue
+ }
+ const dep = allKnownLayers.get(unmetDependency.neededLayer)
+ if (dep === undefined) {
+ const message =
+ ["Loading a dependency failed: layer " + unmetDependency.neededLayer + " is not found, neither as layer of " + themeId + " nor as builtin layer.",
+ "This layer is needed by " + unmetDependency.neededBy,
+ unmetDependency.reason + " (at " + unmetDependency.context + ")",
+ "Loaded layers are: " + alreadyLoaded.map(l => l.id).join(",")
+
+ ]
+ throw message.join("\n\t");
+ }
+ dependenciesToAdd.unshift(dep)
+ loadedLayerIds.add(dep.id);
+ unmetDependencies = unmetDependencies.filter(d => d.neededLayer !== unmetDependency.neededLayer)
+ }
+
+ } while (unmetDependencies.length > 0)
+
+ return dependenciesToAdd;
+ }
+
+ convert(state: DesugaringContext, theme: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors: string[]; warnings: string[] } {
+ const allKnownLayers: Map = state.sharedLayers;
+ const knownTagRenderings: Map = state.tagRenderings;
+ const errors = [];
+ const warnings = [];
+ const layers: LayerConfigJson[] = theme.layers; // Layers should be expanded at this point
+
+ knownTagRenderings.forEach((value, key) => {
+ value.id = key;
+ })
+
+ const dependencies = AddDependencyLayersToTheme.CalculateDependencies(layers, allKnownLayers, theme.id);
+ if(dependencies.length > 0){
+
+ warnings.push(context+": added "+dependencies.map(d => d.id).join(", ")+" to the theme as they are needed")
+ }
+ layers.unshift(...dependencies);
+
+ return {
+ result: {
+ ...theme,
+ layers: layers
+ },
+ errors,
+ warnings
+ };
+ }
+}
+
+export class PrepareLayer extends Fuse {
+ constructor() {
+ super(
+ "Fully prepares and expands a layer for the LayerConfig.",
+ new OnEveryConcat("tagRenderings", new ExpandGroupRewrite()),
+ new OnEveryConcat("tagRenderings", new ExpandTagRendering()),
+ new OnEveryConcat("titleIcons", new ExpandTagRendering())
+ );
+ }
+}
+
+class SubstituteLayer extends Conversion<(string | LayerConfigJson), LayerConfigJson[]> {
+ constructor() {
+ super("Converts the identifier of a builtin layer into the actual layer, or converts a 'builtin' syntax with override in the fully expanded form", []);
+ }
+
+ convert(state: DesugaringContext, json: string | LayerConfigJson, context: string): { result: LayerConfigJson[]; errors: string[]; warnings: string[] } {
+ const errors = []
+ const warnings = []
+ if (typeof json === "string") {
+ const found = state.sharedLayers.get(json)
+ if (found === undefined) {
+ return {
+ result: null,
+ errors: [context + ": The layer with name " + json + " was not found as a builtin layer"],
+ warnings
+ }
+ }
+ return {
+ result: [found],
+ errors, warnings
+ }
+ }
+
+ if (json["builtin"] !== undefined) {
+ let names = json["builtin"]
+ if (typeof names === "string") {
+ names = [names]
+ }
+ const layers = []
+ for (const name of names) {
+ const found = Utils.Clone(state.sharedLayers.get(name))
+ if (found === undefined) {
+ errors.push(context + ": The layer with name " + json + " was not found as a builtin layer")
+ continue
+ }
+ Utils.Merge(json["override"], found);
+ layers.push(found)
+ }
+ return {
+ result: layers,
+ errors, warnings
+ }
+
+ }
+
+ return {
+ result: [json],
+ errors, warnings
+ };
+ }
+
+}
+
+class AddDefaultLayers extends DesugaringStep{
+
+ constructor() {
+ super("Adds the default layers, namely: "+Constants.added_by_default.join(", "),["layers"]);
+ }
+
+ convert(state: DesugaringContext, json: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors: string[]; warnings: string[] } {
+ const errors = []
+ json.layers = [...json.layers]
+ for (const layerName of Constants.added_by_default) {
+ const v = state.sharedLayers.get(layerName)
+ if(v === undefined){
+ errors.push("Default layer "+layerName+" not found")
+ }
+ json.layers.push(v)
+ }
+ return {
+ result: json,
+ errors,
+ warnings: []
+ };
+ }
+
+}
+
+export class PrepareTheme extends Fuse {
+ constructor() {
+ super(
+ "Fully prepares and expands a theme",
+ new OnEveryConcat("layers", new SubstituteLayer()),
+ new AddDefaultLayers(),
+ new AddDependencyLayersToTheme(),
+ new OnEvery("layers", new PrepareLayer()),
+
+ );
+ }
}
\ No newline at end of file
diff --git a/Models/ThemeConfig/TagRenderingConfig.ts b/Models/ThemeConfig/TagRenderingConfig.ts
index c9674097ff..23f6175832 100644
--- a/Models/ThemeConfig/TagRenderingConfig.ts
+++ b/Models/ThemeConfig/TagRenderingConfig.ts
@@ -40,7 +40,6 @@ export default class TagRenderingConfig {
readonly hideInAnswer: boolean | TagsFilter
readonly addExtraTags: Tag[]
}[]
-
constructor(json: string | TagRenderingConfigJson, context?: string) {
if (json === undefined) {
throw "Initing a TagRenderingConfig with undefined in " + context;
@@ -69,7 +68,7 @@ export default class TagRenderingConfig {
}
- this.id = json.id ?? "";
+ this.id = json.id ?? ""; // Some tagrenderings - especially for the map rendering - don't need an ID
if (this.id.match(/^[a-zA-Z0-9 ()?\/=:;,_-]*$/) === null) {
throw "Invalid ID in " + context + ": an id can only contain [a-zA-Z0-0_-] as characters. The offending id is: " + this.id
}
diff --git a/Models/ThemeConfig/WithContextLoader.ts b/Models/ThemeConfig/WithContextLoader.ts
index b6087143dc..5f59e65c84 100644
--- a/Models/ThemeConfig/WithContextLoader.ts
+++ b/Models/ThemeConfig/WithContextLoader.ts
@@ -1,20 +1,10 @@
import TagRenderingConfig from "./TagRenderingConfig";
import SharedTagRenderings from "../../Customizations/SharedTagRenderings";
import {TagRenderingConfigJson} from "./Json/TagRenderingConfigJson";
-import {Utils} from "../../Utils";
export default class WithContextLoader {
protected readonly _context: string;
private readonly _json: any;
-
- public static getKnownTagRenderings : ((id: string) => TagRenderingConfigJson[])= function(id) {
- const found = SharedTagRenderings.SharedTagRenderingJson.get(id)
- if(found !== undefined){
- return [found]
- }else{
- return []
- }
-}
constructor(json: any, context: string) {
this._json = json;
@@ -53,15 +43,15 @@ export default class WithContextLoader {
* A string is interpreted as a name to call
*/
public ParseTagRenderings(
- tagRenderings: (string | { builtin: string, override: any } | TagRenderingConfigJson)[],
- options?:{
+ tagRenderings: TagRenderingConfigJson[],
+ options?: {
/**
* Throw an error if 'question' is defined
*/
readOnlyMode?: boolean,
requiresId?: boolean
prepConfig?: ((config: TagRenderingConfigJson) => TagRenderingConfigJson)
-
+
}
): TagRenderingConfig[] {
if (tagRenderings === undefined) {
@@ -73,62 +63,17 @@ export default class WithContextLoader {
if (options.prepConfig === undefined) {
options.prepConfig = c => c
}
- const preparedConfigs : TagRenderingConfigJson[] = []
- for (let i = 0; i < tagRenderings.length; i++) {
- let renderingJson = tagRenderings[i]
- if(renderingJson === "questions"){
- renderingJson = {
- id: "questions"
- }
- }
- if (typeof renderingJson === "string") {
- renderingJson = {builtin: renderingJson, override: undefined}
- }
-
- if (renderingJson["builtin"] === undefined) {
- const patchedConfig = options.prepConfig(renderingJson)
- preparedConfigs.push(patchedConfig)
- continue
-
- }
-
-
- const renderingId = renderingJson["builtin"]
- let sharedJsons = []
- if(typeof renderingId === "string"){
- sharedJsons = WithContextLoader.getKnownTagRenderings(renderingId)
- }else{
- sharedJsons = [].concat( ...(renderingId).map(id => WithContextLoader.getKnownTagRenderings(id) ) )
- }
-
- if (sharedJsons.length === 0) {
- const keys = Array.from(SharedTagRenderings.SharedTagRenderingJson.keys());
- throw `Predefined tagRendering ${renderingId} not found in ${context}.\n Try one of ${keys.join(
- ", "
- )}\n If you intent to output this text literally, use {\"render\": } instead"}`;
- }
- for (let sharedJson of sharedJsons) {
- if (renderingJson["override"] !== undefined) {
- sharedJson = Utils.Merge(renderingJson["override"], JSON.parse(JSON.stringify(sharedJson)))
- }
-
- const patchedConfig = options.prepConfig(sharedJson)
- preparedConfigs.push(patchedConfig)
- }
-
- }
-
const renderings: TagRenderingConfig[] = []
- for (let i = 0; i < preparedConfigs.length; i++){
- const preparedConfig = preparedConfigs[i];
+ for (let i = 0; i < tagRenderings.length; i++) {
+ const preparedConfig = tagRenderings[i];
const tr = new TagRenderingConfig(preparedConfig, `${context}.tagrendering[${i}]`);
- if(options.readOnlyMode && tr.question !== undefined){
- throw "A question is defined for "+`${context}.tagrendering[${i}], but this is not allowed at this position - probably because this rendering is an icon, badge or label`
+ if (options.readOnlyMode && tr.question !== undefined) {
+ throw "A question is defined for " + `${context}.tagrendering[${i}], but this is not allowed at this position - probably because this rendering is an icon, badge or label`
}
- if(options.requiresId && tr.id === ""){
+ if (options.requiresId && tr.id === "") {
throw `${context}.tagrendering[${i}] has an invalid ID - make sure it is defined and not empty`
}
-
+
renderings.push(tr)
}
diff --git a/UI/AllThemesGui.ts b/UI/AllThemesGui.ts
index 3c83e2a495..ebdc645831 100644
--- a/UI/AllThemesGui.ts
+++ b/UI/AllThemesGui.ts
@@ -1,9 +1,9 @@
+import UserRelatedState from "../Logic/State/UserRelatedState";
import {FixedUiElement} from "./Base/FixedUiElement";
import Combine from "./Base/Combine";
import MoreScreen from "./BigComponents/MoreScreen";
import Translations from "./i18n/Translations";
import Constants from "../Models/Constants";
-import UserRelatedState from "../Logic/State/UserRelatedState";
import {Utils} from "../Utils";
import LanguagePicker from "./LanguagePicker";
import IndexText from "./BigComponents/IndexText";
@@ -13,7 +13,6 @@ import {SubtleButton} from "./Base/SubtleButton";
export default class AllThemesGui {
constructor() {
-
try {
new FixedUiElement("").AttachTo("centermessage")
@@ -41,6 +40,7 @@ export default class AllThemesGui {
.SetStyle("pointer-events: all;")
.AttachTo("topleft-tools");
} catch (e) {
+ console.error(">>>> CRITICAL", e)
new FixedUiElement("Seems like no layers are compiled - check the output of `npm run generate:layeroverview`. Is this visible online? Contact pietervdvn immediately!").SetClass("alert")
.AttachTo("centermessage")
}
diff --git a/UI/AutomatonGui.ts b/UI/AutomatonGui.ts
index 036d64e957..7712191bba 100644
--- a/UI/AutomatonGui.ts
+++ b/UI/AutomatonGui.ts
@@ -27,6 +27,7 @@ import {QueryParameters} from "../Logic/Web/QueryParameters";
import {SubstitutedTranslation} from "./SubstitutedTranslation";
import {AutoAction} from "./Popup/AutoApplyButton";
import DynamicGeoJsonTileSource from "../Logic/FeatureSource/TiledFeatureSource/DynamicGeoJsonTileSource";
+import * as themeOverview from "../assets/generated/theme_overview.json"
class AutomationPanel extends Combine{
@@ -177,7 +178,7 @@ class AutomationPanel extends Combine{
const feature = ffs.feature
const renderingTr = targetAction.GetRenderValue(feature.properties)
const rendering = renderingTr.txt
- log.push(""+feature.properties.id+" : "+new SubstitutedTranslation(renderingTr, new UIEventSource(feature.properties)).ConstructElement().innerText)
+ log.push(""+feature.properties.id+" : "+new SubstitutedTranslation(renderingTr, new UIEventSource(feature.properties), state).ConstructElement().innerText)
const actions = Utils.NoNull(SubstitutedTranslation.ExtractSpecialComponents(rendering)
.map(obj => obj.special))
for (const action of actions) {
@@ -251,7 +252,7 @@ class AutomatonGui {
private static GenerateMainPanel(): BaseUIElement {
const themeSelect = new DropDown("Select a theme",
- AllKnownLayouts.layoutsList.map(l => ({value: l.id, shown: l.id}))
+ Array.from(themeOverview).map(l => ({value: l.id, shown: l.id}))
)
LocalStorageSource.Get("automation-theme-id", "missing_streets").syncWith(themeSelect.GetValue())
diff --git a/UI/Base/SubtleButton.ts b/UI/Base/SubtleButton.ts
index a324505f77..13c3c82a74 100644
--- a/UI/Base/SubtleButton.ts
+++ b/UI/Base/SubtleButton.ts
@@ -8,32 +8,35 @@ import {UIElement} from "../UIElement";
export class SubtleButton extends UIElement {
+ private readonly imageUrl: string | BaseUIElement;
+ private readonly message: string | BaseUIElement;
+ private readonly linkTo: { url: string | UIEventSource; newTab?: boolean };
- private readonly _element: BaseUIElement
constructor(imageUrl: string | BaseUIElement, message: string | BaseUIElement, linkTo: { url: string | UIEventSource, newTab?: boolean } = undefined) {
super();
- this._element = SubtleButton.generateContent(imageUrl, message, linkTo)
- this.SetClass("block flex p-3 my-2 bg-blue-100 rounded-lg hover:shadow-xl hover:bg-blue-200 link-no-underline")
-
+ this.imageUrl = imageUrl;
+ this.message = message;
+ this.linkTo = linkTo;
}
- private static generateContent(imageUrl: string | BaseUIElement, messageT: string | BaseUIElement, linkTo: { url: string | UIEventSource, newTab?: boolean } = undefined): BaseUIElement {
- const message = Translations.W(messageT);
- message
+ protected InnerRender(): string | BaseUIElement {
+ const classes= "block flex p-3 my-2 bg-blue-100 rounded-lg hover:shadow-xl hover:bg-blue-200 link-no-underline";
+ const message = Translations.W(this.message);
let img;
- if ((imageUrl ?? "") === "") {
+ if ((this.imageUrl ?? "") === "") {
img = undefined;
- } else if (typeof (imageUrl) === "string") {
- img = new Img(imageUrl)
+ } else if (typeof (this.imageUrl) === "string") {
+ img = new Img(this.imageUrl)
} else {
- img = imageUrl;
+ img = this.imageUrl;
}
img?.SetClass("block flex items-center justify-center h-11 w-11 flex-shrink0 mr-4")
const image = new Combine([img])
.SetClass("flex-shrink-0");
- if (linkTo == undefined) {
+ if (this.linkTo == undefined) {
+ this.SetClass(classes)
return new Combine([
image,
message?.SetClass("block overflow-ellipsis"),
@@ -46,13 +49,10 @@ export class SubtleButton extends UIElement {
image,
message?.SetClass("block overflow-ellipsis")
]).SetClass("flex group w-full"),
- linkTo.url,
- linkTo.newTab ?? false
- )
- }
+ this.linkTo.url,
+ this.linkTo.newTab ?? false
+ ).SetClass(classes)
- protected InnerRender(): string | BaseUIElement {
- return this._element;
}
diff --git a/UI/BigComponents/FeaturedMessage.ts b/UI/BigComponents/FeaturedMessage.ts
index 508f1be0c7..f1a3af01e0 100644
--- a/UI/BigComponents/FeaturedMessage.ts
+++ b/UI/BigComponents/FeaturedMessage.ts
@@ -3,7 +3,7 @@ import * as welcome_messages from "../../assets/welcome_message.json"
import BaseUIElement from "../BaseUIElement";
import {FixedUiElement} from "../Base/FixedUiElement";
import MoreScreen from "./MoreScreen";
-import {AllKnownLayouts} from "../../Customizations/AllKnownLayouts";
+import * as themeOverview from "../../assets/generated/theme_overview.json"
import Translations from "../i18n/Translations";
import Title from "../Base/Title";
@@ -33,6 +33,12 @@ export default class FeaturedMessage extends Combine {
public static WelcomeMessages(): { start_date: Date, end_date: Date, message: string, featured_theme?: string }[] {
const all_messages: { start_date: Date, end_date: Date, message: string, featured_theme?: string }[] = []
+
+ const themesById = new Map();
+ for (const theme of themeOverview["default"]) {
+ themesById.set(theme.id, theme);
+ }
+
for (const i in welcome_messages) {
if (isNaN(Number(i))) {
continue
@@ -41,7 +47,8 @@ export default class FeaturedMessage extends Combine {
if (wm === null) {
continue
}
- if (AllKnownLayouts.allKnownLayouts.get(wm.featured_theme) === undefined) {
+ if (themesById.get(wm.featured_theme) === undefined) {
+ console.log("THEMES BY ID:", themesById)
console.error("Unkown featured theme for ", wm)
continue
}
@@ -71,7 +78,10 @@ export default class FeaturedMessage extends Combine {
const msg = new FixedUiElement(welcome_message.message).SetClass("link-underline font-lg")
els.push(new Combine([title, msg]).SetClass("m-4"))
if (welcome_message.featured_theme !== undefined) {
- els.push(MoreScreen.createLinkButton({}, AllKnownLayouts.allKnownLayouts.get(welcome_message.featured_theme))
+
+ const theme = themeOverview["default"].filter(th => th.id === welcome_message.featured_theme)[0];
+
+ els.push(MoreScreen.createLinkButton({}, theme)
.SetClass("m-4 self-center md:w-160")
.SetStyle("height: min-content;"))
diff --git a/UI/BigComponents/LicensePicker.ts b/UI/BigComponents/LicensePicker.ts
index 7e92e48455..ffedc5c10d 100644
--- a/UI/BigComponents/LicensePicker.ts
+++ b/UI/BigComponents/LicensePicker.ts
@@ -1,18 +1,18 @@
import {DropDown} from "../Input/DropDown";
import Translations from "../i18n/Translations";
-import State from "../../State";
import {UIEventSource} from "../../Logic/UIEventSource";
+import {OsmConnection} from "../../Logic/Osm/OsmConnection";
export default class LicensePicker extends DropDown {
- constructor() {
+ constructor(state: {osmConnection: OsmConnection}) {
super(Translations.t.image.willBePublished.Clone(),
[
{value: "CC0", shown: Translations.t.image.cco.Clone()},
{value: "CC-BY-SA 4.0", shown: Translations.t.image.ccbs.Clone()},
{value: "CC-BY 4.0", shown: Translations.t.image.ccb.Clone()}
],
- State.state?.osmConnection?.GetPreference("pictures-license") ?? new UIEventSource("CC0")
+ state?.osmConnection?.GetPreference("pictures-license") ?? new UIEventSource("CC0")
)
this.SetClass("flex flex-col sm:flex-row").SetStyle("float:left");
}
diff --git a/UI/BigComponents/MoreScreen.ts b/UI/BigComponents/MoreScreen.ts
index 6e5cbd9889..5cde4729f1 100644
--- a/UI/BigComponents/MoreScreen.ts
+++ b/UI/BigComponents/MoreScreen.ts
@@ -1,5 +1,4 @@
import {VariableUiElement} from "../Base/VariableUIElement";
-import {AllKnownLayouts} from "../../Customizations/AllKnownLayouts";
import Svg from "../../Svg";
import Combine from "../Base/Combine";
import {SubtleButton} from "../Base/SubtleButton";
@@ -15,6 +14,8 @@ import UserRelatedState from "../../Logic/State/UserRelatedState";
import Toggle from "../Input/Toggle";
import {Utils} from "../../Utils";
import Title from "../Base/Title";
+import * as themeOverview from "../../assets/generated/theme_overview.json"
+import {Translation} from "../i18n/Translation";
export default class MoreScreen extends Combine {
@@ -47,7 +48,12 @@ export default class MoreScreen extends Combine {
state: {
locationControl?: UIEventSource,
layoutToUse?: LayoutConfig
- }, layout: LayoutConfig, customThemeDefinition: string = undefined
+ }, layout: {
+ id: string,
+ icon: string,
+ title: any,
+ shortDescription: any
+ }, isCustom: boolean = false
):
BaseUIElement {
if (layout === undefined) {
@@ -73,14 +79,12 @@ export default class MoreScreen extends Combine {
}
let linkPrefix = `${path}/${layout.id.toLowerCase()}.html?`
- let linkSuffix = ""
if (location.hostname === "localhost" || location.hostname === "127.0.0.1") {
- linkPrefix = `${path}/index.html?layout=${layout.id}&`
+ linkPrefix = `${path}/theme.html?layout=${layout.id}&`
}
- if (customThemeDefinition) {
- linkPrefix = `${path}/index.html?userlayout=${layout.id}&`
- linkSuffix = `#${customThemeDefinition}`
+ if (isCustom) {
+ linkPrefix = `${path}/theme.html?userlayout=${layout.id}&`
}
const linkText = currentLocation?.map(currentLocation => {
@@ -91,17 +95,17 @@ export default class MoreScreen extends Combine {
].filter(part => part[1] !== undefined)
.map(part => part[0] + "=" + part[1])
.join("&")
- return `${linkPrefix}${params}${linkSuffix}`;
- }) ?? new UIEventSource(`${linkPrefix}${linkSuffix}`)
+ return `${linkPrefix}${params}`;
+ }) ?? new UIEventSource(`${linkPrefix}`)
return new SubtleButton(layout.icon,
new Combine([
``,
- Translations.WT(layout.title),
+ new Translation(layout.title),
` `,
``,
- Translations.WT(layout.shortDescription)?.SetClass("subtle") ?? "",
+ new Translation(layout.shortDescription)?.SetClass("subtle") ?? "",
` `,
]), {url: linkText, newTab: false});
}
@@ -111,7 +115,7 @@ export default class MoreScreen extends Combine {
if (customThemes.length <= 0) {
return undefined;
}
- const customThemeButtons = customThemes.map(theme => MoreScreen.createLinkButton(state, theme.layout, theme.definition)?.SetClass(buttonClass))
+ const customThemeButtons = customThemes.map(theme => MoreScreen.createLinkButton(state, theme, true)?.SetClass(buttonClass))
return new Combine([
Translations.t.general.customThemeIntro.Clone(),
new Combine(customThemeButtons).SetClass(themeListClasses)
@@ -122,27 +126,29 @@ export default class MoreScreen extends Combine {
private static createPreviouslyVistedHiddenList(state: UserRelatedState, buttonClass: string, themeListStyle: string) {
const t = Translations.t.general.morescreen
const prefix = "mapcomplete-hidden-theme-"
- const hiddenTotal = AllKnownLayouts.layoutsList.filter(layout => layout.hideFromOverview).length
+ const hiddenThemes = themeOverview["default"].filter(layout => layout.hideFromOverview)
+ const hiddenTotal = hiddenThemes.length
+
return new Toggle(
new VariableUiElement(
state.osmConnection.preferencesHandler.preferences.map(allPreferences => {
- const knownThemes = Utils.NoNull(Object.keys(allPreferences)
+ const knownThemes: Set = new Set(Utils.NoNull(Object.keys(allPreferences)
.filter(key => key.startsWith(prefix))
- .map(key => key.substring(prefix.length, key.length - "-enabled".length))
- .map(theme => AllKnownLayouts.allKnownLayouts.get(theme)))
- .filter(theme => theme?.hideFromOverview)
- if (knownThemes.length === 0) {
+ .map(key => key.substring(prefix.length, key.length - "-enabled".length))));
+
+ if(knownThemes.size === 0){
return undefined
}
+
+ const knownThemeDescriptions = hiddenThemes.filter(theme => knownThemes.has(theme.id))
+ .map(theme => MoreScreen.createLinkButton(state, theme)?.SetClass(buttonClass));
- const knownLayouts = new Combine(knownThemes.map(layout =>
- MoreScreen.createLinkButton(state, layout)?.SetClass(buttonClass)
- )).SetClass(themeListStyle)
+ const knownLayouts = new Combine(knownThemeDescriptions).SetClass(themeListStyle)
return new Combine([
new Title(t.previouslyHiddenTitle),
t.hiddenExplanation.Subs({
- hidden_discovered: "" + knownThemes.length,
+ hidden_discovered: "" + knownThemes.size,
total_hidden: "" + hiddenTotal
}),
knownLayouts
@@ -158,7 +164,7 @@ export default class MoreScreen extends Combine {
}
private static createOfficialThemesList(state: { osmConnection: OsmConnection, locationControl?: UIEventSource }, buttonClass: string): BaseUIElement {
- let officialThemes = AllKnownLayouts.layoutsList
+ let officialThemes = themeOverview["default"];
let buttons = officialThemes.map((layout) => {
if (layout === undefined) {
diff --git a/UI/DefaultGUI.ts b/UI/DefaultGUI.ts
index ebce8f1a88..c07dbf7bb9 100644
--- a/UI/DefaultGUI.ts
+++ b/UI/DefaultGUI.ts
@@ -15,13 +15,14 @@ import LeftControls from "./BigComponents/LeftControls";
import RightControls from "./BigComponents/RightControls";
import CenterMessageBox from "./CenterMessageBox";
import ShowDataLayer from "./ShowDataLayer/ShowDataLayer";
-import AllKnownLayers from "../Customizations/AllKnownLayers";
import ScrollableFullScreen from "./Base/ScrollableFullScreen";
import Translations from "./i18n/Translations";
import SimpleAddUI from "./BigComponents/SimpleAddUI";
import StrayClickHandler from "../Logic/Actors/StrayClickHandler";
import Lazy from "./Base/Lazy";
import {DefaultGuiState} from "./DefaultGuiState";
+import LayerConfig from "../Models/ThemeConfig/LayerConfig";
+import * as home_location_json from "../assets/layers/home_location/home_location.json";
/**
@@ -111,7 +112,7 @@ export default class DefaultGUI {
new ShowDataLayer({
leafletMap: state.leafletMap,
- layerToShow: AllKnownLayers.sharedLayers.get("home_location"),
+ layerToShow: new LayerConfig(home_location_json, "all_known_layers", true),
features: state.homeLocation,
enablePopups: false,
})
diff --git a/UI/Image/DeleteImage.ts b/UI/Image/DeleteImage.ts
index baa75d2627..a2e5cfa557 100644
--- a/UI/Image/DeleteImage.ts
+++ b/UI/Image/DeleteImage.ts
@@ -2,20 +2,21 @@ import {UIEventSource} from "../../Logic/UIEventSource";
import Translations from "../i18n/Translations";
import Toggle from "../Input/Toggle";
import Combine from "../Base/Combine";
-import State from "../../State";
import Svg from "../../Svg";
import {Tag} from "../../Logic/Tags/Tag";
import ChangeTagAction from "../../Logic/Osm/Actions/ChangeTagAction";
+import {Changes} from "../../Logic/Osm/Changes";
+import {OsmConnection} from "../../Logic/Osm/OsmConnection";
export default class DeleteImage extends Toggle {
- constructor(key: string, tags: UIEventSource) {
+ constructor(key: string, tags: UIEventSource, state: {changes?: Changes, osmConnection?: OsmConnection}) {
const oldValue = tags.data[key]
const isDeletedBadge = Translations.t.image.isDeleted.Clone()
.SetClass("rounded-full p-1")
.SetStyle("color:white;background:#ff8c8c")
.onClick(async () => {
- await State.state?.changes?.applyAction(new ChangeTagAction(tags.data.id, new Tag(key, oldValue), tags.data, {
+ await state?.changes?.applyAction(new ChangeTagAction(tags.data.id, new Tag(key, oldValue), tags.data, {
changeType: "answer",
theme: "test"
}))
@@ -25,7 +26,7 @@ export default class DeleteImage extends Toggle {
.SetClass("block w-full pl-4 pr-4")
.SetStyle("color:white;background:#ff8c8c; border-top-left-radius:30rem; border-top-right-radius: 30rem;")
.onClick(async () => {
- await State.state?.changes?.applyAction(
+ await state?.changes?.applyAction(
new ChangeTagAction(tags.data.id, new Tag(key, ""), tags.data, {
changeType: "answer",
theme: "test"
@@ -53,7 +54,7 @@ export default class DeleteImage extends Toggle {
tags.map(tags => (tags[key] ?? "") !== "")
),
undefined /*Login (and thus editing) is disabled*/,
- State.state.osmConnection.isLoggedIn
+ state.osmConnection.isLoggedIn
)
this.SetClass("cursor-pointer")
}
diff --git a/UI/Image/ImageCarousel.ts b/UI/Image/ImageCarousel.ts
index c0fc52eda0..1d4d7ef434 100644
--- a/UI/Image/ImageCarousel.ts
+++ b/UI/Image/ImageCarousel.ts
@@ -6,12 +6,14 @@ import {AttributedImage} from "./AttributedImage";
import BaseUIElement from "../BaseUIElement";
import Toggle from "../Input/Toggle";
import ImageProvider from "../../Logic/ImageProviders/ImageProvider";
+import {OsmConnection} from "../../Logic/Osm/OsmConnection";
+import {Changes} from "../../Logic/Osm/Changes";
export class ImageCarousel extends Toggle {
constructor(images: UIEventSource<{ key: string, url: string, provider: ImageProvider }[]>,
tags: UIEventSource,
- keys: string[]) {
+ state: {osmConnection?: OsmConnection, changes?: Changes}) {
const uiElements = images.map((imageURLS: { key: string, url: string, provider: ImageProvider }[]) => {
const uiElements: BaseUIElement[] = [];
for (const url of imageURLS) {
@@ -21,7 +23,7 @@ export class ImageCarousel extends Toggle {
if (url.key !== undefined) {
image = new Combine([
image,
- new DeleteImage(url.key, tags).SetClass("delete-image-marker absolute top-0 left-0 pl-3")
+ new DeleteImage(url.key, tags, state).SetClass("delete-image-marker absolute top-0 left-0 pl-3")
]).SetClass("relative");
}
image
diff --git a/UI/Image/ImageUploadFlow.ts b/UI/Image/ImageUploadFlow.ts
index 5a49d8d723..476a9c2a98 100644
--- a/UI/Image/ImageUploadFlow.ts
+++ b/UI/Image/ImageUploadFlow.ts
@@ -1,5 +1,4 @@
import {UIEventSource} from "../../Logic/UIEventSource";
-import State from "../../State";
import Combine from "../Base/Combine";
import Translations from "../i18n/Translations";
import Svg from "../../Svg";
@@ -13,13 +12,23 @@ import ChangeTagAction from "../../Logic/Osm/Actions/ChangeTagAction";
import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
import {FixedUiElement} from "../Base/FixedUiElement";
import {VariableUiElement} from "../Base/VariableUIElement";
+import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig";
+import {OsmConnection} from "../../Logic/Osm/OsmConnection";
+import {Changes} from "../../Logic/Osm/Changes";
export class ImageUploadFlow extends Toggle {
private static readonly uploadCountsPerId = new Map>()
- constructor(tagsSource: UIEventSource, imagePrefix: string = "image", text: string = undefined) {
+ constructor(tagsSource: UIEventSource,
+ state: {
+ osmConnection: OsmConnection;
+ layoutToUse: LayoutConfig;
+ changes: Changes,
+ featureSwitchUserbadge: UIEventSource;
+ },
+ imagePrefix: string = "image", text: string = undefined) {
const perId = ImageUploadFlow.uploadCountsPerId
const id = tagsSource.data.id
if (!perId.has(id)) {
@@ -41,17 +50,17 @@ export class ImageUploadFlow extends Toggle {
console.log("Adding image:" + key, url);
uploadedCount.data++
uploadedCount.ping()
- Promise.resolve(State.state.changes
+ Promise.resolve(state.changes
.applyAction(new ChangeTagAction(
tags.id, new Tag(key, url), tagsSource.data,
{
changeType: "add-image",
- theme: State.state.layoutToUse.id
+ theme: state.layoutToUse.id
}
)))
})
- const licensePicker = new LicensePicker()
+ const licensePicker = new LicensePicker(state)
const t = Translations.t.image;
@@ -90,7 +99,7 @@ export class ImageUploadFlow extends Toggle {
const tags = tagsSource.data;
- const layout = State.state?.layoutToUse
+ const layout = state?.layoutToUse
let matchingLayer: LayerConfig = undefined
for (const layer of layout?.layers ?? []) {
if (layer.source.osmTags.matchesProperties(tags)) {
@@ -102,7 +111,7 @@ export class ImageUploadFlow extends Toggle {
const title = matchingLayer?.title?.GetRenderValue(tags)?.ConstructElement()?.innerText ?? tags.name ?? "Unknown area";
const description = [
- "author:" + State.state.osmConnection.userDetails.data.name,
+ "author:" + state.osmConnection.userDetails.data.name,
"license:" + license,
"osmid:" + tags.id,
].join("\n");
@@ -146,17 +155,17 @@ export class ImageUploadFlow extends Toggle {
const pleaseLoginButton = t.pleaseLogin.Clone()
- .onClick(() => State.state.osmConnection.AttemptLogin())
+ .onClick(() => state.osmConnection.AttemptLogin())
.SetClass("login-button-friendly");
super(
new Toggle(
/*We can show the actual upload button!*/
uploadFlow,
/* User not logged in*/ pleaseLoginButton,
- State.state?.osmConnection?.isLoggedIn
+ state?.osmConnection?.isLoggedIn
),
undefined /* Nothing as the user badge is disabled*/,
- State.state.featureSwitchUserbadge
+ state.featureSwitchUserbadge
)
}
diff --git a/UI/OpeningHours/OpeningHoursVisualization.ts b/UI/OpeningHours/OpeningHoursVisualization.ts
index 9a2a30800a..2414fc00f2 100644
--- a/UI/OpeningHours/OpeningHoursVisualization.ts
+++ b/UI/OpeningHours/OpeningHoursVisualization.ts
@@ -1,6 +1,5 @@
import {UIEventSource} from "../../Logic/UIEventSource";
import Combine from "../Base/Combine";
-import State from "../../State";
import {FixedUiElement} from "../Base/FixedUiElement";
import {OH} from "./OpeningHours";
import Translations from "../i18n/Translations";
@@ -11,6 +10,7 @@ import Toggle from "../Input/Toggle";
import {VariableUiElement} from "../Base/VariableUIElement";
import Table from "../Base/Table";
import {Translation} from "../i18n/Translation";
+import {OsmConnection} from "../../Logic/Osm/OsmConnection";
export default class OpeningHoursVisualization extends Toggle {
private static readonly weekdays: Translation[] = [
@@ -23,7 +23,7 @@ export default class OpeningHoursVisualization extends Toggle {
Translations.t.general.weekdays.abbreviations.sunday,
]
- constructor(tags: UIEventSource, key: string, prefix = "", postfix = "") {
+ constructor(tags: UIEventSource, state:{osmConnection?: OsmConnection}, key: string, prefix = "", postfix = "") {
const tagsDirect = tags.data;
const ohTable = new VariableUiElement(tags
.map(tags => {
@@ -57,7 +57,7 @@ export default class OpeningHoursVisualization extends Toggle {
new Toggle(
new FixedUiElement(e).SetClass("subtle"),
undefined,
- State.state?.osmConnection?.userDetails.map(userdetails => userdetails.csCount >= Constants.userJourney.tagsVisibleAndWikiLinked)
+ state?.osmConnection?.userDetails.map(userdetails => userdetails.csCount >= Constants.userJourney.tagsVisibleAndWikiLinked)
)
]);
}
diff --git a/UI/Popup/EditableTagRendering.ts b/UI/Popup/EditableTagRendering.ts
index 48cd669efc..d26d5cee11 100644
--- a/UI/Popup/EditableTagRendering.ts
+++ b/UI/Popup/EditableTagRendering.ts
@@ -42,7 +42,7 @@ export default class EditableTagRendering extends Toggle {
}
private static CreateRendering(tags: UIEventSource, configuration: TagRenderingConfig, units: Unit[], editMode: UIEventSource): BaseUIElement {
- const answer: BaseUIElement = new TagRenderingAnswer(tags, configuration)
+ const answer: BaseUIElement = new TagRenderingAnswer(tags, configuration, State.state)
answer.SetClass("w-full")
let rendering = answer;
diff --git a/UI/Popup/FeatureInfoBox.ts b/UI/Popup/FeatureInfoBox.ts
index 5d1d5baf17..56345766c2 100644
--- a/UI/Popup/FeatureInfoBox.ts
+++ b/UI/Popup/FeatureInfoBox.ts
@@ -40,7 +40,7 @@ export default class FeatureInfoBox extends ScrollableFullScreen {
private static GenerateTitleBar(tags: UIEventSource,
layerConfig: LayerConfig): BaseUIElement {
- const title = new TagRenderingAnswer(tags, layerConfig.title ?? new TagRenderingConfig("POI"))
+ const title = new TagRenderingAnswer(tags, layerConfig.title ?? new TagRenderingConfig("POI"), State.state)
.SetClass("break-words font-bold sm:p-0.5 md:p-1 sm:p-1.5 md:p-2");
const titleIcons = new Combine(
layerConfig.titleIcons.map(icon => new TagRenderingAnswer(tags, icon,
@@ -88,7 +88,8 @@ export default class FeatureInfoBox extends ScrollableFullScreen {
if (tr.render !== undefined) {
questionBox.SetClass("text-sm")
- const renderedQuestion = new TagRenderingAnswer(tags, tr, tr.group + " questions", "", {
+ const renderedQuestion = new TagRenderingAnswer(tags, tr,State.state,
+ tr.group + " questions", "", {
specialViz: new Map([["questions", questionBox]])
})
const possiblyHidden = new Toggle(
@@ -163,7 +164,7 @@ export default class FeatureInfoBox extends ScrollableFullScreen {
const hasMinimap = layerConfig.tagRenderings.some(tr => FeatureInfoBox.hasMinimap(tr))
if (!hasMinimap) {
- allRenderings.push(new TagRenderingAnswer(tags, SharedTagRenderings.SharedTagRendering.get("minimap")))
+ allRenderings.push(new TagRenderingAnswer(tags, SharedTagRenderings.SharedTagRendering.get("minimap"), State.state))
}
editElements.push(
@@ -177,7 +178,7 @@ export default class FeatureInfoBox extends ScrollableFullScreen {
return undefined
}
- return new TagRenderingAnswer(tags, SharedTagRenderings.SharedTagRendering.get("last_edit"));
+ return new TagRenderingAnswer(tags, SharedTagRenderings.SharedTagRendering.get("last_edit"), State.state);
}, [State.state.featureSwitchIsDebugging, State.state.featureSwitchIsTesting])
)
diff --git a/UI/Popup/ImportButton.ts b/UI/Popup/ImportButton.ts
index 297b5faee2..4800dce616 100644
--- a/UI/Popup/ImportButton.ts
+++ b/UI/Popup/ImportButton.ts
@@ -19,7 +19,6 @@ import Svg from "../../Svg";
import {Utils} from "../../Utils";
import Minimap from "../Base/Minimap";
import ShowDataLayer from "../ShowDataLayer/ShowDataLayer";
-import AllKnownLayers from "../../Customizations/AllKnownLayers";
import StaticFeatureSource from "../../Logic/FeatureSource/Sources/StaticFeatureSource";
import ShowDataMultiLayer from "../ShowDataLayer/ShowDataMultiLayer";
import CreateWayWithPointReuseAction, {MergePointConfig} from "../../Logic/Osm/Actions/CreateWayWithPointReuseAction";
@@ -35,6 +34,8 @@ import ReplaceGeometryAction from "../../Logic/Osm/Actions/ReplaceGeometryAction
import CreateMultiPolygonWithPointReuseAction from "../../Logic/Osm/Actions/CreateMultiPolygonWithPointReuseAction";
import {Tag} from "../../Logic/Tags/Tag";
import TagApplyButton from "./TagApplyButton";
+import LayerConfig from "../../Models/ThemeConfig/LayerConfig";
+import * as conflation_json from "../../assets/layers/conflation/conflation.json";
import {GeoOperations} from "../../Logic/GeoOperations";
@@ -256,7 +257,7 @@ ${Utils.special_visualizations_importRequirementDocs}
zoomToFeatures: false,
features: changePreview,
allElements: state.allElements,
- layerToShow: AllKnownLayers.sharedLayers.get("conflation")
+ layerToShow: new LayerConfig(conflation_json, "all_known_layers", true)
})
})
diff --git a/UI/Popup/TagRenderingAnswer.ts b/UI/Popup/TagRenderingAnswer.ts
index c7724b5389..0e8bbf57e0 100644
--- a/UI/Popup/TagRenderingAnswer.ts
+++ b/UI/Popup/TagRenderingAnswer.ts
@@ -12,6 +12,7 @@ import TagRenderingConfig from "../../Models/ThemeConfig/TagRenderingConfig";
export default class TagRenderingAnswer extends VariableUiElement {
constructor(tagsSource: UIEventSource, configuration: TagRenderingConfig,
+ state: any,
contentClasses: string = "", contentStyle: string = "", options?:{
specialViz: Map
}) {
@@ -37,7 +38,7 @@ export default class TagRenderingAnswer extends VariableUiElement {
return undefined;
}
- const valuesToRender: BaseUIElement[] = trs.map(tr => new SubstitutedTranslation(tr, tagsSource, options?.specialViz))
+ const valuesToRender: BaseUIElement[] = trs.map(tr => new SubstitutedTranslation(tr, tagsSource, state, options?.specialViz))
if (valuesToRender.length === 1) {
return valuesToRender[0];
} else if (valuesToRender.length > 1) {
diff --git a/UI/Popup/TagRenderingQuestion.ts b/UI/Popup/TagRenderingQuestion.ts
index 9c0c530ab6..fdaf1e5329 100644
--- a/UI/Popup/TagRenderingQuestion.ts
+++ b/UI/Popup/TagRenderingQuestion.ts
@@ -71,7 +71,7 @@ export default class TagRenderingQuestion extends Combine {
}
options = options ?? {}
const applicableUnit = (options.units ?? []).filter(unit => unit.isApplicableToKey(configuration.freeform?.key))[0];
- const question = new SubstitutedTranslation(configuration.question, tags)
+ const question = new SubstitutedTranslation(configuration.question, tags, State.state)
.SetClass("question-text");
@@ -352,7 +352,7 @@ export default class TagRenderingQuestion extends Combine {
}
return new FixedInputElement(
- new SubstitutedTranslation(mapping.then, tagsSource),
+ new SubstitutedTranslation(mapping.then, tagsSource, State.state),
tagging,
(t0, t1) => t1.isEquivalent(t0));
}
diff --git a/UI/Professional.ts b/UI/Professional.ts
deleted file mode 100644
index 9d6d6edeb3..0000000000
--- a/UI/Professional.ts
+++ /dev/null
@@ -1,34 +0,0 @@
-import {FixedUiElement} from "./Base/FixedUiElement";
-import Combine from "./Base/Combine";
-import MoreScreen from "./BigComponents/MoreScreen";
-import Translations from "./i18n/Translations";
-import Constants from "../Models/Constants";
-import UserRelatedState from "../Logic/State/UserRelatedState";
-import {Utils} from "../Utils";
-import LanguagePicker from "./LanguagePicker";
-import IndexText from "./BigComponents/IndexText";
-import FeaturedMessage from "./BigComponents/FeaturedMessage";
-
-export default class Professional {
- constructor() {
- new FixedUiElement("").AttachTo("centermessage")
-
- const state = new UserRelatedState(undefined);
- const intro = new Combine([
- LanguagePicker.CreateLanguagePicker(Translations.t.index.title.SupportedLanguages())
- .SetClass("absolute top-2 right-3"),
- new IndexText()
- ]);
- new Combine([
- intro,
- new FeaturedMessage(),
- new MoreScreen(state, true),
- Translations.t.general.aboutMapcomplete
- .Subs({"osmcha_link": Utils.OsmChaLinkFor(7)})
- .SetClass("link-underline"),
- new FixedUiElement("v" + Constants.vNumber)
- ]).SetClass("block m-5 lg:w-3/4 lg:ml-40")
- .SetStyle("pointer-events: all;")
- .AttachTo("topleft-tools");
- }
-}
\ No newline at end of file
diff --git a/UI/ProfessionalGui.ts b/UI/ProfessionalGui.ts
index 4e305c5241..e6f4476ef5 100644
--- a/UI/ProfessionalGui.ts
+++ b/UI/ProfessionalGui.ts
@@ -97,7 +97,7 @@ export default class ProfessionalGui {
Svg.back_svg().SetStyle("height: 1.5rem;"),
t.backToMapcomplete,
{
- url: window.location.host + "/index.html"
+ url: "./index.html"
}
)]).SetClass("block")
diff --git a/UI/ShowDataLayer/ShowTileInfo.ts b/UI/ShowDataLayer/ShowTileInfo.ts
index e2ba01adf3..702e9d9c83 100644
--- a/UI/ShowDataLayer/ShowTileInfo.ts
+++ b/UI/ShowDataLayer/ShowTileInfo.ts
@@ -8,8 +8,7 @@ import {Tiles} from "../../Models/TileRange";
import * as clusterstyle from "../../assets/layers/cluster_style/cluster_style.json"
export default class ShowTileInfo {
- public static readonly styling = new LayerConfig(
- clusterstyle, "tileinfo", true)
+ public static readonly styling = new LayerConfig(clusterstyle, "ShowTileInfo", true)
constructor(options: {
source: FeatureSource & Tiled, leafletMap: UIEventSource, layer?: LayerConfig,
diff --git a/UI/SpecialVisualizations.ts b/UI/SpecialVisualizations.ts
index df2ff26d60..0855f566e1 100644
--- a/UI/SpecialVisualizations.ts
+++ b/UI/SpecialVisualizations.ts
@@ -27,7 +27,6 @@ import AllImageProviders from "../Logic/ImageProviders/AllImageProviders";
import WikipediaBox from "./Wikipedia/WikipediaBox";
import SimpleMetaTagger from "../Logic/SimpleMetaTagger";
import MultiApply from "./Popup/MultiApply";
-import AllKnownLayers from "../Customizations/AllKnownLayers";
import ShowDataLayer from "./ShowDataLayer/ShowDataLayer";
import {SubtleButton} from "./Base/SubtleButton";
import {DefaultGuiState} from "./DefaultGuiState";
@@ -37,6 +36,7 @@ import FeaturePipelineState from "../Logic/State/FeaturePipelineState";
import {ConflateButton, ImportPointButton, ImportWayButton} from "./Popup/ImportButton";
import TagApplyButton from "./Popup/TagApplyButton";
import AutoApplyButton from "./Popup/AutoApplyButton";
+import * as left_right_style_json from "../assets/layers/left_right_style/left_right_style.json";
import {OpenIdEditor} from "./BigComponents/CopyrightPanel";
export interface SpecialVisualization {
@@ -52,7 +52,6 @@ export default class SpecialVisualizations {
public static specialVisualizations = SpecialVisualizations.init()
-
private static init(){
const specialVisualizations: SpecialVisualization[] =
[
@@ -105,7 +104,7 @@ export default class SpecialVisualizations {
if (args.length > 0) {
imagePrefixes = [].concat(...args.map(a => a.split(",")));
}
- return new ImageCarousel(AllImageProviders.LoadImagesFor(tags, imagePrefixes), tags, imagePrefixes);
+ return new ImageCarousel(AllImageProviders.LoadImagesFor(tags, imagePrefixes), tags, state);
}
},
{
@@ -121,7 +120,7 @@ export default class SpecialVisualizations {
defaultValue: "Add image"
}],
constr: (state: State, tags, args) => {
- return new ImageUploadFlow(tags, args[0], args[1])
+ return new ImageUploadFlow(tags, state, args[0], args[1])
}
},
{
@@ -162,7 +161,7 @@ export default class SpecialVisualizations {
}
],
example: "`{minimap()}`, `{minimap(17, id, _list_of_embedded_feature_ids_calculated_by_calculated_tag):height:10rem; border: 2px solid black}`",
- constr: (state, tagSource, args, defaultGuiState) => {
+ constr: (state, tagSource, args, _) => {
const keys = [...args]
keys.splice(0, 1)
@@ -268,7 +267,7 @@ export default class SpecialVisualizations {
leafletMap: minimap["leafletMap"],
enablePopups: false,
zoomToFeatures: true,
- layerToShow: AllKnownLayers.sharedLayers.get("left_right_style"),
+ layerToShow: new LayerConfig(left_right_style_json, "all_known_layers", true),
features: new StaticFeatureSource([copy], false),
allElements: State.state.allElements
}
@@ -325,7 +324,7 @@ export default class SpecialVisualizations {
}],
example: "A normal opening hours table can be invoked with `{opening_hours_table()}`. A table for e.g. conditional access with opening hours can be `{opening_hours_table(access:conditional, no @ &LPARENS, &RPARENS)}`",
constr: (state: State, tagSource: UIEventSource, args) => {
- return new OpeningHoursVisualization(tagSource, args[0], args[1], args[2])
+ return new OpeningHoursVisualization(tagSource, state, args[0], args[1], args[2])
}
},
{
diff --git a/UI/SubstitutedTranslation.ts b/UI/SubstitutedTranslation.ts
index c753875dc6..98a889952e 100644
--- a/UI/SubstitutedTranslation.ts
+++ b/UI/SubstitutedTranslation.ts
@@ -1,7 +1,6 @@
import {UIEventSource} from "../Logic/UIEventSource";
import {Translation} from "./i18n/Translation";
import Locale from "./i18n/Locale";
-import State from "../State";
import {FixedUiElement} from "./Base/FixedUiElement";
import SpecialVisualizations, {SpecialVisualization} from "./SpecialVisualizations";
import {Utils} from "../Utils";
@@ -15,6 +14,7 @@ export class SubstitutedTranslation extends VariableUiElement {
public constructor(
translation: Translation,
tagsSource: UIEventSource,
+ state,
mapping: Map = undefined) {
const extraMappings: SpecialVisualization[] = [];
@@ -50,7 +50,7 @@ export class SubstitutedTranslation extends VariableUiElement {
}
const viz = proto.special;
try {
- return viz.func.constr(State.state, tagsSource, proto.special.args, DefaultGuiState.state).SetStyle(proto.special.style);
+ return viz.func.constr(state, tagsSource, proto.special.args, DefaultGuiState.state).SetStyle(proto.special.style);
} catch (e) {
console.error("SPECIALRENDERING FAILED for", tagsSource.data?.id, e)
return new FixedUiElement(`Could not generate special rendering for ${viz.func.funcName}(${viz.args.join(", ")}) ${e}`).SetStyle("alert")
diff --git a/UI/i18n/Translation.ts b/UI/i18n/Translation.ts
index d914425a2d..b88cc7b379 100644
--- a/UI/i18n/Translation.ts
+++ b/UI/i18n/Translation.ts
@@ -13,6 +13,9 @@ export class Translation extends BaseUIElement {
if (translations === undefined) {
throw `Translation without content (${context})`
}
+ if(typeof translations === "string"){
+ translations = {"*": translations};
+ }
let count = 0;
for (const translationsKey in translations) {
if (!translations.hasOwnProperty(translationsKey)) {
diff --git a/Utils.ts b/Utils.ts
index 943addefcc..5928fa86fe 100644
--- a/Utils.ts
+++ b/Utils.ts
@@ -191,7 +191,7 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
return newArr;
}
- public static Dupicates(arr: string[]): string[] {
+ public static Dupiclates(arr: string[]): string[] {
if (arr === undefined) {
return undefined;
}
@@ -618,5 +618,17 @@ In the case that MapComplete is pointed to the testing grounds, the edit will be
b: parseInt(hex.substr(5, 2), 16),
}
}
+
+ /**
+ * Deepclone an object by serializing and deserializing it
+ * @param x
+ * @constructor
+ */
+ static Clone(x: T): T {
+ if(x === undefined){
+ return undefined;
+ }
+ return JSON.parse(JSON.stringify(x));
+ }
}
diff --git a/all_themes_index.ts b/all_themes_index.ts
new file mode 100644
index 0000000000..d18acadbd8
--- /dev/null
+++ b/all_themes_index.ts
@@ -0,0 +1,18 @@
+import {Utils} from "./Utils";
+import AllThemesGui from "./UI/AllThemesGui";
+import {QueryParameters} from "./Logic/Web/QueryParameters";
+
+
+const layout = QueryParameters.GetQueryParameter("layout", undefined).data ?? ""
+const customLayout = QueryParameters.GetQueryParameter("userlayout", undefined).data ?? ""
+const l = window.location;
+if( layout !== ""){
+ window.location.replace(l.protocol + "//" + window.location.host+"/"+layout+".html"+ l.search + l.hash);
+}else if (customLayout !== ""){
+ window.location.replace(l.protocol + "//" + window.location.host+"/theme.html"+ l.search + l.hash);
+}
+
+
+Utils.DisableLongPresses()
+document.getElementById("decoration-desktop").remove();
+new AllThemesGui();
\ No newline at end of file
diff --git a/assets/tagRenderings/icons.json b/assets/tagRenderings/icons.json
index 06b97f81ad..864f275397 100644
--- a/assets/tagRenderings/icons.json
+++ b/assets/tagRenderings/icons.json
@@ -1,11 +1,14 @@
{
- "defaultIcons": ["phonelink",
- "emaillink",
- "wikipedialink",
- "osmlink",
- "sharelink"
- ],
-
+ "defaults": {
+ "builtin": [
+ "phonelink",
+ "emaillink",
+ "wikipedialink",
+ "osmlink",
+ "sharelink"
+ ],
+ "override": {}
+ },
"wikipedialink": {
"render": " ",
"condition": {
diff --git a/assets/themes/cyclofix/cyclofix.json b/assets/themes/cyclofix/cyclofix.json
index 23a2adb324..4fb30eea91 100644
--- a/assets/themes/cyclofix/cyclofix.json
+++ b/assets/themes/cyclofix/cyclofix.json
@@ -36,7 +36,7 @@
],
"maintainer": "MapComplete",
"credits": "Originally created during Open Summer of Code by Pieter Fiers, Thibault Declercq, Pierre Barban, Joost Schouppe and Pieter Vander Vennet",
- "icon": "assets/themes/cyclofix/logo.svg",
+ "icon": "./assets/themes/cyclofix/logo.svg",
"version": "0",
"startLat": 0,
"defaultBackgroundId": "CartoDB.Voyager",
diff --git a/assets/themes/natuurpunt/natuurpunt.json b/assets/themes/natuurpunt/natuurpunt.json
index 32fed86b13..7f98d18839 100644
--- a/assets/themes/natuurpunt/natuurpunt.json
+++ b/assets/themes/natuurpunt/natuurpunt.json
@@ -80,6 +80,7 @@
"builtin": "nature_reserve",
"wayHandling": 1,
"override": {
+ "id": "nature_reserve_centerpoints",
"source": {
"osmTags": {
"+and": [
diff --git a/assets/themes/sidewalks/sidewalks.json b/assets/themes/sidewalks/sidewalks.json
index 7217099e88..c750c05fab 100644
--- a/assets/themes/sidewalks/sidewalks.json
+++ b/assets/themes/sidewalks/sidewalks.json
@@ -74,11 +74,11 @@
},
"renderings": [
{
- "id": "sidewalk_minimap",
+ "id": "sidewalk_minimap_left|right",
"render": "{sided_minimap(left|right):height:8rem;border-radius:0.5rem;overflow:hidden}"
},
{
- "id": "has_sidewalk",
+ "id": "has_sidewalk_left|right",
"question": "Is there a sidewalk on this side of the road?",
"mappings": [
{
@@ -92,7 +92,7 @@
]
},
{
- "id": "sidewalk_width",
+ "id": "sidewalk_width_left|right",
"question": "What is the width of the sidewalk on this side of the road?",
"render": "This sidewalk is {sidewalk:left|right:width}m wide",
"condition": "sidewalk:left|right=yes",
diff --git a/assets/themes/speelplekken/speelplekken.json b/assets/themes/speelplekken/speelplekken.json
index 3646004ee4..adf40461a3 100644
--- a/assets/themes/speelplekken/speelplekken.json
+++ b/assets/themes/speelplekken/speelplekken.json
@@ -50,7 +50,6 @@
"geoJsonZoomLevel": 14,
"isOsmCache": true
},
- "icon": "./assets/themes/speelplekken/speelbos.svg",
"minzoom": 12,
"calculatedTags": [
"_is_shadowed=feat.overlapWith('shadow').length > 0 ? 'yes': ''",
@@ -61,7 +60,6 @@
{
"builtin": "playground",
"override": {
- "icon": "./assets/themes/speelplekken/speeltuin.svg",
"minzoom": 14,
"source": {
"geoJsonLocal": "http://127.0.0.1:8080/speelplekken_{layer}_{z}_{x}_{y}.geojson",
@@ -78,7 +76,6 @@
{
"builtin": "village_green",
"override": {
- "icon": "./assets/themes/speelplekken/speelweide.svg",
"minzoom": 14,
"source": {
"geoJsonLocal": "http://127.0.0.1:8080/speelplekken_{layer}_{z}_{x}_{y}.geojson",
@@ -95,7 +92,6 @@
{
"builtin": "grass_in_parks",
"override": {
- "icon": "./assets/themes/speelplekken/speelweide.svg",
"minzoom": 14,
"source": {
"geoJsonLocal": "http://127.0.0.1:8080/speelplekken_{layer}_{z}_{x}_{y}.geojson",
diff --git a/dependencies.svg b/dependencies.svg
deleted file mode 100644
index dfbcce1185..0000000000
--- a/dependencies.svg
+++ /dev/null
@@ -1,7406 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/index.html b/index.html
index 68e3f9fc4b..d7a2734e8c 100644
--- a/index.html
+++ b/index.html
@@ -77,7 +77,7 @@
Below
-
+
-Preferences editor - developers only
-Only use if you know what you're doing. To prevent newbies to make mistakes here, editing a mapcomplete-preference is
-only available if over 500 changes
-Editing any preference -including non-mapcomplete ones- is available when you have more then 2500 changesets. Until that
-point, only editing mapcomplete-preferences is possible.
-'maindiv' not attached
-
-
diff --git a/index.ts b/index.ts
index 8e84847cc3..024b40f795 100644
--- a/index.ts
+++ b/index.ts
@@ -1,5 +1,4 @@
import {FixedUiElement} from "./UI/Base/FixedUiElement";
-import {QueryParameters} from "./Logic/Web/QueryParameters";
import Combine from "./UI/Base/Combine";
import AvailableBaseLayers from "./Logic/Actors/AvailableBaseLayers";
import MinimapImplementation from "./UI/Base/MinimapImplementation";
@@ -18,19 +17,10 @@ MinimapImplementation.initialize()
AvailableBaseLayers.implement(new AvailableBaseLayersImplementation())
ShowOverlayLayerImplementation.Implement();
// Miscelleanous
-
Utils.DisableLongPresses()
-// --------------------- Special actions based on the parameters -----------------
-// @ts-ignore
-if (location.href.startsWith("http://buurtnatuur.be")) {
- // Reload the https version. This is important for the 'locate me' button
- window.location.replace("https://buurtnatuur.be");
-}
-
-
class Init {
- public static Init(layoutToUse: LayoutConfig, encoded: string) {
+ public static Init(layoutToUse: LayoutConfig) {
if (layoutToUse === null) {
// Something went wrong, error message is already on screen
@@ -43,40 +33,15 @@ class Init {
return;
}
- // Workaround/legacy to keep the old paramters working as I renamed some of them
- if (layoutToUse?.id === "cyclofix") {
- const legacy = QueryParameters.GetQueryParameter("layer-bike_shops", "true", "Legacy - keep De Fietsambassade working");
- const correct = QueryParameters.GetQueryParameter("layer-bike_shop", "true", "Legacy - keep De Fietsambassade working")
- if (legacy.data !== "true") {
- correct.setData(legacy.data)
- }
- console.log("layer-bike_shop toggles: legacy:", legacy.data, "new:", correct.data)
-
- const legacyCafe = QueryParameters.GetQueryParameter("layer-bike_cafes", "true", "Legacy - keep De Fietsambassade working")
- const correctCafe = QueryParameters.GetQueryParameter("layer-bike_cafe", "true", "Legacy - keep De Fietsambassade working")
- if (legacyCafe.data !== "true") {
- correctCafe.setData(legacy.data)
- }
- }
-
-
const guiState = new DefaultGuiState()
State.state = new State(layoutToUse);
DefaultGuiState.state = guiState;
// This 'leaks' the global state via the window object, useful for debugging
// @ts-ignore
window.mapcomplete_state = State.state;
-
new DefaultGUI(State.state, guiState)
- if (encoded !== undefined && encoded.length > 10) {
- // We save the layout to the user settings and local storage
- State.state.osmConnection.OnLoggedIn(() => {
- State.state.osmConnection
- .GetLongPreference("installed-theme-" + layoutToUse.id)
- .setData(encoded);
- });
- }
+
}
}
@@ -92,12 +57,12 @@ new Combine(["Initializing... ",
})])
.AttachTo("centermessage"); // Add an initialization and reset button if something goes wrong
-
+// @ts-ignore
DetermineLayout.GetLayout().then(value => {
console.log("Got ", value)
- Init.Init(value[0], value[1])
-}).catch(err => {
- console.error("Error while initializing: ", err, err.stack)
-})
+ Init.Init(value)
+ }).catch(err => {
+ console.error("Error while initializing: ", err, err.stack)
+ })
diff --git a/index_theme.ts.template b/index_theme.ts.template
new file mode 100644
index 0000000000..6d3ac310df
--- /dev/null
+++ b/index_theme.ts.template
@@ -0,0 +1,65 @@
+import {FixedUiElement} from "./UI/Base/FixedUiElement";
+import {QueryParameters} from "./Logic/Web/QueryParameters";
+import Combine from "./UI/Base/Combine";
+import AvailableBaseLayers from "./Logic/Actors/AvailableBaseLayers";
+import MinimapImplementation from "./UI/Base/MinimapImplementation";
+import {Utils} from "./Utils";
+import LayoutConfig from "./Models/ThemeConfig/LayoutConfig";
+import DefaultGUI from "./UI/DefaultGUI";
+import State from "./State";
+import AvailableBaseLayersImplementation from "./Logic/Actors/AvailableBaseLayersImplementation";
+import ShowOverlayLayerImplementation from "./UI/ShowDataLayer/ShowOverlayLayerImplementation";
+import {DefaultGuiState} from "./UI/DefaultGuiState";
+
+
+
+
+
+document.getElementById("decoration-desktop").remove();
+new Combine(["Initializing... ",
+ new FixedUiElement("If this message persist, something went wrong - click here to try again ")
+ .SetClass("link-underline small")
+ .onClick(() => {
+ localStorage.clear();
+ window.location.reload(true);
+
+ })])
+ .AttachTo("centermessage"); // Add an initialization and reset button if something goes wrong
+
+
+
+// Workaround for a stupid crash: inject some functions which would give stupid circular dependencies or crash the other nodejs scripts running from console
+MinimapImplementation.initialize()
+AvailableBaseLayers.implement(new AvailableBaseLayersImplementation())
+ShowOverlayLayerImplementation.Implement();
+// Miscelleanous
+Utils.DisableLongPresses()
+
+const layoutToUse = new LayoutConfig(themeConfig["default"])
+
+
+// Workaround/legacy to keep the old paramters working as I renamed some of them
+if (layoutToUse?.id === "cyclofix") {
+ const legacy = QueryParameters.GetQueryParameter("layer-bike_shops", "true", "Legacy - keep De Fietsambassade working");
+ const correct = QueryParameters.GetQueryParameter("layer-bike_shop", "true", "Legacy - keep De Fietsambassade working")
+ if (legacy.data !== "true") {
+ correct.setData(legacy.data)
+ }
+ console.log("layer-bike_shop toggles: legacy:", legacy.data, "new:", correct.data)
+
+ const legacyCafe = QueryParameters.GetQueryParameter("layer-bike_cafes", "true", "Legacy - keep De Fietsambassade working")
+ const correctCafe = QueryParameters.GetQueryParameter("layer-bike_cafe", "true", "Legacy - keep De Fietsambassade working")
+ if (legacyCafe.data !== "true") {
+ correctCafe.setData(legacy.data)
+ }
+}
+
+
+
+const guiState = new DefaultGuiState()
+State.state = new State(layoutToUse);
+DefaultGuiState.state = guiState;
+// This 'leaks' the global state via the window object, useful for debugging
+// @ts-ignore
+window.mapcomplete_state = State.state;
+new DefaultGUI(State.state, guiState)
diff --git a/notfound.ts b/notfound.ts
new file mode 100644
index 0000000000..12146ef4d9
--- /dev/null
+++ b/notfound.ts
@@ -0,0 +1,11 @@
+import {FixedUiElement} from "./UI/Base/FixedUiElement";
+import Combine from "./UI/Base/Combine";
+import {SubtleButton} from "./UI/Base/SubtleButton";
+import Svg from "./Svg";
+
+new Combine([new FixedUiElement("This page is not found"),
+new SubtleButton(Svg.back_svg(), "Back to index", {
+ url: "./index.html",
+ newTab: false
+})
+]).AttachTo("maindiv")
\ No newline at end of file
diff --git a/package-lock.json b/package-lock.json
index 5ec926bd95..e5c0c4ff7e 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -35,7 +35,7 @@
"leaflet-providers": "^1.13.0",
"leaflet-simple-map-screenshoter": "^0.4.4",
"leaflet.markercluster": "^1.4.1",
- "libphonenumber": "0.0.10",
+ "libphonenumber": "^0.0.9",
"libphonenumber-js": "^1.7.55",
"lz-string": "^1.4.4",
"mangrove-reviews": "^0.1.3",
@@ -43,7 +43,7 @@
"npm-run-all": "^4.1.5",
"opening_hours": "^3.6.0",
"osm-auth": "^1.0.2",
- "osmtogeojson": "^3.0.0-beta.4",
+ "osmtogeojson": "^1.0.0",
"parcel": "^1.2.4",
"prompt-sync": "^4.2.0",
"svg-resizer": "github:vieron/svg-resizer",
@@ -1275,47 +1275,6 @@
"resolved": "https://registry.npmjs.org/@iarna/toml/-/toml-2.2.5.tgz",
"integrity": "sha512-trnsAYxU3xnS1gPHPyU961coFyLkh4gAD/0zQ5mymY4yOZ+CYvsPqUbOFSw0aDM4y0tV7tiFxL/1XfXPNC6IPg=="
},
- "node_modules/@mapbox/geojson-area": {
- "version": "0.2.2",
- "resolved": "https://registry.npmjs.org/@mapbox/geojson-area/-/geojson-area-0.2.2.tgz",
- "integrity": "sha1-GNeBSqNr8j+7zDefjiaiKSfevxA=",
- "dependencies": {
- "wgs84": "0.0.0"
- }
- },
- "node_modules/@mapbox/geojson-rewind": {
- "version": "0.4.0",
- "resolved": "https://registry.npmjs.org/@mapbox/geojson-rewind/-/geojson-rewind-0.4.0.tgz",
- "integrity": "sha512-b+1uPWBERW4Pet/969BNu61ZPDyH2ilIxBjJDFzxyS9TyszF9UrTQyYIl/G38clux3rtpAGGFSGTCSF/qR6UjA==",
- "dependencies": {
- "@mapbox/geojson-area": "0.2.2",
- "concat-stream": "~1.6.0",
- "minimist": "1.2.0",
- "sharkdown": "^0.1.0"
- },
- "bin": {
- "geojson-rewind": "geojson-rewind"
- }
- },
- "node_modules/@mapbox/geojson-rewind/node_modules/concat-stream": {
- "version": "1.6.2",
- "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz",
- "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==",
- "engines": [
- "node >= 0.8"
- ],
- "dependencies": {
- "buffer-from": "^1.0.0",
- "inherits": "^2.0.3",
- "readable-stream": "^2.2.2",
- "typedarray": "^0.0.6"
- }
- },
- "node_modules/@mapbox/geojson-rewind/node_modules/minimist": {
- "version": "1.2.0",
- "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
- "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ="
- },
"node_modules/@mrmlnc/readdir-enhanced": {
"version": "2.2.1",
"resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz",
@@ -1511,24 +1470,6 @@
"node": ">=0.10.0"
}
},
- "node_modules/@parcel/watcher/node_modules/fsevents": {
- "version": "1.2.13",
- "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz",
- "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==",
- "deprecated": "fsevents 1 will break on node v14+ and could be using insecure binaries. Upgrade to fsevents 2.",
- "hasInstallScript": true,
- "optional": true,
- "os": [
- "darwin"
- ],
- "dependencies": {
- "bindings": "^1.5.0",
- "nan": "^2.12.1"
- },
- "engines": {
- "node": ">= 4.0"
- }
- },
"node_modules/@parcel/watcher/node_modules/glob-parent": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz",
@@ -3505,11 +3446,6 @@
"node": "*"
}
},
- "node_modules/ansicolors": {
- "version": "0.2.1",
- "resolved": "https://registry.npmjs.org/ansicolors/-/ansicolors-0.2.1.tgz",
- "integrity": "sha1-vgiVmQl7dKXJxKhKDNvNtivYeu8="
- },
"node_modules/anymatch": {
"version": "3.1.2",
"resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz",
@@ -4411,18 +4347,6 @@
"node": ">=8.0.0"
}
},
- "node_modules/cardinal": {
- "version": "0.4.4",
- "resolved": "https://registry.npmjs.org/cardinal/-/cardinal-0.4.4.tgz",
- "integrity": "sha1-ylu2iltRG5D+k7ms6km97lwyv+I=",
- "dependencies": {
- "ansicolors": "~0.2.1",
- "redeyed": "~0.4.0"
- },
- "bin": {
- "cdl": "bin/cdl.js"
- }
- },
"node_modules/caseless": {
"version": "0.12.0",
"resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
@@ -4714,30 +4638,14 @@
"integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s="
},
"node_modules/concat-stream": {
- "version": "2.0.0",
- "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-2.0.0.tgz",
- "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==",
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.0.1.tgz",
+ "integrity": "sha1-AYsYvBx9BzotyCqkhEI0GixN158=",
"engines": [
- "node >= 6.0"
+ "node >= 0.8.0"
],
"dependencies": {
- "buffer-from": "^1.0.0",
- "inherits": "^2.0.3",
- "readable-stream": "^3.0.2",
- "typedarray": "^0.0.6"
- }
- },
- "node_modules/concat-stream/node_modules/readable-stream": {
- "version": "3.6.0",
- "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz",
- "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==",
- "dependencies": {
- "inherits": "^2.0.3",
- "string_decoder": "^1.1.1",
- "util-deprecate": "^1.0.1"
- },
- "engines": {
- "node": ">= 6"
+ "bops": "0.0.6"
}
},
"node_modules/concaveman": {
@@ -4815,7 +4723,7 @@
"version": "2.6.12",
"resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.12.tgz",
"integrity": "sha512-Kb2wC0fvsWfQrgk8HU5lW6U/Lcs8+9aaYcy4ZFc6DDlo4nZ7n70dEgE5rtR0oG6ufKDUnrwfWL1mXR5ljDatrQ==",
- "deprecated": "core-js@<3.3 is no longer maintained and not recommended for usage due to the number of issues. Because of the V8 engine whims, feature detection in old core-js versions could cause a slowdown up to 100x even if nothing is polyfilled. Please, upgrade your dependencies to the actual version of core-js.",
+ "deprecated": "core-js@<3.4 is no longer maintained and not recommended for usage due to the number of issues. Because of the V8 engine whims, feature detection in old core-js versions could cause a slowdown up to 100x even if nothing is polyfilled. Please, upgrade your dependencies to the actual version of core-js.",
"hasInstallScript": true
},
"node_modules/core-js-compat": {
@@ -6072,9 +5980,9 @@
}
},
"node_modules/domhandler": {
- "version": "2.2.1",
- "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.2.1.tgz",
- "integrity": "sha1-Wd+dzSJ+gIs2Wuc+H2aErD2Ub8I=",
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.0.3.tgz",
+ "integrity": "sha1-iJ+N9iZAOvB4jinWbV1cb36/D9Y=",
"dependencies": {
"domelementtype": "1"
}
@@ -6086,9 +5994,9 @@
"optional": true
},
"node_modules/domutils": {
- "version": "1.3.0",
- "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.3.0.tgz",
- "integrity": "sha1-mtTVm1r2ymhMYv5tdo7xcOcN8ZI=",
+ "version": "1.1.6",
+ "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.1.6.tgz",
+ "integrity": "sha1-vdw94Jm5ou+sxRxiPyj0FuzFdIU=",
"dependencies": {
"domelementtype": "1"
}
@@ -6375,18 +6283,6 @@
"node": ">=0.10.0"
}
},
- "node_modules/esprima": {
- "version": "1.0.4",
- "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.0.4.tgz",
- "integrity": "sha1-n1V+CPw7TSbs6d00+Pv0drYlha0=",
- "bin": {
- "esparse": "bin/esparse.js",
- "esvalidate": "bin/esvalidate.js"
- },
- "engines": {
- "node": ">=0.4.0"
- }
- },
"node_modules/estraverse": {
"version": "4.3.0",
"resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz",
@@ -6887,19 +6783,6 @@
"resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
"integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8="
},
- "node_modules/fsevents": {
- "version": "2.3.2",
- "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
- "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
- "hasInstallScript": true,
- "optional": true,
- "os": [
- "darwin"
- ],
- "engines": {
- "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
- }
- },
"node_modules/function-bind": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
@@ -6973,18 +6856,6 @@
"integrity": "sha1-Lbw2eM0bMbgXnodr2nDNEg3eNcA=",
"deprecated": "This module is now under the @mapbox namespace: install @mapbox/geojson-normalize instead"
},
- "node_modules/geojson-numeric": {
- "version": "0.2.1",
- "resolved": "https://registry.npmjs.org/geojson-numeric/-/geojson-numeric-0.2.1.tgz",
- "integrity": "sha512-rvItMp3W7pe16o2EQTnRw54v6WHdiE4bYjUsdr3FZskFb6oPC7gjLe4zginP+Wd1B/HLl2acTukfn16Lmwn7lg==",
- "dependencies": {
- "concat-stream": "2.0.0",
- "optimist": "~0.3.5"
- },
- "bin": {
- "geojson-numeric": "geojson-numeric"
- }
- },
"node_modules/geojson-random": {
"version": "0.2.2",
"resolved": "https://registry.npmjs.org/geojson-random/-/geojson-random-0.2.2.tgz",
@@ -7164,11 +7035,6 @@
"unicode-trie": "^0.3.1"
}
},
- "node_modules/growl": {
- "version": "1.9.2",
- "resolved": "https://registry.npmjs.org/growl/-/growl-1.9.2.tgz",
- "integrity": "sha1-Dqd0NxXbjY3ixe3hd14bRayFwC8="
- },
"node_modules/handlebars": {
"version": "4.7.7",
"resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz",
@@ -7586,14 +7452,14 @@
}
},
"node_modules/htmlparser2": {
- "version": "3.5.1",
- "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.5.1.tgz",
- "integrity": "sha1-b0L3ZX3RnBP31l3pEYQXOUoL5tA=",
+ "version": "3.0.5",
+ "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.0.5.tgz",
+ "integrity": "sha1-fzBgjLeAnzX/fG3YDinCR6p9r18=",
"dependencies": {
"domelementtype": "1",
- "domhandler": "2.2",
- "domutils": "1.3",
- "readable-stream": "1.1"
+ "domhandler": "2.0",
+ "domutils": "1.1",
+ "readable-stream": "1.0"
}
},
"node_modules/htmlparser2/node_modules/isarray": {
@@ -7602,9 +7468,9 @@
"integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8="
},
"node_modules/htmlparser2/node_modules/readable-stream": {
- "version": "1.1.14",
- "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz",
- "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=",
+ "version": "1.0.34",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz",
+ "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=",
"dependencies": {
"core-util-is": "~1.0.0",
"inherits": "~2.0.1",
@@ -8528,36 +8394,6 @@
"resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
"integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo="
},
- "node_modules/jade": {
- "version": "0.26.3",
- "resolved": "https://registry.npmjs.org/jade/-/jade-0.26.3.tgz",
- "integrity": "sha1-jxDXl32NefL2/4YqgbBRPMslaGw=",
- "deprecated": "Jade has been renamed to pug, please install the latest version of pug instead of jade",
- "dependencies": {
- "commander": "0.6.1",
- "mkdirp": "0.3.0"
- },
- "bin": {
- "jade": "bin/jade"
- }
- },
- "node_modules/jade/node_modules/commander": {
- "version": "0.6.1",
- "resolved": "https://registry.npmjs.org/commander/-/commander-0.6.1.tgz",
- "integrity": "sha1-+mihT2qUXVTbvlDYzbMyDp47GgY=",
- "engines": {
- "node": ">= 0.4.x"
- }
- },
- "node_modules/jade/node_modules/mkdirp": {
- "version": "0.3.0",
- "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.3.0.tgz",
- "integrity": "sha1-G79asbqCevI1dRQ0kEJkVfSB/h4=",
- "deprecated": "Legacy versions of mkdirp are no longer supported. Please update to mkdirp 1.x. (Note that the API surface has changed to use Promises in 1.x.)",
- "engines": {
- "node": "*"
- }
- },
"node_modules/jquery": {
"version": "3.6.0",
"resolved": "https://registry.npmjs.org/jquery/-/jquery-3.6.0.tgz",
@@ -8781,31 +8617,6 @@
"node": "*"
}
},
- "node_modules/jsonparse": {
- "version": "0.0.5",
- "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-0.0.5.tgz",
- "integrity": "sha1-MwVCrT8KZUZlt3jz6y2an6UHrGQ=",
- "engines": [
- "node >= 0.2.0"
- ]
- },
- "node_modules/JSONStream": {
- "version": "0.8.0",
- "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-0.8.0.tgz",
- "integrity": "sha1-78Ri1aW8lOwAf0siVxrNf28q4BM=",
- "dependencies": {
- "jsonparse": "0.0.5",
- "through": "~2.2.7"
- },
- "engines": {
- "node": "*"
- }
- },
- "node_modules/JSONStream/node_modules/through": {
- "version": "2.2.7",
- "resolved": "https://registry.npmjs.org/through/-/through-2.2.7.tgz",
- "integrity": "sha1-bo4hIAGR1OtqmfbwEN9Gqhxusr0="
- },
"node_modules/jsonwebtoken": {
"version": "8.5.1",
"resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
@@ -8984,12 +8795,11 @@
}
},
"node_modules/libphonenumber": {
- "version": "0.0.10",
- "resolved": "https://registry.npmjs.org/libphonenumber/-/libphonenumber-0.0.10.tgz",
- "integrity": "sha1-54u/ZgGYnNCfpDUEdkaI1BeUFIw=",
+ "version": "0.0.9",
+ "resolved": "https://registry.npmjs.org/libphonenumber/-/libphonenumber-0.0.9.tgz",
+ "integrity": "sha1-YuCNsuLJszYL2ISS8uhwTvbErII=",
"dependencies": {
- "closure": "1.0.3",
- "mocha": "^2.4.5"
+ "closure": "1.0.3"
},
"engines": {
"node": ">= 0.8"
@@ -9135,11 +8945,6 @@
"node": ">=0.10.0"
}
},
- "node_modules/lru-cache": {
- "version": "2.7.3",
- "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-2.7.3.tgz",
- "integrity": "sha1-bUUk6LlV+V1PW1iFHOId1y+06VI="
- },
"node_modules/lz-string": {
"version": "1.4.4",
"resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.4.4.tgz",
@@ -9418,103 +9223,6 @@
"resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
"integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0="
},
- "node_modules/mocha": {
- "version": "2.5.3",
- "resolved": "https://registry.npmjs.org/mocha/-/mocha-2.5.3.tgz",
- "integrity": "sha1-FhvlvetJZ3HrmzV0UFC2IrWu/Fg=",
- "dependencies": {
- "commander": "2.3.0",
- "debug": "2.2.0",
- "diff": "1.4.0",
- "escape-string-regexp": "1.0.2",
- "glob": "3.2.11",
- "growl": "1.9.2",
- "jade": "0.26.3",
- "mkdirp": "0.5.1",
- "supports-color": "1.2.0",
- "to-iso-string": "0.0.2"
- },
- "bin": {
- "_mocha": "bin/_mocha",
- "mocha": "bin/mocha"
- },
- "engines": {
- "node": ">= 0.8.x"
- }
- },
- "node_modules/mocha/node_modules/commander": {
- "version": "2.3.0",
- "resolved": "https://registry.npmjs.org/commander/-/commander-2.3.0.tgz",
- "integrity": "sha1-/UMOiJgy7DU7ms0d4hfBHLPu+HM=",
- "engines": {
- "node": ">= 0.6.x"
- }
- },
- "node_modules/mocha/node_modules/debug": {
- "version": "2.2.0",
- "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz",
- "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=",
- "dependencies": {
- "ms": "0.7.1"
- }
- },
- "node_modules/mocha/node_modules/diff": {
- "version": "1.4.0",
- "resolved": "https://registry.npmjs.org/diff/-/diff-1.4.0.tgz",
- "integrity": "sha1-fyjS657nsVqX79ic5j3P2qPMur8=",
- "engines": {
- "node": ">=0.3.1"
- }
- },
- "node_modules/mocha/node_modules/escape-string-regexp": {
- "version": "1.0.2",
- "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.2.tgz",
- "integrity": "sha1-Tbwv5nTnGUnK8/smlc5/LcHZqNE=",
- "engines": {
- "node": ">=0.8.0"
- }
- },
- "node_modules/mocha/node_modules/glob": {
- "version": "3.2.11",
- "resolved": "https://registry.npmjs.org/glob/-/glob-3.2.11.tgz",
- "integrity": "sha1-Spc/Y1uRkPcV0QmH1cAP0oFevj0=",
- "dependencies": {
- "inherits": "2",
- "minimatch": "0.3"
- },
- "engines": {
- "node": "*"
- }
- },
- "node_modules/mocha/node_modules/minimatch": {
- "version": "0.3.0",
- "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-0.3.0.tgz",
- "integrity": "sha1-J12O2qxPG7MyZHIInnlJyDlGmd0=",
- "deprecated": "Please update to minimatch 3.0.2 or higher to avoid a RegExp DoS issue",
- "dependencies": {
- "lru-cache": "2",
- "sigmund": "~1.0.0"
- },
- "engines": {
- "node": "*"
- }
- },
- "node_modules/mocha/node_modules/ms": {
- "version": "0.7.1",
- "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz",
- "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg="
- },
- "node_modules/mocha/node_modules/supports-color": {
- "version": "1.2.0",
- "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-1.2.0.tgz",
- "integrity": "sha1-/x7R5hFp0Gs88tWI4YixjYhH4X4=",
- "bin": {
- "supports-color": "cli.js"
- },
- "engines": {
- "node": ">=0.10.0"
- }
- },
"node_modules/modern-normalize": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/modern-normalize/-/modern-normalize-1.1.0.tgz",
@@ -9553,12 +9261,6 @@
"integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==",
"dev": true
},
- "node_modules/nan": {
- "version": "2.14.2",
- "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz",
- "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==",
- "optional": true
- },
"node_modules/nanocolors": {
"version": "0.1.6",
"resolved": "https://registry.npmjs.org/nanocolors/-/nanocolors-0.1.6.tgz",
@@ -10186,24 +9888,15 @@
"node": ">=8.10.0"
}
},
- "node_modules/osm-polygon-features": {
- "version": "0.9.2",
- "resolved": "https://registry.npmjs.org/osm-polygon-features/-/osm-polygon-features-0.9.2.tgz",
- "integrity": "sha1-IK5BEwxIbkmjsqPCtYoUGcSYZ3g="
- },
"node_modules/osmtogeojson": {
- "version": "3.0.0-beta.4",
- "resolved": "https://registry.npmjs.org/osmtogeojson/-/osmtogeojson-3.0.0-beta.4.tgz",
- "integrity": "sha512-GwNy2w5JKOplOBspagcNhCDhBRV6Du2BCvcLkaA7nX12U86Dl2Ciw9zs/VzFFTXfyZlaK+7bGCWN2SNlfn/jOA==",
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/osmtogeojson/-/osmtogeojson-1.4.0.tgz",
+ "integrity": "sha1-7cFrFDswPAi4PH5JoFgZvaG6cis=",
"dependencies": {
- "@mapbox/geojson-rewind": "0.4.0",
- "concat-stream": "2.0.0",
- "geojson-numeric": "0.2.1",
- "htmlparser2": "3.5.1",
- "JSONStream": "0.8.0",
+ "concat-stream": "~1.0.1",
+ "htmlparser2": "~3.0.5",
+ "lodash": "~2.2.0",
"optimist": "~0.3.5",
- "osm-polygon-features": "^0.9.1",
- "tiny-osmpbf": "^0.1.0",
"xmldom": "~0.1.16"
},
"bin": {
@@ -10211,16 +9904,16 @@
},
"engines": {
"node": ">=0.5"
- },
- "optionalDependencies": {
- "@types/geojson": "^1.0.2"
}
},
- "node_modules/osmtogeojson/node_modules/@types/geojson": {
- "version": "1.0.6",
- "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-1.0.6.tgz",
- "integrity": "sha512-Xqg/lIZMrUd0VRmSRbCAewtwGZiAk3mEUDvV4op1tGl+LvyPcb/MIOSxTl9z+9+J+R4/vpjiCAT4xeKzH9ji1w==",
- "optional": true
+ "node_modules/osmtogeojson/node_modules/lodash": {
+ "version": "2.2.1",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-2.2.1.tgz",
+ "integrity": "sha1-ypNf0UqzwMhyq6zxmLnNpQFECGc=",
+ "engines": [
+ "node",
+ "rhino"
+ ]
},
"node_modules/pako": {
"version": "0.2.9",
@@ -10632,18 +10325,6 @@
"node": ">=0.10.0"
}
},
- "node_modules/pbf": {
- "version": "3.2.1",
- "resolved": "https://registry.npmjs.org/pbf/-/pbf-3.2.1.tgz",
- "integrity": "sha512-ClrV7pNOn7rtmoQVF4TS1vyU0WhYRnP92fzbfF75jAIwpnzdJXf8iTd4CMEqO4yUenH6NDqLiwjqlh6QgZzgLQ==",
- "dependencies": {
- "ieee754": "^1.1.12",
- "resolve-protobuf-schema": "^2.1.0"
- },
- "bin": {
- "pbf": "bin/pbf"
- }
- },
"node_modules/pbkdf2": {
"version": "3.1.2",
"resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz",
@@ -12488,11 +12169,6 @@
"node": ">=6"
}
},
- "node_modules/protocol-buffers-schema": {
- "version": "3.5.1",
- "resolved": "https://registry.npmjs.org/protocol-buffers-schema/-/protocol-buffers-schema-3.5.1.tgz",
- "integrity": "sha512-YVCvdhxWNDP8/nJDyXLuM+UFsuPk4+1PB7WGPVDzm3HTHbzFLxQYeW2iZpS4mmnXrQJGBzt230t/BbEb7PrQaw=="
- },
"node_modules/psl": {
"version": "1.8.0",
"resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz",
@@ -12805,14 +12481,6 @@
"node": ">=0.10.0"
}
},
- "node_modules/redeyed": {
- "version": "0.4.4",
- "resolved": "https://registry.npmjs.org/redeyed/-/redeyed-0.4.4.tgz",
- "integrity": "sha1-N+mQpvKyGyoRwuakj9QTVpjLqX8=",
- "dependencies": {
- "esprima": "~1.0.4"
- }
- },
"node_modules/reduce-css-calc": {
"version": "2.1.8",
"resolved": "https://registry.npmjs.org/reduce-css-calc/-/reduce-css-calc-2.1.8.tgz",
@@ -13073,14 +12741,6 @@
"node": ">=4"
}
},
- "node_modules/resolve-protobuf-schema": {
- "version": "2.1.0",
- "resolved": "https://registry.npmjs.org/resolve-protobuf-schema/-/resolve-protobuf-schema-2.1.0.tgz",
- "integrity": "sha512-kI5ffTiZWmJaS/huM8wZfEMer1eRd7oJQhDuxeCLe3t7N7mX3z94CN0xPxBQxFYQTSNz9T0i+v6inKqSdK8xrQ==",
- "dependencies": {
- "protocol-buffers-schema": "^3.3.1"
- }
- },
"node_modules/resolve-url": {
"version": "0.2.1",
"resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz",
@@ -13457,24 +13117,6 @@
"resolved": "https://registry.npmjs.org/shallow-copy/-/shallow-copy-0.0.1.tgz",
"integrity": "sha1-QV9CcC1z2BAzApLMXuhurhoRoXA="
},
- "node_modules/sharkdown": {
- "version": "0.1.1",
- "resolved": "https://registry.npmjs.org/sharkdown/-/sharkdown-0.1.1.tgz",
- "integrity": "sha512-exwooSpmo5s45lrexgz6Q0rFQM574wYIX3iDZ7RLLqOb7IAoQZu9nxlZODU972g19sR69OIpKP2cpHTzU+PHIg==",
- "dependencies": {
- "cardinal": "~0.4.2",
- "minimist": "0.0.5",
- "split": "~0.2.10"
- },
- "bin": {
- "sharkdown": "sharkdown"
- }
- },
- "node_modules/sharkdown/node_modules/minimist": {
- "version": "0.0.5",
- "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.5.tgz",
- "integrity": "sha1-16oye87PUY+RBqxrjwA/o7zqhWY="
- },
"node_modules/sharp": {
"version": "0.28.3",
"resolved": "https://registry.npmjs.org/sharp/-/sharp-0.28.3.tgz",
@@ -13566,11 +13208,6 @@
"node": ">=0.8.0"
}
},
- "node_modules/sigmund": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/sigmund/-/sigmund-1.0.1.tgz",
- "integrity": "sha1-P/IfGYytIXX587eBhT/ZTQ0ZtZA="
- },
"node_modules/signal-exit": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
@@ -13851,17 +13488,6 @@
"resolved": "https://registry.npmjs.org/splaytree/-/splaytree-3.1.0.tgz",
"integrity": "sha512-gvUGR7xnOy0fLKTCxDeUZYgU/I1Tdf8M/lM1Qrf8L2TIOR5ipZjGk02uYcdv0o2x7WjVRgpm3iS2clLyuVAt0Q=="
},
- "node_modules/split": {
- "version": "0.2.10",
- "resolved": "https://registry.npmjs.org/split/-/split-0.2.10.tgz",
- "integrity": "sha1-Zwl8YB1pfOE2j0GPBs0gHPBSGlc=",
- "dependencies": {
- "through": "2"
- },
- "engines": {
- "node": "*"
- }
- },
"node_modules/split-string": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz",
@@ -14330,6 +13956,7 @@
"version": "1.3.2",
"resolved": "https://registry.npmjs.org/svgo/-/svgo-1.3.2.tgz",
"integrity": "sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw==",
+ "deprecated": "This SVGO version is no longer supported. Upgrade to v2.x.x.",
"dependencies": {
"chalk": "^2.4.1",
"coa": "^2.0.2",
@@ -14734,7 +14361,8 @@
"node_modules/through": {
"version": "2.3.8",
"resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
- "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU="
+ "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
+ "dev": true
},
"node_modules/through2": {
"version": "2.0.5",
@@ -14766,15 +14394,6 @@
"resolved": "https://registry.npmjs.org/tiny-inflate/-/tiny-inflate-1.0.3.tgz",
"integrity": "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw=="
},
- "node_modules/tiny-osmpbf": {
- "version": "0.1.0",
- "resolved": "https://registry.npmjs.org/tiny-osmpbf/-/tiny-osmpbf-0.1.0.tgz",
- "integrity": "sha1-ColXFxE+vmquNjxL5e76js2vuSc=",
- "dependencies": {
- "pbf": "^3.0.4",
- "tiny-inflate": "^1.0.2"
- }
- },
"node_modules/tinyqueue": {
"version": "2.0.3",
"resolved": "https://registry.npmjs.org/tinyqueue/-/tinyqueue-2.0.3.tgz",
@@ -14818,12 +14437,6 @@
"node": ">=4"
}
},
- "node_modules/to-iso-string": {
- "version": "0.0.2",
- "resolved": "https://registry.npmjs.org/to-iso-string/-/to-iso-string-0.0.2.tgz",
- "integrity": "sha1-TcGeZk38y+Jb2NtQiwDG2hWCVdE=",
- "deprecated": "to-iso-string has been deprecated, use @segment/to-iso-string instead."
- },
"node_modules/to-object-path": {
"version": "0.3.0",
"resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz",
@@ -14890,17 +14503,6 @@
"togpx": "togpx"
}
},
- "node_modules/togpx/node_modules/concat-stream": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.0.1.tgz",
- "integrity": "sha1-AYsYvBx9BzotyCqkhEI0GixN158=",
- "engines": [
- "node >= 0.8.0"
- ],
- "dependencies": {
- "bops": "0.0.6"
- }
- },
"node_modules/toidentifier": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
@@ -17786,43 +17388,6 @@
"resolved": "https://registry.npmjs.org/@iarna/toml/-/toml-2.2.5.tgz",
"integrity": "sha512-trnsAYxU3xnS1gPHPyU961coFyLkh4gAD/0zQ5mymY4yOZ+CYvsPqUbOFSw0aDM4y0tV7tiFxL/1XfXPNC6IPg=="
},
- "@mapbox/geojson-area": {
- "version": "0.2.2",
- "resolved": "https://registry.npmjs.org/@mapbox/geojson-area/-/geojson-area-0.2.2.tgz",
- "integrity": "sha1-GNeBSqNr8j+7zDefjiaiKSfevxA=",
- "requires": {
- "wgs84": "0.0.0"
- }
- },
- "@mapbox/geojson-rewind": {
- "version": "0.4.0",
- "resolved": "https://registry.npmjs.org/@mapbox/geojson-rewind/-/geojson-rewind-0.4.0.tgz",
- "integrity": "sha512-b+1uPWBERW4Pet/969BNu61ZPDyH2ilIxBjJDFzxyS9TyszF9UrTQyYIl/G38clux3rtpAGGFSGTCSF/qR6UjA==",
- "requires": {
- "@mapbox/geojson-area": "0.2.2",
- "concat-stream": "~1.6.0",
- "minimist": "1.2.0",
- "sharkdown": "^0.1.0"
- },
- "dependencies": {
- "concat-stream": {
- "version": "1.6.2",
- "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz",
- "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==",
- "requires": {
- "buffer-from": "^1.0.0",
- "inherits": "^2.0.3",
- "readable-stream": "^2.2.2",
- "typedarray": "^0.0.6"
- }
- },
- "minimist": {
- "version": "1.2.0",
- "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
- "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ="
- }
- }
- },
"@mrmlnc/readdir-enhanced": {
"version": "2.2.1",
"resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz",
@@ -17982,16 +17547,6 @@
}
}
},
- "fsevents": {
- "version": "1.2.13",
- "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz",
- "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==",
- "optional": true,
- "requires": {
- "bindings": "^1.5.0",
- "nan": "^2.12.1"
- }
- },
"glob-parent": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz",
@@ -19580,11 +19135,6 @@
"entities": "^1.1.2"
}
},
- "ansicolors": {
- "version": "0.2.1",
- "resolved": "https://registry.npmjs.org/ansicolors/-/ansicolors-0.2.1.tgz",
- "integrity": "sha1-vgiVmQl7dKXJxKhKDNvNtivYeu8="
- },
"anymatch": {
"version": "3.1.2",
"resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz",
@@ -20316,15 +19866,6 @@
"svg-pathdata": "^5.0.5"
}
},
- "cardinal": {
- "version": "0.4.4",
- "resolved": "https://registry.npmjs.org/cardinal/-/cardinal-0.4.4.tgz",
- "integrity": "sha1-ylu2iltRG5D+k7ms6km97lwyv+I=",
- "requires": {
- "ansicolors": "~0.2.1",
- "redeyed": "~0.4.0"
- }
- },
"caseless": {
"version": "0.12.0",
"resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
@@ -20561,26 +20102,11 @@
"integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s="
},
"concat-stream": {
- "version": "2.0.0",
- "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-2.0.0.tgz",
- "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==",
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.0.1.tgz",
+ "integrity": "sha1-AYsYvBx9BzotyCqkhEI0GixN158=",
"requires": {
- "buffer-from": "^1.0.0",
- "inherits": "^2.0.3",
- "readable-stream": "^3.0.2",
- "typedarray": "^0.0.6"
- },
- "dependencies": {
- "readable-stream": {
- "version": "3.6.0",
- "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz",
- "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==",
- "requires": {
- "inherits": "^2.0.3",
- "string_decoder": "^1.1.1",
- "util-deprecate": "^1.0.1"
- }
- }
+ "bops": "0.0.6"
}
},
"concaveman": {
@@ -21630,9 +21156,9 @@
}
},
"domhandler": {
- "version": "2.2.1",
- "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.2.1.tgz",
- "integrity": "sha1-Wd+dzSJ+gIs2Wuc+H2aErD2Ub8I=",
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.0.3.tgz",
+ "integrity": "sha1-iJ+N9iZAOvB4jinWbV1cb36/D9Y=",
"requires": {
"domelementtype": "1"
}
@@ -21644,9 +21170,9 @@
"optional": true
},
"domutils": {
- "version": "1.3.0",
- "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.3.0.tgz",
- "integrity": "sha1-mtTVm1r2ymhMYv5tdo7xcOcN8ZI=",
+ "version": "1.1.6",
+ "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.1.6.tgz",
+ "integrity": "sha1-vdw94Jm5ou+sxRxiPyj0FuzFdIU=",
"requires": {
"domelementtype": "1"
}
@@ -21878,11 +21404,6 @@
}
}
},
- "esprima": {
- "version": "1.0.4",
- "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.0.4.tgz",
- "integrity": "sha1-n1V+CPw7TSbs6d00+Pv0drYlha0="
- },
"estraverse": {
"version": "4.3.0",
"resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz",
@@ -22290,12 +21811,6 @@
"resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
"integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8="
},
- "fsevents": {
- "version": "2.3.2",
- "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
- "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
- "optional": true
- },
"function-bind": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
@@ -22360,15 +21875,6 @@
"resolved": "https://registry.npmjs.org/geojson-normalize/-/geojson-normalize-0.0.0.tgz",
"integrity": "sha1-Lbw2eM0bMbgXnodr2nDNEg3eNcA="
},
- "geojson-numeric": {
- "version": "0.2.1",
- "resolved": "https://registry.npmjs.org/geojson-numeric/-/geojson-numeric-0.2.1.tgz",
- "integrity": "sha512-rvItMp3W7pe16o2EQTnRw54v6WHdiE4bYjUsdr3FZskFb6oPC7gjLe4zginP+Wd1B/HLl2acTukfn16Lmwn7lg==",
- "requires": {
- "concat-stream": "2.0.0",
- "optimist": "~0.3.5"
- }
- },
"geojson-random": {
"version": "0.2.2",
"resolved": "https://registry.npmjs.org/geojson-random/-/geojson-random-0.2.2.tgz",
@@ -22508,11 +22014,6 @@
"unicode-trie": "^0.3.1"
}
},
- "growl": {
- "version": "1.9.2",
- "resolved": "https://registry.npmjs.org/growl/-/growl-1.9.2.tgz",
- "integrity": "sha1-Dqd0NxXbjY3ixe3hd14bRayFwC8="
- },
"handlebars": {
"version": "4.7.7",
"resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz",
@@ -22835,14 +22336,14 @@
}
},
"htmlparser2": {
- "version": "3.5.1",
- "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.5.1.tgz",
- "integrity": "sha1-b0L3ZX3RnBP31l3pEYQXOUoL5tA=",
+ "version": "3.0.5",
+ "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.0.5.tgz",
+ "integrity": "sha1-fzBgjLeAnzX/fG3YDinCR6p9r18=",
"requires": {
"domelementtype": "1",
- "domhandler": "2.2",
- "domutils": "1.3",
- "readable-stream": "1.1"
+ "domhandler": "2.0",
+ "domutils": "1.1",
+ "readable-stream": "1.0"
},
"dependencies": {
"isarray": {
@@ -22851,9 +22352,9 @@
"integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8="
},
"readable-stream": {
- "version": "1.1.14",
- "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz",
- "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=",
+ "version": "1.0.34",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz",
+ "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=",
"requires": {
"core-util-is": "~1.0.0",
"inherits": "~2.0.1",
@@ -23522,27 +23023,6 @@
"resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
"integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo="
},
- "jade": {
- "version": "0.26.3",
- "resolved": "https://registry.npmjs.org/jade/-/jade-0.26.3.tgz",
- "integrity": "sha1-jxDXl32NefL2/4YqgbBRPMslaGw=",
- "requires": {
- "commander": "0.6.1",
- "mkdirp": "0.3.0"
- },
- "dependencies": {
- "commander": {
- "version": "0.6.1",
- "resolved": "https://registry.npmjs.org/commander/-/commander-0.6.1.tgz",
- "integrity": "sha1-+mihT2qUXVTbvlDYzbMyDp47GgY="
- },
- "mkdirp": {
- "version": "0.3.0",
- "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.3.0.tgz",
- "integrity": "sha1-G79asbqCevI1dRQ0kEJkVfSB/h4="
- }
- }
- },
"jquery": {
"version": "3.6.0",
"resolved": "https://registry.npmjs.org/jquery/-/jquery-3.6.0.tgz",
@@ -23712,27 +23192,6 @@
"integrity": "sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM=",
"dev": true
},
- "jsonparse": {
- "version": "0.0.5",
- "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-0.0.5.tgz",
- "integrity": "sha1-MwVCrT8KZUZlt3jz6y2an6UHrGQ="
- },
- "JSONStream": {
- "version": "0.8.0",
- "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-0.8.0.tgz",
- "integrity": "sha1-78Ri1aW8lOwAf0siVxrNf28q4BM=",
- "requires": {
- "jsonparse": "0.0.5",
- "through": "~2.2.7"
- },
- "dependencies": {
- "through": {
- "version": "2.2.7",
- "resolved": "https://registry.npmjs.org/through/-/through-2.2.7.tgz",
- "integrity": "sha1-bo4hIAGR1OtqmfbwEN9Gqhxusr0="
- }
- }
- },
"jsonwebtoken": {
"version": "8.5.1",
"resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
@@ -23887,12 +23346,11 @@
}
},
"libphonenumber": {
- "version": "0.0.10",
- "resolved": "https://registry.npmjs.org/libphonenumber/-/libphonenumber-0.0.10.tgz",
- "integrity": "sha1-54u/ZgGYnNCfpDUEdkaI1BeUFIw=",
+ "version": "0.0.9",
+ "resolved": "https://registry.npmjs.org/libphonenumber/-/libphonenumber-0.0.9.tgz",
+ "integrity": "sha1-YuCNsuLJszYL2ISS8uhwTvbErII=",
"requires": {
- "closure": "1.0.3",
- "mocha": "^2.4.5"
+ "closure": "1.0.3"
}
},
"libphonenumber-js": {
@@ -24022,11 +23480,6 @@
"signal-exit": "^3.0.0"
}
},
- "lru-cache": {
- "version": "2.7.3",
- "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-2.7.3.tgz",
- "integrity": "sha1-bUUk6LlV+V1PW1iFHOId1y+06VI="
- },
"lz-string": {
"version": "1.4.4",
"resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.4.4.tgz",
@@ -24242,76 +23695,6 @@
"integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==",
"dev": true
},
- "mocha": {
- "version": "2.5.3",
- "resolved": "https://registry.npmjs.org/mocha/-/mocha-2.5.3.tgz",
- "integrity": "sha1-FhvlvetJZ3HrmzV0UFC2IrWu/Fg=",
- "requires": {
- "commander": "2.3.0",
- "debug": "2.2.0",
- "diff": "1.4.0",
- "escape-string-regexp": "1.0.2",
- "glob": "3.2.11",
- "growl": "1.9.2",
- "jade": "0.26.3",
- "mkdirp": "0.5.1",
- "supports-color": "1.2.0",
- "to-iso-string": "0.0.2"
- },
- "dependencies": {
- "commander": {
- "version": "2.3.0",
- "resolved": "https://registry.npmjs.org/commander/-/commander-2.3.0.tgz",
- "integrity": "sha1-/UMOiJgy7DU7ms0d4hfBHLPu+HM="
- },
- "debug": {
- "version": "2.2.0",
- "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz",
- "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=",
- "requires": {
- "ms": "0.7.1"
- }
- },
- "diff": {
- "version": "1.4.0",
- "resolved": "https://registry.npmjs.org/diff/-/diff-1.4.0.tgz",
- "integrity": "sha1-fyjS657nsVqX79ic5j3P2qPMur8="
- },
- "escape-string-regexp": {
- "version": "1.0.2",
- "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.2.tgz",
- "integrity": "sha1-Tbwv5nTnGUnK8/smlc5/LcHZqNE="
- },
- "glob": {
- "version": "3.2.11",
- "resolved": "https://registry.npmjs.org/glob/-/glob-3.2.11.tgz",
- "integrity": "sha1-Spc/Y1uRkPcV0QmH1cAP0oFevj0=",
- "requires": {
- "inherits": "2",
- "minimatch": "0.3"
- }
- },
- "minimatch": {
- "version": "0.3.0",
- "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-0.3.0.tgz",
- "integrity": "sha1-J12O2qxPG7MyZHIInnlJyDlGmd0=",
- "requires": {
- "lru-cache": "2",
- "sigmund": "~1.0.0"
- }
- },
- "ms": {
- "version": "0.7.1",
- "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz",
- "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg="
- },
- "supports-color": {
- "version": "1.2.0",
- "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-1.2.0.tgz",
- "integrity": "sha1-/x7R5hFp0Gs88tWI4YixjYhH4X4="
- }
- }
- },
"modern-normalize": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/modern-normalize/-/modern-normalize-1.1.0.tgz",
@@ -24341,12 +23724,6 @@
"integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==",
"dev": true
},
- "nan": {
- "version": "2.14.2",
- "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz",
- "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==",
- "optional": true
- },
"nanocolors": {
"version": "0.1.6",
"resolved": "https://registry.npmjs.org/nanocolors/-/nanocolors-0.1.6.tgz",
@@ -24853,33 +24230,22 @@
"xtend": "~4.0.0"
}
},
- "osm-polygon-features": {
- "version": "0.9.2",
- "resolved": "https://registry.npmjs.org/osm-polygon-features/-/osm-polygon-features-0.9.2.tgz",
- "integrity": "sha1-IK5BEwxIbkmjsqPCtYoUGcSYZ3g="
- },
"osmtogeojson": {
- "version": "3.0.0-beta.4",
- "resolved": "https://registry.npmjs.org/osmtogeojson/-/osmtogeojson-3.0.0-beta.4.tgz",
- "integrity": "sha512-GwNy2w5JKOplOBspagcNhCDhBRV6Du2BCvcLkaA7nX12U86Dl2Ciw9zs/VzFFTXfyZlaK+7bGCWN2SNlfn/jOA==",
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/osmtogeojson/-/osmtogeojson-1.4.0.tgz",
+ "integrity": "sha1-7cFrFDswPAi4PH5JoFgZvaG6cis=",
"requires": {
- "@mapbox/geojson-rewind": "0.4.0",
- "@types/geojson": "^1.0.2",
- "concat-stream": "2.0.0",
- "geojson-numeric": "0.2.1",
- "htmlparser2": "3.5.1",
- "JSONStream": "0.8.0",
+ "concat-stream": "~1.0.1",
+ "htmlparser2": "~3.0.5",
+ "lodash": "~2.2.0",
"optimist": "~0.3.5",
- "osm-polygon-features": "^0.9.1",
- "tiny-osmpbf": "^0.1.0",
"xmldom": "~0.1.16"
},
"dependencies": {
- "@types/geojson": {
- "version": "1.0.6",
- "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-1.0.6.tgz",
- "integrity": "sha512-Xqg/lIZMrUd0VRmSRbCAewtwGZiAk3mEUDvV4op1tGl+LvyPcb/MIOSxTl9z+9+J+R4/vpjiCAT4xeKzH9ji1w==",
- "optional": true
+ "lodash": {
+ "version": "2.2.1",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-2.2.1.tgz",
+ "integrity": "sha1-ypNf0UqzwMhyq6zxmLnNpQFECGc="
}
}
},
@@ -25222,15 +24588,6 @@
"pinkie-promise": "^2.0.0"
}
},
- "pbf": {
- "version": "3.2.1",
- "resolved": "https://registry.npmjs.org/pbf/-/pbf-3.2.1.tgz",
- "integrity": "sha512-ClrV7pNOn7rtmoQVF4TS1vyU0WhYRnP92fzbfF75jAIwpnzdJXf8iTd4CMEqO4yUenH6NDqLiwjqlh6QgZzgLQ==",
- "requires": {
- "ieee754": "^1.1.12",
- "resolve-protobuf-schema": "^2.1.0"
- }
- },
"pbkdf2": {
"version": "3.1.2",
"resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz",
@@ -26615,11 +25972,6 @@
}
}
},
- "protocol-buffers-schema": {
- "version": "3.5.1",
- "resolved": "https://registry.npmjs.org/protocol-buffers-schema/-/protocol-buffers-schema-3.5.1.tgz",
- "integrity": "sha512-YVCvdhxWNDP8/nJDyXLuM+UFsuPk4+1PB7WGPVDzm3HTHbzFLxQYeW2iZpS4mmnXrQJGBzt230t/BbEb7PrQaw=="
- },
"psl": {
"version": "1.8.0",
"resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz",
@@ -26856,14 +26208,6 @@
"strip-indent": "^1.0.1"
}
},
- "redeyed": {
- "version": "0.4.4",
- "resolved": "https://registry.npmjs.org/redeyed/-/redeyed-0.4.4.tgz",
- "integrity": "sha1-N+mQpvKyGyoRwuakj9QTVpjLqX8=",
- "requires": {
- "esprima": "~1.0.4"
- }
- },
"reduce-css-calc": {
"version": "2.1.8",
"resolved": "https://registry.npmjs.org/reduce-css-calc/-/reduce-css-calc-2.1.8.tgz",
@@ -27063,14 +26407,6 @@
"resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz",
"integrity": "sha1-six699nWiBvItuZTM17rywoYh0g="
},
- "resolve-protobuf-schema": {
- "version": "2.1.0",
- "resolved": "https://registry.npmjs.org/resolve-protobuf-schema/-/resolve-protobuf-schema-2.1.0.tgz",
- "integrity": "sha512-kI5ffTiZWmJaS/huM8wZfEMer1eRd7oJQhDuxeCLe3t7N7mX3z94CN0xPxBQxFYQTSNz9T0i+v6inKqSdK8xrQ==",
- "requires": {
- "protocol-buffers-schema": "^3.3.1"
- }
- },
"resolve-url": {
"version": "0.2.1",
"resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz",
@@ -27373,23 +26709,6 @@
"resolved": "https://registry.npmjs.org/shallow-copy/-/shallow-copy-0.0.1.tgz",
"integrity": "sha1-QV9CcC1z2BAzApLMXuhurhoRoXA="
},
- "sharkdown": {
- "version": "0.1.1",
- "resolved": "https://registry.npmjs.org/sharkdown/-/sharkdown-0.1.1.tgz",
- "integrity": "sha512-exwooSpmo5s45lrexgz6Q0rFQM574wYIX3iDZ7RLLqOb7IAoQZu9nxlZODU972g19sR69OIpKP2cpHTzU+PHIg==",
- "requires": {
- "cardinal": "~0.4.2",
- "minimist": "0.0.5",
- "split": "~0.2.10"
- },
- "dependencies": {
- "minimist": {
- "version": "0.0.5",
- "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.5.tgz",
- "integrity": "sha1-16oye87PUY+RBqxrjwA/o7zqhWY="
- }
- }
- },
"sharp": {
"version": "0.28.3",
"resolved": "https://registry.npmjs.org/sharp/-/sharp-0.28.3.tgz",
@@ -27455,11 +26774,6 @@
"resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.2.6.tgz",
"integrity": "sha1-kEktcv/MgVmXa6umL7D2iE8MM3g="
},
- "sigmund": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/sigmund/-/sigmund-1.0.1.tgz",
- "integrity": "sha1-P/IfGYytIXX587eBhT/ZTQ0ZtZA="
- },
"signal-exit": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz",
@@ -27695,14 +27009,6 @@
"resolved": "https://registry.npmjs.org/splaytree/-/splaytree-3.1.0.tgz",
"integrity": "sha512-gvUGR7xnOy0fLKTCxDeUZYgU/I1Tdf8M/lM1Qrf8L2TIOR5ipZjGk02uYcdv0o2x7WjVRgpm3iS2clLyuVAt0Q=="
},
- "split": {
- "version": "0.2.10",
- "resolved": "https://registry.npmjs.org/split/-/split-0.2.10.tgz",
- "integrity": "sha1-Zwl8YB1pfOE2j0GPBs0gHPBSGlc=",
- "requires": {
- "through": "2"
- }
- },
"split-string": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz",
@@ -28039,7 +27345,7 @@
},
"svg-resizer": {
"version": "git+ssh://git@github.com/vieron/svg-resizer.git#00968cb3e7248533ab9451ce7dffa8af288e4f4a",
- "from": "svg-resizer@git+https://github.com/vieron/svg-resizer.git",
+ "from": "svg-resizer@github:vieron/svg-resizer",
"requires": {
"fs-extra": "~0.8.1",
"lodash": "~2.4.1",
@@ -28351,7 +27657,8 @@
"through": {
"version": "2.3.8",
"resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
- "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU="
+ "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
+ "dev": true
},
"through2": {
"version": "2.0.5",
@@ -28380,15 +27687,6 @@
"resolved": "https://registry.npmjs.org/tiny-inflate/-/tiny-inflate-1.0.3.tgz",
"integrity": "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw=="
},
- "tiny-osmpbf": {
- "version": "0.1.0",
- "resolved": "https://registry.npmjs.org/tiny-osmpbf/-/tiny-osmpbf-0.1.0.tgz",
- "integrity": "sha1-ColXFxE+vmquNjxL5e76js2vuSc=",
- "requires": {
- "pbf": "^3.0.4",
- "tiny-inflate": "^1.0.2"
- }
- },
"tinyqueue": {
"version": "2.0.3",
"resolved": "https://registry.npmjs.org/tinyqueue/-/tinyqueue-2.0.3.tgz",
@@ -28422,11 +27720,6 @@
"resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz",
"integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4="
},
- "to-iso-string": {
- "version": "0.0.2",
- "resolved": "https://registry.npmjs.org/to-iso-string/-/to-iso-string-0.0.2.tgz",
- "integrity": "sha1-TcGeZk38y+Jb2NtQiwDG2hWCVdE="
- },
"to-object-path": {
"version": "0.3.0",
"resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz",
@@ -28478,16 +27771,6 @@
"jxon": "~2.0.0-beta.5",
"optimist": "~0.3.5",
"xmldom": "~0.1.17"
- },
- "dependencies": {
- "concat-stream": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.0.1.tgz",
- "integrity": "sha1-AYsYvBx9BzotyCqkhEI0GixN158=",
- "requires": {
- "bops": "0.0.6"
- }
- }
}
},
"toidentifier": {
diff --git a/package.json b/package.json
index fc157041a3..bdac19110e 100644
--- a/package.json
+++ b/package.json
@@ -11,7 +11,7 @@
"start": "npm run start:prepare && npm-run-all --parallel start:parallel:*",
"strt": "npm run start:prepare && npm run start:parallel:parcel",
"start:prepare": "ts-node scripts/generateLayerOverview.ts --no-fail && npm run increase-memory",
- "start:parallel:parcel": "node --max_old_space_size=12000 $(which parcel) serve *.html UI/** Logic/** assets/*.json assets/svg/* assets/generated/* assets/layers/*/*.svg assets/layers/*/*.jpg assets/layers/*/*.png assets/layers/*/*.css assets/tagRenderings/*.json assets/themes/*/*.svg assets/themes/*/*.css assets/themes/*/*.jpg assets/themes/*/*.png vendor/* vendor/*/*",
+ "start:parallel:parcel": "parcel serve *.html UI/** Logic/** assets/*.json assets/svg/* assets/generated/* assets/layers/*/*.svg assets/layers/*/*.jpg assets/layers/*/*.png assets/layers/*/*.css assets/tagRenderings/*.json assets/themes/*/*.svg assets/themes/*/*.css assets/themes/*/*.jpg assets/themes/*/*.png vendor/* vendor/*/*",
"start:parallel:tailwindcli": "tailwindcss -i index.css -o css/index-tailwind-output.css --watch",
"generate:css": "tailwindcss -i index.css -o css/index-tailwind-output.css",
"test": "ts-node test/TestAll.ts",
@@ -28,28 +28,21 @@
"generate:cache:speelplekken:mini": "ts-node scripts/generateCache.ts speelplekken 14 ../MapComplete-data/speelplekken_cache_mini/ 51.181710380278176 4.423413276672363 51.193007664772495 4.444141387939452",
"generate:cache:speelplekken": "npm run generate:layeroverview && ts-node scripts/generateCache.ts speelplekken 14 ../MapComplete-data/speelplekken_cache/ 51.20 4.35 51.09 4.56",
"generate:cache:natuurpunt": "npm run generate:layeroverview && ts-node scripts/generateCache.ts natuurpunt 12 ../MapComplete-data/natuurpunt_cache/ 50.40 2.1 51.54 6.4 --generate-point-overview nature_reserve,visitor_information_centre",
- "generate:layeroverview": "echo {\\\"layers\\\":[], \\\"themes\\\":[]} > ./assets/generated/known_layers_and_themes.json && ts-node scripts/generateLayerOverview.ts --no-fail",
+ "generate:layeroverview": "ts-node scripts/generateLayerOverview.ts --no-fail",
"generate:licenses": "ts-node scripts/generateLicenseInfo.ts --no-fail",
"query:licenses": "ts-node scripts/generateLicenseInfo.ts --query",
"generate:report": "cd Docs/Tools && ./compileStats.sh && git commit . -m 'New statistics ands graphs' && git push",
"generate:contributor-list": "git log --pretty='%aN' | sort | uniq -c | sort -hr | sed 's/ *\\([0-9]*\\) \\(.*\\)$/{\"contributor\":\"\\2\", \"commits\":\\1}/' | tr '\\n' ',' | sed 's/^/{\"contributors\":[/' | sed 's/,$/]}/' | jq > assets/contributors.json",
- "validate:layeroverview": "ts-node scripts/generateLayerOverview.ts --report",
- "validate:licenses": "ts-node scripts/generateLicenseInfo.ts --report",
"generate:schemas": "ts2json-schema -p Models/ThemeConfig/Json/ -o Docs/Schemas/ -t tsconfig.json -R . -m \".*ConfigJson\" && ts-node scripts/fixSchemas.ts ",
"optimize-images": "cd assets/generated/ && find -name '*.png' -exec optipng '{}' \\; && echo 'PNGs are optimized'",
"reset:layeroverview": "echo {\\\"layers\\\":[], \\\"themes\\\":[]} > ./assets/generated/known_layers_and_themes.json",
- "generate": "mkdir -p ./assets/generated && npm run reset:layeroverview && npm run generate:images && npm run generate:charging-stations && npm run generate:translations && npm run generate:licenses && npm run validate:layeroverview",
- "build": "rm -rf dist/ && npm run generate && parcel build --public-url ./ *.html assets/** assets/**/** assets/**/**/** vendor/* vendor/*/*",
+ "generate": "mkdir -p ./assets/generated && npm run reset:layeroverview && npm run generate:images && npm run generate:charging-stations && npm run generate:translations && npm run generate:licenses && npm run generate:layeroverview",
"generate:charging-stations": "cd ./assets/layers/charging_station && ts-node csvToJson.ts && cd -",
- "prepare-deploy": "npm run generate && npm run test && npm run generate:editor-layer-index && npm run generate:layouts && npm run build && rm -rf .cache",
- "deploy:staging": "npm run prepare-deploy && rm -rf ~/git/pietervdvn.github.io/Staging/* && cp -r dist/* ~/git/pietervdvn.github.io/Staging/ && cd ~/git/pietervdvn.github.io/ && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean",
- "deploy:pietervdvn": "cd ~/git/pietervdvn.github.io/ && git pull && cd - && npm run prepare-deploy && rm -rf ~/git/pietervdvn.github.io/MapComplete/* && cp -r dist/* ~/git/pietervdvn.github.io/MapComplete/ && cd ~/git/pietervdvn.github.io/ && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean",
- "deploy:production": "cd ~/git/mapcomplete.github.io/ && git pull && cd - && rm -rf ./assets/generated && npm run prepare-deploy && npm run optimize-images && rm -rf ~/git/mapcomplete.github.io/* && cp -r dist/* ~/git/mapcomplete.github.io/ && cd ~/git/mapcomplete.github.io/ && echo \"mapcomplete.osm.be\" > CNAME && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean && npm run gittag",
+ "prepare-deploy": "./scripts/build.sh",
"gittag": "ts-node scripts/printVersion.ts | bash",
"lint": "tslint --project . -c tslint.json '**.ts' ",
- "clean": "rm -rf .cache/ && (find *.html | grep -v \"\\(index\\|land\\|test\\|preferences\\|customGenerator\\|professional\\|automaton\\).html\" | xargs rm) && rm *.webmanifest",
- "generate:dependency-graph": "node_modules/.bin/depcruise --exclude \"^node_modules\" --output-type dot test/TestAll.ts > dependencies.dot && dot dependencies.dot -T svg -o dependencies.svg && rm dependencies.dot",
- "genPostal": " ts-node ./scripts/postal_code_tools/createRoutablePoint.ts /home/pietervdvn/Downloads/postal_codes/postal_codes_town_hall_points.geojson /home/pietervdvn/Downloads/31370/Postcodes.geojson\n"
+ "clean": "rm -rf .cache/ && (find *.html | grep -v \"\\(404\\|index\\|land\\|test\\|preferences\\|customGenerator\\|professional\\|automaton\\|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"
},
"keywords": [
"OpenStreetMap",
@@ -84,7 +77,7 @@
"leaflet-providers": "^1.13.0",
"leaflet-simple-map-screenshoter": "^0.4.4",
"leaflet.markercluster": "^1.4.1",
- "libphonenumber": "0.0.10",
+ "libphonenumber": "^0.0.9",
"libphonenumber-js": "^1.7.55",
"lz-string": "^1.4.4",
"mangrove-reviews": "^0.1.3",
@@ -92,7 +85,7 @@
"npm-run-all": "^4.1.5",
"opening_hours": "^3.6.0",
"osm-auth": "^1.0.2",
- "osmtogeojson": "^3.0.0-beta.4",
+ "osmtogeojson": "^1.0.0",
"parcel": "^1.2.4",
"prompt-sync": "^4.2.0",
"svg-resizer": "github:vieron/svg-resizer",
diff --git a/preferences.html b/preferences.html
deleted file mode 100644
index 10250978c0..0000000000
--- a/preferences.html
+++ /dev/null
@@ -1,29 +0,0 @@
-
-
-
-
-
Preferences editor
-
-
-
-
-
-
-