lc-to-markdown-txt-html

力扣题目描述,讨论发布内容复制 复制为 markdown、txt、html 等格式

当前为 2024-04-08 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name lc-to-markdown-txt-html
  3. // @author wuxin0011
  4. // @version 0.0.1
  5. // @namespace https://github.com/wuxin0011/tampermonkey-script/tree/main/lc-to-markdown-txt-html
  6. // @description 力扣题目描述,讨论发布内容复制 复制为 markdown、txt、html 等格式
  7. // @icon 
  8. // @match https://leetcode.cn/circle/discuss/*
  9. // @match https://leetcode.cn/problems/*
  10. // @require https://cdn.bootcdn.net/ajax/libs/clipboard.js/2.0.11/clipboard.min.js
  11. // @require https://cdn.bootcdn.net/ajax/libs/turndown/7.1.2/turndown.min.js
  12. // @grant GM_registerMenuCommand
  13. // @grant GM_setValue
  14. // @grant GM_getValue
  15. // @license MIT
  16. // ==/UserScript==
  17.  
  18.  
  19.  
  20.  
  21.  
  22. (function () {
  23. 'use strict';
  24. const url = window.location.href
  25. const HTML_CONVERT = '__HTML_CONVERT__'
  26. const TXT_CONVERT = '__TXT_CONVERT__'
  27. const MARKDOWN_CONVERT = '__MARKDOWN_CONVERT__'
  28. const markdownURL = "https://stonehank.github.io/html-to-md/"
  29.  
  30. const isDiscuss = () => url.indexOf('https://leetcode.cn/circle/discuss') != -1
  31. const isProblem = () => url.indexOf('https://leetcode.cn/problems') != -1
  32.  
  33.  
  34.  
  35. //
  36. const use = (key) => typeof GM_getValue(key) == 'undefined' ? true : GM_getValue(key)
  37. const isUseMarkDown = () => use(MARKDOWN_CONVERT)
  38. const isUseTxt = () => use(TXT_CONVERT)
  39. const isUseHTML = () => use(HTML_CONVERT)
  40. console.log('markdown', isUseMarkDown(), 'txt', isUseTxt(), 'html', isUseHTML())
  41.  
  42.  
  43. const SUPPORT_TYPE = {
  44. 'md': 'md',
  45. 'txt': 'txt',
  46. 'html': 'html'
  47. }
  48.  
  49. const buttons = []
  50. const targetClass = 'my-button-target'
  51. const BUTTON_ID = `#${targetClass}`
  52. for (let i = 0; i < 3; i++) {
  53. const temp = document.createElement('button')
  54. temp.style.marginLeft = '10px'
  55. const type = i == 0 ? SUPPORT_TYPE['md'] : i == 1 ? SUPPORT_TYPE['txt'] : SUPPORT_TYPE['html']
  56. temp.title = `复制为 ${type == 'md' ? 'markdown' : type} 格式`
  57. temp.textContent = type
  58. buttons.push(temp)
  59. }
  60.  
  61. // markdown button
  62. const markdownButton = buttons[0]
  63. markdownButton.id = targetClass
  64. markdownButton.style.display = isUseMarkDown() ? 'inline-block' : 'none'
  65.  
  66. // txt button
  67. const txtButton = buttons[1]
  68. txtButton.style.display = isUseTxt() ? 'inline-block' : 'none'
  69.  
  70. // html button
  71. const htmlButton = buttons[2]
  72. htmlButton.style.display = isUseHTML() ? 'inline-block' : 'none'
  73.  
  74. function getHtmlContent(className) {
  75. const htmlContent = document.querySelector(className)
  76. return htmlContent ? htmlContent.innerHTML : ''
  77. }
  78.  
  79. function updateElementShow(element) {
  80. if (!element instanceof HTMLElement) {
  81. return
  82. }
  83. element.style.display = element.style.display == 'none' ? 'inline-block' : 'none'
  84. }
  85.  
  86.  
  87. function runQuestionActionsContainer() {
  88. const className = '[class$=MarkdownContent]';
  89. const questionActionsContainer = document.querySelector('[class*=QuestionActionsContainer]')
  90. markdownButton.className = 'e11vgnte0 css-yf7o-BaseButtonComponent-ThemedButton ery7n2v0'
  91. htmlButton.className = 'e11vgnte0 css-yf7o-BaseButtonComponent-ThemedButton ery7n2v0'
  92. txtButton.className = 'e11vgnte0 css-yf7o-BaseButtonComponent-ThemedButton ery7n2v0'
  93. const htmlContent = getHtmlContent(className)
  94. runCopy(questionActionsContainer, markdownButton, htmlContent, SUPPORT_TYPE['md'])
  95. runCopy(questionActionsContainer, htmlButton, htmlContent, SUPPORT_TYPE['html'])
  96. }
  97.  
  98.  
  99.  
  100. const toMarkdown = (htmlContent) => {
  101. try {
  102. var turndownService = new TurndownService()
  103. var markdown = turndownService.turndown(htmlContent)
  104. return markdown
  105. } catch (e) {
  106. if (confirm('markdown转换失败,跳转到网站转换?')) {
  107. if (window?.navigator?.clipboard?.writeText) {
  108. window.navigator.clipboard.writeText(htmlContent).then(() => {
  109. window.open(markdownURL, '_blank')
  110. }, () => {
  111.  
  112. })
  113. }
  114. } else {
  115. console.error('convert markdown error default convert txt !', e)
  116. const d = document.createElement('div')
  117. d.innerHTML = content
  118. const txt = handlerText(d.textContent)
  119. return txt
  120. }
  121. }
  122. }
  123.  
  124.  
  125.  
  126. function runProblems() {
  127. const 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'
  128. const className = "[data-track-load=description_content]"
  129. let title = document.querySelector('#qd-content [class*=text-title]')
  130. const titleTxt = title.textContent
  131. title = title ? '<h2>' + (title?.textContent) + '</h2>' : ''
  132. let htmlContent = title + getHtmlContent(className)
  133. let container = document.querySelector(className)
  134. container = container.previousElementSibling
  135. markdownButton.className = buttonClassName
  136. txtButton.className = buttonClassName
  137. htmlButton.className = buttonClassName
  138. runCopy(container, markdownButton, htmlContent, SUPPORT_TYPE['md'])
  139. runCopy(container, txtButton, htmlContent, SUPPORT_TYPE['txt'], titleTxt)
  140. runCopy(container, htmlButton, htmlContent, SUPPORT_TYPE['html'])
  141. }
  142.  
  143.  
  144. function copy(w, element) {
  145. if (!element || !(element instanceof HTMLElement)) {
  146. return
  147. }
  148. try {
  149. const clipboard = new ClipboardJS(element, {
  150. text: function () {
  151. return w;
  152. }
  153. })
  154. clipboard.on('success', function (e) {
  155. updateButtonStatus(element)
  156. })
  157. clipboard.on('error', function (e) {
  158. updateButtonStatus(element, 'copy error!')
  159. })
  160. } catch (error) {
  161. // 如果 clipboardjs 引入失败 使用原生的
  162. // use navigator writeText
  163. element.onclick = () => {
  164. navigator.clipboard.writeText(w).then(() => {
  165. updateButtonStatus(element)
  166. }, () => {
  167. updateButtonStatus(element, 'copy error!')
  168. })
  169. }
  170.  
  171. }
  172.  
  173. }
  174.  
  175. function runCopy(container, ele, htmlContent, type = SUPPORT_TYPE['md'], title = '') {
  176.  
  177. if (!ele || !container || !htmlContent || !type) {
  178. return
  179. }
  180. if (!(container instanceof HTMLElement && ele instanceof HTMLElement)) {
  181. return;
  182. }
  183.  
  184.  
  185.  
  186. // append
  187. container.appendChild(ele)
  188.  
  189.  
  190.  
  191. if (type == SUPPORT_TYPE['md']) {
  192. const markdown = toMarkdown(htmlContent)
  193. copy(markdown, ele)
  194. } else if (type == SUPPORT_TYPE['txt']) {
  195. const d = document.createElement('div')
  196. d.innerHTML = htmlContent
  197. const txt = handlerText(d.textContent, title)
  198. copy(txt, ele)
  199. } else if (type == SUPPORT_TYPE['html']) {
  200. // html
  201. copy(htmlContent, ele)
  202. } else {
  203. console.warn('no support format ' + type)
  204. }
  205.  
  206. }
  207.  
  208. const handlerText = (str, title = '') => {
  209. if (!str) return str
  210. function isIgnore(c) {
  211. return c == '\t' || c == '\b' || c == '\n' || c == '\f'
  212. }
  213. let newstr = ''
  214. let find = str.indexOf('提示')
  215. let tipPos = find == -1 ? str.length : find
  216. for (let i = 0; i < tipPos; i++) {
  217. if (isIgnore(str.charAt(i))) {
  218. continue;
  219. }
  220. newstr = newstr + str.charAt(i)
  221. }
  222. newstr = newstr.replaceAll('示例', '\n\n示例')
  223.  
  224. if (tipPos != str.length) {
  225. newstr = newstr + str.substring(find)
  226. newstr = newstr.replace('提示', '\n\n提示')
  227. }
  228.  
  229. if (title) {
  230. newstr = newstr.replace(title, `${title}\n\n`)
  231. }
  232.  
  233. let i = newstr.length
  234. for (; i >= 0; i--) {
  235. let c = newstr.charAt(i)
  236. if (!(isIgnore(c) || c == ' ')) {
  237. break
  238. }
  239. }
  240. return newstr.substring(0, i)
  241. }
  242.  
  243.  
  244. const updateButtonStatus = (element, newText = 'copy success !', newClass = '', timeout = 1500) => {
  245. if (!element) {
  246. return;
  247. }
  248. const origin = element.textContent
  249. const originClass = element.className
  250. element.textContent = newText
  251. if (newClass) {
  252. element.className = newClass
  253. }
  254. setTimeout(() => {
  255. element.textContent = origin
  256. element.className = originClass
  257. }, timeout)
  258. }
  259.  
  260.  
  261.  
  262.  
  263. const initConmand = () => {
  264. try {
  265.  
  266. // const message = (u, type) => u ? '关闭' : '启用' + (type == 'md' ? ' markdown ' : ` ${type} `)
  267.  
  268. const html_to_markdown = GM_registerMenuCommand(`${isUseMarkDown() ? '关闭' : '启用'} markdown `, () => {
  269. GM_setValue(MARKDOWN_CONVERT, !isUseMarkDown())
  270. updateElementShow(markdownButton)
  271. }, { title: `点击 ${isUseMarkDown() ? '关闭' : '启用'} markdown ` })
  272.  
  273.  
  274. const html_to_txt = GM_registerMenuCommand(`${isUseTxt() ? '关闭' : '启用'} txt `, () => {
  275. GM_setValue(TXT_CONVERT, !isUseTxt())
  276. updateElementShow(txtButton)
  277. }, { title: `点击 ${isUseTxt() ? '关闭' : '启用'} txt ` })
  278.  
  279. const html_to_html = GM_registerMenuCommand(`${isUseHTML() ? '关闭' : '启用'} html `, () => {
  280. GM_setValue(HTML_CONVERT, !isUseHTML())
  281. updateElementShow(htmlButton)
  282. }, { title: `点击 ${isUseHTML() ? '关闭' : '启用'} html ` })
  283.  
  284.  
  285.  
  286.  
  287. const html_to_markdown_web = GM_registerMenuCommand('html转换markdown网站', () => {
  288. window.open(markdownURL, '_blank')
  289. }, { title: '如果格式转换有问题,请复制为 html 然后用这个网站转换' })
  290.  
  291.  
  292.  
  293. } catch (e) {
  294. console.log('init command error', e)
  295. }
  296.  
  297. }
  298.  
  299.  
  300.  
  301. window.onload = () => {
  302. let support = true
  303. let timer = setInterval(() => {
  304.  
  305. if (isDiscuss()) {
  306. runQuestionActionsContainer()
  307. } else if (isProblem()) {
  308. runProblems()
  309. } else {
  310. support = false
  311. }
  312.  
  313. if (support) {
  314. initConmand()
  315. const my_button = document.querySelector(BUTTON_ID)
  316. if (my_button) {
  317. window.clearInterval(timer)
  318. }
  319. } else {
  320. console.warn('No support address ! ', url)
  321. window.clearInterval(timer)
  322. }
  323. }, 3000);
  324. }
  325. })();