// ==UserScript== // @name 力扣题目转Markdown // @name:en LeetCode Promlem to Markdown // @namespace https://gabrielxd.top/ // @version 1.2.0 // @description 转换力扣题目为Markdown格式并复制到剪贴板 // @description:en Convert the LeetCode problems to markdown and copy it to the clipboard. // @author GabrielxD // @match *://leetcode.cn/problems/* // @icon  // @require https://unpkg.com/turndown/dist/turndown.js // @require https://cdn.jsdelivr.net/npm/msg-alert@1.0.0-beta.2/dist/msg-alert.min.js // @grant GM_setClipboard // @grant GM_addStyle // @grant GM_registerMenuCommand // @license MIT // @downloadURL https://update.greasyfork.icu/scripts/448601/%E5%8A%9B%E6%89%A3%E9%A2%98%E7%9B%AE%E8%BD%ACMarkdown.user.js // @updateURL https://update.greasyfork.icu/scripts/448601/%E5%8A%9B%E6%89%A3%E9%A2%98%E7%9B%AE%E8%BD%ACMarkdown.meta.js // ==/UserScript== // 添加复制按钮样式 GM_addStyle(` .copy-btns { display: inline-block; } .copy-btns > button { margin-left: 10px; vertical-align: middle; font-size: 12px; background: transparent; border: none; border-radius: 3px; box-shadow: inset 0px 0px 0px 1px rgba(var(--dsw-green-standard-rgb), 1); color: rgba(var(--dsw-green-standard-rgb), 1); cursor: pointer; outline: none; }`); const turndownService = new TurndownService({ emDelimiter: '*', bulletListMarker: '-' }); turndownService.addRule('strikethrough', { filter: ['pre'], replacement: (content, node) => '\n```txt\n' + node.innerText.trim() + '\n```\n' }); turndownService.addRule('strikethrough', { filter: ['sup'], replacement: content => '^' + content }); // 接收HTML字符串转成Markdown格式 const htmlToMd = htmlStr => { console.log('转换开始...'); return turndownService.turndown(htmlStr.replace(/

 <\/p>/g, '
')); } const getDescMd = () => htmlToMd(descEle.innerHTML); const getCodeMd = () => document.querySelector('input[name=lang]').value + '\n' + document.querySelector('input[name=code]').value + '\n```\n'; const getTitle = () => titleEle.firstElementChild.innerText; // 查询到的节点缓存在变量中 let titleEle, descEle; // 拼接解题模板用到的字符串 const afterDesc = '\n\n \n\n## 解题\n\n### 方法一:\n\n#### 思路\n\n\n\n#### 代码\n\n```'; // 复制标题 const copyTitleBtn = document.createElement('button'); copyTitleBtn.innerText = '复制标题'; copyTitleBtn.addEventListener('click', copyTitleHandler); GM_registerMenuCommand("复制标题", copyTitleHandler); // 复制题目 const copyDescBtn = document.createElement('button'); copyDescBtn.innerText = '复制题目'; copyDescBtn.addEventListener('click', copyDescHandler); GM_registerMenuCommand("复制题目", copyDescHandler); // 生成解题模板并复制 const copySolnTmplBtn = document.createElement('button'); copySolnTmplBtn.innerText = '解题模板'; copySolnTmplBtn.addEventListener('click', copySolnTmplHandler); GM_registerMenuCommand("解题模板", copySolnTmplHandler); // 放入功能按钮 const copyBtnsEle = document.createElement('div'); copyBtnsEle.className = 'copy-btns'; copyBtnsEle.appendChild(copyTitleBtn); copyBtnsEle.appendChild(copyDescBtn); copyBtnsEle.appendChild(copySolnTmplBtn); // 重试计数器 let retryCounter = 0; // 初始化元素 function initElements(targetEle) { titleEle = targetEle.querySelector('h4[data-cypress=QuestionTitle]'); descEle = targetEle.querySelector('div.notranslate:not(#question-detail-main-tabs)'); titleEle.appendChild(copyBtnsEle); } function firstInit() { const targetEle = document.querySelector("div.notranslate#question-detail-main-tabs"); if (targetEle && targetEle.querySelector('h4[data-cypress="QuestionTitle"]') !== null) { initElements(targetEle); } else { if (retryCounter++ > 10) { console.log('获取元素失败, 重试次数过多, 请刷新页面重试或关闭此脚本.'); message.error({ text: '获取元素失败, 重试次数过多, 请刷新页面重试或关闭此脚本.', duration: 2000 }); } else { console.log('获取元素失败, 200ms后重新获取...'); setTimeout(init, 200); } } } const init = () => { // 监听tab栏变化 const observer = new MutationObserver((mutationList, observer) => { mutationList.forEach((mutation) => { const tabEle = mutation.addedNodes[0]; if (mutation.type == 'childList' && typeof(tabEle?.className) === 'string' && tabEle.className.startsWith('description')) { initElements(tabEle); } }); }); observer.observe(document.body, { childList: true, subtree: true }); firstInit(); } function copyTitleHandler() { const title = getTitle(); // 如果标题中有题号, 则舍弃题号取点后的部分作为短标题返回 GM_setClipboard(title.includes('.') ? title.substring(title.lastIndexOf('.') + 2) : title); message.success({ text: '复制标题成功', duration: 800 }); } function copyDescHandler() { GM_setClipboard(getDescMd()); message.success({ text: '复制题目成功', duration: 800 }); } function copySolnTmplHandler() { GM_setClipboard(`## 题目\n\n[${titleEle.firstElementChild.innerText}](${location.href})\n\n---\n\n${getDescMd()}${afterDesc}${getCodeMd()}`); message.success({ text: '复制成功', duration: 800 }); } (() => { 'use strict'; window.addEventListener('load', init); })();