// ==UserScript==
// @name Internet Roadtrip Map History
// @namespace zptr.cc
// @author zeroptr
// @match https://neal.fun/internet-roadtrip/*
// @version 0.1.1
// @license MIT
// @description Shows more history on the Internet Roadtrip minimap (such as past honks)
// @icon https://neal.fun/favicons/internet-roadtrip.png
// @grant unsafeWindow
// @run-at document-end
// @require https://cdn.jsdelivr.net/npm/[email protected]
// ==/UserScript==
(async() => {
const lib = await IRF.modules.maplibre;
const map = (await IRF.vdom.map).data.map;
const data = {
type: "FeatureCollection",
features: []
};
async function loadPastMarkers() {
const req = await fetch(
"https://roadtrip.zptr.cc/data/honks"
);
const markers = await req.json();
markers.sort((a, b) => a.ts - b.ts);
for (const marker of markers) {
addMarker(marker.ts, marker.lat, marker.lon, marker.count || 1, marker.players);
}
}
function addMarker(time, lat, lon, count=1, players=0) {
data.features.push({
type: "Feature",
properties: { count, time, players },
geometry: {
type: "Point",
coordinates: [lon, lat]
}
});
}
async function initEvents() {
const container = await IRF.vdom.container;
const fn = container.state.changeStop;
container.state.changeStop = function (_, e) {
if (e == -2) {
const { lat, lng } = container.data.currentCoords;
addMarker(Math.floor(Date.now() / 1000), lat, lng, 1, container.data.totalUsers);
map.getSource("honks").setData(data);
}
fn.apply(this, arguments);
};
}
initEvents();
await loadPastMarkers();
const img = await map.loadImage("");
map.addImage("honk-image", img.data);
map.addSource("honks", {
type: "geojson",
data,
cluster: true,
clusterMaxZoom: 8,
clusterRadius: 5,
clusterMinPoints: 10,
clusterProperties: {
"sum": ["+", ["get", "count"]]
}
});
map.addLayer({
id: "points",
type: "symbol",
source: "honks",
filter: [">", ["zoom"], 5],
layout: {
"icon-image": "honk-image",
"icon-size": 0.5
},
});
let openedPopup;
map.on("mouseenter", "points", (e) => {
const { count, time, players } = e.features[0].properties;
const date = new Date(time * 1000);
if (openedPopup) {
openedPopup.remove();
}
openedPopup = new lib.Popup()
.setLngLat(e.features[0].geometry.coordinates.slice())
.setHTML(`${date.toLocaleString()}<br>${count} honk${count == 1 ? "" : "s"} | ${players} players`)
.addTo(map);
});
map.on("mouseleave", "points", () => {
if (openedPopup) {
openedPopup.remove();
openedPopup = null;
}
});
setTimeout(() => {
map.moveLayer("points");
}, 1000);
})();