Internet Roadtrip Map History

Shows more history on the Internet Roadtrip minimap (such as past honks)

// ==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);
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址