Compare commits
8 commits
e4bb95c1c8
...
efeec0bf42
Author | SHA1 | Date | |
---|---|---|---|
efeec0bf42 | |||
209eb3c7c2 | |||
f3b2c0245c | |||
4d26db348c | |||
1416481ba9 | |||
2fee843c52 | |||
954e8c4500 | |||
54ba75dd1c |
4 changed files with 222 additions and 75 deletions
14
README.md
14
README.md
|
@ -82,6 +82,20 @@ value also increases the size of the `strecken.pmtiles` file. If not
|
||||||
given explicitly `maxZoom` defaults to 10.
|
given explicitly `maxZoom` defaults to 10.
|
||||||
|
|
||||||
|
|
||||||
|
## Map Editing Mode
|
||||||
|
|
||||||
|
As a new experimental feature, Streckenkarte implements a simple
|
||||||
|
[brouter][brouter] frontend to allow for easy editing, currently only
|
||||||
|
supported on browsers that support
|
||||||
|
[showDirectoryPicker](https://developer.mozilla.org/en-US/docs/Web/API/Window/showDirectoryPicker). It
|
||||||
|
can be accessed by adding `?edit=1` to the URL and then clicking the
|
||||||
|
"edit" button that appears in the top right corner. Then, the
|
||||||
|
directory containing the files for the layer that is to be edited can
|
||||||
|
be opened in the file selection dialog and lines can be drawn as in
|
||||||
|
the normal brouter web frontend. By clicking on "save", the lines are
|
||||||
|
saved locally in the chosen directory and will only appear on the
|
||||||
|
production map if processed correctly by the tile-generating script.
|
||||||
|
|
||||||
## Troubleshooting
|
## Troubleshooting
|
||||||
|
|
||||||
### My pmtiles file is huge (hundreds of megabytes)
|
### My pmtiles file is huge (hundreds of megabytes)
|
||||||
|
|
206
frontend/common/map.js
Normal file
206
frontend/common/map.js
Normal file
|
@ -0,0 +1,206 @@
|
||||||
|
function layer_legend(layer) {
|
||||||
|
return '<span class="dot" style="background: ' + layer["color"] + '" ></span>' + layer["name"];
|
||||||
|
}
|
||||||
|
|
||||||
|
const rules = new Array;
|
||||||
|
const l = new Array;
|
||||||
|
const legend = L.control({position: 'bottomleft'});
|
||||||
|
legend.onAdd = function (map) {
|
||||||
|
const div = L.DomUtil.create('div', 'legend');
|
||||||
|
for (let i = 0; i < l.length; i++) {
|
||||||
|
div.innerHTML += layer_legend(l[i]) + "<br>";
|
||||||
|
}
|
||||||
|
return div;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const map = L.map("map", { center: [52,13], zoom: 3, minZoom: 0 });
|
||||||
|
|
||||||
|
function update_hash() {
|
||||||
|
const {lat, lng} = this.getCenter();
|
||||||
|
const zoom = this.getZoom();
|
||||||
|
const digits = 4;
|
||||||
|
window.history.replaceState(null, '', `#map=${zoom}/${lat.toFixed(digits)}/${lng.toFixed(digits)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function onHashChange() {
|
||||||
|
const hash = document.location.hash;
|
||||||
|
const coords = decodeURIComponent(hash.slice(5)).split("/") // strip off the #map= part
|
||||||
|
const latLng = L.latLng(parseFloat(coords[1]), parseFloat(coords[2]));
|
||||||
|
map.setView(latLng, parseInt(coords[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
map.on("moveend", update_hash);
|
||||||
|
map.on("zoomend", update_hash);
|
||||||
|
|
||||||
|
fetch("layers.json")
|
||||||
|
.then((response) => response.json())
|
||||||
|
.then((data) => {
|
||||||
|
document.title = data["name"]
|
||||||
|
const layers = data["layers"]
|
||||||
|
for (let key in layers) {
|
||||||
|
l.push({ dirname: key , name: layers[key]["humanname"], color: layers[key]["color"] });
|
||||||
|
rules.push({
|
||||||
|
dataLayer: key,
|
||||||
|
symbolizer: new protomapsL.LineSymbolizer(layers[key])
|
||||||
|
});
|
||||||
|
}
|
||||||
|
const tiles = data["tilelayer"]
|
||||||
|
const osm = L.tileLayer(
|
||||||
|
tiles["url_template"],
|
||||||
|
{
|
||||||
|
maxZoom: 19,
|
||||||
|
attribution: tiles["attribution"]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
const strecken = protomapsL.leafletLayer({
|
||||||
|
url: "strecken.pmtiles",
|
||||||
|
maxDataZoom: data["maxZoom"] ?? 10,
|
||||||
|
maxZoom: 19,
|
||||||
|
paintRules: rules,
|
||||||
|
});
|
||||||
|
|
||||||
|
osm.addTo(map);
|
||||||
|
legend.addTo(map);
|
||||||
|
strecken.addTo(map);
|
||||||
|
})
|
||||||
|
|
||||||
|
let dirHandle;
|
||||||
|
let editMode = false;
|
||||||
|
let markers = [];
|
||||||
|
let geojsons = [];
|
||||||
|
let geojson;
|
||||||
|
let editlayer;
|
||||||
|
|
||||||
|
function addGeoJsonToMap(dat) {
|
||||||
|
if (editlayer != undefined) {
|
||||||
|
const style = {
|
||||||
|
"color": editlayer.color
|
||||||
|
};
|
||||||
|
g = L.geoJSON(dat, {style: style});
|
||||||
|
} else {
|
||||||
|
g = L.geoJSON(dat);
|
||||||
|
}
|
||||||
|
g.addTo(map);
|
||||||
|
return g;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updateBrouter () {
|
||||||
|
if (markers.length > 0) {
|
||||||
|
for (i=1; i< markers.length-1; i++) {
|
||||||
|
markers[i]._icon.classList.remove("red");
|
||||||
|
}
|
||||||
|
markers[markers.length-1]._icon.classList.add("red");
|
||||||
|
markers[0]._icon.classList.add("green");
|
||||||
|
}
|
||||||
|
if (markers.length < 2) {
|
||||||
|
if (geojson != undefined) {
|
||||||
|
map.removeLayer(geojson);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
geojsons = [];
|
||||||
|
for (let i = 0; i < markers.length - 1 ; i++) {
|
||||||
|
const marker1 = markers[i].getLatLng();
|
||||||
|
const marker2 = markers[i+1].getLatLng();
|
||||||
|
const url = `https://brouter.de/brouter?lonlats=${marker1.lng},${marker1.lat}|${marker2.lng},${marker2.lat}&profile=rail&alternativeidx=0&format=geojson`;
|
||||||
|
fetch(url).then((response) => response.json())
|
||||||
|
.then((data) => {
|
||||||
|
if (geojson != undefined) {
|
||||||
|
map.removeLayer(geojson);
|
||||||
|
}
|
||||||
|
geojsons.push(data.features[0]);
|
||||||
|
const dat = {type: "FeatureCollection", features: geojsons};
|
||||||
|
geojson = addGeoJsonToMap(dat);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
map.on('click', function(e) {
|
||||||
|
if (!editMode) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
marker = new L.marker(e.latlng, {draggable: true}) ;
|
||||||
|
markers.push(marker);
|
||||||
|
marker.on("click", function(e) {
|
||||||
|
map.removeLayer(this);
|
||||||
|
markers = markers.filter(item => item != this);
|
||||||
|
updateBrouter();
|
||||||
|
})
|
||||||
|
marker.on("dragend", function(e) {
|
||||||
|
updateBrouter();
|
||||||
|
});
|
||||||
|
marker.addTo(map);
|
||||||
|
updateBrouter();
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
async function pickDirectory(e){
|
||||||
|
e.stopPropagation()
|
||||||
|
L.DomEvent.preventDefault(e);
|
||||||
|
if (!editMode) {
|
||||||
|
dirHandle = await window.showDirectoryPicker({ mode: 'readwrite' });
|
||||||
|
for (i = 0; i < l.length ; i++ ) {
|
||||||
|
console.log(l[i].dirname);
|
||||||
|
if (l[i].dirname === dirHandle.name) {
|
||||||
|
editlayer = l[i];
|
||||||
|
this.innerHTML = "Editing layer " + layer_legend(l[i]) + "<br>" + this.innerHTML
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById("edit-mode").style.color = "red";
|
||||||
|
document.getElementById("edit-mode").innerHTML = "save";
|
||||||
|
editMode = true;
|
||||||
|
} else {
|
||||||
|
const filename = window.prompt("Enter filename:", "test");
|
||||||
|
const dat = {type: "FeatureCollection", features: geojsons};
|
||||||
|
const file = await dirHandle.getFileHandle(`${filename}.geojson`, {
|
||||||
|
create: true
|
||||||
|
});
|
||||||
|
const blob = new Blob([JSON.stringify(dat)]);
|
||||||
|
const writableStream = await file.createWritable();
|
||||||
|
await writableStream.write(blob);
|
||||||
|
await writableStream.close();
|
||||||
|
addGeoJsonToMap(dat);
|
||||||
|
for (i=0; i<markers.length; i++) {
|
||||||
|
map.removeLayer(markers[i]);
|
||||||
|
}
|
||||||
|
markers = [];
|
||||||
|
updateBrouter();
|
||||||
|
alert("Saved file!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const searchParams = new URLSearchParams(window.location.search)
|
||||||
|
const edit = searchParams.get("edit");
|
||||||
|
|
||||||
|
if (edit) {
|
||||||
|
const customButton = L.control({ position: 'topright' });
|
||||||
|
customButton.onAdd = () => {
|
||||||
|
const buttonDiv = L.DomUtil.create('div', 'legend');
|
||||||
|
if ("showDirectoryPicker" in window) {
|
||||||
|
buttonDiv.innerHTML = `<button id="edit-mode" >Edit</button>`;
|
||||||
|
buttonDiv.addEventListener('click', pickDirectory, this)
|
||||||
|
} else {
|
||||||
|
buttonDiv.innerHTML = "Your browser does not support editing. <br> As of 2025, editing is supported on Chromium-based browsers only.";
|
||||||
|
}
|
||||||
|
return buttonDiv;
|
||||||
|
};
|
||||||
|
customButton.addTo(map)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function resize() {
|
||||||
|
document.getElementById("map").style.height = window.innerHeight + 'px';
|
||||||
|
}
|
||||||
|
resize();
|
||||||
|
window.addEventListener('resize', () => {
|
||||||
|
resize();
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener("hashchange", onHashChange);
|
||||||
|
onHashChange();
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -19,3 +19,5 @@ body {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
margin-right: 0.7em;
|
margin-right: 0.7em;
|
||||||
}
|
}
|
||||||
|
img.red { filter: hue-rotate(120deg); }
|
||||||
|
img.green { filter: hue-rotate(-120deg); }
|
|
@ -1,75 +0,0 @@
|
||||||
|
|
||||||
const rules = new Array;
|
|
||||||
const l = new Array;
|
|
||||||
var legend = L.control({position: 'bottomleft'});
|
|
||||||
legend.onAdd = function (map) {
|
|
||||||
var div = L.DomUtil.create('div', 'legend');
|
|
||||||
for (var i = 0; i < l.length; i++) {
|
|
||||||
div.innerHTML += '<span class="dot" style="background: ' + l[i]["color"] + '" ></span>' + l[i]["name"] + "<br>";
|
|
||||||
}
|
|
||||||
return div;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
const map = L.map("map", { center: [52,13], zoom: 3, minZoom: 0 });
|
|
||||||
|
|
||||||
function update_hash() {
|
|
||||||
const {lat, lng} = this.getCenter();
|
|
||||||
const zoom = this.getZoom();
|
|
||||||
const digits = 4;
|
|
||||||
window.history.replaceState(null, '', `#map=${zoom}/${lat.toFixed(digits)}/${lng.toFixed(digits)}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
function onHashChange() {
|
|
||||||
const hash = document.location.hash;
|
|
||||||
const coords = decodeURIComponent(hash.slice(5)).split("/") // strip off the #map= part
|
|
||||||
const latLng = L.latLng(parseFloat(coords[1]), parseFloat(coords[2]));
|
|
||||||
map.setView(latLng, parseInt(coords[0]));
|
|
||||||
}
|
|
||||||
|
|
||||||
map.on("moveend", update_hash);
|
|
||||||
map.on("zoomend", update_hash);
|
|
||||||
|
|
||||||
fetch("layers.json")
|
|
||||||
.then((response) => response.json())
|
|
||||||
.then((data) => {
|
|
||||||
document.title = data["name"]
|
|
||||||
layers = data["layers"]
|
|
||||||
for (let key in layers) {
|
|
||||||
l.push({ name: layers[key]["humanname"], color: layers[key]["color"] });
|
|
||||||
rules.push({
|
|
||||||
dataLayer: key,
|
|
||||||
symbolizer: new protomapsL.LineSymbolizer(layers[key])
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const tiles = data["tilelayer"]
|
|
||||||
var osm = L.tileLayer(
|
|
||||||
tiles["url_template"],
|
|
||||||
{
|
|
||||||
maxZoom: 19,
|
|
||||||
attribution: tiles["attribution"]
|
|
||||||
}
|
|
||||||
);
|
|
||||||
var strecken = protomapsL.leafletLayer({
|
|
||||||
url: "strecken.pmtiles",
|
|
||||||
maxDataZoom: data["maxZoom"] ?? 10,
|
|
||||||
maxZoom: 19,
|
|
||||||
paintRules: rules,
|
|
||||||
});
|
|
||||||
|
|
||||||
osm.addTo(map);
|
|
||||||
legend.addTo(map);
|
|
||||||
strecken.addTo(map);
|
|
||||||
})
|
|
||||||
|
|
||||||
function resize() {
|
|
||||||
document.getElementById("map").style.height = window.innerHeight + 'px';
|
|
||||||
}
|
|
||||||
resize();
|
|
||||||
window.addEventListener('resize', () => {
|
|
||||||
resize();
|
|
||||||
});
|
|
||||||
|
|
||||||
window.addEventListener("hashchange", onHashChange);
|
|
||||||
onHashChange();
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue