2024-11-26 20:41:37 -06:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html lang="en">
|
|
|
|
<head>
|
|
|
|
<meta charset="UTF-8">
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
|
|
<title>Lawrence Deer Club Maps</title>
|
|
|
|
<link rel="stylesheet" href="/leaflet/leaflet.css" />
|
|
|
|
<script src="/leaflet/leaflet.js"></script>
|
|
|
|
<style>
|
|
|
|
html, body {
|
|
|
|
margin: 0;
|
|
|
|
padding: 0;
|
|
|
|
}
|
|
|
|
#map {
|
|
|
|
height: 100vh;
|
|
|
|
width: 100vw;
|
|
|
|
}
|
|
|
|
|
|
|
|
div.popup dt {
|
|
|
|
font-weight: bold;
|
|
|
|
}
|
|
|
|
|
2024-11-27 20:35:22 -06:00
|
|
|
.label {
|
|
|
|
font-weight: bold;
|
|
|
|
text-align: center;
|
|
|
|
margin-top: -1em;
|
|
|
|
}
|
|
|
|
.label div {
|
|
|
|
position: relative;
|
|
|
|
left: -50%;
|
2025-01-03 23:42:46 -06:00
|
|
|
top: -50%;
|
2024-11-27 20:35:22 -06:00
|
|
|
text-shadow: 0 0 2px white;
|
|
|
|
}
|
2024-11-30 12:26:39 -06:00
|
|
|
.leaflet-popup-content-wrapper {
|
|
|
|
height: min(800px, 60vh);
|
|
|
|
overflow-y: auto;
|
|
|
|
}
|
|
|
|
#popupcontainer {
|
|
|
|
position: relative;
|
|
|
|
z-index: 1000;
|
|
|
|
}
|
|
|
|
#popupcontainer .leaflet-popup-content-wrapper {
|
|
|
|
height: unset;
|
|
|
|
}
|
|
|
|
#popupcontainer .leaflet-popup {
|
|
|
|
transform: none !important;
|
|
|
|
position: fixed;
|
|
|
|
width: 100vw;
|
|
|
|
height: 100vh;
|
|
|
|
right: 0;
|
|
|
|
left: 0 !important;
|
|
|
|
top: 1em;
|
|
|
|
bottom: 1em !important;
|
|
|
|
overflow: auto;
|
|
|
|
}
|
|
|
|
#popupcontainer .leaflet-popup-content {
|
|
|
|
width: initial;
|
|
|
|
}
|
2025-01-03 00:02:04 -06:00
|
|
|
|
|
|
|
/* BEGIN lightbox */
|
|
|
|
|
|
|
|
/* simplified from https://jekyllcodex.org/without-plugin/lightbox/ */
|
|
|
|
#lightbox {
|
|
|
|
width: 100%;
|
|
|
|
height: 100%;
|
|
|
|
position: fixed;
|
|
|
|
top: 0;
|
|
|
|
left: 0;
|
|
|
|
background: rgba(0,0,0,0.85);
|
|
|
|
z-index: 9999999;
|
|
|
|
line-height: 0;
|
|
|
|
cursor: pointer;
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
#lightbox .img {
|
|
|
|
position: relative;
|
|
|
|
top: 50%;
|
|
|
|
left: 50%;
|
|
|
|
transform: translate(-50%,-50%);
|
|
|
|
max-width: 100%;
|
|
|
|
max-height: 100%;
|
|
|
|
}
|
|
|
|
#lightbox .img img {
|
|
|
|
opacity: 0;
|
|
|
|
pointer-events: none;
|
|
|
|
width: auto;
|
|
|
|
}
|
|
|
|
@media screen and (min-width: 1200px) {
|
|
|
|
#lightbox .img {
|
|
|
|
max-width: 1200px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@media screen and (min-height: 1200px) {
|
|
|
|
#lightbox .img {
|
|
|
|
max-height: 1200px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#lightbox #close {
|
|
|
|
height: 50px;
|
|
|
|
width: 50px;
|
|
|
|
position: fixed;
|
|
|
|
cursor: pointer;
|
|
|
|
text-decoration: none;
|
|
|
|
z-index: 99;
|
|
|
|
right: 0;
|
|
|
|
top: 0;
|
|
|
|
}
|
|
|
|
#lightbox #close:after, #lightbox #close:before {
|
|
|
|
position: absolute;
|
|
|
|
margin-top: 22px;
|
|
|
|
margin-left: 14px;
|
|
|
|
content: "";
|
|
|
|
height: 3px;
|
|
|
|
background: white;
|
|
|
|
width: 23px;
|
|
|
|
transform-origin: 50% 50%;
|
|
|
|
transform: rotate(-45deg);
|
|
|
|
}
|
|
|
|
#lightbox #close:after {
|
|
|
|
transform: rotate(45deg);
|
|
|
|
}
|
|
|
|
#lightbox, #lightbox * {
|
|
|
|
user-select: none;
|
|
|
|
}
|
|
|
|
/* END lightbox */
|
2024-11-26 20:41:37 -06:00
|
|
|
</style>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<div id="map"></div>
|
2024-11-30 12:26:39 -06:00
|
|
|
<div id="popupcontainer" class="leaflet-container"></div>
|
2025-01-03 00:02:04 -06:00
|
|
|
<div id="lightbox" onclick="this.style.display = 'none';"></div>
|
2024-11-26 20:41:37 -06:00
|
|
|
<script>
|
2025-01-03 00:02:04 -06:00
|
|
|
function displayLightboxOnClick(target, event) {
|
|
|
|
event.preventDefault();
|
|
|
|
document.getElementById('lightbox').innerHTML = `
|
|
|
|
<a id="close"></a>
|
|
|
|
<div class="img" style="background: url('${target.getAttribute('href')}') center center / contain no-repeat;">
|
|
|
|
<img src="${target.getAttribute('href')}">
|
|
|
|
</div>`;
|
|
|
|
document.getElementById('lightbox').style.display = 'block';
|
|
|
|
}
|
2025-01-03 23:42:46 -06:00
|
|
|
function calculateCentroid(points) {
|
|
|
|
// https://en.wikipedia.org/wiki/Centroid#Of_a_polygon
|
|
|
|
let area = 0;
|
|
|
|
let centroidLat = 0;
|
|
|
|
let centroidLng = 0;
|
|
|
|
|
|
|
|
for (let i = 0; i < points.length; i++) {
|
|
|
|
const { lat: lat1, lng: lng1 } = points[i];
|
|
|
|
const { lat: lat2, lng: lng2 } = points[(i + 1) % points.length]; // Next vertex, wrapping around
|
|
|
|
|
|
|
|
const determinant = lat1 * lng2 - lng1 * lat2;
|
|
|
|
area += determinant;
|
|
|
|
centroidLat += (lat1 + lat2) * determinant;
|
|
|
|
centroidLng += (lng1 + lng2) * determinant;
|
|
|
|
}
|
|
|
|
|
|
|
|
area *= 0.5;
|
|
|
|
centroidLat /= (6 * area);
|
|
|
|
centroidLng /= (6 * area);
|
|
|
|
|
|
|
|
return [centroidLat, centroidLng];
|
|
|
|
}
|
2024-11-26 20:41:37 -06:00
|
|
|
(async function() {
|
2025-01-02 22:08:40 -06:00
|
|
|
const map = L.map('map', {
|
2025-01-03 22:59:05 -06:00
|
|
|
minZoom: 15,
|
2025-01-02 22:08:40 -06:00
|
|
|
maxZoom: 21,
|
|
|
|
maxBounds: [
|
|
|
|
[47.517085, -93.427584],
|
|
|
|
[47.457925, -93.340026],
|
|
|
|
],
|
2025-01-03 22:59:43 -06:00
|
|
|
});
|
2024-11-26 20:41:37 -06:00
|
|
|
// L.tileLayer('https://tile.openstreetmap.org/{z}/{x}/{y}.png', {
|
|
|
|
// maxZoom: 19,
|
|
|
|
// attribution: '© <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a>'
|
|
|
|
// }).addTo(map);
|
2025-01-02 23:23:45 -06:00
|
|
|
L.tileLayer('./satellite/{z}/{x}/{y}.jpg', {
|
2025-01-02 22:07:10 -06:00
|
|
|
minNativeZoom: 12,
|
|
|
|
maxNativeZoom: 20,
|
2025-01-02 23:53:39 -06:00
|
|
|
maxZoom: 21,
|
2025-01-02 22:07:10 -06:00
|
|
|
bounds: [
|
|
|
|
[47.517085, -93.427584],
|
|
|
|
[47.457925, -93.340026],
|
|
|
|
],
|
|
|
|
}).addTo(map);
|
|
|
|
|
2024-11-26 20:41:37 -06:00
|
|
|
|
|
|
|
// Make these requests in parallel
|
|
|
|
const track_req_promise = fetch('data/track.geojson');
|
|
|
|
const plats_req_promise = fetch('data/plats.geojson');
|
|
|
|
const data_req_promise = fetch('data/data.json');
|
|
|
|
const track_req = await track_req_promise;
|
|
|
|
const plats_req = await plats_req_promise;
|
|
|
|
const data_req = await data_req_promise;
|
|
|
|
const track = await track_req.json();
|
|
|
|
const plats = await plats_req.json();
|
|
|
|
const data = await data_req.json();
|
|
|
|
|
|
|
|
L.geoJSON(plats, {
|
2024-11-30 12:33:41 -06:00
|
|
|
filter: feature => !feature.properties.TAO_NAME.toLowerCase().includes("lawrence deer club"),
|
2024-11-27 20:35:22 -06:00
|
|
|
onEachFeature: function(feature, layer) {
|
2025-01-03 23:42:46 -06:00
|
|
|
const centroid = calculateCentroid(layer.getLatLngs()[0]);
|
|
|
|
const label = L.marker(centroid, {
|
2024-11-27 20:35:22 -06:00
|
|
|
icon: L.divIcon({
|
|
|
|
iconSize: null,
|
|
|
|
className: "label",
|
|
|
|
html: "<div>" + feature.properties.TAO_NAME + "</div>"
|
|
|
|
})
|
|
|
|
}).addTo(map);
|
|
|
|
},
|
2024-11-26 20:41:37 -06:00
|
|
|
}).addTo(map);
|
|
|
|
|
2025-01-02 23:24:51 -06:00
|
|
|
L.geoJSON(track, {
|
|
|
|
style: {
|
|
|
|
color: "maroon",
|
|
|
|
},
|
|
|
|
}).addTo(map);
|
|
|
|
|
2024-11-26 20:41:37 -06:00
|
|
|
for (let stand of data.stands) {
|
|
|
|
if (stand.location) {
|
|
|
|
const marker = L.marker(stand.location).addTo(map);
|
|
|
|
const popupContentWrapper = document.createElement('div');
|
|
|
|
popupContentWrapper.classList.add('popup');
|
|
|
|
let attributesString = '<dl>';
|
|
|
|
for (let [attribute, value] of Object.entries(stand.attributes)) {
|
|
|
|
attributesString += `<dt>${attribute}</dt><dd>${value}</dd>`;
|
|
|
|
}
|
|
|
|
attributesString += '</dl>';
|
|
|
|
popupContentWrapper.innerHTML = `
|
|
|
|
<h2>${stand.name}</h2>
|
2025-01-03 00:02:04 -06:00
|
|
|
<a href="images/${stand.image}" onclick="displayLightboxOnClick(this, event)"><img style="width: min(80vw, 300px);" src="images/thumbs/${stand.image}"></a>
|
2024-11-26 20:41:37 -06:00
|
|
|
${attributesString}
|
|
|
|
`;
|
2024-11-30 12:26:39 -06:00
|
|
|
let popupOptions = {};
|
|
|
|
// TODO: evaluate this when creating popup instead of at
|
|
|
|
// page instantiation time -- if the page is resized, this
|
|
|
|
// won't keep up with those changes.
|
|
|
|
if (window.visualViewport.width < 600) {
|
|
|
|
popupOptions.pane = document.getElementById('popupcontainer');
|
|
|
|
}
|
|
|
|
marker.bindPopup(popupContentWrapper, popupOptions);
|
2024-11-26 20:41:37 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
map.fitBounds([
|
|
|
|
[47.4865,-93.4068],
|
|
|
|
[47.4992,-93.3746],
|
|
|
|
]);
|
|
|
|
})();
|
|
|
|
</script>
|
|
|
|
</body>
|
|
|
|
</html>
|