// ==UserScript==
// @name Komodo - Mods for Komoot
// @namespace https://github.com/jerboa88
// @version 2.0.1
// @author John Goodliff
// @description A userscript that adds additional features for route planning on Komoot.com.
// @license MIT
// @icon 
// @homepage https://johng.io/p/komodo
// @homepageURL https://johng.io/p/komodo
// @source https://github.com/jerboa88/komodo.git
// @supportURL https://github.com/jerboa88/komodo/issues
// @match https://www.komoot.com/user/*/routes
// @match https://www.komoot.com/tour/*
// @grant none
// ==/UserScript==
(function () {
'use strict';
const d=new Set;const importCSS = async e=>{d.has(e)||(d.add(e),(t=>{typeof GM_addStyle=="function"?GM_addStyle(t):document.head.appendChild(document.createElement("style")).append(t);})(e));};
const styleCss = ':root{--komodo-spacing: .375rem;--komodo-pill-bg-color: var(--theme-ui-colors-primary);--komodo-pill-text-color: var(--theme-ui-colors-textOnDark);--komodo-button-bg-color: var(--theme-ui-colors-white);--komodo-button-border-color: var(--komodo-button-bg-color);--komodo-button-text-color: var(--theme-ui-colors-secondary);--komodo-button-hover-bg-color: rgba(0, 119, 217, .1);--komodo-button-hover-border-color: #0065b8;--komodo-button-hover-text-color: #0065b8;--komodo-button-disabled-bg-color: var(--theme-ui-colors-muted);--komodo-button-disabled-border-color: var(--komodo-button-disabled-bg-color);--komodo-button-disabled-text-color: var(--theme-ui-colors-disabled)}dialog[data-test-id=rename-tour-dialog]>div{width:100%;max-width:64rem}.komodo-filter-container{flex-wrap:wrap;gap:var(--komodo-spacing)}.komodo-filter-container>button{margin-right:0!important}div:has(>a[href="/upload"]){align-items:center}.komodo-hide{display:none}.komodo-tag-filter-container{flex:1 1 auto;display:flex;flex-wrap:wrap;gap:var(--komodo-spacing)}.komodo-tag-filter{border-width:1px;font-weight:700;border-radius:8px;flex:1 1 0%;background-color:var(--theme-ui-colors-card);color:var(--theme-ui-colors-text);border-color:var(--theme-ui-colors-black20)}.komodo-tag-filter:hover{border-color:var(--theme-ui-colors-black30)}.komodo-tag-filter>p{align-items:center;display:flex;flex-direction:row;gap:1.5rem;justify-content:space-between;padding:1rem;width:initial;align-self:stretch}.komodo-tag-filter>div{border-bottom-width:1px;border-color:var(--theme-ui-colors-border);border-style:solid;width:100%;justify-self:stretch}.komodo-tag-filter>fieldset{align-items:stretch;display:flex;flex-direction:column;gap:.75rem;justify-content:end;padding:1rem;width:initial;align-self:stretch}.komodo-tag-filter>fieldset>label{align-items:center;border-color:var(--theme-ui-colors-border);border-radius:8px;border-style:solid;color:var(--theme-ui-colors-text);cursor:pointer;display:flex;flex-direction:row;gap:1.5rem;grid-area:grid-item-0;justify-content:space-between;padding:.5rem;width:initial;align-self:stretch;border-width:1px}.komodo-tag-filter>fieldset>label:hover{border-color:var(--theme-ui-colors-whisper)}.komodo-tag-filter>fieldset>label:has(input[type=checkbox][value=true]){color:var(--theme-ui-colors-primary)}.komodo-tag-filter>fieldset>label:has(input[type=checkbox][value=false]){color:var(--theme-ui-colors-error)}.komodo-tag-filter>fieldset>label>input[type=checkbox][value=true]{accent-color:var(--komodo-pill-bg-color)}.komodo-tag-filter>fieldset>label>input[type=checkbox][value=false]{accent-color:var(--theme-ui-colors-error)}.komodo-tag-filter>select{display:block;width:fit-content;margin-top:var(--komodo-spacing)}.komodo-pill{align-items:center;background-color:var(--komodo-pill-bg-color);border-radius:4px;color:var(--komodo-pill-text-color);display:inline-flex;justify-content:center;min-width:2em;text-align:center;font-size:12px;font-weight:700;padding:.25em .5em;text-transform:inherit;flex-shrink:0}.komodo-tag-pill-container{display:flex;flex-wrap:wrap;margin-top:var(--komodo-spacing);gap:var(--komodo-spacing)}.komodo-tag-pill-container>.komodo-pill>div>span:nth-child(2){white-space:pre}.komodo-button{align-items:center;appearance:none;background-color:var(--komodo-button-bg-color);border-color:var(--komodo-button-border-color);border-radius:8px;border-style:solid;color:var(--komodo-button-text-color);cursor:pointer;display:inline-flex;justify-content:center;pointer-events:auto;text-align:center;width:unset;border-width:.0625rem;text-decoration:none;transition:all .2s;font-size:16px;font-weight:700;line-height:1.5rem;padding:.4375rem .6875rem}.komodo-button:hover{background-color:var(--komodo-button-hover-bg-color);border-color:var(--komodo-button-hover-border-color);color:var(--komodo-button-hover-text-color)}.komodo-button:disabled{cursor:default;background-color:var(--komodo-button-disabled-bg-color);border-color:var(--komodo-button-disabled-border-color);color:var(--komodo-button-disabled-text-color)}.komodo-button>svg{color:inherit}.komodo-button>span{display:inline-flex;text-align:center;flex-flow:column;padding-left:.25rem;padding-right:0}.komodo-new{position:relative}.komodo-new:after{content:"🦎";position:absolute;top:0;right:calc(var(--komodo-spacing) * -1);z-index:1;font-size:small;line-height:0}';
importCSS(styleCss);
const PROJECT = {
EMOJI: "🦎",
NAME: "Komodo"
};
const prefix = PROJECT.NAME.toLowerCase();
const CLASS = {
NEW: `${prefix}-new`,
HIDE: `${prefix}-hide`,
FILTER_CONTAINER: `${prefix}-filter-container`,
TAG_FILTER_CONTAINER: `${prefix}-tag-filter-container`,
TAG_FILTER: `${prefix}-tag-filter`,
TAG_PILL_CONTAINER: `${prefix}-tag-pill-container`,
PILL: `${prefix}-pill`,
BUTTON: `${prefix}-button`
};
const DATA_ATTRIBUTE = {
TOUR_ID: "tourId",
TAG_NAME: `${prefix}TagName`,
TAG_VALUE: `${prefix}TagValue`
};
const TAG_DELIMITER = {
START: "[",
END: "]",
KEY_VALUE: ":",
VALUE: ","
};
const SCRIPT_NAME = `${PROJECT.EMOJI} ${PROJECT.NAME}`;
const buildLogPrefix = (() => {
const htmlNode = window.getComputedStyle(document.documentElement);
const colorMap = {
primary: htmlNode.getPropertyValue("--theme-ui-colors-primaryOnDark"),
debug: htmlNode.getPropertyValue("--theme-ui-colors-info"),
info: htmlNode.getPropertyValue("--theme-ui-colors-success"),
warn: htmlNode.getPropertyValue("--theme-ui-colors-warning"),
error: htmlNode.getPropertyValue("--theme-ui-colors-error")
};
return (severity) => [
`%c${SCRIPT_NAME} %c${severity}`,
`font-style:italic;color:${colorMap.primary};`,
`color:${colorMap[severity]};`
];
})();
const buildLogFn = (severity) => {
const logFn = console[severity];
const logPrefix = buildLogPrefix(severity);
return (...args) => logFn(...logPrefix, ...args);
};
const debug = buildLogFn("debug");
const warn = buildLogFn("warn");
const assertDefined = (value, message = "Value is not defined") => {
if (value == null) throw new Error(message);
return value;
};
const toElementId = (value) => {
if (!value) {
return "id_empty";
}
const validChar = /^[a-zA-Z0-9\-_:.]+$/;
let result = "";
for (const ch of value) {
if (validChar.test(ch)) {
result += ch;
} else {
const code = ch.codePointAt(0)?.toString(16).padStart(4, "0");
result += `_u${code}_`;
}
}
if (!/^[a-zA-Z]/.test(result)) {
result = `id_${result}`;
}
return result;
};
const createElementTemplate = (nullableReferenceElement) => {
const referenceElement = assertDefined(
nullableReferenceElement,
"Unable to create element template. Reference element not found"
);
const elementTemplate = referenceElement.cloneNode(true);
elementTemplate.classList.add(CLASS.NEW);
return elementTemplate;
};
const createPill = (text) => {
const div = document.createElement("div");
div.classList.add(CLASS.NEW, CLASS.PILL);
return div;
};
const createButton = (text, icon, handleClick) => {
const button = document.createElement("button");
const span = document.createElement("span");
span.textContent = text;
button.onclick = (event) => {
debug("Button clicked");
handleClick(event, button, span, icon);
};
button.classList.add(CLASS.NEW, CLASS.BUTTON);
button.appendChild(icon);
button.appendChild(span);
return button;
};
const createTriStateCheckbox = (() => {
const stateMap = {
undefined: void 0,
true: true,
false: false
};
const states = Object.values(stateMap);
const updateCheckboxState = (checkbox, checkedState) => {
checkbox.checked = checkedState === true;
checkbox.indeterminate = checkedState === false;
checkbox.value = String(checkedState);
};
return (id, initialCheckedState, onClick) => {
const checkbox = document.createElement("input");
checkbox.type = "checkbox";
checkbox.id = id;
checkbox.addEventListener("click", () => {
let checkedState = stateMap[checkbox.value];
const newCheckedStateIndex = (states.indexOf(checkedState) + 1) % states.length;
checkedState = states[newCheckedStateIndex];
updateCheckboxState(checkbox, checkedState);
onClick(checkedState);
});
updateCheckboxState(checkbox, initialCheckedState);
return checkbox;
};
})();
const showElement = (element, visible) => {
const shouldHide = !visible;
const isHidden = element.classList.contains(CLASS.HIDE);
if (isHidden === shouldHide) {
return false;
}
element.classList.toggle(CLASS.HIDE, shouldHide);
return true;
};
const onReactMounted = (callback) => {
const canaryClassName = "ReactModalPortal";
const continueCall = () => {
debug("React has been mounted");
callback();
};
const canaries = document.body.getElementsByClassName(canaryClassName);
if (canaries.length > 0) {
continueCall();
return;
}
const observer = new MutationObserver((mutations) => {
debug("Mutations observed on body", mutations);
for (const mutation of mutations) {
for (const newNode of mutation.addedNodes) {
if (newNode instanceof HTMLElement && newNode.classList.contains(canaryClassName)) {
observer.disconnect();
continueCall();
return;
}
}
}
});
debug("Waiting for React to be mounted");
observer.observe(document.body, { childList: true });
};
class TagMap {
tagMap = new Map();
startDelimiter;
endDelimiter;
keyValueDelimiter;
valueDelimiter;
constructor(startDelimiter = "[", endDelimiter = "]", keyValueDelimiter = ":", valueDelimiter = ",") {
this.startDelimiter = startDelimiter;
this.endDelimiter = endDelimiter;
this.keyValueDelimiter = keyValueDelimiter;
this.valueDelimiter = valueDelimiter;
}
getValueToInclusionMap(name) {
const valueToInclusionMap = this.tagMap.get(name);
if (!valueToInclusionMap) {
throw new Error(
"TagMap: Expected valueToInclusionMap to be defined, but it was not"
);
}
return valueToInclusionMap;
}
add(name, value) {
if (!this.tagMap.has(name)) {
this.tagMap.set(name, new Map());
}
const valueToInclusionMap = this.getValueToInclusionMap(name);
if (valueToInclusionMap.has(value)) {
return false;
}
valueToInclusionMap.set(value, void 0);
return true;
}
setInclusion(name, value, isIncluded) {
const valueToInclusionMap = this.tagMap.get(name);
if (!valueToInclusionMap || !valueToInclusionMap.has(value)) {
return false;
}
const current = valueToInclusionMap.get(value);
if (current === isIncluded) {
return false;
}
valueToInclusionMap.set(value, isIncluded);
return true;
}
getAsMap = () => {
return this.tagMap;
};
*[Symbol.iterator]() {
const sortedNames = Array.from(this.tagMap.keys()).sort();
for (const name of sortedNames) {
const valueToInclusionMap = this.getValueToInclusionMap(name);
const sortedValues = Array.from(valueToInclusionMap.keys()).sort();
for (const value of sortedValues) {
yield { name, value, isIncluded: valueToInclusionMap.get(value) };
}
}
}
parseAndAdd(input) {
const parsedTagMap = new TagMap(
TAG_DELIMITER.START,
TAG_DELIMITER.END,
TAG_DELIMITER.KEY_VALUE,
TAG_DELIMITER.VALUE
);
let text = "";
let wasUpdated = false;
let i = 0;
while (i < input.length) {
if (input[i] === this.startDelimiter) {
i++;
let inside = "";
while (i < input.length && input[i] !== this.endDelimiter) {
inside += input[i++];
}
if (i < input.length && input[i] === this.endDelimiter) {
i++;
}
const kvIndex = inside.indexOf(this.keyValueDelimiter);
let tagName;
let values = [];
if (kvIndex >= 0) {
tagName = inside.slice(0, kvIndex).trim();
values = inside.slice(kvIndex + 1).split(this.valueDelimiter).map((v) => v.trim()).filter((v) => v.length > 0);
} else {
values = inside.split(this.valueDelimiter).map((v) => v.trim()).filter((v) => v.length > 0);
}
for (const v of values) {
const wasAdded = this.add(tagName, v);
parsedTagMap.add(tagName, v);
if (wasAdded) wasUpdated = true;
}
} else {
text += input[i++];
}
}
return { text, parsedTagMap, wasUpdated };
}
matches(candidate) {
for (const { name, value, isIncluded } of this) {
const candidateValueToInclusionMap = candidate.tagMap.get(name);
const existsInCandidate = candidateValueToInclusionMap?.has(value) ?? false;
if (isIncluded === true && !existsInCandidate) {
debug(
`TagMap.matches: ${name}:${value} is included in reference but not in candidate`
);
return false;
}
if (isIncluded === false && existsInCandidate) {
debug(
`TagMap.matches: ${name}:${value} is excluded in reference but exists in candidate`
);
return false;
}
}
return true;
}
}
const pattern$1 = /^\/user\/\d*?\/routes$/;
const init$1 = async () => {
const tagMap = new TagMap(
TAG_DELIMITER.START,
TAG_DELIMITER.END,
TAG_DELIMITER.KEY_VALUE,
TAG_DELIMITER.VALUE
);
const savedRoutesAnchor = assertDefined(
document.querySelector(
'a[href^="/user/"][href$="/routes"]'
),
"No saved routes link found"
);
const ul = assertDefined(
document.querySelector(
'ul[data-test-id="tours-list"]'
),
"No route list found"
);
const getLis = () => [...ul.children].filter((li) => li.nodeName === "LI");
const scrollToLoadAll = async () => {
debug("Force loading all routes");
const initialScrollPos = window.scrollY;
const totalNumOfRoutes = Number(
assertDefined(
savedRoutesAnchor.lastElementChild?.textContent,
"Unable to get total number of routes. Required element not found"
)
);
debug(`Found ${totalNumOfRoutes} total routes`);
const loadMore = async () => {
ul.scrollTop = ul.scrollHeight;
window.scrollTo(0, document.body.scrollHeight);
await new Promise((r) => setTimeout(r, 100));
return totalNumOfRoutes > getLis().length;
};
while (await loadMore()) ;
debug(`Restoring scroll position: ${initialScrollPos}`);
window.scrollTo(0, initialScrollPos);
};
const addLoadAllRoutesButton = () => {
debug("Adding load all routes button to page");
const importLinkAnchor = document.querySelector(
'a[href="/upload"]'
);
const container = assertDefined(importLinkAnchor.parentElement);
const icon = createElementTemplate(
savedRoutesAnchor.firstElementChild
);
const loadAllRoutesbutton = createButton(
"Load All Routes",
icon,
async (_event, button, span) => {
button.disabled = true;
span.textContent = "Loading...";
await scrollToLoadAll();
span.textContent = "Loaded";
}
);
container.insertBefore(loadAllRoutesbutton, importLinkAnchor);
};
const createTagFilterSet = (tagName, tagValueToInclusionMap) => {
const container = document.createElement("fieldset");
const sortedTagValueEntries = [...tagValueToInclusionMap.entries()].sort(
([tagValueA], [tagValueB]) => tagValueA.localeCompare(tagValueB)
);
for (const [tagValue, isIncluded] of sortedTagValueEntries) {
const handleClick = (checkedState) => {
tagMap.setInclusion(tagName, tagValue, checkedState);
applyFilters();
};
const checkboxId = `${toElementId(tagName)}-${toElementId(tagValue)}`;
const checkbox = createTriStateCheckbox(
checkboxId,
isIncluded,
handleClick
);
const label = document.createElement("label");
const span = document.createElement("span");
span.textContent = tagValue;
label.dataset[DATA_ATTRIBUTE.TAG_VALUE] = tagValue;
label.appendChild(span);
label.appendChild(checkbox);
container.appendChild(label);
}
return container;
};
const createTagFiltersContainer = () => {
debug("Creating tag filters container");
const tagFiltersContainer = document.createElement("form");
tagFiltersContainer.classList.add(CLASS.TAG_FILTER_CONTAINER);
for (const [tagName, tagValueToInclusionMap] of tagMap.getAsMap()) {
const tagFilter = document.createElement("div");
tagFilter.classList.add(CLASS.NEW, CLASS.TAG_FILTER);
tagFilter.dataset[DATA_ATTRIBUTE.TAG_NAME] = tagName;
const filterSetTitle = document.createElement("p");
const divider = document.createElement("div");
filterSetTitle.textContent = tagName ?? "...";
tagFilter.appendChild(filterSetTitle);
tagFilter.appendChild(divider);
const container = createTagFilterSet(tagName, tagValueToInclusionMap);
tagFilter.appendChild(container);
tagFiltersContainer.appendChild(tagFilter);
}
return tagFiltersContainer;
};
const updateTagFilterControls = () => {
debug("Updating tag filter controls on page");
const filterContainer = document.querySelector(
'#js-filter-anchor div:not([data-bottomsheet-scroll-ignore="true"]):has(> button:not([type="button"])'
);
const existingTagFilterContainer = filterContainer?.getElementsByClassName(
CLASS.TAG_FILTER_CONTAINER
)?.[0];
const tagFilterControls = createTagFiltersContainer();
existingTagFilterContainer ? existingTagFilterContainer.replaceWith(tagFilterControls) : filterContainer?.appendChild(tagFilterControls);
filterContainer?.classList.add(CLASS.FILTER_CONTAINER);
};
const updateLiTitle = (a) => {
if (!a) {
warn("No a element found in li element", a);
return {
routeTagMap: new TagMap(),
updated: false
};
}
const originalTitle = assertDefined(
a.textContent,
"Expected a.textContent to be defined, but it was not"
);
const {
text,
parsedTagMap: routeTagMap,
wasUpdated
} = tagMap.parseAndAdd(originalTitle);
a.textContent = text;
a.title = originalTitle;
return { routeTagMap, wasUpdated };
};
const parseLiTagPills = (li) => {
const pills = li.getElementsByClassName(
CLASS.PILL
);
const routeTagMap = new TagMap();
for (const pill of pills) {
const name = pill.dataset[DATA_ATTRIBUTE.TAG_NAME];
const value = assertDefined(
pill.dataset[DATA_ATTRIBUTE.TAG_VALUE],
`No tag value found in pill: ${pill.textContent}`
);
routeTagMap.add(name, value);
}
return routeTagMap;
};
const createTagPill = (tag) => {
const pill = createPill();
const container = document.createElement("div");
const valueSpan = document.createElement("span");
valueSpan.textContent = tag.value;
pill.dataset[DATA_ATTRIBUTE.TAG_VALUE] = tag.value;
if (tag.name) {
const nameSpan = document.createElement("span");
const separatorSpan = document.createElement("span");
nameSpan.textContent = tag.name;
separatorSpan.textContent = ": ";
pill.dataset[DATA_ATTRIBUTE.TAG_NAME] = tag.name;
container.appendChild(nameSpan);
container.appendChild(separatorSpan);
}
container.appendChild(valueSpan);
pill.appendChild(container);
return pill;
};
const createTagPillContainer = (routeTagMap) => {
const div = document.createElement("div");
for (const tag of routeTagMap) {
div.appendChild(createTagPill(tag));
}
div.classList.add(CLASS.TAG_PILL_CONTAINER);
return div;
};
const updateLi = (li) => {
debug("Updating li element");
const a = assertDefined(
li.querySelector(
'a[data-test-id="tours_list_item_title"]'
),
"No a element found in li element"
);
const { routeTagMap, wasUpdated } = updateLiTitle(a);
a.parentElement?.appendChild(createTagPillContainer(routeTagMap));
if (wasUpdated) {
updateTagFilterControls();
}
filterLi(li, routeTagMap);
};
const filterLi = (li, routeTagMap) => {
const doesMatchFilter = tagMap.matches(routeTagMap);
const wasVisibilityChanged = showElement(li, doesMatchFilter);
if (wasVisibilityChanged) {
const msgPrefix = doesMatchFilter ? "Showing" : "Hiding";
debug(`${msgPrefix} li element: ${li.dataset[DATA_ATTRIBUTE.TOUR_ID]}`);
}
};
const applyFilters = () => {
debug("Applying filters");
const lis = getLis();
for (const li of lis) {
const routeTagMap = parseLiTagPills(li);
filterLi(li, routeTagMap);
}
};
debug("Setting up route list page");
const observer = new MutationObserver((mutations) => {
debug("Mutations observed on ul", mutations);
for (const mutation of mutations) {
for (const newNode of mutation.addedNodes) {
if (newNode.nodeName === "LI") {
updateLi(newNode);
}
}
}
});
debug("Waiting for li elements to be added to the list");
observer.observe(ul, { childList: true });
getLis().forEach(updateLi);
addLoadAllRoutesButton();
updateTagFilterControls();
};
const handler$1 = () => onReactMounted(init$1);
const routeListRoute = {
pattern: pattern$1,
handler: handler$1
};
const pattern = /^\/tour\/\d*?/;
const handler = async () => {
debug("Setting up route page");
};
const routeViewRoute = {
pattern,
handler
};
const registerRouteHandlers = (routes) => {
const path = location.pathname;
for (const { pattern: pattern2, handler: handler2 } of routes) {
if (pattern2.test(path)) {
handler2();
break;
}
}
};
const init = () => {
debug("Script loaded");
registerRouteHandlers([routeViewRoute, routeListRoute]);
debug("Script unloaded");
};
init();
})();