├── LICENSE ├── README.md ├── img └── image-20220503194601369.png └── leetcode-helper.js /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2022 tonngw 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | - ## 介绍 2 | 3 | 此油猴脚本是一个针对 LeetCode 题库,方便写题解的小工具。 4 | 5 | > 如果对你有帮助别忘了点个 `Star` 吖~。 6 | 7 | ## 功能 8 | 9 | 1. **复制**题目描述,并存入剪切板 10 | 2. 复制题目描述**生成**当前题目的题解模板,并存入剪切板 11 | 12 | 大多数情况下一道题目只会写一种做法,这里提供了一套简洁的模板,模板来自 `AcWing`。 13 | 14 | ## 快速开始 15 | 16 | `Alt + C` 生成当前题目的题解模板,打开写题解页面,`Ctrl + A` + `Ctrl + V` 一气呵成,开始写自己的逻辑。 17 | 18 | ## 安装 19 | 20 | **环境:** 21 | 22 | 1. Chrome 浏览器 / Edge 浏览器 23 | 2. 安装 `Tampermonkey` 油猴插件,这块大家自行搜索安装,网上有很多教程。 24 | 25 | **Option 1**:直接在 [https://greasyfork.org/zh-CN](https://greasyfork.org/zh-CN) 搜索 `leetcode-helper` 安装即可,或者通过 [此链接](https://greasyfork.org/zh-CN/scripts/444408-leetcode-helper) 直接安装。「推荐」 26 | 27 | **Option 2:**脚本源代码已经放在了[我的 GitHub 仓库](https://github.com/tonngw/leetcode-helper)中,可以复制自行安装。 28 | 29 | ## 如何使用 30 | 31 | ### 快捷键「推荐」 32 | 33 | - 功能 1 - `Alt + T(timu)` 34 | - 功能 2 - `Alt + C(copy)` 35 | 36 | 大家可以自行在源代码中修改快捷键 37 | 38 | ### 页面按钮 39 | 40 | ![demo.jpg](https://cdn.acwing.com/media/article/image/2023/05/30/52520_3f3cb130fe-demo.jpg) 41 | 42 | ### 右键菜单 43 | 44 | 右键 - `Tampermonkey`,可以看到 `leetcode-helper` 提供的功能。 45 | 46 | ### 插件栏 47 | 48 | 点击油猴插件,可以看到 `leetcode-helper` 提供的功能。 49 | 50 | ## 实现效果 51 | 52 | ![demo2.jpg](https://cdn.acwing.com/media/article/image/2023/05/30/52520_868ee009fe-demo2.jpg) 53 | 54 | ## 致谢 55 | 56 | 感谢以下作者提供的思路 57 | 58 | - [https://github.com/ZimoLoveShuang/leetcode-helper](https://github.com/ZimoLoveShuang/leetcode-helper) 59 | 60 | ## 参考资料 61 | 62 | - https://github.com/mixmark-io/turndown 63 | - https://bbs.tampermonkey.net.cn/ 64 | - https://github.com/t4t5/sweetalert 65 | - https://blog.csdn.net/u010598445/article/details/108880602 -------------------------------------------------------------------------------- /img/image-20220503194601369.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tonngw/leetcode-helper/bc3d9ca9fe28c21fcc7088a5780db3107d84c1f4/img/image-20220503194601369.png -------------------------------------------------------------------------------- /leetcode-helper.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name leetcode-helper 3 | // @namespace https://github.com/tonngw 4 | // @version 1.1.2 5 | // @description LeetCode 题解助手 | 复制 LeetCode 题目描述 | 生成当前题目题解模板 6 | // @author tonngw 7 | // @run-at document-end 8 | // @match https://leetcode.cn/problems/* 9 | // @icon  10 | // @require https://cdn.staticfile.org/jquery/3.4.1/jquery.min.js 11 | // @require https://unpkg.com/sweetalert/dist/sweetalert.min.js 12 | // @require https://unpkg.com/turndown/dist/turndown.js 13 | // @grant GM_registerMenuCommand 14 | // @grant GM_setClipboard 15 | // @license MIT 16 | // ==/UserScript== 17 | 18 | (function () { 19 | 'use strict'; 20 | 21 | // 初始化 html to markdown 转换工具 22 | var turndownService = new TurndownService(); 23 | 24 | const window = unsafeWindow; 25 | const description = '.description__2b0C'; 26 | var content = ''; 27 | 28 | // 注入菜单 29 | GM_registerMenuCommand("复制 LeetCode 题目为 Markdown,并存入剪切板", copy); 30 | GM_registerMenuCommand("生成当前题目的题解模板,并存入剪切板", generateSolution); 31 | 32 | // 添加复制按钮 33 | var copyBtn = document.createElement("button"); //创建一个 input 对象(提示框按钮) 34 | copyBtn.id = "copyBtn"; 35 | copyBtn.innerHTML = ' 复制'; 36 | copyBtn.style.width = "50px"; 37 | copyBtn.style.height = "25px"; 38 | copyBtn.style.align = "center"; 39 | copyBtn.style.marginLeft = "10px"; 40 | copyBtn.title = "复制题目为 Markdown 格式"; 41 | 42 | // 添加生成题解按钮 43 | var generateSolutionBtn = document.createElement("button"); // 创建一个input对象(提示框按钮) 44 | generateSolutionBtn.id = "generateSolutionBtn"; 45 | generateSolutionBtn.innerHTML = ' 生成'; 46 | generateSolutionBtn.style.width = "50px"; 47 | generateSolutionBtn.style.height = "25px"; 48 | generateSolutionBtn.style.align = "center"; 49 | generateSolutionBtn.style.marginLeft = "10px"; 50 | generateSolutionBtn.title = "生成 Markdown 格式题解"; 51 | 52 | window.onload = setTimeout(function () { 53 | var x = document.getElementsByClassName("mr-2 text-label-1 dark:text-dark-label-1 text-lg font-medium")[0]; 54 | console.log("I was invoked..."); 55 | x.parentNode.appendChild(copyBtn); 56 | x.parentNode.appendChild(generateSolutionBtn); 57 | }, 1500); 58 | 59 | // 为复制按钮绑定点击功能 60 | copyBtn.onclick = function (e) { 61 | e.preventDefault(); 62 | copy(); 63 | }; 64 | 65 | // 为复制按钮绑定点击功能 66 | generateSolutionBtn.onclick = function (e) { 67 | e.preventDefault(); 68 | generateSolution(); 69 | }; 70 | 71 | // 监听键盘按键,为功能绑定快捷键 72 | unsafeWindow.addEventListener("keydown", (evt) => { 73 | // console.log('evt', evt); 74 | if (evt.altKey) { 75 | // Alt + T 复制题目 76 | if (evt.keyCode == 84) { 77 | copy(); 78 | } 79 | // Alt + C 生成当前题目题解模板 80 | if (evt.keyCode == 67) { 81 | generateSolution(); 82 | } 83 | } 84 | }); 85 | 86 | // 题目复制功能实现 87 | function copy() { 88 | copyImpl(); 89 | GM_setClipboard(content); 90 | swal({ 91 | icon: "success", 92 | title: "复制成功", 93 | }); 94 | } 95 | 96 | function copyImpl() { 97 | // 题目描述 内容 Dom 98 | var contentDom = $('._1l1MA')[0].outerHTML; 99 | console.log(contentDom); 100 | content = handleHtml(contentDom); 101 | } 102 | 103 | // 生成题解功能实现 104 | function generateSolution() { 105 | generateSolutionImpl(); 106 | swal({ 107 | icon: "success", 108 | title: "生成成功", 109 | }); 110 | } 111 | 112 | function generateSolutionImpl() { 113 | var solutionTemplate = ""; 114 | var problemDescConst = "### 题目描述\n"; 115 | copyImpl(); 116 | var problemDesc = content; 117 | var splitLine = "\n\n---\n"; 118 | var algorithmConst = "### 算法\n" 119 | var specificAlgorithmConst = "#### (暴力枚举) $O(n^2)$"; 120 | var solution = "\nwrite here...\n\n" 121 | var timeComplexityConst = "#### 时间复杂度"; 122 | var timeComplexity = "\nwrite here...\n\n" 123 | var spaceComplexityConst = "#### 空间复杂度"; 124 | var spaceComplexity = "\nwrite here...\n\n"; 125 | var codeConst = "#### C++ 代码\n"; 126 | var code = "```\n" + "my code...\n" + "```"; 127 | solutionTemplate = problemDescConst + problemDesc + splitLine + algorithmConst + specificAlgorithmConst + 128 | solution + timeComplexityConst + timeComplexity + spaceComplexityConst + spaceComplexity + codeConst + code; 129 | GM_setClipboard(solutionTemplate); 130 | } 131 | 132 | /** 133 | * html 转 markdown 134 | * @param html 135 | * @returns {void|*} 136 | */ 137 | function handleHtml(html) { 138 | turndownService.addRule('strikethrough', { 139 | filter: ['pre'], 140 | replacement: function (content, node) { 141 | // console.log(node.innerText); 142 | return '\n```\n' + node.innerText.trim() + '\n```\n\n'; 143 | } 144 | }); 145 | turndownService.addRule('strikethrough', { 146 | filter: ['strong'], 147 | replacement: function (content) { 148 | return '**' + content + "**" 149 | } 150 | }); 151 | turndownService.addRule('strikethrough', { 152 | filter: ['code'], 153 | replacement: function (content) { 154 | return '$' + content + "$" 155 | } 156 | }); 157 | turndownService.addRule('strikethrough', { 158 | filter: ['sup'], 159 | replacement: function (content) { 160 | return '^{' + content + "}" 161 | } 162 | }); 163 | 164 | var markdown = turndownService.turndown(html); 165 | return markdown 166 | } 167 | })(); --------------------------------------------------------------------------------