您需要先安装一个扩展,例如 篡改猴、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 0.2.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== (o=>{if(typeof GM_addStyle=="function"){GM_addStyle(o);return}const t=document.createElement("style");t.textContent=o,document.head.append(t)})(' :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;padding:.5rem 1rem;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>label{font-weight:700}.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:"\u{1F98E}";position:absolute;top:0;right:calc(var(--komodo-spacing) * -1);z-index:1;font-size:small;line-height:0} '); (function () { 'use strict'; 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 = { // Built-in TOUR_ID: "tourId", TAG_NAME: `${prefix}TagName`, TAG_VALUE: `${prefix}TagValue` }; 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 createMultiSelect = (id, optionObjs, handleChange) => { const select = document.createElement("select"); select.id = id; select.multiple = true; select.size = Math.min(8, optionObjs.length); const sortedValues = [...optionObjs].sort( (a, b) => a.value.localeCompare(b.value) ); for (const { value, selected } of sortedValues) { const option = document.createElement("option"); option.value = value; option.textContent = value; option.selected = selected; select.appendChild(option); } select.onchange = handleChange; return select; }; 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 Tag { constructor(name, value) { this.name = name; this.value = value; } toString() { return `${this.name}: ${this.value}`; } } const TAG_REGEX = /\[\s*([^:[\]]+?)\s*:\s*([^:[\]]+?)\s*\]/g; class TagManager { tagValuesMap = /* @__PURE__ */ new Map(); filteredTagValuesMap = /* @__PURE__ */ new Map(); /** * Parses a string for tags and returns the remaining text and the extracted tags. * * @param text - The text to parse. * @returns An object containing the remaining text and the extracted tags. */ static extractTags(text) { const tags = []; const matches = text.matchAll(TAG_REGEX); for (const match of matches) { tags.push(new Tag(match[1], match[2])); } return { text: text.replace(TAG_REGEX, "").trim(), tags }; } /** * Adds a single tag to the tag manager. * * @param tag - The tag to add. * @returns `true` if the tag was added, `false` if it already existed. */ add(tag) { let updated = false; if (this.tagValuesMap.has(tag.name)) { const values = this.tagValuesMap.get(tag.name); if (!values.has(tag.value)) { values.add(tag.value); updated = true; } } else { this.tagValuesMap.set(tag.name, /* @__PURE__ */ new Set([tag.value])); updated = true; } return updated; } /** * Adds multiple tags to the tag manager. * * @param tags - The tags to add. * @returns `true` if any of the tags were added, `false` if all of them already existed. */ addMultiple(tags) { return tags.map((t) => this.add(t)).some(Boolean); } /** * Gets all tags and their values. * * @returns An array of all tags and their values. */ getAll() { return [...this.tagValuesMap.entries()]; } /** * Sets a list of values to filter by for a given tag name. * * @param tagName - The name of the tag to filter by. * @param values - The values to filter by. * @returns The new Map of filtered values. */ setFilteredValuesSet(tagName, values) { this.filteredTagValuesMap.set(tagName, values); } /** * Gets the list of values to filter by for a given tag name. * * @param tagName - The name of the tag to filter by. * @returns The list of values to filter by. */ getFilteredValuesSet(tagName) { return this.filteredTagValuesMap.get(tagName) ?? /* @__PURE__ */ new Set(); } /** * Checks if a list of tags matches the current filters. * * @param tags - The list of tags to check. * @returns `true` if the tags match the filters, `false` otherwise. */ matchesFilters(tags) { for (const [tagName, values] of this.filteredTagValuesMap.entries()) { if (values.size === 0) continue; if (!tags.some((tag) => tag.name === tagName && values.has(tag.value))) { return false; } } return true; } } const pattern$1 = /^\/user\/\d*?\/routes$/; const init$1 = async () => { const tagManager = new TagManager(); 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 createTagSelect = (name, values, id) => { const optionObjs = [...values].map((value) => ({ value, selected: tagManager.getFilteredValuesSet(name).has(value) ?? false })); const select = createMultiSelect(id, optionObjs, (event) => { const target = event.currentTarget; const selectedValuesSet = new Set( [...target.selectedOptions].map((o) => o.value) ); tagManager.setFilteredValuesSet(name, selectedValuesSet); applyFilters(); }); return select; }; const createTagSelectContainer = () => { debug("Creating tag select container"); const tagFilterContainer = document.createElement("div"); tagFilterContainer.classList.add(CLASS.TAG_FILTER_CONTAINER); for (const [name, values] of tagManager.getAll()) { const tagFilter = document.createElement("div"); tagFilter.classList.add(CLASS.NEW, CLASS.TAG_FILTER); const label = document.createElement("label"); const id = toElementId(name); label.textContent = name; label.htmlFor = id; tagFilter.appendChild(label); const select = createTagSelect(name, values, id); tagFilter.appendChild(select); tagFilterContainer.appendChild(tagFilter); } return tagFilterContainer; }; 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 = createTagSelectContainer(); existingTagFilterContainer ? existingTagFilterContainer.replaceWith(tagFilterControls) : filterContainer?.appendChild(tagFilterControls); filterContainer?.classList.add(CLASS.FILTER_CONTAINER); }; const parseLiTitle = (a) => { if (!a) { warn("No a element found in li element", a); return []; } const { text, tags } = TagManager.extractTags(a.textContent); a.textContent = text; return tags; }; const parseLiTagPills = (li) => { const pills = li.getElementsByClassName( CLASS.PILL ); if (!pills.length) return []; return [...pills].map((pill) => { 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}` ); return new Tag(name, value); }); }; const createTagPill = (tag) => { const pill = createPill(); const container = document.createElement("div"); const nameSpan = document.createElement("span"); const separatorSpan = document.createElement("span"); const valueSpan = document.createElement("span"); nameSpan.textContent = tag.name; separatorSpan.textContent = ": "; valueSpan.textContent = tag.value; pill.dataset[DATA_ATTRIBUTE.TAG_NAME] = tag.name; pill.dataset[DATA_ATTRIBUTE.TAG_VALUE] = tag.value; container.appendChild(nameSpan); container.appendChild(separatorSpan); container.appendChild(valueSpan); pill.appendChild(container); return pill; }; const createTagPillContainer = (tags) => { const div = document.createElement("div"); tags.forEach((tag) => 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 tags = parseLiTitle(a); const wasTagMapUpdated = tagManager.addMultiple(tags); a.parentElement?.appendChild(createTagPillContainer(tags)); if (wasTagMapUpdated) { updateTagFilterControls(); } filterLi(li, tags); }; const filterLi = (li, tags) => { const doesMatchFilter = tagManager.matchesFilters(tags); 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 tags = parseLiTagPills(li); filterLi(li, tags); } }; 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(); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址