lc-to-markdown-txt-html

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

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

  1. // ==UserScript==
  2. // @name lc-to-markdown-txt-html
  3. // @author wuxin0011
  4. // @version 0.0.2-3.beta
  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. (function () {
  19. 'use strict';
  20. const url = window.location.href
  21. const HTML_CONVERT = '__HTML_CONVERT__'
  22. const TXT_CONVERT = '__TXT_CONVERT__'
  23. const MARKDOWN_CONVERT = '__MARKDOWN_CONVERT__'
  24. const markdownURL = "https://stonehank.github.io/html-to-md/"
  25.  
  26. const isDiscuss = () => url.indexOf('https://leetcode.cn/circle/discuss') != -1
  27. const isProblem = () => url.indexOf('https://leetcode.cn/problems') != -1
  28. //
  29. const use = (key) => typeof GM_getValue(key) == 'undefined' ? true : GM_getValue(key)
  30. const isUseMarkDown = () => use(MARKDOWN_CONVERT)
  31. const isUseTxt = () => use(TXT_CONVERT)
  32. const isUseHTML = () => use(HTML_CONVERT)
  33. let timerId = null
  34. let loadOk = false
  35. console.log('markdown', isUseMarkDown(), 'txt', isUseTxt(), 'html', isUseHTML())
  36.  
  37.  
  38. const SUPPORT_TYPE = {
  39. 'md': 'md',
  40. 'txt': 'txt',
  41. 'html': 'html'
  42. }
  43.  
  44.  
  45.  
  46. const buttons = []
  47. const targetClass = 'my-button-target'
  48. const BUTTON_ID = `#${targetClass}`
  49. for (let i = 0; i < 3; i++) {
  50. const temp = document.createElement('button')
  51. temp.style.marginLeft = '10px'
  52. const type = i == 0 ? SUPPORT_TYPE['md'] : i == 1 ? SUPPORT_TYPE['txt'] : SUPPORT_TYPE['html']
  53. temp.title = `复制为 ${type == 'md' ? 'markdown' : type} 格式`
  54. temp.id = `${BUTTON_ID}-${type}`
  55. temp.textContent = type
  56. temp.copytype = type
  57. buttons.push(temp)
  58. }
  59.  
  60.  
  61. const updateDisplay = (element, u) => element && element instanceof HTMLElement ? (element.style.display = u ? 'inline-block' : 'none') : ''
  62. // markdown button
  63. const markdownButton = buttons[0]
  64. updateDisplay(markdownButton, isUseMarkDown())
  65.  
  66. // txt button
  67. const txtButton = buttons[1]
  68. updateDisplay(txtButton, isUseTxt())
  69.  
  70. // html button
  71. const htmlButton = buttons[2]
  72. updateDisplay(htmlButton, isUseHTML())
  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. // console.log('run problem', url)
  128. 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'
  129. const className = "[data-track-load=description_content]"
  130. let title = document.querySelector('#qd-content [class*=text-title]')
  131. const titleTxt = title?.textContent
  132. title = title ? '<h2>' + (title?.textContent) + '</h2>' : ''
  133. let u = window.location.href
  134. let orginUrl = title ? `<a href="${u}">` + (u) + '</a>' : ''
  135. let htmlContent = title + getHtmlContent(className) + orginUrl
  136. let container = document.querySelector(className)
  137. if (!container) {
  138. console.warn('找不到 容器!', url)
  139. return;
  140. }
  141. container = container.previousElementSibling
  142. markdownButton.className = buttonClassName
  143. txtButton.className = buttonClassName
  144. htmlButton.className = buttonClassName
  145. runCopy(container, txtButton, htmlContent, SUPPORT_TYPE['txt'], titleTxt)
  146. runCopy(container, htmlButton, htmlContent, SUPPORT_TYPE['html'])
  147. runCopy(container, markdownButton, htmlContent, SUPPORT_TYPE['md'])
  148. }
  149.  
  150.  
  151. function copy(w, element) {
  152. if (!element || !(element instanceof HTMLElement)) {
  153. return
  154. }
  155.  
  156. try {
  157. let clipboard = element?.clipboardObject
  158. if (clipboard) {
  159. //console.log('clipboard destroy')
  160. clipboard.destroy();
  161. }
  162. clipboard = new ClipboardJS(element, {
  163. text: function () {
  164. return w;
  165. }
  166. })
  167. // console.log('update txt >>>>>>>>>')
  168. element.clipboardObject = clipboard
  169. clipboard.on('success', function (e) {
  170. updateButtonStatus(element)
  171. })
  172. clipboard.on('error', function (e) {
  173. updateButtonStatus(element, 'copy error!')
  174. })
  175.  
  176.  
  177. } catch (error) {
  178. // 如果 clipboardjs 引入失败 使用原生的
  179. // use navigator writeText
  180. element.onclick = () => {
  181. navigator.clipboard.writeText(w).then(() => {
  182. //updateButtonStatus(element)
  183. }, () => {
  184. updateButtonStatus(element, 'copy error!')
  185. })
  186. }
  187.  
  188. }
  189.  
  190. }
  191.  
  192.  
  193.  
  194.  
  195. function runCopy(container, ele, htmlContent, type = SUPPORT_TYPE['md'], title = '') {
  196.  
  197. if (!ele || !container || !htmlContent || !type) {
  198. return
  199. }
  200. if (!(container instanceof HTMLElement && ele instanceof HTMLElement)) {
  201. return;
  202. }
  203. // append
  204. if (!document.getElementById(ele.id)) {
  205. ele.originClass = ele.className
  206. container.appendChild(ele)
  207. } else {
  208. if (timerId != null) {
  209. window.clearInterval(timerId)
  210. timerId = null
  211. }
  212. // 加载完成 初始化
  213. loadOk = true
  214. initConmand()
  215. updateButtonStatus(ele, ele.copytype, '', 100)
  216. }
  217.  
  218. if (type == SUPPORT_TYPE['md']) {
  219. const markdown = toMarkdown(htmlContent)
  220. copy(markdown, ele)
  221. } else if (type == SUPPORT_TYPE['txt']) {
  222. const d = document.createElement('div')
  223. d.innerHTML = htmlContent
  224. const txt = handlerText(d.textContent, title)
  225. copy(txt, ele)
  226. } else if (type == SUPPORT_TYPE['html']) {
  227. // html
  228. copy(htmlContent, ele)
  229. } else {
  230. console.warn('no support format ' + type)
  231. }
  232.  
  233. }
  234.  
  235.  
  236. const MAX_LEN = 80
  237.  
  238.  
  239. const handlerText = (str, title = '') => {
  240. if (!str) return str
  241. // 移出空白字符
  242. str = str.replaceAll(' ', '')
  243. str = str.replaceAll('&nbsp;', '')
  244. str = str.replace('。', "。\n")
  245. str = str.replace('231', '2^31')
  246. str = str.replace(/10(\d?)/g, '10^$1')
  247. str = str.replace(/10\^0000/g, '100000')
  248. str = str.replace(/10\^000/g, '10000')
  249. str = str.replace(/10\^00/g, '1000')
  250.  
  251. // 这个位分界线
  252. str = str.replace(/\n{2,}/g, "\n")
  253.  
  254.  
  255. str = str.replace('http','\n\nhttp')
  256. str = str.replaceAll('示例', "\n示例")
  257. str = str.replace('提示', "\n提示")
  258. if(title != '') {
  259. str = str.replace(title, title + "\n\n")
  260. }
  261. return str
  262. }
  263.  
  264.  
  265. const updateButtonStatus = (element, newText = 'copied!', newClass = '', timeout = 1500) => {
  266. if (!element) {
  267. return;
  268. }
  269. // console.log('update button status', element, newText)
  270. element.textContent = newText
  271. if (newClass) {
  272. element.className = newClass
  273. }
  274. setTimeout(() => {
  275. element.textContent = element.copytype
  276. element.className = element.originClass
  277. }, timeout)
  278. }
  279.  
  280.  
  281.  
  282.  
  283. const initConmand = () => {
  284. try {
  285.  
  286. // const message = (u, type) => u ? '关闭' : '启用' + (type == 'md' ? ' markdown ' : ` ${type} `)
  287.  
  288. const html_to_markdown = GM_registerMenuCommand(`${isUseMarkDown() ? '关闭' : '启用'} markdown `, () => {
  289. GM_setValue(MARKDOWN_CONVERT, !isUseMarkDown())
  290. updateElementShow(markdownButton)
  291. }, { title: `点击 ${isUseMarkDown() ? '关闭' : '启用'} markdown ` })
  292.  
  293.  
  294. const html_to_txt = GM_registerMenuCommand(`${isUseTxt() ? '关闭' : '启用'} txt `, () => {
  295. GM_setValue(TXT_CONVERT, !isUseTxt())
  296. updateElementShow(txtButton)
  297. }, { title: `点击 ${isUseTxt() ? '关闭' : '启用'} txt ` })
  298.  
  299. const html_to_html = GM_registerMenuCommand(`${isUseHTML() ? '关闭' : '启用'} html `, () => {
  300. GM_setValue(HTML_CONVERT, !isUseHTML())
  301. updateElementShow(htmlButton)
  302. }, { title: `点击 ${isUseHTML() ? '关闭' : '启用'} html ` })
  303.  
  304.  
  305.  
  306.  
  307. const html_to_markdown_web = GM_registerMenuCommand('html转换markdown网站', () => {
  308. window.open(markdownURL, '_blank')
  309. }, { title: '如果格式转换有问题,请复制为 html 然后用这个网站转换' })
  310.  
  311.  
  312.  
  313. } catch (e) {
  314. console.log('init command error', e)
  315. }
  316.  
  317. }
  318.  
  319.  
  320. let times = 0
  321.  
  322.  
  323. const start = () => {
  324. timerId = setInterval(() => {
  325. let support = true
  326. if (isDiscuss()) {
  327. runQuestionActionsContainer()
  328. } else if (isProblem()) {
  329. runProblems()
  330. } else {
  331. support = false
  332. }
  333. times += 1
  334. if (times > 10 && timerId != null) {
  335. window.clearInterval(timerId)
  336. timerId = null
  337. }
  338. if (!support) {
  339. console.warn('No support address ! ', url)
  340. if (timerId != null) {
  341. window.clearInterval(timerId)
  342. }
  343. return
  344. }
  345. }, 3000);
  346.  
  347.  
  348. }
  349.  
  350.  
  351.  
  352. window.onload = () => {
  353.  
  354. times = 0
  355. start()
  356. }
  357.  
  358.  
  359.  
  360. // 监听地址改变
  361. // 重新修改描述
  362. window.addEventListener("urlchange", () => {
  363.  
  364. if (!loadOk) {
  365. return
  366. }
  367. // console.log('url is change ...')
  368. // console.log('ok ok')
  369. let pretitle = document.querySelector('title').textContent
  370. let titleId = null
  371. let updateTimes = 0
  372. titleId = setInterval(() => {
  373. let curTitle = document.querySelector('title').textContent
  374. if (isDiscuss()) {
  375. runQuestionActionsContainer()
  376. } else if (isProblem()) {
  377. runProblems()
  378. }
  379. if (pretitle == curTitle && updateTimes >= 3) {
  380. window.clearInterval(titleId)
  381. return
  382. }
  383. updateTimes += 1
  384. }, 1500);
  385.  
  386. })
  387.  
  388. })();