为 Codeberg 的 README 添加目录。
// ==UserScript== // @name Codeberg README TOC // @name:zh-CN Codeberg README 目录 // @name:zh-TW Codeberg README 目錄 // @name:ja Codeberg README 目次 // @name:ko Codeberg README 목차 // @name:es Codeberg README Tabla de Contenidos // @name:fr Codeberg README Table des Matières // @name:de Codeberg README Inhaltsverzeichnis // @name:ru Codeberg README Оглавление // @name:pt Codeberg README Índice // @name:pt-BR Codeberg README Índice // @name:it Codeberg README Indice // @name:pl Codeberg README Spis Treści // @name:nl Codeberg README Inhoudsopgave // @name:tr Codeberg README İçindekiler // @name:ar Codeberg README جدول المحتويات // @description Add table of contents(TOC) for README in Codeberg. // @description:zh-CN 为 Codeberg 的 README 添加目录。 // @description:zh-TW 為 Codeberg 的 README 添加目錄。 // @description:ja Codeberg の README に目次を追加します。 // @description:ko Codeberg의 README에 목차를 추가합니다. // @description:es Agrega una tabla de contenidos para README en Codeberg. // @description:fr Ajoute une table des matières pour README dans Codeberg. // @description:de Fügt ein Inhaltsverzeichnis für README in Codeberg hinzu. // @description:ru Добавляет оглавление для README в Codeberg. // @description:pt Adiciona um índice para README no Codeberg. // @description:pt-BR Adiciona um índice para README no Codeberg. // @description:it Aggiunge un indice per README in Codeberg. // @description:pl Dodaje spis treści dla README w Codeberg. // @description:nl Voegt een inhoudsopgave toe voor README in Codeberg. // @description:tr Codeberg'deki README için içindekiler tablosu ekler. // @description:ar يضيف جدول محتويات لـ README في Codeberg. // @namespace tampermonkey // @version 0.1.7 // @author aspen138, Claude Code(Sonnet 4.5) // @license MIT // @icon  // @exclude https://codeberg.org/*/*/commits/* // @exclude https://codeberg.org/*/*/branches // @exclude https://codeberg.org/*/*/tags // @exclude https://codeberg.org/*/*/issues // @exclude https://codeberg.org/*/*/pulls // @exclude https://codeberg.org/*/*/activity // @exclude https://codeberg.org/*/*/actions // @match https://codeberg.org/** // @require https://cdn.jsdelivr.net/npm/[email protected]/umd/react.production.min.js // @require https://cdn.jsdelivr.net/npm/[email protected]/umd/react-dom.production.min.js // ==/UserScript== (o=>{const t=document.createElement("style");t.dataset.source="vite-plugin-monkey",t.textContent=o,document.head.append(t)})(" /* Main container for two-column layout */ .codeberg-toc-layout-container { display: flex !important; gap: 20px !important; width: 100% !important; } /* README content - left side with increased width */ .codeberg-toc-main-content { flex: 0 0 72% !important; max-width: 72% !important; min-width: 0 !important; } /* TOC container - right side with reduced width */ #codeberg-readme-toc { flex: 0 0 25% !important; max-width: 25% !important; position: sticky !important; top: 20px !important; max-height: calc(100vh - 40px) !important; padding: 16px !important; padding-right: 0px !important; background: var(--color-canvas-subtle, #f6f8fa) !important; border: 1px solid var(--color-border-muted, #d1d9e0) !important; border-radius: 6px !important; display: flex !important; flex-direction: column !important; } /* TOC title */ #codeberg-readme-toc h2 { margin: 0 0 12px 0 !important; font-size: 16px !important; font-weight: 600 !important; color: var(--color-fg-default, #1f2328) !important; border-bottom: 1px solid var(--color-border-muted, #d1d9e0) !important; padding-bottom: 8px !important; } /* TOC list */ #codeberg-readme-toc ul { list-style: none !important; margin: 0 !important; padding: 0 !important; padding-right: 0px !important; overflow-y: auto !important; flex: 1 !important; } #codeberg-readme-toc ul li { margin-bottom: 4px !important; line-height: 1.4 !important; } /* TOC links */ #codeberg-readme-toc a { color: var(--color-fg-default, #1f2328) !important; text-decoration: none !important; display: block !important; padding: 4px 8px !important; border-radius: 4px !important; font-size: 13px !important; transition: background-color 0.2s ease !important; cursor: pointer !important; } #codeberg-readme-toc a:hover { color: var(--color-accent-fg, #0969da) !important; background-color: var(--color-canvas-default, #ffffff) !important; } /* Responsive design */ @media (max-width: 1200px) { .codeberg-toc-layout-container { flex-direction: column !important; } .codeberg-toc-main-content { flex: 1 !important; max-width: 100% !important; } #codeberg-readme-toc { flex: none !important; max-width: 100% !important; position: static !important; margin-top: 20px !important; } } "); (function (require$$0, require$$0$1) { 'use strict'; var jsxRuntimeExports = {}; var jsxRuntime = { get exports() { return jsxRuntimeExports; }, set exports(v) { jsxRuntimeExports = v; } }; var reactJsxRuntime_production_min = {}; /** * @license React * react-jsx-runtime.production.min.js * * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ var f = require$$0, k = Symbol.for("react.element"), l = Symbol.for("react.fragment"), m$1 = Object.prototype.hasOwnProperty, n = f.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner, p = { key: true, ref: true, __self: true, __source: true }; function q(c, a, g) { var b, d = {}, e = null, h = null; void 0 !== g && (e = "" + g); void 0 !== a.key && (e = "" + a.key); void 0 !== a.ref && (h = a.ref); for (b in a) m$1.call(a, b) && !p.hasOwnProperty(b) && (d[b] = a[b]); if (c && c.defaultProps) for (b in a = c.defaultProps, a) void 0 === d[b] && (d[b] = a[b]); return { $$typeof: k, type: c, key: e, ref: h, props: d, _owner: n.current }; } reactJsxRuntime_production_min.Fragment = l; reactJsxRuntime_production_min.jsx = q; reactJsxRuntime_production_min.jsxs = q; (function(module) { { module.exports = reactJsxRuntime_production_min; } })(jsxRuntime); var client = {}; var m = require$$0$1; { client.createRoot = m.createRoot; client.hydrateRoot = m.hydrateRoot; } const name = "codeberg-readme-toc"; function ensureElements() { const readmeSection = document.querySelector("#readme"); if (!readmeSection) { return null; } const container = readmeSection.querySelector(".file-view.markup.markdown"); if (!container) { return null; } const headings = container.querySelectorAll("h1, h2, h3, h4, h5, h6"); if (!container || !headings.length) { return null; } return { container, headings, readmeSection }; } function getToc() { const elements = ensureElements(); if (!elements) { return []; } const tocItems = [...elements.headings].map((heading) => { var _a; const depth = Number(heading.tagName.slice(1)); const anchor = heading.querySelector("a.anchor"); const text = (_a = heading.textContent) == null ? void 0 : _a.trim(); let hash = null; if (anchor && anchor.href) { const url = new URL(anchor.href); hash = url.hash; } else if (heading.id) { hash = '#' + heading.id; } return { depth, text, hash, element: heading }; }).filter(item => item.text && item.hash); return tocItems; } const handleTocClick = (e, element) => { e.preventDefault(); if (element) { element.scrollIntoView({ behavior: 'smooth', block: 'start' }); const hash = element.id ? '#' + element.id : ''; if (hash && window.history.pushState) { window.history.pushState(null, null, hash); } } }; const Toc = ({ toc }) => { return /* @__PURE__ */ jsxRuntimeExports.jsx("ul", { children: toc.map((h, i) => /* @__PURE__ */ jsxRuntimeExports.jsx("li", { style: { paddingLeft: (h.depth - 1) * 16 }, children: /* @__PURE__ */ jsxRuntimeExports.jsx( "a", { href: h.hash, onClick: (e) => handleTocClick(e, h.element), children: h.text } ) }, i)) }); }; function App() { const toc = getToc(); if (!toc.length) { return null; } return /* @__PURE__ */ jsxRuntimeExports.jsxs(jsxRuntimeExports.Fragment, { children: [ /* @__PURE__ */ jsxRuntimeExports.jsx("h2", { className: "h4 mb-3", children: "Table of Contents" }), /* @__PURE__ */ jsxRuntimeExports.jsx(Toc, { toc }) ] }); } async function render() { let root = document.querySelector(`#${name}`); if (root) { return; } const elements = ensureElements(); if (!elements) { return; } const { readmeSection } = elements; const fileHeader = readmeSection.querySelector(".file-header"); const fileView = readmeSection.querySelector(".file-view"); if (!fileHeader || !fileView) { if (readmeSection.firstElementChild) { root = document.createElement("div"); root.id = name; readmeSection.insertBefore(root, readmeSection.firstElementChild); const reactRoot = client.createRoot(root); const app = /* @__PURE__ */ jsxRuntimeExports.jsx(require$$0.StrictMode, { children: /* @__PURE__ */ jsxRuntimeExports.jsx(App, {}) }); reactRoot.render(app); return; } return; } const layoutContainer = document.createElement("div"); layoutContainer.className = "codeberg-toc-layout-container"; const contentWrapper = document.createElement("div"); contentWrapper.className = "codeberg-toc-main-content"; root = document.createElement("div"); root.id = name; if (fileHeader.nextElementSibling) { fileHeader.parentNode.insertBefore(layoutContainer, fileHeader.nextElementSibling); } else { fileHeader.parentNode.appendChild(layoutContainer); } contentWrapper.appendChild(fileView); layoutContainer.appendChild(contentWrapper); layoutContainer.appendChild(root); const reactRoot = client.createRoot(root); const app = /* @__PURE__ */ jsxRuntimeExports.jsx(require$$0.StrictMode, { children: /* @__PURE__ */ jsxRuntimeExports.jsx(App, {}) }); reactRoot.render(app); } function run() { const isRepoPage = window.location.pathname.includes('/src/') || window.location.href.includes('codeberg.org') && !window.location.pathname.includes('/api/'); if (!isRepoPage) { return; } render(); } function initTOC() { run(); const observer = new MutationObserver((mutations) => { let shouldRun = false; mutations.forEach((mutation) => { if (mutation.type === 'childList' && mutation.addedNodes.length > 0) { const hasReadme = Array.from(mutation.addedNodes).some(node => node.nodeType === 1 && (node.querySelector && node.querySelector('#readme')) ); if (hasReadme) { shouldRun = true; } } }); if (shouldRun) { setTimeout(run, 100); } }); observer.observe(document.body, { childList: true, subtree: true }); } if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', () => { initTOC(); }); } else { initTOC(); } })(React, ReactDOM);