Internet Roadtrip Pathfinder

A somewhat reliable pathfinder for neal.fun's Internet Roadtrip.

// ==UserScript==
// @name        Internet Roadtrip Pathfinder
// @namespace   internet-roadtrip-pathfinder
// @match       https://neal.fun/*
// @version     1.7.0
// @author      mat
// @description A somewhat reliable pathfinder for neal.fun's Internet Roadtrip.
// @license     0BSD
// @run-at      document-start
// @resource    flagCheckerboardPng 
// @resource    flagSvg 
// @resource    flagWithCrossSvg 
// @require     https://cdn.jsdelivr.net/npm/[email protected]
// @grant       GM_addStyle
// @grant       GM_getResourceURL
// @grant       GM_getValue
// @grant       GM_setValue
// @grant       unsafeWindow
// ==/UserScript==

var Pathfinder = (function (exports) {
    'use strict';

    // geojson uses [lng,lat] and we mostly do the same, so we handle all of that here to avoid mistakes
    function getLat(pos) {
        return pos[1];
    }
    function getLng(pos) {
        return pos[0];
    }
    function newPosition(lat, lng) {
        return [lng, lat];
    }
    /**
     * @param coords formatted like "lat,lng"
     * @returns [lng, lat]
     */
    function parseCoordinatesString(coords) {
        if (!coords.includes(","))
            return null;
        const [lat, lng] = coords.split(",").map(Number);
        if (lat === undefined || lng === undefined)
            return null;
        return newPosition(lat, lng);
    }

    const LOG_PREFIX = "[Pathfinder]";

    let map;
    async function initMap() {
        map = await IRF.vdom.map.then((mapVDOM) => mapVDOM.state.map);
    }

    /**
     * in meters, from Google Maps
     */
    const EARTH_RADIUS = 6_378_137;
    /**
     * @param origin lat, lng
     * @param dest lat, lng
     * @returns in meters
     */
    function calculateDistance(origin, dest) {
        const aLat = getLat(origin);
        const bLat = getLat(dest);
        const aLng = getLng(origin);
        const bLng = getLng(dest);
        const theta1 = toRadians(aLat);
        const theta2 = toRadians(bLat);
        const deltaTheta = toRadians(bLat - aLat);
        const deltaLambda = toRadians(bLng - aLng);
        const a = Math.pow(Math.sin(deltaTheta / 2), 2) +
            Math.cos(theta1) *
                Math.cos(theta2) *
                Math.pow(Math.sin(deltaLambda / 2), 2);
        const c = 2 * Math.asin(Math.sqrt(a));
        return EARTH_RADIUS * c;
    }
    /**
     * @param origin lat, lng
     * @param dest lat, lng
     * @returns in degrees
     */
    function calculateHeading(origin, dest) {
        const [aLng, aLat] = [toRadians(getLng(origin)), toRadians(getLat(origin))];
        const [bLng, bLat] = [toRadians(getLng(dest)), toRadians(getLat(dest))];
        const deltaLng = bLng - aLng;
        const [aLatSin, aLatCos] = [Math.sin(aLat), Math.cos(aLat)];
        const [bLatSin, bLatCos] = [Math.sin(bLat), Math.cos(bLat)];
        const [deltaLngSin, deltaLngCos] = [Math.sin(deltaLng), Math.cos(deltaLng)];
        const s = deltaLngSin * bLatCos;
        const c = aLatCos * bLatSin - aLatSin * bLatCos * deltaLngCos;
        return (toDegrees(Math.atan2(s, c)) + 360) % 360;
    }
    /**
     *
     * @param a in degrees
     * @param in degrees
     * @returns in degrees, between 0 and 360
     */
    function calculateHeadingDiff(a, b) {
        a = (a + 360) % 360;
        b = (b + 360) % 360;
        let diff = Math.abs(a - b);
        if (diff > 180) {
            diff = 360 - diff;
        }
        return diff;
    }
    function toRadians(degrees) {
        return degrees * (Math.PI / 180);
    }
    function toDegrees(radians) {
        return radians * (180 / Math.PI);
    }

    /**
     * Code related to picking the best option and finding panoramas in the pathfinder's path.
     */
    function showBestOption() {
        if (!exports.currentData) {
            throw Error("called showBestOption when currentData was still null");
        }
        const currentPos = newPosition(exports.currentData.lat, exports.currentData.lng);
        const firstPath = getFirstPath();
        if (firstPath.length < panosAdvancedInFirstPath + 2)
            return;
        const bestNextPos = firstPath[panosAdvancedInFirstPath + 1];
        const bestHeading = calculateHeading(currentPos, bestNextPos);
        console.debug(LOG_PREFIX, "option bestHeading", bestHeading);
        let bestOptionIndex = -1;
        let bestOptionHeadingDiff = Infinity;
        const options = exports.currentData.options;
        // first, check only the option that have lat+lng present (since those are more reliable)
        for (let optionIndex = 0; optionIndex < options.length; optionIndex++) {
            const option = options[optionIndex];
            if (!option.lat || !option.lng)
                continue;
            const optionPos = newPosition(option.lat, option.lng);
            const firstPathSliced = firstPath.slice(panosAdvancedInFirstPath, panosAdvancedInFirstPath + 2);
            const [matchingPanoInPathIndex, matchingPanoInPathDistance] = findClosestPanoInPath(optionPos, firstPathSliced);
            console.debug(LOG_PREFIX, "option with lat+lng", firstPathSliced[matchingPanoInPathIndex], matchingPanoInPathDistance);
            // heading diff and distance in meters aren't really comparable, but if a pano had a distance
            // of less than 1m then it's almost guaranteed to be the one we want anyways.
            if (matchingPanoInPathDistance < 1 &&
                matchingPanoInPathDistance < bestOptionHeadingDiff) {
                bestOptionIndex = optionIndex;
                bestOptionHeadingDiff = matchingPanoInPathDistance;
            }
        }
        // if nothing was found from the lat+lng check, do the less reliable heading check instead
        if (bestOptionIndex < 0) {
            for (let optionIndex = 0; optionIndex < options.length; optionIndex++) {
                const option = options[optionIndex];
                const optionHeading = option.heading;
                const optionHeadingDiff = calculateHeadingDiff(optionHeading, bestHeading);
                if (optionHeadingDiff < bestOptionHeadingDiff) {
                    bestOptionIndex = optionIndex;
                    bestOptionHeadingDiff = optionHeadingDiff;
                }
            }
        }
        if (bestOptionHeadingDiff > 100) {
            console.warn(LOG_PREFIX, "all of the options are bad!");
        }
        else {
            console.debug(LOG_PREFIX, "best option is", options[bestOptionIndex], `(diff: ${bestOptionHeadingDiff})`);
            highlightOptionIndex(bestOptionIndex);
        }
    }
    function highlightOptionIndex(optionIndex) {
        const optionArrowEls = Array.from(document.querySelectorAll(".option"));
        for (let i = 0; i < optionArrowEls.length; i++) {
            const optionArrowEl = optionArrowEls[i];
            if (i === optionIndex)
                optionArrowEl.classList.add("pathfinder-chosen-option");
            else
                optionArrowEl.classList.remove("pathfinder-chosen-option");
        }
    }
    function clearOptionHighlights() {
        for (const optionArrowEl of document.querySelectorAll(".option")) {
            optionArrowEl.classList.remove("pathfinder-chosen-option");
        }
    }
    /**
     * @returns The index of the closest pano in the path, and its distance.
     * Also, panos after the first one with a heading difference greater than 100 degrees are ignored.
     */
    function findClosestPanoInPath(targetPos, path) {
        let closestPanoInFirstPathIndex = -1;
        let closestPanoInFirstPathDistance = Infinity;
        for (let i = 0; i < path.length; i++) {
            const candidatePos = path[i];
            const distanceToCur = calculateDistance(candidatePos, targetPos);
            if (i > 0 && exports.currentData !== null) {
                // heading check
                const prevPos = path[i - 1];
                const candidateHeading = calculateHeading(prevPos, candidatePos);
                const headingDiff = calculateHeadingDiff(exports.currentData.heading, candidateHeading);
                if (headingDiff > 100) {
                    console.debug(LOG_PREFIX, "skipping due to heading diff:", headingDiff);
                    continue;
                }
            }
            if (distanceToCur < closestPanoInFirstPathDistance) {
                closestPanoInFirstPathIndex = i;
                closestPanoInFirstPathDistance = distanceToCur;
            }
        }
        return [closestPanoInFirstPathIndex, closestPanoInFirstPathDistance];
    }

    const DEFAULT_SETTINGS = {
        current_searching_path: true,
        allow_long_jumps: true,
        remove_reached_stops: false,
        // advanced settings
        use_option_cache: true,
        backend_url: "https://ir.matdoes.dev",
        heuristic_factor: 3.3,
        forward_penalty_on_intersections: 0,
    };
    const SETTINGS = JSON.parse(JSON.stringify(DEFAULT_SETTINGS));
    function loadSettingsFromSave() {
        const savedSettings = GM_getValue("settings") ?? "{}";
        for (const [k, v] of Object.entries(JSON.parse(savedSettings))) {
            // @ts-ignore assume that settings has the correct types
            SETTINGS[k] = v;
        }
    }
    function saveSettings() {
        GM_setValue("settings", JSON.stringify(SETTINGS));
    }
    function initSettingsTab() {
        loadSettingsFromSave();
        console.log(LOG_PREFIX, "loaded settings:", SETTINGS);
        const settingsTab = IRF.ui.panel.createTabFor(GM.info, {
            tabName: "Pathfinder",
            style: `
        .pathfinder-settings-tab-content {
            *, *::before, *::after {
                box-sizing: border-box;
            }

            .field-group {
                margin-block: 1rem;
            }
            .field-group-right-aligned {
                float: right;
                display: flex;
            }
            button {
                margin-left: 0.5rem;
            }
            i {
                margin-top: 0;
            }
            h2 {
                margin-bottom: 0;
            }
        }
        `,
            className: "pathfinder-settings-tab-content",
        });
        function addSetting(opts) {
            const id = `pathfinder-${opts.key}`;
            opts.inputEl.id = id;
            function setDisplayedValue(v) {
                if (typeof v === "boolean")
                    opts.inputEl.checked = v;
                else
                    opts.inputEl.value = v.toString();
            }
            setDisplayedValue(SETTINGS[opts.key]);
            opts.inputEl.addEventListener("change", (e) => {
                let newValue;
                const settingType = typeof DEFAULT_SETTINGS[opts.key];
                if (settingType === "boolean")
                    newValue = opts.inputEl.checked;
                else if (settingType === "number")
                    newValue = Number(opts.inputEl.value);
                else
                    newValue = opts.inputEl.value;
                SETTINGS[opts.key] = newValue;
                updateLabel();
                saveSettings();
                opts.cb?.(newValue);
            });
            opts.inputEl.addEventListener("input", (e) => {
                updateLabel(opts.inputEl.value);
            });
            const labelEl = document.createElement("label");
            labelEl.htmlFor = id;
            function updateLabel(shownValue) {
                const newLabelText = typeof opts.label === "string"
                    ? opts.label
                    : opts.label(shownValue ?? SETTINGS[opts.key]);
                labelEl.textContent = newLabelText;
            }
            updateLabel();
            const fieldGroupEl = document.createElement("div");
            fieldGroupEl.classList.add("field-group");
            fieldGroupEl.append(labelEl);
            const rightAlignedContentEl = document.createElement("div");
            rightAlignedContentEl.classList.add("field-group-right-aligned");
            if (!opts.isInputSeparate)
                rightAlignedContentEl.append(opts.inputEl);
            if (opts.hasResetBtn) {
                const resetBtnEl = document.createElement("button");
                resetBtnEl.textContent = "Reset";
                rightAlignedContentEl.append(resetBtnEl);
                resetBtnEl.addEventListener("click", () => {
                    const newValue = DEFAULT_SETTINGS[opts.key];
                    SETTINGS[opts.key] = newValue;
                    setDisplayedValue(newValue);
                    updateLabel();
                    saveSettings();
                    opts.cb?.(newValue);
                });
            }
            fieldGroupEl.append(rightAlignedContentEl);
            if (opts.isInputSeparate)
                fieldGroupEl.append(opts.inputEl);
            settingsTab.container.appendChild(fieldGroupEl);
        }
        function addToggle(label, key, cb) {
            const inputEl = document.createElement("input");
            inputEl.type = "checkbox";
            inputEl.classList.add(IRF.ui.panel.styles.toggle);
            addSetting({ inputEl, label, key, hasResetBtn: true, cb });
        }
        function addTextInput(label, key, cb) {
            const inputEl = document.createElement("input");
            addSetting({ inputEl, label, key, hasResetBtn: true, cb });
        }
        function addSlider(label, key, min, max, cb) {
            const inputEl = document.createElement("input");
            inputEl.type = "range";
            inputEl.classList.add(IRF.ui.panel.styles.slider);
            inputEl.min = min.toString();
            inputEl.max = max.toString();
            if (max - min < 10)
                inputEl.step = "0.01";
            function getLabel(label, value) {
                return `${label}: ${value}`;
            }
            addSetting({
                inputEl,
                label: (value) => getLabel(label, value),
                key,
                hasResetBtn: true,
                isInputSeparate: true,
                cb: (value) => {
                    cb?.(value);
                },
            });
        }
        addToggle("Show currently searching path", "current_searching_path", () => {
            rerenderPath("current_searching_path");
        });
        addToggle("Allow long jumps", "allow_long_jumps", () => {
            clearCachedPaths();
            refreshPath();
        });
        addToggle("Remove stops as they are reached", "remove_reached_stops");
        const advancedSettingsHeaderEl = document.createElement("h2");
        advancedSettingsHeaderEl.textContent = "Advanced settings";
        const advancedSettingsDescEl = document.createElement("i");
        advancedSettingsDescEl.textContent =
            "NOTE: These settings can make the pathfinder stop working, mess up your ETAs, and significantly hurt performance. You should reset them if something breaks.";
        settingsTab.container.append(advancedSettingsHeaderEl, advancedSettingsDescEl);
        addTextInput("Custom backend URL", "backend_url", () => {
            pfWs.close();
            clearCachedPaths();
            refreshPath();
        });
        addToggle("Use option cache", "use_option_cache", () => {
            clearCachedPaths();
            refreshPath();
        });
        addSlider("Heuristic factor", "heuristic_factor", 1, 4, () => {
            clearCachedPaths();
            refreshPath();
        });
        addSlider("Forward penalty on intersections (in seconds)", "forward_penalty_on_intersections", 0, 600, () => {
            clearCachedPaths();
            refreshPath();
        });
    }

    let tricksControl = undefined;
    /**
     * Tries to initialize the Minimap Tricks integration, if possible.
     */
    function tryInitMmt() {
        Promise.all([waitForMmtControl, waitForMmtAddContextFn]).then(([newTricksControl, addContext]) => {
            tricksControl = newTricksControl;
            onMmtFound(addContext);
        });
    }
    /**
     * Called when the Minimap Tricks userscript is found.
     */
    async function onMmtFound(addContext) {
        if (!tricksControl) {
            throw Error("tricksControl must be set");
        }
        document.body.classList.add("pathfinder-found-minimap-tricks");
        function setAndSaveDestination(pos) {
            updateDestinationFromString(`${getLat(pos)},${getLng(pos)}`);
        }
        function clearAndSaveDestination() {
            updateDestinationFromString("");
        }
        // Map button
        tricksControl.addButton(GM_getResourceURL("flagSvg"), "Find path to location", (control) => setAndSaveDestination(newPosition(control.lat, control.lng)), 
        // contexts
        ["Map"]);
        // Add stop button
        const addStopBtn = tricksControl.addButton(GM_getResourceURL("flagSvg"), "Add stop to path", (control) => {
            addStopToPath(newPosition(control.lat, control.lng));
        }, 
        // contexts
        ["Map", "Marker", "Pathfinder"]);
        addStopBtn.context_button.classList.add("pathfinder-add-stop-mmt-context-menu-button");
        // Marker button
        tricksControl.addButton(GM_getResourceURL("flagSvg"), "Set as pathfinder destination", (control) => setAndSaveDestination(newPosition(control.lat, control.lng)), 
        // contexts
        ["Marker"]);
        // Remove buttons
        const removePathBtn = tricksControl.addButton(GM_getResourceURL("flagWithCrossSvg"), "Clear found path", () => clearAndSaveDestination(), 
        // contexts
        ["Side", "Map", "Car", "Pathfinder", "Pathfinder destination"]);
        removePathBtn.side_button.classList.add("pathfinder-clear-path-mmt-side-button");
        removePathBtn.context_button.classList.add("pathfinder-clear-path-mmt-context-menu-button");
        tricksControl.addButton(GM_getResourceURL("flagWithCrossSvg"), "Remove stop", (control) => {
            removeStop(newPosition(control.lat, control.lng));
        }, 
        // contexts
        ["Pathfinder stop"]);
        addContext("Pathfinder", [
            // New buttons
            "Find path to location",
            "Add stop to path",
            "Clear found path",
            // Grandfathered buttons from Minimap Tricks
            "Copy coordinates",
            "Add marker",
        ]);
        addContext("Pathfinder destination", [
            "Clear found path",
            "Copy coordinates",
            "Add marker",
        ]);
        addContext("Pathfinder stop", [
            "Remove stop",
            "Copy coordinates",
            "Add marker",
        ]);
        map.on("contextmenu", "best_path", (event) => {
            event.preventDefault();
            openContextMenu(event.originalEvent, event.lngLat, "Pathfinder");
        });
        map.on("contextmenu", "best_path_segments", (event) => {
            event.preventDefault();
            openContextMenu(event.originalEvent, event.lngLat, "Pathfinder");
        });
        setNewInfoDisplay();
    }
    function setNewInfoDisplay() {
        const pathfinderInfoControl = new (class {
            _map;
            _container;
            onAdd(map) {
                this._map = map;
                this._container = this.insertDom();
                return this._container;
            }
            insertDom() {
                const containerEl = document.createElement("div");
                containerEl.id = "minimap-controls";
                containerEl.className =
                    "maplibregl-ctrl maplibregl-ctrl-scale pathfinder-info";
                containerEl.style.marginRight = "36px";
                return containerEl;
            }
            onRemove() {
                if (this._container) {
                    this._container.parentNode?.removeChild(this._container);
                }
                this._map = undefined;
            }
        })();
        map.addControl(pathfinderInfoControl, "bottom-right");
        replacePathfinderInfoEl(pathfinderInfoControl._container);
    }
    const waitForMmtControl = new Promise((resolve) => {
        if (unsafeWindow._MMT_control) {
            resolve(unsafeWindow._MMT_control);
            return;
        }
        let _tricksControl;
        Object.defineProperty(unsafeWindow, "_MMT_control", {
            get() {
                return _tricksControl;
            },
            set(tricksControl) {
                _tricksControl = tricksControl;
                resolve(tricksControl);
            },
            configurable: true,
            enumerable: true,
        });
    });
    const waitForMmtAddContextFn = new Promise((resolve) => {
        if (unsafeWindow._MMT_addContext) {
            resolve(unsafeWindow._MMT_addContext);
            return;
        }
        let _contexts;
        Object.defineProperty(unsafeWindow, "_MMT_addContext", {
            get() {
                return _contexts;
            },
            set(contexts) {
                _contexts = contexts;
                resolve(contexts);
            },
            configurable: true,
            enumerable: true,
        });
    });
    function addMarkerContextMenuListener(marker, contextName) {
        marker.getElement().addEventListener("contextmenu", (event) => {
            openContextMenu(event, marker.getLngLat(), contextName);
        });
    }
    function openContextMenu(event, pos, contextName) {
        if (!tricksControl)
            return;
        event.stopPropagation();
        event.preventDefault();
        const data = {};
        tricksControl.openMenu(contextName, pos.lat, pos.lng, event.clientX, event.clientY, data);
    }

    let destinationMarker;
    /**
     * should be the same length as the number of stops
     */
    let stopMarkers = [];
    async function initMarkers() {
        const maplibre = await IRF.modules.maplibre;
        destinationMarker = new maplibre.Marker({
            element: (() => {
                const imgEl = document.createElement("img");
                imgEl.className = "pathfinder-destination-marker";
                imgEl.src = GM_getResourceURL("flagCheckerboardPng");
                return imgEl;
            })(),
            anchor: "bottom-left",
        });
        addMarkerContextMenuListener(destinationMarker, "Pathfinder destination");
        rerenderStopMarkers();
    }
    function updateDestinationMarker(position) {
        if (!position) {
            // if no coords are passed then the point is removed
            destinationMarker.remove();
            return;
        }
        destinationMarker
            .setLngLat([getLng(position), getLat(position)])
            .addTo(map);
    }
    async function newStopMarker() {
        const maplibre = await IRF.modules.maplibre;
        const marker = new maplibre.Marker({
            element: (() => {
                const imgEl = document.createElement("img");
                imgEl.className = "pathfinder-stop-marker";
                imgEl.src = GM_getResourceURL("flagCheckerboardPng");
                return imgEl;
            })(),
            anchor: "bottom-left",
        });
        addMarkerContextMenuListener(marker, "Pathfinder stop");
        return marker;
    }
    async function rerenderStopMarkers() {
        const unorderedStops = getUnorderedStops();
        while (stopMarkers.length > unorderedStops.length) {
            stopMarkers.pop().remove();
        }
        while (unorderedStops.length > stopMarkers.length) {
            const stopMarker = await newStopMarker();
            // coordinates are required, but they'll get updated in a moment
            stopMarker.setLngLat([0, 0]).addTo(map);
            stopMarkers.push(stopMarker);
        }
        // stopMarkers and unorderedStops are now the same length, now update the lat/lng for all of them
        for (let i = 0; i < stopMarkers.length; i++) {
            const marker = stopMarkers[i];
            const stopPos = unorderedStops[i];
            marker.setLngLat([getLng(stopPos), getLat(stopPos)]);
        }
    }

    /**
     * Returns the list of destinations for the path we're following. Usually this will just be the one
     * destination, but can include more if the path has stops in it.
     */
    function getPathDestinations() {
        const remainingStops = getUnorderedStops();
        const lastDestination = parseCoordinatesString(getDestinationString());
        if (!exports.currentData || !lastDestination)
            return [];
        const stops = [];
        // find the closest stop in remainingStops
        let currentPosition = newPosition(exports.currentData.lat, exports.currentData.lng);
        while (remainingStops.length > 0) {
            let closestIndex = -1;
            let closestDistance = Number.POSITIVE_INFINITY;
            for (const [candidateIndex, candidate] of remainingStops.entries()) {
                const candidateDistance = calculateDistance(currentPosition, candidate);
                if (candidateDistance < closestDistance) {
                    closestIndex = candidateIndex;
                    closestDistance = candidateDistance;
                }
            }
            const closestStop = remainingStops[closestIndex];
            if (!closestStop) {
                throw Error(`closestStop should\'ve been set. closestIndex: ${closestIndex}, remainingStops: ${remainingStops}`);
            }
            stops.push(closestStop);
            remainingStops.splice(closestIndex, 1);
        }
        stops.push(lastDestination);
        return stops;
    }
    function addStopToPath(pos) {
        const currentStops = getUnorderedStops();
        if (currentStops.find((s) => s[0] === pos[0] && s[1] === pos[1])) {
            // stop is already present
            return;
        }
        currentStops.push(pos);
        setStops(currentStops);
        rerenderCompletePathSegments();
        rerenderStopMarkers();
        refreshPath();
    }
    function removeStop(pos) {
        const oldStops = getUnorderedStops();
        const newStops = oldStops.filter((s) => s[0] !== pos[0] || s[1] !== pos[1]);
        if (newStops.length === oldStops.length) {
            console.warn(LOG_PREFIX, "failed to remove stop at", pos, "currentStops:", oldStops);
            return false;
        }
        setStops(newStops);
        rerenderCompletePathSegments();
        rerenderStopMarkers();
        refreshPath();
        return true;
    }
    /**
     * @returns {GeoJSON.Position[]} array of [lng, lat]
     */
    function getUnorderedStops() {
        return JSON.parse(GM_getValue("stops") || "[]");
    }
    function setStops(stops) {
        GM_setValue("stops", JSON.stringify(stops));
    }

    const sleep = (ms) => new Promise((res) => setTimeout(res, ms));
    function prettyTime(seconds) {
        if (seconds < 0)
            return "now";
        const hours = Math.floor(seconds / 3600);
        const minutes = Math.floor((seconds % 3600) / 60);
        const secondsLeft = Math.floor(seconds % 60);
        const msLeft = Math.floor((seconds * 1000) % 1000);
        if (hours > 0) {
            return `${hours}h ${minutes}m ${secondsLeft}s`;
        }
        else if (minutes > 0) {
            return `${minutes}m ${secondsLeft}s`;
        }
        else if (secondsLeft > 0) {
            return `${secondsLeft}s`;
        }
        else {
            return `${msLeft}ms`;
        }
    }

    /**
     * A map of path IDs to the best_paths that we calculated to completion.
     */
    const completePathSegments = new Map();
    /**
     * Path ID to its cost (which is the duration in seconds).
     */
    const pathSegmentCosts = new Map();
    const pathIdsToPathSegmentMetadata = new Map();
    /**
     * `lat,lng` to the path id, this includes destinations that aren't currently being used
     */
    const destinationToPathIdMap = new Map();
    /**
     * A map of path IDs to their expected destinations. It'll likely be a few meters from the actual destination.
     */
    const pathIdToDestination = new Map();
    const calculatingPaths = new Map();
    async function setupPathSources() {
        console.debug(LOG_PREFIX, "waiting for old-route to render");
        let waitedCount = 0;
        // alternatively just wait 2 seconds, in case internet-roadtrip.neal.fun/route is broken
        while (map.getSource("old-route") === undefined && waitedCount < 20) {
            await sleep(100);
            waitedCount += 1;
        }
        console.debug(LOG_PREFIX, "setting up path sources");
        setupPathSource("current_searching_path", "#00f");
        setupPathSource("best_path", "#f0f");
        setupPathSource("best_path_segments", "#f0f");
    }
    function setupPathSource(pathSourceId, color) {
        map.addSource(pathSourceId, {
            type: "geojson",
            data: {
                type: "Feature",
                properties: {},
                geometry: {
                    type: "LineString",
                    coordinates: [],
                },
            },
        });
        map.addLayer({
            id: pathSourceId,
            type: "line",
            source: pathSourceId,
            layout: {
                "line-join": "round",
                "line-cap": "round",
            },
            paint: {
                "line-color": color,
                "line-width": 4,
            },
        });
    }
    async function updatePathSource(pathSourceId, keepPrefixLength, append) {
        let curPath = calculatingPaths.get(pathSourceId) ?? [];
        curPath = curPath.slice(0, keepPrefixLength);
        curPath.push(...append);
        calculatingPaths.set(pathSourceId, curPath);
        rerenderPath(pathSourceId);
    }
    function updateCurrentLocation(curPos) {
        // check if the new location is near the front of our pathfinder's best path
        const firstPath = getFirstPath().slice(0, 10);
        const [closestPanoInBestPathIndex, closestPanoInBestPathDistance] = findClosestPanoInPath(curPos, firstPath);
        if (closestPanoInBestPathIndex === -1) {
            // this is usually fine, but can sometimes happen if we were stuck and got teleported out
            return false;
        }
        if (closestPanoInBestPathDistance > 20) {
            return false;
        }
        const i = closestPanoInBestPathIndex;
        panosAdvancedInFirstPath += i;
        updateLastCostRecalculationTimestamp();
        console.debug(LOG_PREFIX, "close enough, updating panosAdvancedInBestPath to", panosAdvancedInFirstPath);
        rerenderPath("best_path");
        rerenderPath("current_searching_path");
        rerenderCompletePathSegments();
        return true;
    }
    function getFirstPath() {
        const firstPathDest = getPathDestinations()[0];
        const firstPathId = convertDestinationToPathId(firstPathDest);
        if (firstPathId === undefined) {
            console.debug(LOG_PREFIX, "called updateCurrentLocation before the current path was requested");
            return [];
        }
        const unskipped = calculatingPathId === firstPathId
            ? calculatingPaths.get("best_path")
            : completePathSegments.get(firstPathId);
        return unskipped?.slice(panosAdvancedInFirstPath) ?? [];
    }
    let panosAdvancedInFirstPath = 0;
    function rerenderPath(pathSourceId) {
        let skip = 0;
        if (calculatingPathId !== undefined) {
            const pathDestination = pathIdsToPathSegmentMetadata.get(calculatingPathId).destination;
            const paths = getPathDestinations();
            const firstDestInPath = paths[0];
            if (firstDestInPath[0] === pathDestination[0] &&
                firstDestInPath[1] === pathDestination[1]) {
                // we've confirmed that this is the first path, so skip some panos
                skip = panosAdvancedInFirstPath;
            }
        }
        console.debug(LOG_PREFIX, "rendering", pathSourceId, "and skipping", skip);
        let path = calculatingPaths.get(pathSourceId)?.slice(skip) ?? [];
        // hide the current_path if the setting is checked
        // hide the current_searching_path if the setting is checked
        if (pathSourceId === "current_searching_path" &&
            !SETTINGS.current_searching_path) {
            path = [];
        }
        const pathSource = map.getSource(pathSourceId);
        pathSource?.setData({
            type: "Feature",
            properties: {},
            geometry: {
                type: "LineString",
                coordinates: path,
            },
        });
    }
    function rerenderCompletePathSegments() {
        const multiLines = [];
        for (const [index, stopDest] of getPathDestinations().entries()) {
            const pathId = convertDestinationToPathId(stopDest);
            if (pathId === undefined) {
                console.debug(LOG_PREFIX, "failed rendering path segment because it hasn't been requested yet:", destinationToPathIdMap, stopDest);
                continue;
            }
            let skip = 0;
            if (index === 0) {
                skip = panosAdvancedInFirstPath;
            }
            console.debug(LOG_PREFIX, "rendering best_path_segments and skipping", skip);
            const lines = completePathSegments.get(pathId)?.slice(skip);
            if (lines) {
                multiLines.push(lines);
            }
            else {
                console.warn(LOG_PREFIX, "stop destination", stopDest, "not present in completePathSegments. this probably just means that we haven't finished calculating it");
                break;
            }
        }
        const pathSource = map.getSource("best_path_segments");
        pathSource.setData({
            type: "Feature",
            properties: {},
            geometry: {
                type: "MultiLineString",
                coordinates: multiLines,
            },
        });
    }
    /**
     * Remove the pathfinder's lines from the map and forget their current state. You should probably
     * use `clearAllPaths` instead.
     */
    function resetRenderedPath() {
        panosAdvancedInFirstPath = 0;
        clearCalculatingPaths();
        rerenderCompletePathSegments();
        rerenderPath("best_path");
        rerenderPath("current_searching_path");
    }
    function updateCompletePathSegment(pathId, path, cost) {
        completePathSegments.set(pathId, path);
        pathSegmentCosts.set(pathId, cost);
        // no path is being calculated at this point anymore
        clearCalculatingPathId();
        // the path is already in completePathSegments, so remove it from
        // calculatingPaths to make it so we don't render the same path twice
        calculatingPaths.clear();
    }
    function convertDestinationToPathId(pos) {
        return destinationToPathIdMap.get(`${getLat(pos)},${getLng(pos)}`);
    }
    function clearCalculatingPaths() {
        if (getUnorderedStops().length === 0) {
            // persist these so we can avoid recalculating paths with many stops
            completePathSegments.clear();
            pathSegmentCosts.clear();
            pathIdsToPathSegmentMetadata.clear();
            destinationToPathIdMap.clear();
            pathIdToDestination.clear();
        }
        calculatingPaths.clear();
        clearCalculatingPathId();
    }
    function clearCachedPaths() {
        completePathSegments.clear();
        pathSegmentCosts.clear();
        pathIdsToPathSegmentMetadata.clear();
        destinationToPathIdMap.clear();
        pathIdToDestination.clear();
        calculatingPaths.clear();
        clearCalculatingPathId();
    }

    let pfWs;
    let queuedWebSocketMessages = [];
    let calculatingPathId = undefined;
    let nextPathId = 0;
    /**
     *
     * @param heading in degrees
     * @param start lng, lat
     * @param end lng, lat
     * @param startPano The ID of the current pano. If not passed, the start coords will get
     * snapped to the nearest pano instead.
     */
    function requestNewPath(heading, start, end, startPano) {
        const pathMetadata = {
            source: {
                pos: start,
                heading,
            },
            destination: end,
        };
        let alreadyKnownPathNodes = undefined;
        let alreadyKnownPathCost = undefined;
        const previousPathIdToSameDest = convertDestinationToPathId(end);
        if (previousPathIdToSameDest !== undefined) {
            console.debug(LOG_PREFIX, "we previously calculated a path to", end, "that we might be able to reuse");
            // save the data in a variable for a bit just in case
            const oldPathMetadata = pathIdsToPathSegmentMetadata.get(previousPathIdToSameDest);
            const oldPathCost = pathSegmentCosts.get(previousPathIdToSameDest);
            const oldPathNodes = completePathSegments.get(previousPathIdToSameDest);
            // we're calculating a new path to the same destination, so forget everything we have
            // stored about the old path to avoid a memory leak
            pathIdsToPathSegmentMetadata.delete(previousPathIdToSameDest);
            pathSegmentCosts.delete(previousPathIdToSameDest);
            completePathSegments.delete(previousPathIdToSameDest);
            pathIdToDestination.delete(previousPathIdToSameDest);
            // we might be able to copy that old path (to avoid recalculating) if it had the same source too
            if (oldPathNodes !== undefined &&
                JSON.stringify(oldPathMetadata) === JSON.stringify(pathMetadata)) {
                alreadyKnownPathNodes = oldPathNodes;
                alreadyKnownPathCost = oldPathCost;
                console.debug(LOG_PREFIX, "reusing old path", previousPathIdToSameDest, "to", end);
            }
        }
        console.debug(LOG_PREFIX, "destinationToPathIdMap", destinationToPathIdMap, end);
        // request a new path
        const pathId = nextPathId;
        calculatingPathId = pathId;
        nextPathId++;
        pathIdsToPathSegmentMetadata.set(pathId, pathMetadata);
        destinationToPathIdMap.set(`${getLat(end)},${getLng(end)}`, pathId);
        pathIdToDestination.set(pathId, end);
        if (alreadyKnownPathNodes !== undefined) {
            onProgress({
                id: calculatingPathId,
                percent_done: 1,
                best_path_cost: alreadyKnownPathCost,
                best_path_keep_prefix_length: 0,
                best_path_append: alreadyKnownPathNodes,
                current_path_keep_prefix_length: 0,
                current_path_append: [],
            });
            return;
        }
        sendWebSocketMessage({
            kind: "path",
            start: [getLat(start), getLng(start)],
            end: [getLat(end), getLng(end)],
            heading,
            start_pano: startPano,
            id: pathId,
            no_long_jumps: !SETTINGS.allow_long_jumps,
            heuristic_factor: SETTINGS.heuristic_factor,
            forward_penalty_on_intersections: SETTINGS.forward_penalty_on_intersections,
            use_option_cache: SETTINGS.use_option_cache,
        });
    }
    /**
     * Send a message to the pathfinder's WebSocket, queuing it for later if the WebSocket is currently
     * closed.
     *
     * @param msg The object that will get converted into JSON and sent to
     * the server.
     */
    function sendWebSocketMessage(msg) {
        console.debug(LOG_PREFIX, "sending", msg, "to pathfinder websocket");
        if (pfWs.readyState !== 1) {
            console.debug(LOG_PREFIX, "websocket is closed, adding message to queue");
            queuedWebSocketMessages.push(JSON.stringify(msg));
        }
        else {
            pfWs.send(JSON.stringify(msg));
        }
    }
    async function waitAndReconnect() {
        console.debug(LOG_PREFIX, "reconnecting to WebSocket");
        // this timeout is 10 seconds because of a firefox quirk that makes it delay creating websockets if you do it too fast
        await new Promise((r) => setTimeout(r, 10000));
        console.debug(LOG_PREFIX, "reconnecting...");
        connect();
    }
    function connect() {
        console.debug(LOG_PREFIX, "connecting to websocket");
        pfWs = new WebSocket(SETTINGS.backend_url.replace("http", "ws").replace(/\/$/, "") + "/path");
        console.debug(LOG_PREFIX, "websocket created:", pfWs);
        pfWs.addEventListener("close", async () => {
            console.debug(LOG_PREFIX, "Pathfinder WebSocket closed.");
            waitAndReconnect();
        });
        pfWs.addEventListener("error", (e) => {
            console.error(LOG_PREFIX, "Pathfinder WebSocket error:", e);
            pfWs.close();
        });
        pfWs.addEventListener("open", () => {
            console.debug(LOG_PREFIX, "Pathfinder WebSocket connected.");
            for (const msg of queuedWebSocketMessages) {
                pfWs.send(msg);
            }
            queuedWebSocketMessages = [];
        });
        pfWs.addEventListener("message", (e) => {
            const data = JSON.parse(e.data);
            if (data.type === "progress") {
                onProgress(data);
            }
            else if (data.type === "error") {
                alert(data.message);
            }
        });
    }
    async function waitUntilConnected() {
        if (pfWs.readyState !== 1) {
            await new Promise((res) => pfWs.addEventListener("open", res));
        }
    }
    async function clearCalculatingPathId() {
        calculatingPathId = undefined;
    }

    exports.pathfinderInfoEl = void 0;
    let pathfinderRefreshBtnEl;
    let destinationInputEl;
    exports.lastCostRecalculationTimestamp = Date.now();
    async function init() {
        const vdomContainer = await IRF.vdom.container;
        await initMap();
        injectStylesheet();
        const pathfinderContainerEl = document.createElement("div");
        pathfinderContainerEl.id = "pathfinder-container";
        destinationInputEl = document.createElement("input");
        destinationInputEl.classList.add("pathfinder-destination-input");
        destinationInputEl.placeholder = "lat, lng";
        destinationInputEl.value = getDestinationString();
        pathfinderRefreshBtnEl = document.createElement("button");
        pathfinderRefreshBtnEl.textContent = "🗘";
        pathfinderRefreshBtnEl.classList.add("pathfinder-refresh-btn");
        pathfinderRefreshBtnEl.disabled = true;
        exports.pathfinderInfoEl = document.createElement("span");
        exports.pathfinderInfoEl.classList.add("pathfinder-info");
        setInterval(() => {
            let totalCost = 0;
            let totalPanos = 0;
            for (const dest of getPathDestinations()) {
                const pathSegmentId = convertDestinationToPathId(dest);
                if (pathSegmentId === undefined) {
                    // means we haven't calculated this path yet, so we can't set an eta!
                    return;
                }
                const pathSegmentCost = pathSegmentCosts.get(pathSegmentId);
                if (pathSegmentCost === undefined) {
                    // means the path hasn't finished being calculated
                    return;
                }
                totalCost += pathSegmentCost;
                const actualPath = completePathSegments.get(pathSegmentId);
                totalPanos += actualPath.length;
            }
            if (totalCost) {
                const secondsSinceLastCostRecalculation = (Date.now() - exports.lastCostRecalculationTimestamp) / 1000;
                totalCost -= secondsSinceLastCostRecalculation;
                const advancedPercentage = panosAdvancedInFirstPath / totalPanos;
                const adjustedCost = totalCost * (1 - advancedPercentage);
                const prettyEta = prettyTime(adjustedCost);
                exports.pathfinderInfoEl.textContent = `ETA: ${prettyEta}`;
            }
        }, 1000);
        pathfinderContainerEl.appendChild(destinationInputEl);
        pathfinderContainerEl.appendChild(pathfinderRefreshBtnEl);
        pathfinderContainerEl.appendChild(exports.pathfinderInfoEl);
        vdomContainer.state.updateData = new Proxy(vdomContainer.state.updateData, {
            apply(oldUpdateData, thisArg, args) {
                // onUpdateData is a promise so errors won't propagate to here
                onUpdateData(args[0]);
                return oldUpdateData.apply(thisArg, args);
            },
        });
        const mapContainerEl = map.getContainer().parentElement;
        mapContainerEl.appendChild(pathfinderContainerEl);
        await initMarkers();
        destinationInputEl.addEventListener("change", () => {
            console.debug(LOG_PREFIX, "destination input changed");
            refreshPath();
        });
        pathfinderRefreshBtnEl.addEventListener("click", () => {
            console.debug(LOG_PREFIX, "refresh button clicked");
            refreshPath();
        });
        tryInitMmt();
        initSettingsTab();
        // this has to be done after settings are loaded so the backend url is correct
        connect();
        await setupPathSources();
        // wait until the websocket is open
        await waitUntilConnected();
        // now wait until we've received data from the internet roadtrip ws
        while (!exports.currentData) {
            await sleep(100);
        }
        console.debug(LOG_PREFIX, "start called");
        refreshPath();
    }
    function updateLastCostRecalculationTimestamp() {
        exports.lastCostRecalculationTimestamp = Date.now();
    }
    /** called when we receive a progress update from the pathfinder server */
    function onProgress(data) {
        pathfinderRefreshBtnEl.disabled = false;
        if (data.percent_done < 0) {
            // means the path was cleared
            clearAllPaths();
            rerenderCompletePathSegments();
            return;
        }
        updatePathSource("best_path", data.best_path_keep_prefix_length, data.best_path_append);
        updatePathSource("current_searching_path", data.current_path_keep_prefix_length, data.current_path_append);
        if (data.percent_done < 1) {
            // round to 5 decimal places but truncate to 1
            const percentDoneString = (data.percent_done * 100)
                .toFixed(5)
                .match(/^-?\d+(?:\.\d{0,1})?/)[0];
            exports.pathfinderInfoEl.textContent = `${percentDoneString}%`;
            return;
        }
        // path is done
        const pathId = data.id;
        updateCompletePathSegment(pathId, calculatingPaths.get("best_path"), data.best_path_cost);
        console.debug(LOG_PREFIX, `finished path ${pathId}, updated in completePathSegments`);
        rerenderCompletePathSegments();
        updateLastCostRecalculationTimestamp();
        // find the next segment if possible!
        const lastPosition = completePathSegments.get(pathId).at(-1);
        const secondLastPosition = completePathSegments.get(pathId).at(-2);
        const lastHeading = calculateHeading(secondLastPosition, lastPosition);
        const expectedDestination = pathIdToDestination.get(pathId);
        const allDestinations = getPathDestinations();
        const destinationIndex = allDestinations.findIndex((d) => d[0] === expectedDestination[0] && d[1] === expectedDestination[1]);
        console.debug(LOG_PREFIX, "allDestinations:", allDestinations, "expectedDestination:", expectedDestination, "destinationIndex", destinationIndex);
        if (destinationIndex === -1) {
            console.warn(LOG_PREFIX, "the path we just found wasn't in getPathDestinations():", allDestinations, "expectedDestination:", expectedDestination);
            return;
        }
        if (destinationIndex === allDestinations.length - 1) {
            console.debug(LOG_PREFIX, "found last segment in path");
            return;
        }
        const nextDestination = allDestinations[destinationIndex + 1];
        requestNewPath(lastHeading, lastPosition, nextDestination, 
        // pathfinder server doesn't send us pano ids
        undefined);
        rerenderCompletePathSegments();
    }
    /**
     * Send a message to stop calculating a path, and remove the lines from the map.
     */
    function abortPathfinding() {
        // note that this will get set again when the server sends us a progress update with percent_done being -1
        if (exports.pathfinderInfoEl)
            exports.pathfinderInfoEl.textContent = "";
        if (calculatingPathId !== undefined) {
            sendWebSocketMessage({
                kind: "abort",
                id: calculatingPathId,
            });
        }
        clearAllPaths();
    }
    /**
     * Remove the pathfinder's lines from the map, without aborting the current path.
     */
    function clearAllPaths() {
        exports.pathfinderInfoEl.textContent = "";
        resetRenderedPath();
    }
    function updateDestinationFromString(destString) {
        setDestinationString(destString);
        const dest = parseCoordinatesString(destString);
        updateDestinationMarker(dest);
        if (!dest) {
            document.body.classList.remove("pathfinder-has-destination");
            setStops([]);
            // abortPathfinding has to happen before clearAllPaths so the calculatingPathId is still set
            abortPathfinding();
            clearCalculatingPaths();
            rerenderCompletePathSegments();
            rerenderStopMarkers();
            clearAllPaths();
            return;
        }
        clearAllPaths();
        document.body.classList.add("pathfinder-has-destination");
        if (!exports.currentData) {
            // if we haven't received any data from the game yet then we can't know our current location
            return;
        }
        const curPos = newPosition(exports.currentData.lat, exports.currentData.lng);
        // at least one destination must be present because we just set the destination string and
        // checked that it was valid
        const firstDestination = getPathDestinations()[0];
        requestNewPath(exports.currentData.heading, curPos, firstDestination, exports.currentData.pano);
    }
    exports.previousData = null;
    exports.currentData = null;
    /**
     * The number of times in a row that the current position wasn't found in the best path.
     * This exists so we can recalculate the path if this value gets too high.
     */
    let lostPathCount = 0;
    /**
     * Called whenever we receive a message from the game WebSocket.
     */
    async function onUpdateData(msg) {
        [exports.previousData, exports.currentData] = [exports.currentData, msg];
        const curPos = newPosition(exports.currentData.lat, exports.currentData.lng);
        const locationChanged = exports.previousData?.lat !== getLat(curPos) ||
            exports.previousData?.lng !== getLng(curPos) ||
            exports.previousData?.heading !== exports.currentData.heading;
        if (locationChanged)
            clearOptionHighlights();
        if (SETTINGS.remove_reached_stops) {
            const remainingStops = getUnorderedStops();
            for (const stop of [...remainingStops]) {
                if (calculateDistance(stop, curPos) < 15 /* meters */) {
                    removeStop(stop);
                    break;
                }
            }
        }
        if (locationChanged && getPathDestinations().length > 0) {
            const isPathFound = updateCurrentLocation(curPos);
            if (isPathFound) {
                lostPathCount = 0;
                // wait a bit to make sure that any new elements are created
                await sleep(1100);
                showBestOption();
            }
            else {
                console.warn(LOG_PREFIX, `lost path? (#${lostPathCount})`);
                lostPathCount += 1;
                if (lostPathCount >= 3) {
                    lostPathCount = 0;
                    refreshPath();
                }
            }
        }
    }
    async function refreshPath() {
        pathfinderRefreshBtnEl.disabled = true;
        const destinationValue = getDestinationString();
        const hasDestination = destinationValue.trim() !== "";
        document.body.classList.toggle("pathfinder-has-destination", hasDestination);
        if (!hasDestination) {
            updateDestinationMarker(null);
            abortPathfinding();
            return;
        }
        updateDestinationFromString(destinationValue);
        // makes it so we don't wait until the next location change to highlight the new best option
        await sleep(2000);
        showBestOption();
    }
    /**
     * @returns A string that should be formatted as `lat,lng`, but might not be.
     */
    function getDestinationString() {
        return GM_getValue("destination") ?? "";
    }
    /**
     * @param value A string that should be formatted as `lat,lng`, but might not be.
     */
    function setDestinationString(value) {
        GM_setValue("destination", value.trim());
    }
    function injectStylesheet() {
        GM_addStyle(`
    body:not(.pathfinder-found-minimap-tricks) {
      & .map-container .info-button {
        /* overlaps with our ui */
        display: none;
      }
      & .pathfinder-refresh-btn {
        line-height: 1;
        padding: 0.2em;
      }
      & .pathfinder-info {
        background-color: #fff;
        padding: 0.1em 0.3em;
      }
    }

    body.pathfinder-found-minimap-tricks {
      & #pathfinder-container {
        display: none;
      }

      & .pathfinder-info {
        margin-right: 36px;

        &:empty {
          display: none;
        }
      }
    }

    .pathfinder-destination-marker {
      width: 25px;
      cursor: default;
    }
    .pathfinder-stop-marker {
      width: 15px;
      cursor: default;
    }
    .pathfinder-chosen-option path {
      fill: #f0f !important;
    }

    body:not(.pathfinder-has-destination) {
      & .pathfinder-clear-path-mmt-side-button,
      .pathfinder-clear-path-mmt-context-menu-button,
      .pathfinder-add-stop-mmt-context-menu-button {
        display: none !important;
      }
    }
  `);
    }
    function replacePathfinderInfoEl(newEl) {
        exports.pathfinderInfoEl.remove();
        exports.pathfinderInfoEl = newEl;
    }
    init();

    exports.clearAllPaths = clearAllPaths;
    exports.getDestinationString = getDestinationString;
    exports.onProgress = onProgress;
    exports.refreshPath = refreshPath;
    exports.replacePathfinderInfoEl = replacePathfinderInfoEl;
    exports.updateDestinationFromString = updateDestinationFromString;
    exports.updateLastCostRecalculationTimestamp = updateLastCostRecalculationTimestamp;

    return exports;

})({});