// ==UserScript== // @name acwing-helper // @namespace https://github.com/tonngw // @version 1.1.6 // @description AcWing 助手,学算法就上 AcWing!| 题目复制 | 生成题解模板 | 切换页面风格 (AcWing <-> LeetCode) | 复制代码 | 题目直接跳转 | 一键填写样例 // @author tonngw // @match https://www.acwing.com/problem/content/*/ // @match https://www.acwing.com/activity/content/*/ // @match https://www.acwing.com/activity/content/punch_the_clock/*/ // @match https://www.acwing.com/activity/content/code/content/*/ // @match https://www.acwing.com/solution/*/ // @match https://www.acwing.com/blog/*/ // @match https://www.acwing.com/community/*/ // @exclude https://www.acwing.com/problem/content/submission/*/ // @exclude https://www.acwing.com/problem/content/discussion/*/ // @exclude https://www.acwing.com/problem/content/solution/*/ // @exclude https://www.acwing.com/problem/content/video/*/ // @icon  // @require https://cdn.staticfile.org/jquery/3.4.1/jquery.min.js // @require https://unpkg.com/sweetalert/dist/sweetalert.min.js // @require https://unpkg.com/turndown/dist/turndown.js // @require https://unpkg.com/turndown-plugin-gfm/dist/turndown-plugin-gfm.js // @require https://cdn.bootcdn.net/ajax/libs/bootstrap-switch/4.0.0-alpha.1/js/bootstrap-switch.min.js // @grant GM_registerMenuCommand // @grant GM_setClipboard // @grant GM_addStyle // @license MIT // @downloadURL none // ==/UserScript== (function () { "use strict"; $("head").append($(``)); // 使用 turndown-plugin-gfm 修复 标签解析错误,https://github.com/mixmark-io/turndown-plugin-gfm var gfm = turndownPluginGfm.gfm; // 初始化 html to markdown 转换工具 var turndownService = new TurndownService(); turndownService.use(gfm); const window = unsafeWindow; const description = ".row"; var content = ""; // 判断路径中是否包含 code, solution, blog, community,代码复制功能只在指定路径下下生效 var url = window.location.href; if (url.includes("code") || url.includes("solution") || url.includes("blog") || url.includes("community")) { // 插入复制代码按钮,并设置位置 $(".hljs").each(function() { $(this).before( "" ); }); // $(".copyCodeBtn").css("position", "absolute"); // $(".copyCodeBtn").css("top", "10px"); // $(".copyCodeBtn").css("right", "10px"); // 去除按钮默认样式 $(".copyCodeBtn").css("border", "none"); $(".copyCodeBtn").css("background-color", "transparent"); $(".copyCodeBtn").css("outline", "none"); turndownService.addRule('strikethrough', { filter: ['pre'], replacement: function (content) { return '' + content.trim() + "" } }); // 为按钮绑定点击事件 $(".copyCodeBtn").click(function() { let target = $(this).next(); // console.log($(target).html()); target.markdown = turndownService.turndown($(target)[0].outerHTML); GM_setClipboard(target.markdown); $(this).text("已复制到剪贴板"); }); return; } // 拦截带有 ? 的路径直接打开题目 if (url.includes("?")) { // alert(url); // alert($(".label-info").get(0).href); location.href = $(".label-info").get(0).href; return; } // 在题目内容页面添加在当前页面打开题目按钮 if (url.includes("activity/content/problem/content")) { var gotoHref = $(".label-info").get(0).href; // console.log(gotoHref); var gotoA = '    '; $(".label-info").after(gotoA); return; } // 获取所有的打卡题目,并添加直达题目按钮 if (url.includes("punch_the_clock") || url.includes("activity")) { $(".punch-line").each(function () { var gotoHref = $(this).children("a").get(0).href + '?'; // console.log(gotoHref); var gotoA = ''; $(this).append(gotoA); }); return; } // 题目页面的相关功能 if (url.includes("problem/content")) { /* 一键填写样例功能 start */ GM_addStyle(` .fillSmapleBtn { background-color: #5cb85c; /* 设置按钮背景颜色 */ border: none; /* 去除边框 */ color: white; /* 设置文字颜色 */ text-align: center; /* 文字居中 */ text-decoration: none; /* 去除下划线 */ display: inline-block; /* 将按钮显示为行内元素 */ font-size: 13px; /* 设置字体大小 */ cursor: pointer; /* 鼠标悬停样式 */ border-radius: 4px; /* 圆角设置 */ transition-duration: 0.4s; /* 过渡动画时间 */ } .fillSmapleBtn:hover { background-color: #3e8e41; /* 设置鼠标悬停时的背景颜色 */ text: 填入样例; } .fillSmapleBtn:hover::after { content: " 填入样例"; /* 悬停时显示的文本 */ } `) // 获取包含样例两个字的

var sz = $(".hljs").length; // 插入填入样例按钮,并设置位置 $('h4:contains("输入样例")').filter(function() { return $(this).text().includes("输入样例"); }).each(function() { $(this).after( "" ); }); $('h4:contains("样例输入")').filter(function() { return $(this).text().includes("样例输入"); }).each(function() { $(this).after( "" ); }); turndownService.addRule('strikethrough', { filter: ['pre'], replacement: function (content) { return '' + content.trim() + "" } }); // Auto resize input textarea height when user clicks fill smaple button. function autoResizeTextarea(textarea) { $(textarea).css('height', 'auto'); // Reset the height $(textarea).css('height', textarea.scrollHeight + 'px'); // Set the new height based on the content } // 为填入样例按钮绑定点击事件 $(".fillSmapleBtn").click(function() { let target = $(this).next(); target.markdown = turndownService.turndown($(target)[0].outerHTML); GM_setClipboard(target.markdown); // $(this).text("已复制到剪贴板"); $("#run-code-stdin").val(target.markdown); autoResizeTextarea($('#run-code-stdin')[0]); // Call the autoResizeTextarea function with the DOM element // 当页面风格为 vertical 时才生效 if (page_style == "vertical") { // 页面滑动到调试按钮位置 $('html, body').animate({ scrollTop: $("#submit_code_btn").offset().top }, 500); } }); /* 一键填写样例功能 end */ // 添加复制按钮 console.log("acwing helper..."); var copyBtn = document.createElement("button"); //创建一个 input 对象(提示框按钮) copyBtn.id = "copyBtn"; copyBtn.textContent = "复制"; copyBtn.style.width = "50px"; copyBtn.style.height = "30px"; copyBtn.style.align = "center"; var x = document.getElementsByClassName("problem-content-sub-btn")[0]; // 在浏览器控制台可以查看所有函数,ctrl+shift+I 调出控制台,在 Console 窗口进行实验测试 x.appendChild(copyBtn); var page_style = "vertical"; // 添加切换按钮 $("#open_ac_saber_btn").after(''); // name 值和 input 标签的 name 值一样 $("[name='switchBtn']").bootstrapSwitch({ onText : "Right", // 设置ON文本 offText : "Bottom  ", // 设置OFF文本 onColor : "success",// 设置ON文本颜色(info/success/warning/danger/primary) offColor : "info", // 设置OFF文本颜色 (info/success/warning/danger/primary) size : "normal", // 设置控件大小,从小到大 (mini/small/normal/large) // 当开关状态改变时触发 onSwitchChange : function(event, state) { if (state == true){ setTimeout(() => window.location.reload(), 100); page_style = "vertical"; } else { page_style = "horizontal"; switchPageStyle(); } } }); $(".bootstrap-switch-on").css("top", "10px"); $(".bootstrap-switch-on").css("left", "10px"); // 添加生成题解按钮 var generateSolutionBtn = document.createElement("button"); // 创建一个input对象(提示框按钮) generateSolutionBtn.id = "generateSolutionBtn"; generateSolutionBtn.textContent = "生成"; generateSolutionBtn.style.width = "50px"; generateSolutionBtn.style.height = "30px"; generateSolutionBtn.style.align = "center"; var y = document.getElementsByClassName("problem-content-sub-btn")[3]; y.appendChild(generateSolutionBtn); // 监听键盘按键,为功能绑定快捷键 unsafeWindow.addEventListener("keydown", (evt) => { // console.log('evt', evt); if (evt.altKey) { // Alt + T 复制题目 if (evt.keyCode == 84) { copy(); } // Alt + S 切换页面风格 if (evt.keyCode == 83) { $("[name='switchBtn']").click(); } // Alt + C 生成当前题目题解模板 if (evt.keyCode == 67) { generateSolution(); } } // F9 调试代码 if (evt.keyCode == 120) { debugCode(); } // F10 提交代码 if (evt.keyCode == 121) { submitCode(); } if (evt.keyCode == 119) { win = window.open(); //打开新的空白窗口 win.document.write ("

这是新打开的窗口

"); //在新窗口中输出提示信息 win.focus (); //让原窗口获取焦点 win.opener.document.write ("

这是原来窗口

"); //在原窗口中输出提示信息 console.log(win.opener == window); //检测window.opener属性值 } }); // 注入右键菜单 GM_registerMenuCommand("复制 AcWing 题目为 Markdown,并存入剪切板", copy); GM_registerMenuCommand("切换页面风格", function(){$("[name='switchBtn']").click()}); GM_registerMenuCommand("生成当前题目的题解模板,并存入剪切板", generateSolution); // 为复制按钮绑定点击功能 copyBtn.onclick = function (e) { e.preventDefault(); copy(); }; // 为复制题解按钮绑定按键点击功能 generateSolutionBtn.onclick = function (e) { e.preventDefault(); generateSolution(); }; } // 题目复制功能实现 function copy() { copyImpl(); swal({ icon: "success", title: "复制成功", }); } function copyImpl() { // 内容 Dom var contentDom = $(".section-martor")[0].outerHTML; // 将题目描述 html 转换为 markdown content = handleHtml(contentDom); var str = content/* + "\n" + "来源:AcWing\n" + "链接:" + window.location.href + "\n" + "著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。"*/; GM_setClipboard(str); } // 切换页面风格功能实现 function switchPageStyle() { $(".col-sm-3").attr("id", "right"); $(".col-sm-9").append($(".table-responsive")); $(".col-sm-9").attr("class", "col-sm-4 col-xs-12"); $(".col-sm-3").attr("class", "col-sm-8"); $(".container").css("width", "1430px"); $("#right").html(""); $("#right").append($("#code_tool_bar")); $("#right").append($("#code_editor")); $("#right").append($("#data-augmentation-div")); $("#right").append($("#submit_code_btn")); $("#right").append($("#run_code_btn")); $("#right").append($("#submit-code-status-block")); $("#right").append($("#run-code-status-block")); } // @Deprecated 复制代码功能实现 function copyCode() { turndownService.addRule('pre', { filter: 'pre', replacement: function (content) { return "\n" + content.trim() + "\n"; } }); let target = $("div[data-tab='preview-tab-content']"); console.log(target.html()); target.markdown = turndownService.turndown($(target).html()); GM_setClipboard(target.markdown); swal({ icon: "success", title: "复制成功", }); } // 生成题解功能实现 function generateSolution() { generateSolutionImpl(); swal({ icon: "success", title: "生成成功", }); } function generateSolutionImpl() { var solutionTemplate = ""; var problemDescConst = "### 题目描述\n"; copyImpl(); var problemDesc = content; var splitLine = "\n\n---\n"; var algorithmConst = "### 算法\n" var specificAlgorithmConst = "#### (暴力枚举) $O(n^2)$"; var solution = "\n\nwrite here...\n\n" var timeComplexityConst = "#### 时间复杂度"; var timeComplexity = "\n\nwrite here...\n\n" var spaceComplexityConst = "#### 空间复杂度"; var spaceComplexity = "\n\nwrite here...\n\n"; var codeConst = "#### C++ 代码\n"; var code = "```\n" + "my code...\n" + "```"; solutionTemplate = problemDescConst + problemDesc + splitLine + algorithmConst + specificAlgorithmConst + solution + timeComplexityConst + timeComplexity + spaceComplexityConst + spaceComplexity + codeConst + code; GM_setClipboard(solutionTemplate); } // 调试代码 function debugCode() { $("#run_code_btn").click(); } // 提交代码 function submitCode() { $("#submit_code_btn").click(); } /** * html 转 markdown * @param html * @returns {void|*} */ function handleHtml(html) { // 处理数学公式 turndownService.addRule("strikethrough", { filter: ["script"], replacement: function (content) { return "$" + content + "$"; }, }); // 跳过 span 标签,不进行处理 turndownService.remove("span"); var markdown = turndownService.turndown(html); return markdown; } })();