303 lines
15 KiB
HTML
303 lines
15 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<meta name="color-scheme" content="dark light">
|
|
<title>IP Map</title>
|
|
<script src="https://unpkg.com/maplibre-gl@4.1.2/dist/maplibre-gl.js"></script>
|
|
<link rel="stylesheet" href="https://unpkg.com/maplibre-gl@4.1.2/dist/maplibre-gl.css" />
|
|
<link rel="shortcut icon" href="" />
|
|
<style>
|
|
body {
|
|
margin: 0;
|
|
padding: 0;
|
|
}
|
|
html, body {
|
|
width: 100%;
|
|
height: 100%;
|
|
}
|
|
main {
|
|
width: 100vh;
|
|
height: 100%;
|
|
margin: 0 auto;
|
|
}
|
|
#map {
|
|
width: 100%;
|
|
height: 100%;
|
|
}
|
|
.maplibregl-canvas {
|
|
cursor: pointer;
|
|
}
|
|
.maplibregl-popup-content {
|
|
background-color: #222;
|
|
font-size: 1rem;
|
|
padding: 0.8rem 1.2rem;
|
|
}
|
|
.map-overlay {
|
|
position: absolute;
|
|
top: 1rem;
|
|
right: 1rem;
|
|
padding: 1.5rem;
|
|
background-color: #222;
|
|
color: #eee;
|
|
box-shadow: 3px 3px 2px rgba(0, 0, 0, 0.8);
|
|
border-radius: 3px;
|
|
}
|
|
.map-overlay h2 {
|
|
display: block;
|
|
margin: 0;
|
|
margin-bottom: 1rem;
|
|
}
|
|
#map-style-controls ul {
|
|
padding-left: 1em;
|
|
list-style-type: none;
|
|
}
|
|
#map-style-controls > ul {
|
|
padding-left: 0;
|
|
}
|
|
#map-style-controls ul.hidden {
|
|
display: none;
|
|
}
|
|
#map-style-controls label {
|
|
display: block;
|
|
padding: 0.5rem;
|
|
font-weight: bold;
|
|
}
|
|
#map-style-controls input[type=radio] {
|
|
padding: 0;
|
|
margin: 0;
|
|
border: 0;
|
|
margin-right: 0.5rem;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<main>
|
|
<div id="map"></div>
|
|
<div class="map-overlay">
|
|
<h2>Map Style</h2>
|
|
<div id="map-style-controls"><p>Loading available styles...</p></div>
|
|
</div>
|
|
</main>
|
|
<script>
|
|
const coordsToHilbert = ({ x, y }) => {
|
|
let rotation = 0
|
|
let reflection = 0
|
|
let index = 0
|
|
for (let b = 15; b >= 0; b--) {
|
|
let bits = reflection
|
|
reflection = (y >>> b) & 1
|
|
reflection |= ((x >>> b) & 1) << 1
|
|
bits = bits ^ reflection
|
|
bits = ((bits >>> rotation) | (bits << (2 - rotation))) & 3
|
|
index |= bits << (b << 1)
|
|
reflection ^= (1 << rotation)
|
|
bits = bits & (-bits) & 1
|
|
while (bits) {
|
|
++rotation
|
|
bits >>>= 1
|
|
}
|
|
if (++rotation >= 2)
|
|
rotation -= 2
|
|
}
|
|
index ^= 0x2aaaaaaa
|
|
for (let d = 1; d < 32; d *= 2) {
|
|
let t = index >>> d
|
|
if (!t) break
|
|
index ^= t
|
|
}
|
|
return index
|
|
}
|
|
|
|
const hilbertToCoords = index => {
|
|
let rotation = 0
|
|
let reflection = 0
|
|
let coord = { x: 0, y: 0 }
|
|
index ^= (index >>> 1) ^ 0x2aaaaaaa
|
|
for (let b = 15; b >= 0; b--) {
|
|
var bits = index >>> (2 * b) & 3
|
|
reflection ^= ((bits >>> (2 - rotation)) | (bits << rotation)) & 3
|
|
coord.x |= (reflection & 1) << b
|
|
coord.y |= ((reflection >>> 1) & 1) << b
|
|
reflection ^= (1 << rotation)
|
|
bits &= (-bits) & 1
|
|
while (bits) {
|
|
bits >>>= 1
|
|
++rotation
|
|
}
|
|
if (++rotation >= 2)
|
|
rotation -= 2
|
|
}
|
|
return coord
|
|
}
|
|
|
|
const tilesDir = "tiles"
|
|
const sourceId = "ipmap-tiles-source"
|
|
const styleControlsDiv = document.getElementById("map-style-controls")
|
|
|
|
const dateDir = (date = new Date()) => `${date.getFullYear()}-${(date.getMonth() + 1).toString().padStart(2, "0")}-${date.getDate().toString().padStart(2, "0")}`
|
|
const getId = (date, variant, colormap) => `${date.replaceAll("-", "")}-${variant}-${colormap}`
|
|
const getTilesUrl = (date, variant, colormap) => `${tilesDir}/${date}/${variant}/${colormap}/{z}/{y}/{x}.png`
|
|
const getSourceId = (date, variant, colormap) => `ipmap-tiles-source-${getId(date, variant, colormap)}`
|
|
const getLayerId = (date, variant, colormap) => `ipmap-tiles-layer-${getId(date, variant, colormap)}`
|
|
|
|
const map = new maplibregl.Map({
|
|
container: "map",
|
|
attributionControl: false,
|
|
renderWorldCopies: false,
|
|
doubleClickZoom: false,
|
|
dragRotate: false,
|
|
style: {
|
|
version: 8,
|
|
sources: {},
|
|
layers: []
|
|
},
|
|
center: [0, 0],
|
|
minZoom: -1,
|
|
maxZoom: 12,
|
|
zoom: 0
|
|
})
|
|
map.painter.context.extTextureFilterAnisotropic = undefined
|
|
|
|
const dataP = fetch(`${tilesDir}/tiles.json`).then(res => res.json())
|
|
|
|
map.once("style.load", async () => {
|
|
const data = await dataP
|
|
const flatData = Object.entries(data)
|
|
.sort(([a], [b]) => a.localeCompare(b))
|
|
.flatMap(([date, variantData]) =>
|
|
Object.entries(variantData)
|
|
.sort(([a], [b]) => a.localeCompare(b))
|
|
.flatMap(([variant, colormaps]) => colormaps
|
|
.sort((a, b) => a.localeCompare(b))
|
|
.flatMap(colormap => ({ date, variant, colormap }))))
|
|
if (flatData.length === 0) {
|
|
console.log("no data found")
|
|
return
|
|
}
|
|
|
|
let { date: curDate, variant: curVariant, colormap: curColormap } = flatData[flatData.length - 1]
|
|
|
|
map.addSource(sourceId, {
|
|
type: "raster",
|
|
tiles: [getTilesUrl(curDate, curVariant, curColormap)],
|
|
tileSize: 256,
|
|
minzoom: 0,
|
|
maxzoom: 8,
|
|
})
|
|
map.addLayer({
|
|
id: "ipmap-tiles-layer",
|
|
type: "raster",
|
|
source: sourceId,
|
|
paint: {
|
|
"raster-resampling": "nearest"
|
|
}
|
|
})
|
|
|
|
const setStyle = (date, variant, colormap) => {
|
|
if (date === curDate && variant === curVariant && colormap === curColormap || !data[date]?.[variant]?.includes(colormap))
|
|
return
|
|
map.getSource(sourceId)?.setTiles([getTilesUrl(date, variant, colormap)])
|
|
curDate = date
|
|
curVariant = variant
|
|
curColormap = colormap
|
|
}
|
|
|
|
const dateList = document.createElement("ul")
|
|
for (const [date, variantData] of Object.entries(data).sort(([a], [b]) => a.localeCompare(b))) {
|
|
const isCurDate = date === curDate
|
|
const dateInput = document.createElement("input")
|
|
dateInput.type = "radio"
|
|
dateInput.name = "date"
|
|
dateInput.value = date
|
|
dateInput.checked = isCurDate
|
|
const dateLabel = document.createElement("label")
|
|
dateLabel.appendChild(dateInput)
|
|
dateLabel.appendChild(document.createTextNode(date))
|
|
const dateItem = document.createElement("li")
|
|
dateItem.appendChild(dateLabel)
|
|
const variantList = document.createElement("ul")
|
|
if (!isCurDate) variantList.className = "hidden"
|
|
for (const [variant, colormaps] of Object.entries(variantData).sort(([a], [b]) => a.localeCompare(b))) {
|
|
const isCurVariant = variant === curVariant
|
|
const variantInput = document.createElement("input")
|
|
variantInput.type = "radio"
|
|
variantInput.name = `${date}-variant`
|
|
variantInput.value = variant
|
|
variantInput.checked = isCurVariant
|
|
const variantLabel = document.createElement("label")
|
|
variantLabel.appendChild(variantInput)
|
|
variantLabel.appendChild(document.createTextNode(variant))
|
|
const variantItem = document.createElement("li")
|
|
variantItem.appendChild(variantLabel)
|
|
const colormapList = document.createElement("ul")
|
|
if (!isCurVariant) colormapList.classList.add("hidden")
|
|
for (const colormap of colormaps.sort((a, b) => a.localeCompare(b))) {
|
|
const isCurColormap = colormap === curColormap
|
|
const colormapInput = document.createElement("input")
|
|
colormapInput.type = "radio"
|
|
colormapInput.name = `${date}-${variant}-colormap`
|
|
colormapInput.value = colormap
|
|
colormapInput.checked = isCurColormap
|
|
const colormapLabel = document.createElement("label")
|
|
colormapLabel.appendChild(colormapInput)
|
|
colormapLabel.appendChild(document.createTextNode(colormap))
|
|
const colormapItem = document.createElement("li")
|
|
colormapItem.appendChild(colormapLabel)
|
|
colormapItem.addEventListener("click", e => {
|
|
;[...dateList.children].forEach(el =>
|
|
[...el.lastChild.children].forEach(el =>
|
|
[...el.lastChild.children].forEach(el => {
|
|
const cb = el.firstChild.firstChild
|
|
cb.checked = cb.value === colormap
|
|
})))
|
|
setStyle(curDate, curVariant, colormap)
|
|
})
|
|
colormapList.appendChild(colormapItem)
|
|
}
|
|
variantInput.addEventListener("click", e => {
|
|
;[...dateList.children].forEach(el =>
|
|
[...el.lastChild.children].forEach(el => {
|
|
const cb = el.firstChild.firstChild
|
|
const isCur = cb.value === variant
|
|
el.lastChild.className = isCur ? "" : "hidden"
|
|
cb.checked = isCur
|
|
}))
|
|
setStyle(curDate, variant, curColormap)
|
|
})
|
|
variantItem.appendChild(colormapList)
|
|
variantList.appendChild(variantItem)
|
|
}
|
|
dateInput.addEventListener("click", e => {
|
|
;[...dateList.children].forEach(el => {
|
|
const cb = el.firstChild.firstChild
|
|
const isCur = cb.value === date
|
|
el.lastChild.className = isCur ? "" : "hidden"
|
|
})
|
|
setStyle(date, curVariant, curColormap)
|
|
})
|
|
dateItem.appendChild(variantList)
|
|
dateList.appendChild(dateItem)
|
|
}
|
|
styleControlsDiv.replaceChildren(dateList)
|
|
})
|
|
|
|
map.addControl(new maplibregl.NavigationControl({ showCompass: false }), "top-left")
|
|
const toIp = v => `${v >> 24 & 0xFF}.${v >> 16 & 0xFF}.${v >> 8 & 0xFF}.${v >> 0 & 0xFF}`
|
|
map.on("click", (e) => {
|
|
const { x, y } = maplibregl.MercatorCoordinate.fromLngLat(e.lngLat, 0)
|
|
const rawIp = coordsToHilbert({ x: Math.floor(0x10000 * x), y: Math.floor(0x10000 * y) })
|
|
const subnet = Math.min(32, Math.round(map.getZoom()) * 2 + 18)
|
|
const text = subnet < 32 ?
|
|
`Range: ${toIp((rawIp >> (32 - subnet)) << (32 - subnet))}/${subnet}` :
|
|
`IP: ${toIp(rawIp)}`
|
|
new maplibregl.Popup()
|
|
.setHTML(text)
|
|
.setLngLat(e.lngLat)
|
|
.addTo(map)
|
|
})
|
|
</script>
|
|
</body>
|
|
</html>
|