// ==UserScript== // @name lc-to-markdown-txt-html // @author wuxin0011 // @version 0.0.3.1 // @namespace https://github.com/wuxin0011/tampermonkey-script/tree/main/lc-to-markdown-txt-html // @description 力扣题目描述,讨论发布内容复制 复制为 markdown、txt、html 等格式 // @icon  // @match https://leetcode.cn/circle/discuss/* // @match https://leetcode.cn/problems/* // @match https://leetcode.cn/contest/weekly-contest-*/problems/* // @match https://leetcode.cn/contest/biweekly-contest-*/problems/* // @require https://cdn.bootcdn.net/ajax/libs/clipboard.js/2.0.11/clipboard.min.js // @require https://cdn.bootcdn.net/ajax/libs/turndown/7.1.2/turndown.min.js // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_getValue // @license MIT // ==/UserScript== (function () { 'use strict'; const url = window.location.href const HTML_CONVERT = '__HTML_CONVERT__' const TXT_CONVERT = '__TXT_CONVERT__' const MARKDOWN_CONVERT = '__MARKDOWN_CONVERT__' const markdownURL = "https://stonehank.github.io/html-to-md/" const isDiscuss = () => url.indexOf('https://leetcode.cn/circle/discuss') != -1 const isProblem = () => url.indexOf('https://leetcode.cn/problems') != -1 const isContest = () => url.indexOf('https://leetcode.cn/contest/weekly-contest') != -1 || url.indexOf('https://leetcode.cn/contest/biweekly-contest') != -1 const isAutoKey = '__auto_pluging_key' + (isDiscuss() ? '__Discuss__' : isProblem() ? '__Problem__' : '__Contest__') // const use = (key) => typeof GM_getValue(key) == 'undefined' ? true : GM_getValue(key) const isUseMarkDown = () => use(MARKDOWN_CONVERT) const isUseTxt = () => use(TXT_CONVERT) const isUseHTML = () => use(HTML_CONVERT) let timerId = null let loadOk = false console.log('markdown', isUseMarkDown(), 'txt', isUseTxt(), 'html', isUseHTML()) const isUsePlugins = () => isUseHTML() || isUseMarkDown() || isUseTxt() const isUsePluginInThis = () => use(isAutoKey) // 当前页面是否使用该插件 let isFindButtonContainer = false const SUPPORT_TYPE = { 'md': 'md', 'txt': 'txt', 'html': 'html' } const buttons = [] const targetClass = 'my-button-target' const BUTTON_ID = `#${targetClass}` for (let i = 0; i < 3; i++) { const temp = document.createElement('button') temp.style.marginLeft = '10px' const type = i == 0 ? SUPPORT_TYPE['md'] : i == 1 ? SUPPORT_TYPE['txt'] : SUPPORT_TYPE['html'] temp.title = `复制为 ${type == 'md' ? 'markdown' : type} 格式` temp.id = `${BUTTON_ID}-${type}` temp.textContent = type temp.copytype = type buttons.push(temp) } const updateDisplay = (element, u) => element && element instanceof HTMLElement ? (element.style.display = u ? 'inline-block' : 'none') : '' // markdown button const markdownButton = buttons[0] updateDisplay(markdownButton, isUseMarkDown()) // txt button const txtButton = buttons[1] updateDisplay(txtButton, isUseTxt()) // html button const htmlButton = buttons[2] updateDisplay(htmlButton, isUseHTML()) function getHtmlContent(className) { const htmlContent = document.querySelector(className) return htmlContent ? htmlContent.innerHTML : '' } function updateElementShow(element) { if (!element instanceof HTMLElement) { return } element.style.display = element.style.display == 'none' ? 'inline-block' : 'none' } function runQuestionActionsContainer() { const className = '[class$=MarkdownContent]'; const questionActionsContainer = document.querySelector('[class*=QuestionActionsContainer]') markdownButton.className = 'e11vgnte0 css-yf7o-BaseButtonComponent-ThemedButton ery7n2v0' htmlButton.className = 'e11vgnte0 css-yf7o-BaseButtonComponent-ThemedButton ery7n2v0' txtButton.className = 'e11vgnte0 css-yf7o-BaseButtonComponent-ThemedButton ery7n2v0' const htmlContent = getHtmlContent(className) runCopy(questionActionsContainer, markdownButton, htmlContent, SUPPORT_TYPE['md']) runCopy(questionActionsContainer, htmlButton, htmlContent, SUPPORT_TYPE['html']) } const toMarkdown = (htmlContent) => { try { var turndownService = new TurndownService() var markdown = turndownService.turndown(htmlContent) return markdown } catch (e) { if (confirm('markdown转换失败,跳转到网站转换?')) { if (window?.navigator?.clipboard?.writeText) { window.navigator.clipboard.writeText(htmlContent).then(() => { window.open(markdownURL, '_blank') }, () => { }) } } else { console.error('convert markdown error default convert txt !', e) const d = document.createElement('div') d.innerHTML = content const txt = handlerText(d.textContent) return txt } } } function runProblems() { // console.log('run problem', url) let buttonClassName = 'relative inline-flex items-center justify-center text-caption px-2 py-1 gap-1 rounded-full bg-fill-secondary text-difficulty-easy dark:text-difficulty-easy' let className = "[data-track-load=description_content]" let titleClassName = '#qd-content [class*=text-title]' const isFlexMode = !!document.querySelector('#__next') if (isContest()) { // console.log('isFlexMode', isFlexMode) if (isFlexMode) { className = ".FN9Jv" titleClassName = '#qd-content a' } else { className = '#base_content .question-content' titleClassName = '#base_content .question-title h3' } } else { // LCP 老版本的 容器 https://leetcode.cn/problems/1ybDKD/description/ if (!document.querySelector(className)) { className = ".FN9Jv" titleClassName = '#qd-content a' } } let title = document.querySelector(titleClassName) const titleTxt = title?.textContent title = title ? '

' + (title?.textContent) + '

' : '' let u = window.location.href let orginUrl = title ? `` + (u) + '' : '' let htmlContent = title + getHtmlContent(className) + orginUrl let container = null // https://leetcode.cn/contest/weekly-contest-312 if (isContest() && !isFlexMode) { if (!isFindButtonContainer) { const c = document.querySelector('.contest-question-info') if (c && !c.querySelector('#lx-markdown-plugins')) { const str = `
  • 插件
  • ` c.innerHTML = c.innerHTML + str container = c.querySelector('.lx-markdown-plugins') if (container) { isFindButtonContainer = true } } } } else { container = document.querySelector(className) if (container) { container = container.previousElementSibling } } if (!container) { console.warn('找不到 容器!', url) return; } markdownButton.className = buttonClassName txtButton.className = buttonClassName htmlButton.className = buttonClassName runCopy(container, txtButton, htmlContent, SUPPORT_TYPE['txt'], titleTxt) runCopy(container, htmlButton, htmlContent, SUPPORT_TYPE['html']) runCopy(container, markdownButton, htmlContent, SUPPORT_TYPE['md']) } function copy(w, element) { if (!element || !(element instanceof HTMLElement)) { return } try { let clipboard = element?.clipboardObject if (clipboard) { //console.log('clipboard destroy') clipboard.destroy(); } clipboard = new ClipboardJS(element, { text: function () { return w; } }) // console.log('update txt >>>>>>>>>') element.clipboardObject = clipboard clipboard.on('success', function (e) { updateButtonStatus(element) }) clipboard.on('error', function (e) { updateButtonStatus(element, 'copy error!') }) } catch (error) { // 如果 clipboardjs 引入失败 使用原生的 // use navigator writeText element.onclick = () => { navigator.clipboard.writeText(w).then(() => { //updateButtonStatus(element) }, () => { updateButtonStatus(element, 'copy error!') }) } } } function runCopy(container, ele, htmlContent, type = SUPPORT_TYPE['md'], title = '') { if (!ele || !container || !htmlContent || !type) { return } if (!(container instanceof HTMLElement && ele instanceof HTMLElement)) { return; } // append if (!document.getElementById(ele.id)) { ele.originClass = ele.className container.appendChild(ele) } else { if (timerId != null) { window.clearInterval(timerId) timerId = null } // 加载完成 初始化 loadOk = true // initConmand() updateButtonStatus(ele, ele.copytype, '', 100) } if (type == SUPPORT_TYPE['md']) { const markdown = toMarkdown(htmlContent) copy(markdown, ele) } else if (type == SUPPORT_TYPE['txt']) { const d = document.createElement('div') d.innerHTML = htmlContent const txt = handlerText(d.textContent, title) copy(txt, ele) } else if (type == SUPPORT_TYPE['html']) { // html copy(htmlContent, ele) } else { console.warn('no support format ' + type) } } const MAX_LEN = 80 const handlerText = (str, title = '') => { if (!str) return str // 移出空白字符 str = str.replaceAll(' ', '') str = str.replaceAll('​​​​​​​​​​​​​​​​​​​​​​​', '') str = str.replaceAll(' ', '') str = str.replace('。', "。\n") str = str.replace('231', '2^31') str = str.replace(/10(?!0)(\d+)/g, '10^$1') // 这个位分界线 str = str.replace(/\n{2,}/g, "\n") str = str.replace('http', '\n\nhttp') str = str.replaceAll('示例', "\n示例") str = str.replace('提示', "\n提示") if (title != '') { str = str.replace(title, title + "\n\n") } return str } const updateButtonStatus = (element, newText = 'copied!', newClass = '', timeout = 1500) => { if (!element) { return; } // console.log('update button status', element, newText) element.textContent = newText if (newClass) { element.className = newClass } setTimeout(() => { element.textContent = element.copytype element.className = element.originClass }, timeout) } const initConmand = () => { try { const isAutoPluginCommand = GM_registerMenuCommand(`当前页面 ${isUsePluginInThis() ? '关闭' : '启用'} 插件 `, () => { GM_setValue(isAutoKey, !isUsePluginInThis()) window.location.reload() }, { title: `当前页面 ${isUseHTML() ? '关闭' : '启用'} 插件 ` }) if (!isUsePluginInThis()) { return; } // const message = (u, type) => u ? '关闭' : '启用' + (type == 'md' ? ' markdown ' : ` ${type} `) const html_to_markdown = GM_registerMenuCommand(`${isUseMarkDown() ? '关闭' : '启用'} markdown `, () => { GM_setValue(MARKDOWN_CONVERT, !isUseMarkDown()) updateElementShow(markdownButton) }, { title: `点击 ${isUseMarkDown() ? '关闭' : '启用'} markdown ` }) const html_to_txt = GM_registerMenuCommand(`${isUseTxt() ? '关闭' : '启用'} txt `, () => { GM_setValue(TXT_CONVERT, !isUseTxt()) updateElementShow(txtButton) }, { title: `点击 ${isUseTxt() ? '关闭' : '启用'} txt ` }) const html_to_html = GM_registerMenuCommand(`${isUseHTML() ? '关闭' : '启用'} html `, () => { GM_setValue(HTML_CONVERT, !isUseHTML()) updateElementShow(htmlButton) }, { title: `点击 ${isUseHTML() ? '关闭' : '启用'} html ` }) const html_to_markdown_web = GM_registerMenuCommand('html转换markdown网站', () => { window.open(markdownURL, '_blank') }, { title: '如果格式转换有问题,请复制为 html 然后用这个网站转换' }) } catch (e) { console.log('init command error', e) } } let times = 0 const MAX_CNT = 10 initConmand() // console.log('isUsePlugins()',isUsePlugins()) const start = () => { if (!isUsePlugins()) { return; } timerId = setInterval(() => { let support = true if (isDiscuss()) { runQuestionActionsContainer() } else if (isProblem() || isContest()) { runProblems() } else { support = false } times += 1 if (times > MAX_CNT && timerId != null) { window.clearInterval(timerId) timerId = null } if (!support) { console.warn('No support address ! ', url) if (timerId != null) { window.clearInterval(timerId) } return } }, 3000); } window.onload = () => { times = 0 if (!isUsePluginInThis()) { return; } start() } // 监听地址改变 // 重新修改描述 window.addEventListener("urlchange", () => { if (!loadOk) { return } let pretitle = document.querySelector('title').textContent let titleId = null let updateTimes = 0 titleId = setInterval(() => { let curTitle = document.querySelector('title').textContent if (isDiscuss()) { runQuestionActionsContainer() } else if (isProblem()) { runProblems() } if (pretitle == curTitle && updateTimes >= 3) { window.clearInterval(titleId) return } updateTimes += 1 }, 1500); }) })();