├── 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 | 
41 |
42 | ### 右键菜单
43 |
44 | 右键 - `Tampermonkey`,可以看到 `leetcode-helper` 提供的功能。
45 |
46 | ### 插件栏
47 |
48 | 点击油猴插件,可以看到 `leetcode-helper` 提供的功能。
49 |
50 | ## 实现效果
51 |
52 | 
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 | })();
--------------------------------------------------------------------------------