您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
A userscript that adds additional features for route planning on Komoot.com.
// ==UserScript== // @name Komodo - Mods for Komoot // @namespace https://github.com/jerboa88 // @version 2.0.0 // @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 key = name ?? ""; const candidateValueToInclusionMap = candidate.tagMap.get(key); 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 = assertDefined( pill.dataset[DATA_ATTRIBUTE.TAG_NAME], `No tag name found in pill: ${pill.textContent}` ); 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(); })();