// ==UserScript== // @name lc-to-markdown-txt-html // @author wuxin0011 // @version 0.0.5 // @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://cdnjs.cloudflare.com/ajax/libs/clipboard.js/2.0.11/clipboard.min.js // @require https://unpkg.com/turndown@7.2.0/dist/turndown.js // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_getValue // @license MIT // @downloadURL https://update.greasyfork.icu/scripts/499334/lc-to-markdown-txt-html.user.js // @updateURL https://update.greasyfork.icu/scripts/499334/lc-to-markdown-txt-html.meta.js // ==/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') console.log('is find', !!document.querySelector(className)) 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 ? '