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